1c6202d85SSimon Glass /* 2c6202d85SSimon Glass * Copyright (c) 2014 Google, Inc 3c6202d85SSimon Glass * 4c6202d85SSimon Glass * SPDX-License-Identifier: GPL-2.0+ 5c6202d85SSimon Glass */ 6c6202d85SSimon Glass 7c6202d85SSimon Glass #include <common.h> 8c6202d85SSimon Glass #include <dm.h> 9c6202d85SSimon Glass #include <errno.h> 10c6202d85SSimon Glass #include <fdtdec.h> 11c6202d85SSimon Glass #include <i2c.h> 12c6202d85SSimon Glass #include <malloc.h> 13c6202d85SSimon Glass #include <dm/device-internal.h> 14c6202d85SSimon Glass #include <dm/lists.h> 15c6202d85SSimon Glass 16c6202d85SSimon Glass DECLARE_GLOBAL_DATA_PTR; 17c6202d85SSimon Glass 18c6202d85SSimon Glass #define I2C_MAX_OFFSET_LEN 4 19c6202d85SSimon Glass 207d7db222SSimon Glass /* Useful debugging function */ 217d7db222SSimon Glass void i2c_dump_msgs(struct i2c_msg *msg, int nmsgs) 227d7db222SSimon Glass { 237d7db222SSimon Glass int i; 247d7db222SSimon Glass 257d7db222SSimon Glass for (i = 0; i < nmsgs; i++) { 267d7db222SSimon Glass struct i2c_msg *m = &msg[i]; 277d7db222SSimon Glass 287d7db222SSimon Glass printf(" %s %x len=%x", m->flags & I2C_M_RD ? "R" : "W", 297d7db222SSimon Glass msg->addr, msg->len); 307d7db222SSimon Glass if (!(m->flags & I2C_M_RD)) 317d7db222SSimon Glass printf(": %x", m->buf[0]); 327d7db222SSimon Glass printf("\n"); 337d7db222SSimon Glass } 347d7db222SSimon Glass } 357d7db222SSimon Glass 36c6202d85SSimon Glass /** 37c6202d85SSimon Glass * i2c_setup_offset() - Set up a new message with a chip offset 38c6202d85SSimon Glass * 39c6202d85SSimon Glass * @chip: Chip to use 40c6202d85SSimon Glass * @offset: Byte offset within chip 41c6202d85SSimon Glass * @offset_buf: Place to put byte offset 42c6202d85SSimon Glass * @msg: Message buffer 43c6202d85SSimon Glass * @return 0 if OK, -EADDRNOTAVAIL if the offset length is 0. In that case the 44c6202d85SSimon Glass * message is still set up but will not contain an offset. 45c6202d85SSimon Glass */ 46c6202d85SSimon Glass static int i2c_setup_offset(struct dm_i2c_chip *chip, uint offset, 47c6202d85SSimon Glass uint8_t offset_buf[], struct i2c_msg *msg) 48c6202d85SSimon Glass { 49c6202d85SSimon Glass int offset_len; 50c6202d85SSimon Glass 51c6202d85SSimon Glass msg->addr = chip->chip_addr; 52c6202d85SSimon Glass msg->flags = chip->flags & DM_I2C_CHIP_10BIT ? I2C_M_TEN : 0; 53c6202d85SSimon Glass msg->len = chip->offset_len; 54c6202d85SSimon Glass msg->buf = offset_buf; 55c6202d85SSimon Glass if (!chip->offset_len) 56c6202d85SSimon Glass return -EADDRNOTAVAIL; 57c6202d85SSimon Glass assert(chip->offset_len <= I2C_MAX_OFFSET_LEN); 58c6202d85SSimon Glass offset_len = chip->offset_len; 59c6202d85SSimon Glass while (offset_len--) 60c6202d85SSimon Glass *offset_buf++ = offset >> (8 * offset_len); 61c6202d85SSimon Glass 62c6202d85SSimon Glass return 0; 63c6202d85SSimon Glass } 64c6202d85SSimon Glass 65c6202d85SSimon Glass static int i2c_read_bytewise(struct udevice *dev, uint offset, 66c6202d85SSimon Glass uint8_t *buffer, int len) 67c6202d85SSimon Glass { 68e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 69c6202d85SSimon Glass struct udevice *bus = dev_get_parent(dev); 70c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 71c6202d85SSimon Glass struct i2c_msg msg[2], *ptr; 72c6202d85SSimon Glass uint8_t offset_buf[I2C_MAX_OFFSET_LEN]; 73c6202d85SSimon Glass int ret; 74c6202d85SSimon Glass int i; 75c6202d85SSimon Glass 76c6202d85SSimon Glass for (i = 0; i < len; i++) { 77c6202d85SSimon Glass if (i2c_setup_offset(chip, offset + i, offset_buf, msg)) 78c6202d85SSimon Glass return -EINVAL; 79c6202d85SSimon Glass ptr = msg + 1; 80c6202d85SSimon Glass ptr->addr = chip->chip_addr; 81c6202d85SSimon Glass ptr->flags = msg->flags | I2C_M_RD; 82c6202d85SSimon Glass ptr->len = 1; 83c6202d85SSimon Glass ptr->buf = &buffer[i]; 84c6202d85SSimon Glass ptr++; 85c6202d85SSimon Glass 86c6202d85SSimon Glass ret = ops->xfer(bus, msg, ptr - msg); 87c6202d85SSimon Glass if (ret) 88c6202d85SSimon Glass return ret; 89c6202d85SSimon Glass } 90c6202d85SSimon Glass 91c6202d85SSimon Glass return 0; 92c6202d85SSimon Glass } 93c6202d85SSimon Glass 94c6202d85SSimon Glass static int i2c_write_bytewise(struct udevice *dev, uint offset, 95c6202d85SSimon Glass const uint8_t *buffer, int len) 96c6202d85SSimon Glass { 97e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 98c6202d85SSimon Glass struct udevice *bus = dev_get_parent(dev); 99c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 100c6202d85SSimon Glass struct i2c_msg msg[1]; 101c6202d85SSimon Glass uint8_t buf[I2C_MAX_OFFSET_LEN + 1]; 102c6202d85SSimon Glass int ret; 103c6202d85SSimon Glass int i; 104c6202d85SSimon Glass 105c6202d85SSimon Glass for (i = 0; i < len; i++) { 106c6202d85SSimon Glass if (i2c_setup_offset(chip, offset + i, buf, msg)) 107c6202d85SSimon Glass return -EINVAL; 108c6202d85SSimon Glass buf[msg->len++] = buffer[i]; 109c6202d85SSimon Glass 110c6202d85SSimon Glass ret = ops->xfer(bus, msg, 1); 111c6202d85SSimon Glass if (ret) 112c6202d85SSimon Glass return ret; 113c6202d85SSimon Glass } 114c6202d85SSimon Glass 115c6202d85SSimon Glass return 0; 116c6202d85SSimon Glass } 117c6202d85SSimon Glass 118f9a4c2daSSimon Glass int dm_i2c_read(struct udevice *dev, uint offset, uint8_t *buffer, int len) 119c6202d85SSimon Glass { 120e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 121c6202d85SSimon Glass struct udevice *bus = dev_get_parent(dev); 122c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 123c6202d85SSimon Glass struct i2c_msg msg[2], *ptr; 124c6202d85SSimon Glass uint8_t offset_buf[I2C_MAX_OFFSET_LEN]; 125c6202d85SSimon Glass int msg_count; 126c6202d85SSimon Glass 127c6202d85SSimon Glass if (!ops->xfer) 128c6202d85SSimon Glass return -ENOSYS; 129c6202d85SSimon Glass if (chip->flags & DM_I2C_CHIP_RD_ADDRESS) 130c6202d85SSimon Glass return i2c_read_bytewise(dev, offset, buffer, len); 131c6202d85SSimon Glass ptr = msg; 132c6202d85SSimon Glass if (!i2c_setup_offset(chip, offset, offset_buf, ptr)) 133c6202d85SSimon Glass ptr++; 134c6202d85SSimon Glass 135c6202d85SSimon Glass if (len) { 136c6202d85SSimon Glass ptr->addr = chip->chip_addr; 137c6202d85SSimon Glass ptr->flags = chip->flags & DM_I2C_CHIP_10BIT ? I2C_M_TEN : 0; 138c6202d85SSimon Glass ptr->flags |= I2C_M_RD; 139c6202d85SSimon Glass ptr->len = len; 140c6202d85SSimon Glass ptr->buf = buffer; 141c6202d85SSimon Glass ptr++; 142c6202d85SSimon Glass } 143c6202d85SSimon Glass msg_count = ptr - msg; 144c6202d85SSimon Glass 145c6202d85SSimon Glass return ops->xfer(bus, msg, msg_count); 146c6202d85SSimon Glass } 147c6202d85SSimon Glass 148f9a4c2daSSimon Glass int dm_i2c_write(struct udevice *dev, uint offset, const uint8_t *buffer, 149f9a4c2daSSimon Glass int len) 150c6202d85SSimon Glass { 151e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 152c6202d85SSimon Glass struct udevice *bus = dev_get_parent(dev); 153c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 154c6202d85SSimon Glass struct i2c_msg msg[1]; 155c6202d85SSimon Glass 156c6202d85SSimon Glass if (!ops->xfer) 157c6202d85SSimon Glass return -ENOSYS; 158c6202d85SSimon Glass 159c6202d85SSimon Glass if (chip->flags & DM_I2C_CHIP_WR_ADDRESS) 160c6202d85SSimon Glass return i2c_write_bytewise(dev, offset, buffer, len); 161c6202d85SSimon Glass /* 162c6202d85SSimon Glass * The simple approach would be to send two messages here: one to 163c6202d85SSimon Glass * set the offset and one to write the bytes. However some drivers 164c6202d85SSimon Glass * will not be expecting this, and some chips won't like how the 165c6202d85SSimon Glass * driver presents this on the I2C bus. 166c6202d85SSimon Glass * 167c6202d85SSimon Glass * The API does not support separate offset and data. We could extend 168c6202d85SSimon Glass * it with a flag indicating that there is data in the next message 169c6202d85SSimon Glass * that needs to be processed in the same transaction. We could 170c6202d85SSimon Glass * instead add an additional buffer to each message. For now, handle 171c6202d85SSimon Glass * this in the uclass since it isn't clear what the impact on drivers 172c6202d85SSimon Glass * would be with this extra complication. Unfortunately this means 173c6202d85SSimon Glass * copying the message. 174c6202d85SSimon Glass * 175c6202d85SSimon Glass * Use the stack for small messages, malloc() for larger ones. We 176c6202d85SSimon Glass * need to allow space for the offset (up to 4 bytes) and the message 177c6202d85SSimon Glass * itself. 178c6202d85SSimon Glass */ 179c6202d85SSimon Glass if (len < 64) { 180c6202d85SSimon Glass uint8_t buf[I2C_MAX_OFFSET_LEN + len]; 181c6202d85SSimon Glass 182c6202d85SSimon Glass i2c_setup_offset(chip, offset, buf, msg); 183c6202d85SSimon Glass msg->len += len; 184c6202d85SSimon Glass memcpy(buf + chip->offset_len, buffer, len); 185c6202d85SSimon Glass 186c6202d85SSimon Glass return ops->xfer(bus, msg, 1); 187c6202d85SSimon Glass } else { 188c6202d85SSimon Glass uint8_t *buf; 189c6202d85SSimon Glass int ret; 190c6202d85SSimon Glass 191c6202d85SSimon Glass buf = malloc(I2C_MAX_OFFSET_LEN + len); 192c6202d85SSimon Glass if (!buf) 193c6202d85SSimon Glass return -ENOMEM; 194c6202d85SSimon Glass i2c_setup_offset(chip, offset, buf, msg); 195c6202d85SSimon Glass msg->len += len; 196c6202d85SSimon Glass memcpy(buf + chip->offset_len, buffer, len); 197c6202d85SSimon Glass 198c6202d85SSimon Glass ret = ops->xfer(bus, msg, 1); 199c6202d85SSimon Glass free(buf); 200c6202d85SSimon Glass return ret; 201c6202d85SSimon Glass } 202c6202d85SSimon Glass } 203c6202d85SSimon Glass 204df358c6bSSimon Glass int dm_i2c_xfer(struct udevice *dev, struct i2c_msg *msg, int nmsgs) 205df358c6bSSimon Glass { 206df358c6bSSimon Glass struct udevice *bus = dev_get_parent(dev); 207df358c6bSSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 208df358c6bSSimon Glass 209df358c6bSSimon Glass if (!ops->xfer) 210df358c6bSSimon Glass return -ENOSYS; 211df358c6bSSimon Glass 212df358c6bSSimon Glass return ops->xfer(bus, msg, nmsgs); 213df358c6bSSimon Glass } 214df358c6bSSimon Glass 215ba3864f8SSimon Glass int dm_i2c_reg_read(struct udevice *dev, uint offset) 216ba3864f8SSimon Glass { 217ba3864f8SSimon Glass uint8_t val; 218ba3864f8SSimon Glass int ret; 219ba3864f8SSimon Glass 220ba3864f8SSimon Glass ret = dm_i2c_read(dev, offset, &val, 1); 221ba3864f8SSimon Glass if (ret < 0) 222ba3864f8SSimon Glass return ret; 223ba3864f8SSimon Glass 224ba3864f8SSimon Glass return val; 225ba3864f8SSimon Glass } 226ba3864f8SSimon Glass 227ba3864f8SSimon Glass int dm_i2c_reg_write(struct udevice *dev, uint offset, uint value) 228ba3864f8SSimon Glass { 229ba3864f8SSimon Glass uint8_t val = value; 230ba3864f8SSimon Glass 231ba3864f8SSimon Glass return dm_i2c_write(dev, offset, &val, 1); 232ba3864f8SSimon Glass } 233ba3864f8SSimon Glass 234c6202d85SSimon Glass /** 235c6202d85SSimon Glass * i2c_probe_chip() - probe for a chip on a bus 236c6202d85SSimon Glass * 237c6202d85SSimon Glass * @bus: Bus to probe 238c6202d85SSimon Glass * @chip_addr: Chip address to probe 239c6202d85SSimon Glass * @flags: Flags for the chip 240c6202d85SSimon Glass * @return 0 if found, -ENOSYS if the driver is invalid, -EREMOTEIO if the chip 241c6202d85SSimon Glass * does not respond to probe 242c6202d85SSimon Glass */ 243c6202d85SSimon Glass static int i2c_probe_chip(struct udevice *bus, uint chip_addr, 244c6202d85SSimon Glass enum dm_i2c_chip_flags chip_flags) 245c6202d85SSimon Glass { 246c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 247c6202d85SSimon Glass struct i2c_msg msg[1]; 248c6202d85SSimon Glass int ret; 249c6202d85SSimon Glass 250c6202d85SSimon Glass if (ops->probe_chip) { 251c6202d85SSimon Glass ret = ops->probe_chip(bus, chip_addr, chip_flags); 252c6202d85SSimon Glass if (!ret || ret != -ENOSYS) 253c6202d85SSimon Glass return ret; 254c6202d85SSimon Glass } 255c6202d85SSimon Glass 256c6202d85SSimon Glass if (!ops->xfer) 257c6202d85SSimon Glass return -ENOSYS; 258c6202d85SSimon Glass 259c6202d85SSimon Glass /* Probe with a zero-length message */ 260c6202d85SSimon Glass msg->addr = chip_addr; 261c6202d85SSimon Glass msg->flags = chip_flags & DM_I2C_CHIP_10BIT ? I2C_M_TEN : 0; 262c6202d85SSimon Glass msg->len = 0; 263c6202d85SSimon Glass msg->buf = NULL; 264c6202d85SSimon Glass 265c6202d85SSimon Glass return ops->xfer(bus, msg, 1); 266c6202d85SSimon Glass } 267c6202d85SSimon Glass 26825ab4b03SSimon Glass static int i2c_bind_driver(struct udevice *bus, uint chip_addr, uint offset_len, 269c6202d85SSimon Glass struct udevice **devp) 270c6202d85SSimon Glass { 271e6f66ec0SSimon Glass struct dm_i2c_chip *chip; 272c6202d85SSimon Glass char name[30], *str; 273c6202d85SSimon Glass struct udevice *dev; 274c6202d85SSimon Glass int ret; 275c6202d85SSimon Glass 276c6202d85SSimon Glass snprintf(name, sizeof(name), "generic_%x", chip_addr); 277c6202d85SSimon Glass str = strdup(name); 2785c2d23bfSSimon Glass if (!str) 2795c2d23bfSSimon Glass return -ENOMEM; 280c6202d85SSimon Glass ret = device_bind_driver(bus, "i2c_generic_chip_drv", str, &dev); 281c6202d85SSimon Glass debug("%s: device_bind_driver: ret=%d\n", __func__, ret); 282c6202d85SSimon Glass if (ret) 283c6202d85SSimon Glass goto err_bind; 284c6202d85SSimon Glass 285c6202d85SSimon Glass /* Tell the device what we know about it */ 286e6f66ec0SSimon Glass chip = dev_get_parent_platdata(dev); 287e6f66ec0SSimon Glass chip->chip_addr = chip_addr; 288e6f66ec0SSimon Glass chip->offset_len = offset_len; 289e6f66ec0SSimon Glass ret = device_probe(dev); 290e6f66ec0SSimon Glass debug("%s: device_probe: ret=%d\n", __func__, ret); 291c6202d85SSimon Glass if (ret) 292c6202d85SSimon Glass goto err_probe; 293c6202d85SSimon Glass 294c6202d85SSimon Glass *devp = dev; 295c6202d85SSimon Glass return 0; 296c6202d85SSimon Glass 297c6202d85SSimon Glass err_probe: 298e6f66ec0SSimon Glass /* 299e6f66ec0SSimon Glass * If the device failed to probe, unbind it. There is nothing there 300e6f66ec0SSimon Glass * on the bus so we don't want to leave it lying around 301e6f66ec0SSimon Glass */ 302c6202d85SSimon Glass device_unbind(dev); 303c6202d85SSimon Glass err_bind: 304c6202d85SSimon Glass free(str); 305c6202d85SSimon Glass return ret; 306c6202d85SSimon Glass } 307c6202d85SSimon Glass 30825ab4b03SSimon Glass int i2c_get_chip(struct udevice *bus, uint chip_addr, uint offset_len, 30925ab4b03SSimon Glass struct udevice **devp) 310c6202d85SSimon Glass { 311c6202d85SSimon Glass struct udevice *dev; 312c6202d85SSimon Glass 313c6202d85SSimon Glass debug("%s: Searching bus '%s' for address %02x: ", __func__, 314c6202d85SSimon Glass bus->name, chip_addr); 315c6202d85SSimon Glass for (device_find_first_child(bus, &dev); dev; 316c6202d85SSimon Glass device_find_next_child(&dev)) { 317e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 318c6202d85SSimon Glass int ret; 319c6202d85SSimon Glass 320c6202d85SSimon Glass if (chip->chip_addr == chip_addr) { 321c6202d85SSimon Glass ret = device_probe(dev); 322c6202d85SSimon Glass debug("found, ret=%d\n", ret); 323c6202d85SSimon Glass if (ret) 324c6202d85SSimon Glass return ret; 325c6202d85SSimon Glass *devp = dev; 326c6202d85SSimon Glass return 0; 327c6202d85SSimon Glass } 328c6202d85SSimon Glass } 329c6202d85SSimon Glass debug("not found\n"); 33025ab4b03SSimon Glass return i2c_bind_driver(bus, chip_addr, offset_len, devp); 331c6202d85SSimon Glass } 332c6202d85SSimon Glass 33325ab4b03SSimon Glass int i2c_get_chip_for_busnum(int busnum, int chip_addr, uint offset_len, 33425ab4b03SSimon Glass struct udevice **devp) 335c6202d85SSimon Glass { 336c6202d85SSimon Glass struct udevice *bus; 337c6202d85SSimon Glass int ret; 338c6202d85SSimon Glass 339c6202d85SSimon Glass ret = uclass_get_device_by_seq(UCLASS_I2C, busnum, &bus); 340c6202d85SSimon Glass if (ret) { 341c6202d85SSimon Glass debug("Cannot find I2C bus %d\n", busnum); 342c6202d85SSimon Glass return ret; 343c6202d85SSimon Glass } 34425ab4b03SSimon Glass ret = i2c_get_chip(bus, chip_addr, offset_len, devp); 345c6202d85SSimon Glass if (ret) { 346c6202d85SSimon Glass debug("Cannot find I2C chip %02x on bus %d\n", chip_addr, 347c6202d85SSimon Glass busnum); 348c6202d85SSimon Glass return ret; 349c6202d85SSimon Glass } 350c6202d85SSimon Glass 351c6202d85SSimon Glass return 0; 352c6202d85SSimon Glass } 353c6202d85SSimon Glass 354f9a4c2daSSimon Glass int dm_i2c_probe(struct udevice *bus, uint chip_addr, uint chip_flags, 355c6202d85SSimon Glass struct udevice **devp) 356c6202d85SSimon Glass { 357c6202d85SSimon Glass int ret; 358c6202d85SSimon Glass 359c6202d85SSimon Glass *devp = NULL; 360c6202d85SSimon Glass 361c6202d85SSimon Glass /* First probe that chip */ 362c6202d85SSimon Glass ret = i2c_probe_chip(bus, chip_addr, chip_flags); 363c6202d85SSimon Glass debug("%s: bus='%s', address %02x, ret=%d\n", __func__, bus->name, 364c6202d85SSimon Glass chip_addr, ret); 365c6202d85SSimon Glass if (ret) 366c6202d85SSimon Glass return ret; 367c6202d85SSimon Glass 368c6202d85SSimon Glass /* The chip was found, see if we have a driver, and probe it */ 36925ab4b03SSimon Glass ret = i2c_get_chip(bus, chip_addr, 1, devp); 370c6202d85SSimon Glass debug("%s: i2c_get_chip: ret=%d\n", __func__, ret); 371c6202d85SSimon Glass 372c6202d85SSimon Glass return ret; 373c6202d85SSimon Glass } 374c6202d85SSimon Glass 375ca88b9b9SSimon Glass int dm_i2c_set_bus_speed(struct udevice *bus, unsigned int speed) 376c6202d85SSimon Glass { 377c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 378e564f054SSimon Glass struct dm_i2c_bus *i2c = dev_get_uclass_priv(bus); 379c6202d85SSimon Glass int ret; 380c6202d85SSimon Glass 381c6202d85SSimon Glass /* 382c6202d85SSimon Glass * If we have a method, call it. If not then the driver probably wants 383c6202d85SSimon Glass * to deal with speed changes on the next transfer. It can easily read 384c6202d85SSimon Glass * the current speed from this uclass 385c6202d85SSimon Glass */ 386c6202d85SSimon Glass if (ops->set_bus_speed) { 387c6202d85SSimon Glass ret = ops->set_bus_speed(bus, speed); 388c6202d85SSimon Glass if (ret) 389c6202d85SSimon Glass return ret; 390c6202d85SSimon Glass } 391c6202d85SSimon Glass i2c->speed_hz = speed; 392c6202d85SSimon Glass 393c6202d85SSimon Glass return 0; 394c6202d85SSimon Glass } 395c6202d85SSimon Glass 396ca88b9b9SSimon Glass int dm_i2c_get_bus_speed(struct udevice *bus) 397c6202d85SSimon Glass { 398c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 399e564f054SSimon Glass struct dm_i2c_bus *i2c = dev_get_uclass_priv(bus); 400c6202d85SSimon Glass 401c6202d85SSimon Glass if (!ops->get_bus_speed) 402c6202d85SSimon Glass return i2c->speed_hz; 403c6202d85SSimon Glass 404c6202d85SSimon Glass return ops->get_bus_speed(bus); 405c6202d85SSimon Glass } 406c6202d85SSimon Glass 407c6202d85SSimon Glass int i2c_set_chip_flags(struct udevice *dev, uint flags) 408c6202d85SSimon Glass { 409c6202d85SSimon Glass struct udevice *bus = dev->parent; 410e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 411c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 412c6202d85SSimon Glass int ret; 413c6202d85SSimon Glass 414c6202d85SSimon Glass if (ops->set_flags) { 415c6202d85SSimon Glass ret = ops->set_flags(dev, flags); 416c6202d85SSimon Glass if (ret) 417c6202d85SSimon Glass return ret; 418c6202d85SSimon Glass } 419c6202d85SSimon Glass chip->flags = flags; 420c6202d85SSimon Glass 421c6202d85SSimon Glass return 0; 422c6202d85SSimon Glass } 423c6202d85SSimon Glass 424c6202d85SSimon Glass int i2c_get_chip_flags(struct udevice *dev, uint *flagsp) 425c6202d85SSimon Glass { 426e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 427c6202d85SSimon Glass 428c6202d85SSimon Glass *flagsp = chip->flags; 429c6202d85SSimon Glass 430c6202d85SSimon Glass return 0; 431c6202d85SSimon Glass } 432c6202d85SSimon Glass 433c6202d85SSimon Glass int i2c_set_chip_offset_len(struct udevice *dev, uint offset_len) 434c6202d85SSimon Glass { 435e6f66ec0SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 436c6202d85SSimon Glass 437c6202d85SSimon Glass if (offset_len > I2C_MAX_OFFSET_LEN) 438c6202d85SSimon Glass return -EINVAL; 439c6202d85SSimon Glass chip->offset_len = offset_len; 440c6202d85SSimon Glass 441c6202d85SSimon Glass return 0; 442c6202d85SSimon Glass } 443c6202d85SSimon Glass 44401501804SSimon Glass int i2c_get_chip_offset_len(struct udevice *dev) 44501501804SSimon Glass { 44601501804SSimon Glass struct dm_i2c_chip *chip = dev_get_parent_platdata(dev); 44701501804SSimon Glass 44801501804SSimon Glass return chip->offset_len; 44901501804SSimon Glass } 45001501804SSimon Glass 451c6202d85SSimon Glass int i2c_deblock(struct udevice *bus) 452c6202d85SSimon Glass { 453c6202d85SSimon Glass struct dm_i2c_ops *ops = i2c_get_ops(bus); 454c6202d85SSimon Glass 455c6202d85SSimon Glass /* 456c6202d85SSimon Glass * We could implement a software deblocking here if we could get 457c6202d85SSimon Glass * access to the GPIOs used by I2C, and switch them to GPIO mode 458c6202d85SSimon Glass * and then back to I2C. This is somewhat beyond our powers in 459c6202d85SSimon Glass * driver model at present, so for now just fail. 460c6202d85SSimon Glass * 461c6202d85SSimon Glass * See https://patchwork.ozlabs.org/patch/399040/ 462c6202d85SSimon Glass */ 463c6202d85SSimon Glass if (!ops->deblock) 464c6202d85SSimon Glass return -ENOSYS; 465c6202d85SSimon Glass 466c6202d85SSimon Glass return ops->deblock(bus); 467c6202d85SSimon Glass } 468c6202d85SSimon Glass 4695142ac79SMugunthan V N #if CONFIG_IS_ENABLED(OF_CONTROL) 470c6202d85SSimon Glass int i2c_chip_ofdata_to_platdata(const void *blob, int node, 471c6202d85SSimon Glass struct dm_i2c_chip *chip) 472c6202d85SSimon Glass { 4737132b9fdSSimon Glass chip->offset_len = fdtdec_get_int(gd->fdt_blob, node, 4747132b9fdSSimon Glass "u-boot,i2c-offset-len", 1); 475c6202d85SSimon Glass chip->flags = 0; 476c6202d85SSimon Glass chip->chip_addr = fdtdec_get_int(gd->fdt_blob, node, "reg", -1); 477c6202d85SSimon Glass if (chip->chip_addr == -1) { 478c6202d85SSimon Glass debug("%s: I2C Node '%s' has no 'reg' property\n", __func__, 479c6202d85SSimon Glass fdt_get_name(blob, node, NULL)); 480c6202d85SSimon Glass return -EINVAL; 481c6202d85SSimon Glass } 482c6202d85SSimon Glass 483c6202d85SSimon Glass return 0; 484c6202d85SSimon Glass } 4855142ac79SMugunthan V N #endif 486c6202d85SSimon Glass 487c6202d85SSimon Glass static int i2c_post_probe(struct udevice *dev) 488c6202d85SSimon Glass { 4895142ac79SMugunthan V N #if CONFIG_IS_ENABLED(OF_CONTROL) 490e564f054SSimon Glass struct dm_i2c_bus *i2c = dev_get_uclass_priv(dev); 491c6202d85SSimon Glass 492c6202d85SSimon Glass i2c->speed_hz = fdtdec_get_int(gd->fdt_blob, dev->of_offset, 493c6202d85SSimon Glass "clock-frequency", 100000); 494c6202d85SSimon Glass 495ca88b9b9SSimon Glass return dm_i2c_set_bus_speed(dev, i2c->speed_hz); 4965142ac79SMugunthan V N #else 4975142ac79SMugunthan V N return 0; 4985142ac79SMugunthan V N #endif 499c6202d85SSimon Glass } 500c6202d85SSimon Glass 501e6f66ec0SSimon Glass static int i2c_child_post_bind(struct udevice *dev) 502e6f66ec0SSimon Glass { 5035142ac79SMugunthan V N #if CONFIG_IS_ENABLED(OF_CONTROL) 504e6f66ec0SSimon Glass struct dm_i2c_chip *plat = dev_get_parent_platdata(dev); 505e6f66ec0SSimon Glass 506e6f66ec0SSimon Glass if (dev->of_offset == -1) 507e6f66ec0SSimon Glass return 0; 508e6f66ec0SSimon Glass 509e6f66ec0SSimon Glass return i2c_chip_ofdata_to_platdata(gd->fdt_blob, dev->of_offset, plat); 5105142ac79SMugunthan V N #else 5115142ac79SMugunthan V N return 0; 5125142ac79SMugunthan V N #endif 513e6f66ec0SSimon Glass } 514e6f66ec0SSimon Glass 515c6202d85SSimon Glass UCLASS_DRIVER(i2c) = { 516c6202d85SSimon Glass .id = UCLASS_I2C, 517c6202d85SSimon Glass .name = "i2c", 5189cc36a2bSSimon Glass .flags = DM_UC_FLAG_SEQ_ALIAS, 519*91195485SSimon Glass #if CONFIG_IS_ENABLED(OF_CONTROL) 520*91195485SSimon Glass .post_bind = dm_scan_fdt_dev, 521*91195485SSimon Glass #endif 522c6202d85SSimon Glass .post_probe = i2c_post_probe, 523e6f66ec0SSimon Glass .per_device_auto_alloc_size = sizeof(struct dm_i2c_bus), 524e6f66ec0SSimon Glass .per_child_platdata_auto_alloc_size = sizeof(struct dm_i2c_chip), 525e6f66ec0SSimon Glass .child_post_bind = i2c_child_post_bind, 526c6202d85SSimon Glass }; 527c6202d85SSimon Glass 528c6202d85SSimon Glass UCLASS_DRIVER(i2c_generic) = { 529c6202d85SSimon Glass .id = UCLASS_I2C_GENERIC, 530c6202d85SSimon Glass .name = "i2c_generic", 531c6202d85SSimon Glass }; 532c6202d85SSimon Glass 533c6202d85SSimon Glass U_BOOT_DRIVER(i2c_generic_chip_drv) = { 534c6202d85SSimon Glass .name = "i2c_generic_chip_drv", 535c6202d85SSimon Glass .id = UCLASS_I2C_GENERIC, 536c6202d85SSimon Glass }; 537