blob: 784c1c874018a737b23f8c7decd743a748e8edd8 [file] [log] [blame]
David Daney512254b2009-09-16 14:54:18 -07001/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (C) 2004-2009 Cavium Networks
7 * Copyright (C) 2008 Wind River Systems
8 */
9
10#include <linux/init.h>
11#include <linux/irq.h>
12#include <linux/module.h>
13#include <linux/platform_device.h>
14
15#include <asm/octeon/octeon.h>
16#include <asm/octeon/cvmx-rnm-defs.h>
17
18static struct octeon_cf_data octeon_cf_data;
19
20static int __init octeon_cf_device_init(void)
21{
22 union cvmx_mio_boot_reg_cfgx mio_boot_reg_cfg;
23 unsigned long base_ptr, region_base, region_size;
24 struct platform_device *pd;
25 struct resource cf_resources[3];
26 unsigned int num_resources;
27 int i;
28 int ret = 0;
29
30 /* Setup octeon-cf platform device if present. */
31 base_ptr = 0;
32 if (octeon_bootinfo->major_version == 1
33 && octeon_bootinfo->minor_version >= 1) {
34 if (octeon_bootinfo->compact_flash_common_base_addr)
35 base_ptr =
36 octeon_bootinfo->compact_flash_common_base_addr;
37 } else {
38 base_ptr = 0x1d000800;
39 }
40
41 if (!base_ptr)
42 return ret;
43
44 /* Find CS0 region. */
45 for (i = 0; i < 8; i++) {
46 mio_boot_reg_cfg.u64 = cvmx_read_csr(CVMX_MIO_BOOT_REG_CFGX(i));
47 region_base = mio_boot_reg_cfg.s.base << 16;
48 region_size = (mio_boot_reg_cfg.s.size + 1) << 16;
49 if (mio_boot_reg_cfg.s.en && base_ptr >= region_base
50 && base_ptr < region_base + region_size)
51 break;
52 }
53 if (i >= 7) {
54 /* i and i + 1 are CS0 and CS1, both must be less than 8. */
55 goto out;
56 }
57 octeon_cf_data.base_region = i;
58 octeon_cf_data.is16bit = mio_boot_reg_cfg.s.width;
59 octeon_cf_data.base_region_bias = base_ptr - region_base;
60 memset(cf_resources, 0, sizeof(cf_resources));
61 num_resources = 0;
62 cf_resources[num_resources].flags = IORESOURCE_MEM;
63 cf_resources[num_resources].start = region_base;
64 cf_resources[num_resources].end = region_base + region_size - 1;
65 num_resources++;
66
67
68 if (!(base_ptr & 0xfffful)) {
69 /*
70 * Boot loader signals availability of DMA (true_ide
71 * mode) by setting low order bits of base_ptr to
72 * zero.
73 */
74
75 /* Asume that CS1 immediately follows. */
76 mio_boot_reg_cfg.u64 =
77 cvmx_read_csr(CVMX_MIO_BOOT_REG_CFGX(i + 1));
78 region_base = mio_boot_reg_cfg.s.base << 16;
79 region_size = (mio_boot_reg_cfg.s.size + 1) << 16;
80 if (!mio_boot_reg_cfg.s.en)
81 goto out;
82
83 cf_resources[num_resources].flags = IORESOURCE_MEM;
84 cf_resources[num_resources].start = region_base;
85 cf_resources[num_resources].end = region_base + region_size - 1;
86 num_resources++;
87
88 octeon_cf_data.dma_engine = 0;
89 cf_resources[num_resources].flags = IORESOURCE_IRQ;
90 cf_resources[num_resources].start = OCTEON_IRQ_BOOTDMA;
91 cf_resources[num_resources].end = OCTEON_IRQ_BOOTDMA;
92 num_resources++;
93 } else {
94 octeon_cf_data.dma_engine = -1;
95 }
96
97 pd = platform_device_alloc("pata_octeon_cf", -1);
98 if (!pd) {
99 ret = -ENOMEM;
100 goto out;
101 }
102 pd->dev.platform_data = &octeon_cf_data;
103
104 ret = platform_device_add_resources(pd, cf_resources, num_resources);
105 if (ret)
106 goto fail;
107
108 ret = platform_device_add(pd);
109 if (ret)
110 goto fail;
111
112 return ret;
113fail:
114 platform_device_put(pd);
115out:
116 return ret;
117}
118device_initcall(octeon_cf_device_init);
119
120/* Octeon Random Number Generator. */
121static int __init octeon_rng_device_init(void)
122{
123 struct platform_device *pd;
124 int ret = 0;
125
126 struct resource rng_resources[] = {
127 {
128 .flags = IORESOURCE_MEM,
129 .start = XKPHYS_TO_PHYS(CVMX_RNM_CTL_STATUS),
130 .end = XKPHYS_TO_PHYS(CVMX_RNM_CTL_STATUS) + 0xf
131 }, {
132 .flags = IORESOURCE_MEM,
133 .start = cvmx_build_io_address(8, 0),
134 .end = cvmx_build_io_address(8, 0) + 0x7
135 }
136 };
137
138 pd = platform_device_alloc("octeon_rng", -1);
139 if (!pd) {
140 ret = -ENOMEM;
141 goto out;
142 }
143
144 ret = platform_device_add_resources(pd, rng_resources,
145 ARRAY_SIZE(rng_resources));
146 if (ret)
147 goto fail;
148
149 ret = platform_device_add(pd);
150 if (ret)
151 goto fail;
152
153 return ret;
154fail:
155 platform_device_put(pd);
156
157out:
158 return ret;
159}
160device_initcall(octeon_rng_device_init);
161
David Daneyf41c3c12010-01-07 13:23:41 -0800162
163#define OCTEON_I2C_IO_BASE 0x1180000001000ull
164#define OCTEON_I2C_IO_UNIT_OFFSET 0x200
165
166static struct octeon_i2c_data octeon_i2c_data[2];
167
168static int __init octeon_i2c_device_init(void)
169{
170 struct platform_device *pd;
171 int ret = 0;
172 int port, num_ports;
173
174 struct resource i2c_resources[] = {
175 {
176 .flags = IORESOURCE_MEM,
177 }, {
178 .flags = IORESOURCE_IRQ,
179 }
180 };
181
182 if (OCTEON_IS_MODEL(OCTEON_CN56XX) || OCTEON_IS_MODEL(OCTEON_CN52XX))
183 num_ports = 2;
184 else
185 num_ports = 1;
186
187 for (port = 0; port < num_ports; port++) {
188 octeon_i2c_data[port].sys_freq = octeon_get_clock_rate();
189 /*FIXME: should be examined. At the moment is set for 100Khz */
190 octeon_i2c_data[port].i2c_freq = 100000;
191
192 pd = platform_device_alloc("i2c-octeon", port);
193 if (!pd) {
194 ret = -ENOMEM;
195 goto out;
196 }
197
198 pd->dev.platform_data = octeon_i2c_data + port;
199
200 i2c_resources[0].start =
201 OCTEON_I2C_IO_BASE + (port * OCTEON_I2C_IO_UNIT_OFFSET);
202 i2c_resources[0].end = i2c_resources[0].start + 0x1f;
203 switch (port) {
204 case 0:
205 i2c_resources[1].start = OCTEON_IRQ_TWSI;
206 i2c_resources[1].end = OCTEON_IRQ_TWSI;
207 break;
208 case 1:
209 i2c_resources[1].start = OCTEON_IRQ_TWSI2;
210 i2c_resources[1].end = OCTEON_IRQ_TWSI2;
211 break;
212 default:
213 BUG();
214 }
215
216 ret = platform_device_add_resources(pd,
217 i2c_resources,
218 ARRAY_SIZE(i2c_resources));
219 if (ret)
220 goto fail;
221
222 ret = platform_device_add(pd);
223 if (ret)
224 goto fail;
225 }
226 return ret;
227fail:
228 platform_device_put(pd);
229out:
230 return ret;
231}
232device_initcall(octeon_i2c_device_init);
233
David Daney0f7e64a2009-10-14 12:04:37 -0700234/* Octeon SMI/MDIO interface. */
235static int __init octeon_mdiobus_device_init(void)
236{
237 struct platform_device *pd;
238 int ret = 0;
239
240 if (octeon_is_simulation())
241 return 0; /* No mdio in the simulator. */
242
243 /* The bus number is the platform_device id. */
244 pd = platform_device_alloc("mdio-octeon", 0);
245 if (!pd) {
246 ret = -ENOMEM;
247 goto out;
248 }
249
250 ret = platform_device_add(pd);
251 if (ret)
252 goto fail;
253
254 return ret;
255fail:
256 platform_device_put(pd);
257
258out:
259 return ret;
260
261}
262device_initcall(octeon_mdiobus_device_init);
263
David Daney24479d92009-10-14 12:04:39 -0700264/* Octeon mgmt port Ethernet interface. */
265static int __init octeon_mgmt_device_init(void)
266{
267 struct platform_device *pd;
268 int ret = 0;
269 int port, num_ports;
270
271 struct resource mgmt_port_resource = {
272 .flags = IORESOURCE_IRQ,
273 .start = -1,
274 .end = -1
275 };
276
277 if (!OCTEON_IS_MODEL(OCTEON_CN56XX) && !OCTEON_IS_MODEL(OCTEON_CN52XX))
278 return 0;
279
280 if (OCTEON_IS_MODEL(OCTEON_CN56XX))
281 num_ports = 1;
282 else
283 num_ports = 2;
284
285 for (port = 0; port < num_ports; port++) {
286 pd = platform_device_alloc("octeon_mgmt", port);
287 if (!pd) {
288 ret = -ENOMEM;
289 goto out;
290 }
291 switch (port) {
292 case 0:
293 mgmt_port_resource.start = OCTEON_IRQ_MII0;
294 break;
295 case 1:
296 mgmt_port_resource.start = OCTEON_IRQ_MII1;
297 break;
298 default:
299 BUG();
300 }
301 mgmt_port_resource.end = mgmt_port_resource.start;
302
303 ret = platform_device_add_resources(pd, &mgmt_port_resource, 1);
304
305 if (ret)
306 goto fail;
307
308 ret = platform_device_add(pd);
309 if (ret)
310 goto fail;
311 }
312 return ret;
313fail:
314 platform_device_put(pd);
315
316out:
317 return ret;
318
319}
320device_initcall(octeon_mgmt_device_init);
321
David Daney512254b2009-09-16 14:54:18 -0700322MODULE_AUTHOR("David Daney <ddaney@caviumnetworks.com>");
323MODULE_LICENSE("GPL");
324MODULE_DESCRIPTION("Platform driver for Octeon SOC");