xref: /linux/drivers/bcma/host_soc.c (revision 621cde16e49b3ecf7d59a8106a20aaebfb4a59a9)
1ecd177c2SHauke Mehrtens /*
2ecd177c2SHauke Mehrtens  * Broadcom specific AMBA
3ecd177c2SHauke Mehrtens  * System on Chip (SoC) Host
4ecd177c2SHauke Mehrtens  *
5ecd177c2SHauke Mehrtens  * Licensed under the GNU/GPL. See COPYING for details.
6ecd177c2SHauke Mehrtens  */
7ecd177c2SHauke Mehrtens 
8ecd177c2SHauke Mehrtens #include "bcma_private.h"
9ecd177c2SHauke Mehrtens #include "scan.h"
102101e533SHauke Mehrtens #include <linux/slab.h>
112101e533SHauke Mehrtens #include <linux/module.h>
122101e533SHauke Mehrtens #include <linux/of_address.h>
13ecd177c2SHauke Mehrtens #include <linux/bcma/bcma.h>
14ecd177c2SHauke Mehrtens #include <linux/bcma/bcma_soc.h>
15ecd177c2SHauke Mehrtens 
bcma_host_soc_read8(struct bcma_device * core,u16 offset)16ecd177c2SHauke Mehrtens static u8 bcma_host_soc_read8(struct bcma_device *core, u16 offset)
17ecd177c2SHauke Mehrtens {
18ecd177c2SHauke Mehrtens 	return readb(core->io_addr + offset);
19ecd177c2SHauke Mehrtens }
20ecd177c2SHauke Mehrtens 
bcma_host_soc_read16(struct bcma_device * core,u16 offset)21ecd177c2SHauke Mehrtens static u16 bcma_host_soc_read16(struct bcma_device *core, u16 offset)
22ecd177c2SHauke Mehrtens {
23ecd177c2SHauke Mehrtens 	return readw(core->io_addr + offset);
24ecd177c2SHauke Mehrtens }
25ecd177c2SHauke Mehrtens 
bcma_host_soc_read32(struct bcma_device * core,u16 offset)26ecd177c2SHauke Mehrtens static u32 bcma_host_soc_read32(struct bcma_device *core, u16 offset)
27ecd177c2SHauke Mehrtens {
28ecd177c2SHauke Mehrtens 	return readl(core->io_addr + offset);
29ecd177c2SHauke Mehrtens }
30ecd177c2SHauke Mehrtens 
bcma_host_soc_write8(struct bcma_device * core,u16 offset,u8 value)31ecd177c2SHauke Mehrtens static void bcma_host_soc_write8(struct bcma_device *core, u16 offset,
32ecd177c2SHauke Mehrtens 				 u8 value)
33ecd177c2SHauke Mehrtens {
34ecd177c2SHauke Mehrtens 	writeb(value, core->io_addr + offset);
35ecd177c2SHauke Mehrtens }
36ecd177c2SHauke Mehrtens 
bcma_host_soc_write16(struct bcma_device * core,u16 offset,u16 value)37ecd177c2SHauke Mehrtens static void bcma_host_soc_write16(struct bcma_device *core, u16 offset,
38ecd177c2SHauke Mehrtens 				 u16 value)
39ecd177c2SHauke Mehrtens {
40ecd177c2SHauke Mehrtens 	writew(value, core->io_addr + offset);
41ecd177c2SHauke Mehrtens }
42ecd177c2SHauke Mehrtens 
bcma_host_soc_write32(struct bcma_device * core,u16 offset,u32 value)43ecd177c2SHauke Mehrtens static void bcma_host_soc_write32(struct bcma_device *core, u16 offset,
44ecd177c2SHauke Mehrtens 				 u32 value)
45ecd177c2SHauke Mehrtens {
46ecd177c2SHauke Mehrtens 	writel(value, core->io_addr + offset);
47ecd177c2SHauke Mehrtens }
48ecd177c2SHauke Mehrtens 
49ecd177c2SHauke Mehrtens #ifdef CONFIG_BCMA_BLOCKIO
bcma_host_soc_block_read(struct bcma_device * core,void * buffer,size_t count,u16 offset,u8 reg_width)50ecd177c2SHauke Mehrtens static void bcma_host_soc_block_read(struct bcma_device *core, void *buffer,
51ecd177c2SHauke Mehrtens 				     size_t count, u16 offset, u8 reg_width)
52ecd177c2SHauke Mehrtens {
53ecd177c2SHauke Mehrtens 	void __iomem *addr = core->io_addr + offset;
54ecd177c2SHauke Mehrtens 
55ecd177c2SHauke Mehrtens 	switch (reg_width) {
56ecd177c2SHauke Mehrtens 	case sizeof(u8): {
57ecd177c2SHauke Mehrtens 		u8 *buf = buffer;
58ecd177c2SHauke Mehrtens 
59ecd177c2SHauke Mehrtens 		while (count) {
60ecd177c2SHauke Mehrtens 			*buf = __raw_readb(addr);
61ecd177c2SHauke Mehrtens 			buf++;
62ecd177c2SHauke Mehrtens 			count--;
63ecd177c2SHauke Mehrtens 		}
64ecd177c2SHauke Mehrtens 		break;
65ecd177c2SHauke Mehrtens 	}
66ecd177c2SHauke Mehrtens 	case sizeof(u16): {
67ecd177c2SHauke Mehrtens 		__le16 *buf = buffer;
68ecd177c2SHauke Mehrtens 
69ecd177c2SHauke Mehrtens 		WARN_ON(count & 1);
70ecd177c2SHauke Mehrtens 		while (count) {
71ecd177c2SHauke Mehrtens 			*buf = (__force __le16)__raw_readw(addr);
72ecd177c2SHauke Mehrtens 			buf++;
73ecd177c2SHauke Mehrtens 			count -= 2;
74ecd177c2SHauke Mehrtens 		}
75ecd177c2SHauke Mehrtens 		break;
76ecd177c2SHauke Mehrtens 	}
77ecd177c2SHauke Mehrtens 	case sizeof(u32): {
78ecd177c2SHauke Mehrtens 		__le32 *buf = buffer;
79ecd177c2SHauke Mehrtens 
80ecd177c2SHauke Mehrtens 		WARN_ON(count & 3);
81ecd177c2SHauke Mehrtens 		while (count) {
82ecd177c2SHauke Mehrtens 			*buf = (__force __le32)__raw_readl(addr);
83ecd177c2SHauke Mehrtens 			buf++;
84ecd177c2SHauke Mehrtens 			count -= 4;
85ecd177c2SHauke Mehrtens 		}
86ecd177c2SHauke Mehrtens 		break;
87ecd177c2SHauke Mehrtens 	}
88ecd177c2SHauke Mehrtens 	default:
89ecd177c2SHauke Mehrtens 		WARN_ON(1);
90ecd177c2SHauke Mehrtens 	}
91ecd177c2SHauke Mehrtens }
92ecd177c2SHauke Mehrtens 
bcma_host_soc_block_write(struct bcma_device * core,const void * buffer,size_t count,u16 offset,u8 reg_width)93ecd177c2SHauke Mehrtens static void bcma_host_soc_block_write(struct bcma_device *core,
94ecd177c2SHauke Mehrtens 				      const void *buffer,
95ecd177c2SHauke Mehrtens 				      size_t count, u16 offset, u8 reg_width)
96ecd177c2SHauke Mehrtens {
97ecd177c2SHauke Mehrtens 	void __iomem *addr = core->io_addr + offset;
98ecd177c2SHauke Mehrtens 
99ecd177c2SHauke Mehrtens 	switch (reg_width) {
100ecd177c2SHauke Mehrtens 	case sizeof(u8): {
101ecd177c2SHauke Mehrtens 		const u8 *buf = buffer;
102ecd177c2SHauke Mehrtens 
103ecd177c2SHauke Mehrtens 		while (count) {
104ecd177c2SHauke Mehrtens 			__raw_writeb(*buf, addr);
105ecd177c2SHauke Mehrtens 			buf++;
106ecd177c2SHauke Mehrtens 			count--;
107ecd177c2SHauke Mehrtens 		}
108ecd177c2SHauke Mehrtens 		break;
109ecd177c2SHauke Mehrtens 	}
110ecd177c2SHauke Mehrtens 	case sizeof(u16): {
111ecd177c2SHauke Mehrtens 		const __le16 *buf = buffer;
112ecd177c2SHauke Mehrtens 
113ecd177c2SHauke Mehrtens 		WARN_ON(count & 1);
114ecd177c2SHauke Mehrtens 		while (count) {
115ecd177c2SHauke Mehrtens 			__raw_writew((__force u16)(*buf), addr);
116ecd177c2SHauke Mehrtens 			buf++;
117ecd177c2SHauke Mehrtens 			count -= 2;
118ecd177c2SHauke Mehrtens 		}
119ecd177c2SHauke Mehrtens 		break;
120ecd177c2SHauke Mehrtens 	}
121ecd177c2SHauke Mehrtens 	case sizeof(u32): {
122ecd177c2SHauke Mehrtens 		const __le32 *buf = buffer;
123ecd177c2SHauke Mehrtens 
124ecd177c2SHauke Mehrtens 		WARN_ON(count & 3);
125ecd177c2SHauke Mehrtens 		while (count) {
126ecd177c2SHauke Mehrtens 			__raw_writel((__force u32)(*buf), addr);
127ecd177c2SHauke Mehrtens 			buf++;
128ecd177c2SHauke Mehrtens 			count -= 4;
129ecd177c2SHauke Mehrtens 		}
130ecd177c2SHauke Mehrtens 		break;
131ecd177c2SHauke Mehrtens 	}
132ecd177c2SHauke Mehrtens 	default:
133ecd177c2SHauke Mehrtens 		WARN_ON(1);
134ecd177c2SHauke Mehrtens 	}
135ecd177c2SHauke Mehrtens }
136ecd177c2SHauke Mehrtens #endif /* CONFIG_BCMA_BLOCKIO */
137ecd177c2SHauke Mehrtens 
bcma_host_soc_aread32(struct bcma_device * core,u16 offset)138ecd177c2SHauke Mehrtens static u32 bcma_host_soc_aread32(struct bcma_device *core, u16 offset)
139ecd177c2SHauke Mehrtens {
140ecf47e9bSHauke Mehrtens 	if (WARN_ONCE(!core->io_wrap, "Accessed core has no wrapper/agent\n"))
141ecf47e9bSHauke Mehrtens 		return ~0;
142ecd177c2SHauke Mehrtens 	return readl(core->io_wrap + offset);
143ecd177c2SHauke Mehrtens }
144ecd177c2SHauke Mehrtens 
bcma_host_soc_awrite32(struct bcma_device * core,u16 offset,u32 value)145ecd177c2SHauke Mehrtens static void bcma_host_soc_awrite32(struct bcma_device *core, u16 offset,
146ecd177c2SHauke Mehrtens 				  u32 value)
147ecd177c2SHauke Mehrtens {
148ecf47e9bSHauke Mehrtens 	if (WARN_ONCE(!core->io_wrap, "Accessed core has no wrapper/agent\n"))
149ecf47e9bSHauke Mehrtens 		return;
150ecd177c2SHauke Mehrtens 	writel(value, core->io_wrap + offset);
151ecd177c2SHauke Mehrtens }
152ecd177c2SHauke Mehrtens 
15394f3457fSHauke Mehrtens static const struct bcma_host_ops bcma_host_soc_ops = {
154ecd177c2SHauke Mehrtens 	.read8		= bcma_host_soc_read8,
155ecd177c2SHauke Mehrtens 	.read16		= bcma_host_soc_read16,
156ecd177c2SHauke Mehrtens 	.read32		= bcma_host_soc_read32,
157ecd177c2SHauke Mehrtens 	.write8		= bcma_host_soc_write8,
158ecd177c2SHauke Mehrtens 	.write16	= bcma_host_soc_write16,
159ecd177c2SHauke Mehrtens 	.write32	= bcma_host_soc_write32,
160ecd177c2SHauke Mehrtens #ifdef CONFIG_BCMA_BLOCKIO
161ecd177c2SHauke Mehrtens 	.block_read	= bcma_host_soc_block_read,
162ecd177c2SHauke Mehrtens 	.block_write	= bcma_host_soc_block_write,
163ecd177c2SHauke Mehrtens #endif
164ecd177c2SHauke Mehrtens 	.aread32	= bcma_host_soc_aread32,
165ecd177c2SHauke Mehrtens 	.awrite32	= bcma_host_soc_awrite32,
166ecd177c2SHauke Mehrtens };
167ecd177c2SHauke Mehrtens 
bcma_host_soc_register(struct bcma_soc * soc)168ecd177c2SHauke Mehrtens int __init bcma_host_soc_register(struct bcma_soc *soc)
169ecd177c2SHauke Mehrtens {
170ecd177c2SHauke Mehrtens 	struct bcma_bus *bus = &soc->bus;
171ecd177c2SHauke Mehrtens 
172ecd177c2SHauke Mehrtens 	/* iomap only first core. We have to read some register on this core
173ecd177c2SHauke Mehrtens 	 * to scan the bus.
174ecd177c2SHauke Mehrtens 	 */
1754bdc0d67SChristoph Hellwig 	bus->mmio = ioremap(BCMA_ADDR_BASE, BCMA_CORE_SIZE * 1);
176ecd177c2SHauke Mehrtens 	if (!bus->mmio)
177ecd177c2SHauke Mehrtens 		return -ENOMEM;
178ecd177c2SHauke Mehrtens 
179ecd177c2SHauke Mehrtens 	/* Host specific */
180ecd177c2SHauke Mehrtens 	bus->hosttype = BCMA_HOSTTYPE_SOC;
181ecd177c2SHauke Mehrtens 	bus->ops = &bcma_host_soc_ops;
182ecd177c2SHauke Mehrtens 
183dc8ecdd3SRafał Miłecki 	/* Initialize struct, detect chip */
184dc8ecdd3SRafał Miłecki 	bcma_init_bus(bus);
185dc8ecdd3SRafał Miłecki 
186a395135dSRafał Miłecki 	return 0;
187a395135dSRafał Miłecki }
188a395135dSRafał Miłecki 
bcma_host_soc_init(struct bcma_soc * soc)189a395135dSRafał Miłecki int __init bcma_host_soc_init(struct bcma_soc *soc)
190a395135dSRafał Miłecki {
191a395135dSRafał Miłecki 	struct bcma_bus *bus = &soc->bus;
192a395135dSRafał Miłecki 	int err;
193a395135dSRafał Miłecki 
194a395135dSRafał Miłecki 	/* Scan bus and initialize it */
195c5ed1df7SRafał Miłecki 	err = bcma_bus_early_register(bus);
196ecd177c2SHauke Mehrtens 	if (err)
197ecd177c2SHauke Mehrtens 		iounmap(bus->mmio);
198ecd177c2SHauke Mehrtens 
199ecd177c2SHauke Mehrtens 	return err;
200ecd177c2SHauke Mehrtens }
2012101e533SHauke Mehrtens 
2022101e533SHauke Mehrtens #ifdef CONFIG_OF
bcma_host_soc_probe(struct platform_device * pdev)2032101e533SHauke Mehrtens static int bcma_host_soc_probe(struct platform_device *pdev)
2042101e533SHauke Mehrtens {
2052101e533SHauke Mehrtens 	struct device *dev = &pdev->dev;
2062101e533SHauke Mehrtens 	struct device_node *np = dev->of_node;
2072101e533SHauke Mehrtens 	struct bcma_bus *bus;
2082101e533SHauke Mehrtens 	int err;
2092101e533SHauke Mehrtens 
2102101e533SHauke Mehrtens 	/* Alloc */
2112101e533SHauke Mehrtens 	bus = devm_kzalloc(dev, sizeof(*bus), GFP_KERNEL);
2122101e533SHauke Mehrtens 	if (!bus)
2132101e533SHauke Mehrtens 		return -ENOMEM;
2142101e533SHauke Mehrtens 
2155a1c18b7SRafał Miłecki 	bus->dev = dev;
2165a1c18b7SRafał Miłecki 
2172101e533SHauke Mehrtens 	/* Map MMIO */
2182101e533SHauke Mehrtens 	bus->mmio = of_iomap(np, 0);
2192101e533SHauke Mehrtens 	if (!bus->mmio)
2202101e533SHauke Mehrtens 		return -ENOMEM;
2212101e533SHauke Mehrtens 
2222101e533SHauke Mehrtens 	/* Host specific */
2232101e533SHauke Mehrtens 	bus->hosttype = BCMA_HOSTTYPE_SOC;
2242101e533SHauke Mehrtens 	bus->ops = &bcma_host_soc_ops;
2252101e533SHauke Mehrtens 
2262101e533SHauke Mehrtens 	/* Initialize struct, detect chip */
2272101e533SHauke Mehrtens 	bcma_init_bus(bus);
2282101e533SHauke Mehrtens 
2292101e533SHauke Mehrtens 	/* Register */
2302101e533SHauke Mehrtens 	err = bcma_bus_register(bus);
2312101e533SHauke Mehrtens 	if (err)
2322101e533SHauke Mehrtens 		goto err_unmap_mmio;
2332101e533SHauke Mehrtens 
2342101e533SHauke Mehrtens 	platform_set_drvdata(pdev, bus);
2352101e533SHauke Mehrtens 
2362101e533SHauke Mehrtens 	return err;
2372101e533SHauke Mehrtens 
2382101e533SHauke Mehrtens err_unmap_mmio:
2392101e533SHauke Mehrtens 	iounmap(bus->mmio);
2402101e533SHauke Mehrtens 	return err;
2412101e533SHauke Mehrtens }
2422101e533SHauke Mehrtens 
bcma_host_soc_remove(struct platform_device * pdev)243*370e6231SUwe Kleine-König static void bcma_host_soc_remove(struct platform_device *pdev)
2442101e533SHauke Mehrtens {
2452101e533SHauke Mehrtens 	struct bcma_bus *bus = platform_get_drvdata(pdev);
2462101e533SHauke Mehrtens 
2472101e533SHauke Mehrtens 	bcma_bus_unregister(bus);
2482101e533SHauke Mehrtens 	iounmap(bus->mmio);
2492101e533SHauke Mehrtens 	platform_set_drvdata(pdev, NULL);
2502101e533SHauke Mehrtens }
2512101e533SHauke Mehrtens 
2522101e533SHauke Mehrtens static const struct of_device_id bcma_host_soc_of_match[] = {
2532101e533SHauke Mehrtens 	{ .compatible = "brcm,bus-axi", },
2542101e533SHauke Mehrtens 	{},
2552101e533SHauke Mehrtens };
2562101e533SHauke Mehrtens MODULE_DEVICE_TABLE(of, bcma_host_soc_of_match);
2572101e533SHauke Mehrtens 
2582101e533SHauke Mehrtens static struct platform_driver bcma_host_soc_driver = {
2592101e533SHauke Mehrtens 	.driver = {
2602101e533SHauke Mehrtens 		.name = "bcma-host-soc",
2612101e533SHauke Mehrtens 		.of_match_table = bcma_host_soc_of_match,
2622101e533SHauke Mehrtens 	},
2632101e533SHauke Mehrtens 	.probe		= bcma_host_soc_probe,
264*370e6231SUwe Kleine-König 	.remove_new	= bcma_host_soc_remove,
2652101e533SHauke Mehrtens };
2662101e533SHauke Mehrtens 
bcma_host_soc_register_driver(void)2672101e533SHauke Mehrtens int __init bcma_host_soc_register_driver(void)
2682101e533SHauke Mehrtens {
2692101e533SHauke Mehrtens 	return platform_driver_register(&bcma_host_soc_driver);
2702101e533SHauke Mehrtens }
2712101e533SHauke Mehrtens 
bcma_host_soc_unregister_driver(void)2722101e533SHauke Mehrtens void __exit bcma_host_soc_unregister_driver(void)
2732101e533SHauke Mehrtens {
2742101e533SHauke Mehrtens 	platform_driver_unregister(&bcma_host_soc_driver);
2752101e533SHauke Mehrtens }
2762101e533SHauke Mehrtens #endif /* CONFIG_OF */
277