1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (C) 2009 Wolfgang Grandegger <wg@grandegger.com>
6 #include <linux/kernel.h>
7 #include <linux/module.h>
8 #include <linux/platform_device.h>
9 #include <linux/interrupt.h>
10 #include <linux/netdevice.h>
11 #include <linux/delay.h>
12 #include <linux/irq.h>
14 #include <linux/can/dev.h>
15 #include <linux/can/platform/sja1000.h>
19 #define DRV_NAME "sja1000_isa"
23 MODULE_AUTHOR("Wolfgang Grandegger <wg@grandegger.com>");
24 MODULE_DESCRIPTION("Socket-CAN driver for SJA1000 on the ISA bus");
25 MODULE_LICENSE("GPL v2");
27 #define CLK_DEFAULT 16000000 /* 16 MHz */
28 #define CDR_DEFAULT (CDR_CBP | CDR_CLK_OFF)
29 #define OCR_DEFAULT OCR_TX0_PUSHPULL
31 static unsigned long port
[MAXDEV
];
32 static unsigned long mem
[MAXDEV
];
33 static int irq
[MAXDEV
];
34 static int clk
[MAXDEV
];
35 static unsigned char cdr
[MAXDEV
] = {[0 ... (MAXDEV
- 1)] = 0xff};
36 static unsigned char ocr
[MAXDEV
] = {[0 ... (MAXDEV
- 1)] = 0xff};
37 static int indirect
[MAXDEV
] = {[0 ... (MAXDEV
- 1)] = -1};
38 static spinlock_t indirect_lock
[MAXDEV
]; /* lock for indirect access mode */
40 module_param_hw_array(port
, ulong
, ioport
, NULL
, 0444);
41 MODULE_PARM_DESC(port
, "I/O port number");
43 module_param_hw_array(mem
, ulong
, iomem
, NULL
, 0444);
44 MODULE_PARM_DESC(mem
, "I/O memory address");
46 module_param_hw_array(indirect
, int, ioport
, NULL
, 0444);
47 MODULE_PARM_DESC(indirect
, "Indirect access via address and data port");
49 module_param_hw_array(irq
, int, irq
, NULL
, 0444);
50 MODULE_PARM_DESC(irq
, "IRQ number");
52 module_param_array(clk
, int, NULL
, 0444);
53 MODULE_PARM_DESC(clk
, "External oscillator clock frequency "
54 "(default=16000000 [16 MHz])");
56 module_param_array(cdr
, byte
, NULL
, 0444);
57 MODULE_PARM_DESC(cdr
, "Clock divider register "
58 "(default=0x48 [CDR_CBP | CDR_CLK_OFF])");
60 module_param_array(ocr
, byte
, NULL
, 0444);
61 MODULE_PARM_DESC(ocr
, "Output control register "
62 "(default=0x18 [OCR_TX0_PUSHPULL])");
64 #define SJA1000_IOSIZE 0x20
65 #define SJA1000_IOSIZE_INDIRECT 0x02
67 static struct platform_device
*sja1000_isa_devs
[MAXDEV
];
69 static u8
sja1000_isa_mem_read_reg(const struct sja1000_priv
*priv
, int reg
)
71 return readb(priv
->reg_base
+ reg
);
74 static void sja1000_isa_mem_write_reg(const struct sja1000_priv
*priv
,
77 writeb(val
, priv
->reg_base
+ reg
);
80 static u8
sja1000_isa_port_read_reg(const struct sja1000_priv
*priv
, int reg
)
82 return inb((unsigned long)priv
->reg_base
+ reg
);
85 static void sja1000_isa_port_write_reg(const struct sja1000_priv
*priv
,
88 outb(val
, (unsigned long)priv
->reg_base
+ reg
);
91 static u8
sja1000_isa_port_read_reg_indirect(const struct sja1000_priv
*priv
,
94 unsigned long flags
, base
= (unsigned long)priv
->reg_base
;
97 spin_lock_irqsave(&indirect_lock
[priv
->dev
->dev_id
], flags
);
99 readval
= inb(base
+ 1);
100 spin_unlock_irqrestore(&indirect_lock
[priv
->dev
->dev_id
], flags
);
105 static void sja1000_isa_port_write_reg_indirect(const struct sja1000_priv
*priv
,
108 unsigned long flags
, base
= (unsigned long)priv
->reg_base
;
110 spin_lock_irqsave(&indirect_lock
[priv
->dev
->dev_id
], flags
);
113 spin_unlock_irqrestore(&indirect_lock
[priv
->dev
->dev_id
], flags
);
116 static int sja1000_isa_probe(struct platform_device
*pdev
)
118 struct net_device
*dev
;
119 struct sja1000_priv
*priv
;
120 void __iomem
*base
= NULL
;
121 int iosize
= SJA1000_IOSIZE
;
125 dev_dbg(&pdev
->dev
, "probing idx=%d: port=%#lx, mem=%#lx, irq=%d\n",
126 idx
, port
[idx
], mem
[idx
], irq
[idx
]);
129 if (!request_mem_region(mem
[idx
], iosize
, DRV_NAME
)) {
133 base
= ioremap(mem
[idx
], iosize
);
139 if (indirect
[idx
] > 0 ||
140 (indirect
[idx
] == -1 && indirect
[0] > 0))
141 iosize
= SJA1000_IOSIZE_INDIRECT
;
142 if (!request_region(port
[idx
], iosize
, DRV_NAME
)) {
148 dev
= alloc_sja1000dev(0);
153 priv
= netdev_priv(dev
);
156 priv
->irq_flags
= IRQF_SHARED
;
158 priv
->reg_base
= base
;
159 dev
->base_addr
= mem
[idx
];
160 priv
->read_reg
= sja1000_isa_mem_read_reg
;
161 priv
->write_reg
= sja1000_isa_mem_write_reg
;
163 priv
->reg_base
= (void __iomem
*)port
[idx
];
164 dev
->base_addr
= port
[idx
];
166 if (iosize
== SJA1000_IOSIZE_INDIRECT
) {
167 priv
->read_reg
= sja1000_isa_port_read_reg_indirect
;
168 priv
->write_reg
= sja1000_isa_port_write_reg_indirect
;
169 spin_lock_init(&indirect_lock
[idx
]);
171 priv
->read_reg
= sja1000_isa_port_read_reg
;
172 priv
->write_reg
= sja1000_isa_port_write_reg
;
177 priv
->can
.clock
.freq
= clk
[idx
] / 2;
179 priv
->can
.clock
.freq
= clk
[0] / 2;
181 priv
->can
.clock
.freq
= CLK_DEFAULT
/ 2;
183 if (ocr
[idx
] != 0xff)
184 priv
->ocr
= ocr
[idx
];
185 else if (ocr
[0] != 0xff)
188 priv
->ocr
= OCR_DEFAULT
;
190 if (cdr
[idx
] != 0xff)
191 priv
->cdr
= cdr
[idx
];
192 else if (cdr
[0] != 0xff)
195 priv
->cdr
= CDR_DEFAULT
;
197 platform_set_drvdata(pdev
, dev
);
198 SET_NETDEV_DEV(dev
, &pdev
->dev
);
201 err
= register_sja1000dev(dev
);
203 dev_err(&pdev
->dev
, "registering %s failed (err=%d)\n",
208 dev_info(&pdev
->dev
, "%s device registered (reg_base=0x%p, irq=%d)\n",
209 DRV_NAME
, priv
->reg_base
, dev
->irq
);
213 free_sja1000dev(dev
);
219 release_mem_region(mem
[idx
], iosize
);
221 release_region(port
[idx
], iosize
);
226 static void sja1000_isa_remove(struct platform_device
*pdev
)
228 struct net_device
*dev
= platform_get_drvdata(pdev
);
229 struct sja1000_priv
*priv
= netdev_priv(dev
);
232 unregister_sja1000dev(dev
);
235 iounmap(priv
->reg_base
);
236 release_mem_region(mem
[idx
], SJA1000_IOSIZE
);
238 if (priv
->read_reg
== sja1000_isa_port_read_reg_indirect
)
239 release_region(port
[idx
], SJA1000_IOSIZE_INDIRECT
);
241 release_region(port
[idx
], SJA1000_IOSIZE
);
243 free_sja1000dev(dev
);
246 static struct platform_driver sja1000_isa_driver
= {
247 .probe
= sja1000_isa_probe
,
248 .remove
= sja1000_isa_remove
,
254 static int __init
sja1000_isa_init(void)
258 for (idx
= 0; idx
< MAXDEV
; idx
++) {
259 if ((port
[idx
] || mem
[idx
]) && irq
[idx
]) {
260 sja1000_isa_devs
[idx
] =
261 platform_device_alloc(DRV_NAME
, idx
);
262 if (!sja1000_isa_devs
[idx
]) {
264 goto exit_free_devices
;
266 err
= platform_device_add(sja1000_isa_devs
[idx
]);
268 platform_device_put(sja1000_isa_devs
[idx
]);
269 goto exit_free_devices
;
271 pr_debug("%s: platform device %d: port=%#lx, mem=%#lx, "
273 DRV_NAME
, idx
, port
[idx
], mem
[idx
], irq
[idx
]);
274 } else if (idx
== 0 || port
[idx
] || mem
[idx
]) {
275 pr_err("%s: insufficient parameters supplied\n",
278 goto exit_free_devices
;
282 err
= platform_driver_register(&sja1000_isa_driver
);
284 goto exit_free_devices
;
286 pr_info("Legacy %s driver for max. %d devices registered\n",
293 if (sja1000_isa_devs
[idx
])
294 platform_device_unregister(sja1000_isa_devs
[idx
]);
300 static void __exit
sja1000_isa_exit(void)
304 platform_driver_unregister(&sja1000_isa_driver
);
305 for (idx
= 0; idx
< MAXDEV
; idx
++) {
306 if (sja1000_isa_devs
[idx
])
307 platform_device_unregister(sja1000_isa_devs
[idx
]);
311 module_init(sja1000_isa_init
);
312 module_exit(sja1000_isa_exit
);