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