xref: /freebsd/sys/compat/linuxkpi/common/src/linux_i2c.c (revision 6c05f3a74f30934ee60919cc97e16ec69b542b06)
1 /*-
2  * Copyright (c) 2021 Beckhoff Automation GmbH & Co. KG
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23  * SUCH DAMAGE.
24  *
25  */
26 
27 #include <sys/param.h>
28 #include <sys/systm.h>
29 #include <sys/bus.h>
30 #include <sys/malloc.h>
31 
32 #include <dev/iicbus/iicbus.h>
33 #include <dev/iicbus/iiconf.h>
34 
35 #include <linux/device.h>
36 #include <linux/i2c.h>
37 #include <linux/i2c-algo-bit.h>
38 #include <linux/list.h>
39 #include <linux/pci.h>
40 
41 #include "iicbus_if.h"
42 #include "iicbb_if.h"
43 #include "lkpi_iic_if.h"
44 
45 static int lkpi_i2c_transfer(device_t dev, struct iic_msg *msgs, uint32_t nmsgs);
46 static int lkpi_i2c_reset(device_t dev, u_char speed, u_char addr, u_char *oldaddr);
47 
48 struct lkpi_iic_softc {
49 	device_t		iicbus;
50 	struct i2c_adapter	*adapter;
51 };
52 
53 static struct sx lkpi_sx_i2c;
54 
55 static void
56 lkpi_sysinit_i2c(void *arg __unused)
57 {
58 
59 	sx_init(&lkpi_sx_i2c, "lkpi-i2c");
60 }
61 
62 static void
63 lkpi_sysuninit_i2c(void *arg __unused)
64 {
65 
66 	sx_destroy(&lkpi_sx_i2c);
67 }
68 
69 SYSINIT(lkpi_i2c, SI_SUB_DRIVERS, SI_ORDER_ANY,
70     lkpi_sysinit_i2c, NULL);
71 SYSUNINIT(lkpi_i2c, SI_SUB_DRIVERS, SI_ORDER_ANY,
72     lkpi_sysuninit_i2c, NULL);
73 
74 static int
75 lkpi_iic_probe(device_t dev)
76 {
77 
78 	device_set_desc(dev, "LinuxKPI I2C");
79 	return (BUS_PROBE_NOWILDCARD);
80 }
81 
82 static int
83 lkpi_iic_attach(device_t dev)
84 {
85 	struct lkpi_iic_softc *sc;
86 
87 	sc = device_get_softc(dev);
88 	sc->iicbus = device_add_child(dev, "iicbus", -1);
89 	if (sc->iicbus == NULL) {
90 		device_printf(dev, "Couldn't add iicbus child, aborting\n");
91 		return (ENXIO);
92 	}
93 	bus_attach_children(dev);
94 	return (0);
95 }
96 
97 static int
98 lkpi_iic_detach(device_t dev)
99 {
100 	struct lkpi_iic_softc *sc;
101 
102 	sc = device_get_softc(dev);
103 	if (sc->iicbus)
104 		device_delete_child(dev, sc->iicbus);
105 	return (0);
106 }
107 
108 static int
109 lkpi_iic_add_adapter(device_t dev, struct i2c_adapter *adapter)
110 {
111 	struct lkpi_iic_softc *sc;
112 
113 	sc = device_get_softc(dev);
114 	sc->adapter = adapter;
115 
116 	return (0);
117 }
118 
119 static struct i2c_adapter *
120 lkpi_iic_get_adapter(device_t dev)
121 {
122 	struct lkpi_iic_softc *sc;
123 
124 	sc = device_get_softc(dev);
125 	return (sc->adapter);
126 }
127 
128 static device_method_t lkpi_iic_methods[] = {
129 	/* device interface */
130 	DEVMETHOD(device_probe,		lkpi_iic_probe),
131 	DEVMETHOD(device_attach,	lkpi_iic_attach),
132 	DEVMETHOD(device_detach,	lkpi_iic_detach),
133 	DEVMETHOD(device_suspend,	bus_generic_suspend),
134 	DEVMETHOD(device_resume,	bus_generic_resume),
135 
136 	/* iicbus interface */
137 	DEVMETHOD(iicbus_transfer,	lkpi_i2c_transfer),
138 	DEVMETHOD(iicbus_reset,		lkpi_i2c_reset),
139 	DEVMETHOD(iicbus_callback,	iicbus_null_callback),
140 
141 	/* lkpi_iic interface */
142 	DEVMETHOD(lkpi_iic_add_adapter,	lkpi_iic_add_adapter),
143 	DEVMETHOD(lkpi_iic_get_adapter,	lkpi_iic_get_adapter),
144 
145 	DEVMETHOD_END
146 };
147 
148 driver_t lkpi_iic_driver = {
149 	"lkpi_iic",
150 	lkpi_iic_methods,
151 	sizeof(struct lkpi_iic_softc),
152 };
153 
154 DRIVER_MODULE(lkpi_iic, drmn, lkpi_iic_driver, 0, 0);
155 DRIVER_MODULE(lkpi_iic, drm, lkpi_iic_driver, 0, 0);
156 DRIVER_MODULE(iicbus, lkpi_iic, iicbus_driver, 0, 0);
157 MODULE_DEPEND(linuxkpi, iicbus, IICBUS_MINVER, IICBUS_PREFVER, IICBUS_MAXVER);
158 
159 static int
160 lkpi_i2c_reset(device_t dev, u_char speed, u_char addr, u_char *oldaddr)
161 {
162 
163 	/* That doesn't seems to be supported in linux */
164 	return (0);
165 }
166 
167 static int i2c_check_for_quirks(struct i2c_adapter *adapter,
168     struct iic_msg *msgs, uint32_t nmsgs)
169 {
170 	const struct i2c_adapter_quirks *quirks;
171 	device_t dev;
172 	int i, max_nmsgs;
173 	bool check_len;
174 
175 	dev = adapter->dev.parent->bsddev;
176 	quirks = adapter->quirks;
177 	if (quirks == NULL)
178 		return (0);
179 
180 	check_len = true;
181 	max_nmsgs = quirks->max_num_msgs;
182 
183 	if (quirks->flags & I2C_AQ_COMB) {
184 		max_nmsgs = 2;
185 
186 		if (nmsgs == 2) {
187 			if (quirks->flags & I2C_AQ_COMB_WRITE_FIRST &&
188 			    msgs[0].flags & IIC_M_RD) {
189 				device_printf(dev,
190 				    "Error: "
191 				    "first combined message must be write\n");
192 				return (EOPNOTSUPP);
193 			}
194 			if (quirks->flags & I2C_AQ_COMB_READ_SECOND &&
195 			    !(msgs[1].flags & IIC_M_RD)) {
196 				device_printf(dev,
197 				    "Error: "
198 				    "second combined message must be read\n");
199 				return (EOPNOTSUPP);
200 			}
201 
202 			if (quirks->flags & I2C_AQ_COMB_SAME_ADDR &&
203 			    msgs[0].slave != msgs[1].slave) {
204 				device_printf(dev,
205 				    "Error: "
206 				    "combined message must be use the same "
207 				    "address\n");
208 				return (EOPNOTSUPP);
209 			}
210 
211 			if (quirks->max_comb_1st_msg_len &&
212 			    msgs[0].len > quirks->max_comb_1st_msg_len) {
213 				device_printf(dev,
214 				    "Error: "
215 				    "message too long: %hu > %hu max\n",
216 				    msgs[0].len,
217 				    quirks->max_comb_1st_msg_len);
218 				return (EOPNOTSUPP);
219 			}
220 			if (quirks->max_comb_2nd_msg_len &&
221 			    msgs[1].len > quirks->max_comb_2nd_msg_len) {
222 				device_printf(dev,
223 				    "Error: "
224 				    "message too long: %hu > %hu max\n",
225 				    msgs[1].len,
226 				    quirks->max_comb_2nd_msg_len);
227 				return (EOPNOTSUPP);
228 			}
229 
230 			check_len = false;
231 		}
232 	}
233 
234 	if (max_nmsgs && nmsgs > max_nmsgs) {
235 		device_printf(dev,
236 		    "Error: too many messages: %d > %d max\n",
237 		    nmsgs, max_nmsgs);
238 		return (EOPNOTSUPP);
239 	}
240 
241 	for (i = 0; i < nmsgs; i++) {
242 		if (msgs[i].flags & IIC_M_RD) {
243 			if (check_len && quirks->max_read_len &&
244 			    msgs[i].len > quirks->max_read_len) {
245 				device_printf(dev,
246 				    "Error: "
247 				    "message %d too long: %hu > %hu max\n",
248 				    i, msgs[i].len, quirks->max_read_len);
249 				return (EOPNOTSUPP);
250 			}
251 			if (quirks->flags & I2C_AQ_NO_ZERO_LEN_READ &&
252 			    msgs[i].len == 0) {
253 				device_printf(dev,
254 				    "Error: message %d of length 0\n", i);
255 				return (EOPNOTSUPP);
256 			}
257 		} else {
258 			if (check_len && quirks->max_write_len &&
259 			    msgs[i].len > quirks->max_write_len) {
260 				device_printf(dev,
261 				    "Message %d too long: %hu > %hu max\n",
262 				    i, msgs[i].len, quirks->max_write_len);
263 				return (EOPNOTSUPP);
264 			}
265 			if (quirks->flags & I2C_AQ_NO_ZERO_LEN_WRITE &&
266 			    msgs[i].len == 0) {
267 				device_printf(dev,
268 				    "Error: message %d of length 0\n", i);
269 				return (EOPNOTSUPP);
270 			}
271 		}
272 	}
273 
274 	return (0);
275 }
276 
277 static int
278 lkpi_i2c_transfer(device_t dev, struct iic_msg *msgs, uint32_t nmsgs)
279 {
280 	struct lkpi_iic_softc *sc;
281 	struct i2c_msg *linux_msgs;
282 	int i, ret = 0;
283 
284 	sc = device_get_softc(dev);
285 	if (sc->adapter == NULL)
286 		return (ENXIO);
287 	ret = i2c_check_for_quirks(sc->adapter, msgs, nmsgs);
288 	if (ret != 0)
289 		return (ret);
290 	linux_set_current(curthread);
291 
292 	linux_msgs = malloc(sizeof(struct i2c_msg) * nmsgs,
293 	    M_DEVBUF, M_WAITOK | M_ZERO);
294 
295 	for (i = 0; i < nmsgs; i++) {
296 		linux_msgs[i].addr = msgs[i].slave >> 1;
297 		linux_msgs[i].len = msgs[i].len;
298 		linux_msgs[i].buf = msgs[i].buf;
299 		if (msgs[i].flags & IIC_M_RD) {
300 			linux_msgs[i].flags |= I2C_M_RD;
301 			for (int j = 0; j < msgs[i].len; j++)
302 				msgs[i].buf[j] = 0;
303 		}
304 		if (msgs[i].flags & IIC_M_NOSTART)
305 			linux_msgs[i].flags |= I2C_M_NOSTART;
306 	}
307 	ret = i2c_transfer(sc->adapter, linux_msgs, nmsgs);
308 	free(linux_msgs, M_DEVBUF);
309 
310 	if (ret < 0)
311 		return (-ret);
312 	return (0);
313 }
314 
315 int
316 lkpi_i2c_add_adapter(struct i2c_adapter *adapter)
317 {
318 	device_t lkpi_iic;
319 
320 	if (adapter->name[0] == '\0')
321 		return (-EINVAL);
322 	if (bootverbose)
323 		device_printf(adapter->dev.parent->bsddev,
324 		    "Adding i2c adapter %s\n", adapter->name);
325 	sx_xlock(&lkpi_sx_i2c);
326 	lkpi_iic = device_add_child(adapter->dev.parent->bsddev, "lkpi_iic", -1);
327 	if (lkpi_iic == NULL) {
328 		device_printf(adapter->dev.parent->bsddev, "Couldn't add lkpi_iic\n");
329 		sx_xunlock(&lkpi_sx_i2c);
330 		return (ENXIO);
331 	}
332 
333 	bus_topo_lock();
334 	bus_attach_children(adapter->dev.parent->bsddev);
335 	bus_topo_unlock();
336 	LKPI_IIC_ADD_ADAPTER(lkpi_iic, adapter);
337 	sx_xunlock(&lkpi_sx_i2c);
338 	return (0);
339 }
340 
341 int
342 lkpi_i2c_del_adapter(struct i2c_adapter *adapter)
343 {
344 	device_t child;
345 	int unit, rv;
346 
347 	if (adapter == NULL)
348 		return (-EINVAL);
349 	if (bootverbose)
350 		device_printf(adapter->dev.parent->bsddev,
351 		    "Removing i2c adapter %s\n", adapter->name);
352 	sx_xlock(&lkpi_sx_i2c);
353 	unit = 0;
354 	while ((child = device_find_child(adapter->dev.parent->bsddev, "lkpi_iic", unit++)) != NULL) {
355 
356 		if (adapter == LKPI_IIC_GET_ADAPTER(child)) {
357 			bus_topo_lock();
358 			device_delete_child(adapter->dev.parent->bsddev, child);
359 			bus_topo_unlock();
360 			rv = 0;
361 			goto out;
362 		}
363 	}
364 
365 	unit = 0;
366 	while ((child = device_find_child(adapter->dev.parent->bsddev, "lkpi_iicbb", unit++)) != NULL) {
367 
368 		if (adapter == LKPI_IIC_GET_ADAPTER(child)) {
369 			bus_topo_lock();
370 			device_delete_child(adapter->dev.parent->bsddev, child);
371 			bus_topo_unlock();
372 			rv = 0;
373 			goto out;
374 		}
375 	}
376 	rv = -EINVAL;
377 out:
378 	sx_xunlock(&lkpi_sx_i2c);
379 	return (rv);
380 }
381