1 /*- 2 * Copyright (c) 1998, 2001 Nicolas Souchu 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24 * SUCH DAMAGE. 25 * 26 * $FreeBSD$ 27 * 28 */ 29 #include <sys/param.h> 30 #include <sys/kernel.h> 31 #include <sys/systm.h> 32 #include <sys/module.h> 33 #include <sys/bus.h> 34 #include <sys/conf.h> 35 #include <sys/uio.h> 36 #include <sys/fcntl.h> 37 38 39 #include <dev/iicbus/iiconf.h> 40 #include <dev/iicbus/iicbus.h> 41 42 #include <machine/iic.h> 43 44 #include "iicbus_if.h" 45 46 #define BUFSIZE 1024 47 48 struct iic_softc { 49 50 u_char sc_addr; /* 7 bit address on iicbus */ 51 int sc_count; /* >0 if device opened */ 52 53 char sc_buffer[BUFSIZE]; /* output buffer */ 54 char sc_inbuf[BUFSIZE]; /* input buffer */ 55 56 dev_t sc_devnode; 57 }; 58 59 #define IIC_SOFTC(unit) \ 60 ((struct iic_softc *)devclass_get_softc(iic_devclass, (unit))) 61 62 #define IIC_DEVICE(unit) \ 63 (devclass_get_device(iic_devclass, (unit))) 64 65 static int iic_probe(device_t); 66 static int iic_attach(device_t); 67 static int iic_detach(device_t); 68 static void iic_identify(driver_t *driver, device_t parent); 69 70 static devclass_t iic_devclass; 71 72 static device_method_t iic_methods[] = { 73 /* device interface */ 74 DEVMETHOD(device_identify, iic_identify), 75 DEVMETHOD(device_probe, iic_probe), 76 DEVMETHOD(device_attach, iic_attach), 77 DEVMETHOD(device_detach, iic_detach), 78 79 /* iicbus interface */ 80 DEVMETHOD(iicbus_intr, iicbus_generic_intr), 81 82 { 0, 0 } 83 }; 84 85 static driver_t iic_driver = { 86 "iic", 87 iic_methods, 88 sizeof(struct iic_softc), 89 }; 90 91 static d_open_t iicopen; 92 static d_close_t iicclose; 93 static d_write_t iicwrite; 94 static d_read_t iicread; 95 static d_ioctl_t iicioctl; 96 97 #define CDEV_MAJOR 105 98 static struct cdevsw iic_cdevsw = { 99 /* open */ iicopen, 100 /* close */ iicclose, 101 /* read */ iicread, 102 /* write */ iicwrite, 103 /* ioctl */ iicioctl, 104 /* poll */ nopoll, 105 /* mmap */ nommap, 106 /* strategy */ nostrategy, 107 /* name */ "iic", 108 /* maj */ CDEV_MAJOR, 109 /* dump */ nodump, 110 /* psize */ nopsize, 111 /* flags */ 0, 112 }; 113 114 static void 115 iic_identify(driver_t *driver, device_t parent) 116 { 117 BUS_ADD_CHILD(parent, 0, "iic", 0); 118 } 119 120 static int 121 iic_probe(device_t dev) 122 { 123 device_set_desc(dev, "I2C generic I/O"); 124 return (0); 125 } 126 127 static int 128 iic_attach(device_t dev) 129 { 130 struct iic_softc *sc = (struct iic_softc *)device_get_softc(dev); 131 132 if (!sc) 133 return (ENOMEM); 134 135 bzero(sc, sizeof(struct iic_softc)); 136 137 sc->sc_devnode = make_dev(&iic_cdevsw, device_get_unit(dev), 138 UID_ROOT, GID_WHEEL, 139 0600, "iic%d", device_get_unit(dev)); 140 141 return (0); 142 } 143 144 static int 145 iic_detach(device_t dev) 146 { 147 struct iic_softc *sc = (struct iic_softc *)device_get_softc(dev); 148 149 if (sc->sc_devnode) 150 destroy_dev(sc->sc_devnode); 151 152 return (0); 153 } 154 155 static int 156 iicopen (dev_t dev, int flags, int fmt, struct thread *td) 157 { 158 struct iic_softc *sc = IIC_SOFTC(minor(dev)); 159 160 if (!sc) 161 return (EINVAL); 162 163 if (sc->sc_count > 0) 164 return (EBUSY); 165 166 sc->sc_count++; 167 168 return (0); 169 } 170 171 static int 172 iicclose(dev_t dev, int flags, int fmt, struct thread *td) 173 { 174 struct iic_softc *sc = IIC_SOFTC(minor(dev)); 175 176 if (!sc) 177 return (EINVAL); 178 179 if (!sc->sc_count) 180 return (EINVAL); 181 182 sc->sc_count--; 183 184 if (sc->sc_count < 0) 185 panic("%s: iic_count < 0!", __func__); 186 187 return (0); 188 } 189 190 static int 191 iicwrite(dev_t dev, struct uio * uio, int ioflag) 192 { 193 device_t iicdev = IIC_DEVICE(minor(dev)); 194 struct iic_softc *sc = IIC_SOFTC(minor(dev)); 195 int sent, error, count; 196 197 if (!sc || !iicdev || !sc->sc_addr) 198 return (EINVAL); 199 200 if (sc->sc_count == 0) 201 return (EINVAL); 202 203 if ((error = iicbus_request_bus(device_get_parent(iicdev), iicdev, IIC_DONTWAIT))) 204 return (error); 205 206 count = min(uio->uio_resid, BUFSIZE); 207 uiomove(sc->sc_buffer, count, uio); 208 209 error = iicbus_block_write(device_get_parent(iicdev), sc->sc_addr, 210 sc->sc_buffer, count, &sent); 211 212 iicbus_release_bus(device_get_parent(iicdev), iicdev); 213 214 return(error); 215 } 216 217 static int 218 iicread(dev_t dev, struct uio * uio, int ioflag) 219 { 220 device_t iicdev = IIC_DEVICE(minor(dev)); 221 struct iic_softc *sc = IIC_SOFTC(minor(dev)); 222 int len, error = 0; 223 int bufsize; 224 225 if (!sc || !iicdev || !sc->sc_addr) 226 return (EINVAL); 227 228 if (sc->sc_count == 0) 229 return (EINVAL); 230 231 if ((error = iicbus_request_bus(device_get_parent(iicdev), iicdev, IIC_DONTWAIT))) 232 return (error); 233 234 /* max amount of data to read */ 235 len = min(uio->uio_resid, BUFSIZE); 236 237 if ((error = iicbus_block_read(device_get_parent(iicdev), sc->sc_addr, 238 sc->sc_inbuf, len, &bufsize))) 239 return (error); 240 241 if (bufsize > uio->uio_resid) 242 panic("%s: too much data read!", __func__); 243 244 iicbus_release_bus(device_get_parent(iicdev), iicdev); 245 246 return (uiomove(sc->sc_inbuf, bufsize, uio)); 247 } 248 249 static int 250 iicioctl(dev_t dev, u_long cmd, caddr_t data, int flags, struct thread *td) 251 { 252 device_t iicdev = IIC_DEVICE(minor(dev)); 253 struct iic_softc *sc = IIC_SOFTC(minor(dev)); 254 device_t parent = device_get_parent(iicdev); 255 struct iiccmd *s = (struct iiccmd *)data; 256 int error, count; 257 258 if (!sc) 259 return (EINVAL); 260 261 if ((error = iicbus_request_bus(device_get_parent(iicdev), iicdev, 262 (flags & O_NONBLOCK) ? IIC_DONTWAIT : 263 (IIC_WAIT | IIC_INTR)))) 264 return (error); 265 266 switch (cmd) { 267 case I2CSTART: 268 error = iicbus_start(parent, s->slave, 0); 269 270 /* 271 * Implicitly set the chip addr to the slave addr passed as 272 * parameter. Consequently, start/stop shall be called before 273 * the read or the write of a block. 274 */ 275 if (!error) 276 sc->sc_addr = s->slave; 277 278 break; 279 280 case I2CSTOP: 281 error = iicbus_stop(parent); 282 break; 283 284 case I2CRSTCARD: 285 error = iicbus_reset(parent, 0, 0, NULL); 286 break; 287 288 case I2CWRITE: 289 error = iicbus_write(parent, s->buf, s->count, &count, 10); 290 break; 291 292 case I2CREAD: 293 error = iicbus_read(parent, s->buf, s->count, &count, s->last, 10); 294 break; 295 296 default: 297 error = ENODEV; 298 } 299 300 iicbus_release_bus(device_get_parent(iicdev), iicdev); 301 302 return (error); 303 } 304 305 DRIVER_MODULE(iic, iicbus, iic_driver, iic_devclass, 0, 0); 306 MODULE_DEPEND(iic, iicbus, IICBUS_MINVER, IICBUS_PREFVER, IICBUS_MAXVER); 307 MODULE_VERSION(iic, 1); 308