| /* |
| * QEMU SMBus device emulation. |
| * |
| * Copyright (c) 2007 CodeSourcery. |
| * Written by Paul Brook |
| * |
| * This code is licenced under the LGPL. |
| */ |
| |
| /* TODO: Implement PEC. */ |
| |
| #include "hw.h" |
| #include "i2c.h" |
| #include "smbus.h" |
| |
| //#define DEBUG_SMBUS 1 |
| |
| #ifdef DEBUG_SMBUS |
| #define DPRINTF(fmt, ...) \ |
| do { printf("smbus(%02x): " fmt , dev->i2c.address, ## __VA_ARGS__); } while (0) |
| #define BADF(fmt, ...) \ |
| do { fprintf(stderr, "smbus: error: " fmt , ## __VA_ARGS__); exit(1);} while (0) |
| #else |
| #define DPRINTF(fmt, ...) do {} while(0) |
| #define BADF(fmt, ...) \ |
| do { fprintf(stderr, "smbus: error: " fmt , ## __VA_ARGS__);} while (0) |
| #endif |
| |
| enum { |
| SMBUS_IDLE, |
| SMBUS_WRITE_DATA, |
| SMBUS_RECV_BYTE, |
| SMBUS_READ_DATA, |
| SMBUS_DONE, |
| SMBUS_CONFUSED = -1 |
| }; |
| |
| static void smbus_do_quick_cmd(SMBusDevice *dev, int recv) |
| { |
| SMBusDeviceInfo *t = container_of(dev->i2c.info, SMBusDeviceInfo, i2c); |
| |
| DPRINTF("Quick Command %d\n", recv); |
| if (t->quick_cmd) |
| t->quick_cmd(dev, recv); |
| } |
| |
| static void smbus_do_write(SMBusDevice *dev) |
| { |
| SMBusDeviceInfo *t = container_of(dev->i2c.info, SMBusDeviceInfo, i2c); |
| |
| if (dev->data_len == 0) { |
| smbus_do_quick_cmd(dev, 0); |
| } else if (dev->data_len == 1) { |
| DPRINTF("Send Byte\n"); |
| if (t->send_byte) { |
| t->send_byte(dev, dev->data_buf[0]); |
| } |
| } else { |
| dev->command = dev->data_buf[0]; |
| DPRINTF("Command %d len %d\n", dev->command, dev->data_len - 1); |
| if (t->write_data) { |
| t->write_data(dev, dev->command, dev->data_buf + 1, |
| dev->data_len - 1); |
| } |
| } |
| } |
| |
| static void smbus_i2c_event(i2c_slave *s, enum i2c_event event) |
| { |
| SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, s); |
| |
| switch (event) { |
| case I2C_START_SEND: |
| switch (dev->mode) { |
| case SMBUS_IDLE: |
| DPRINTF("Incoming data\n"); |
| dev->mode = SMBUS_WRITE_DATA; |
| break; |
| default: |
| BADF("Unexpected send start condition in state %d\n", dev->mode); |
| dev->mode = SMBUS_CONFUSED; |
| break; |
| } |
| break; |
| |
| case I2C_START_RECV: |
| switch (dev->mode) { |
| case SMBUS_IDLE: |
| DPRINTF("Read mode\n"); |
| dev->mode = SMBUS_RECV_BYTE; |
| break; |
| case SMBUS_WRITE_DATA: |
| if (dev->data_len == 0) { |
| BADF("Read after write with no data\n"); |
| dev->mode = SMBUS_CONFUSED; |
| } else { |
| if (dev->data_len > 1) { |
| smbus_do_write(dev); |
| } else { |
| dev->command = dev->data_buf[0]; |
| DPRINTF("%02x: Command %d\n", dev->i2c.address, |
| dev->command); |
| } |
| DPRINTF("Read mode\n"); |
| dev->data_len = 0; |
| dev->mode = SMBUS_READ_DATA; |
| } |
| break; |
| default: |
| BADF("Unexpected recv start condition in state %d\n", dev->mode); |
| dev->mode = SMBUS_CONFUSED; |
| break; |
| } |
| break; |
| |
| case I2C_FINISH: |
| switch (dev->mode) { |
| case SMBUS_WRITE_DATA: |
| smbus_do_write(dev); |
| break; |
| case SMBUS_RECV_BYTE: |
| smbus_do_quick_cmd(dev, 1); |
| break; |
| case SMBUS_READ_DATA: |
| BADF("Unexpected stop during receive\n"); |
| break; |
| default: |
| /* Nothing to do. */ |
| break; |
| } |
| dev->mode = SMBUS_IDLE; |
| dev->data_len = 0; |
| break; |
| |
| case I2C_NACK: |
| switch (dev->mode) { |
| case SMBUS_DONE: |
| /* Nothing to do. */ |
| break; |
| case SMBUS_READ_DATA: |
| dev->mode = SMBUS_DONE; |
| break; |
| default: |
| BADF("Unexpected NACK in state %d\n", dev->mode); |
| dev->mode = SMBUS_CONFUSED; |
| break; |
| } |
| } |
| } |
| |
| static int smbus_i2c_recv(i2c_slave *s) |
| { |
| SMBusDeviceInfo *t = container_of(s->info, SMBusDeviceInfo, i2c); |
| SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, s); |
| int ret; |
| |
| switch (dev->mode) { |
| case SMBUS_RECV_BYTE: |
| if (t->receive_byte) { |
| ret = t->receive_byte(dev); |
| } else { |
| ret = 0; |
| } |
| DPRINTF("Receive Byte %02x\n", ret); |
| dev->mode = SMBUS_DONE; |
| break; |
| case SMBUS_READ_DATA: |
| if (t->read_data) { |
| ret = t->read_data(dev, dev->command, dev->data_len); |
| dev->data_len++; |
| } else { |
| ret = 0; |
| } |
| DPRINTF("Read data %02x\n", ret); |
| break; |
| default: |
| BADF("Unexpected read in state %d\n", dev->mode); |
| dev->mode = SMBUS_CONFUSED; |
| ret = 0; |
| break; |
| } |
| return ret; |
| } |
| |
| static int smbus_i2c_send(i2c_slave *s, uint8_t data) |
| { |
| SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, s); |
| |
| switch (dev->mode) { |
| case SMBUS_WRITE_DATA: |
| DPRINTF("Write data %02x\n", data); |
| dev->data_buf[dev->data_len++] = data; |
| break; |
| default: |
| BADF("Unexpected write in state %d\n", dev->mode); |
| break; |
| } |
| return 0; |
| } |
| |
| static void smbus_device_init(i2c_slave *i2c) |
| { |
| SMBusDeviceInfo *t = container_of(i2c->info, SMBusDeviceInfo, i2c); |
| SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, i2c); |
| |
| t->init(dev); |
| } |
| |
| void smbus_register_device(const char *name, int size, SMBusDeviceInfo *info) |
| { |
| assert(size >= sizeof(SMBusDevice)); |
| info->i2c.init = smbus_device_init; |
| info->i2c.event = smbus_i2c_event; |
| info->i2c.recv = smbus_i2c_recv; |
| info->i2c.send = smbus_i2c_send; |
| i2c_register_slave(name, size, &info->i2c); |
| } |
| |
| /* Master device commands. */ |
| void smbus_quick_command(i2c_bus *bus, int addr, int read) |
| { |
| i2c_start_transfer(bus, addr, read); |
| i2c_end_transfer(bus); |
| } |
| |
| uint8_t smbus_receive_byte(i2c_bus *bus, int addr) |
| { |
| uint8_t data; |
| |
| i2c_start_transfer(bus, addr, 1); |
| data = i2c_recv(bus); |
| i2c_nack(bus); |
| i2c_end_transfer(bus); |
| return data; |
| } |
| |
| void smbus_send_byte(i2c_bus *bus, int addr, uint8_t data) |
| { |
| i2c_start_transfer(bus, addr, 0); |
| i2c_send(bus, data); |
| i2c_end_transfer(bus); |
| } |
| |
| uint8_t smbus_read_byte(i2c_bus *bus, int addr, uint8_t command) |
| { |
| uint8_t data; |
| i2c_start_transfer(bus, addr, 0); |
| i2c_send(bus, command); |
| i2c_start_transfer(bus, addr, 1); |
| data = i2c_recv(bus); |
| i2c_nack(bus); |
| i2c_end_transfer(bus); |
| return data; |
| } |
| |
| void smbus_write_byte(i2c_bus *bus, int addr, uint8_t command, uint8_t data) |
| { |
| i2c_start_transfer(bus, addr, 0); |
| i2c_send(bus, command); |
| i2c_send(bus, data); |
| i2c_end_transfer(bus); |
| } |
| |
| uint16_t smbus_read_word(i2c_bus *bus, int addr, uint8_t command) |
| { |
| uint16_t data; |
| i2c_start_transfer(bus, addr, 0); |
| i2c_send(bus, command); |
| i2c_start_transfer(bus, addr, 1); |
| data = i2c_recv(bus); |
| data |= i2c_recv(bus) << 8; |
| i2c_nack(bus); |
| i2c_end_transfer(bus); |
| return data; |
| } |
| |
| void smbus_write_word(i2c_bus *bus, int addr, uint8_t command, uint16_t data) |
| { |
| i2c_start_transfer(bus, addr, 0); |
| i2c_send(bus, command); |
| i2c_send(bus, data & 0xff); |
| i2c_send(bus, data >> 8); |
| i2c_end_transfer(bus); |
| } |
| |
| int smbus_read_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data) |
| { |
| int len; |
| int i; |
| |
| i2c_start_transfer(bus, addr, 0); |
| i2c_send(bus, command); |
| i2c_start_transfer(bus, addr, 1); |
| len = i2c_recv(bus); |
| if (len > 32) |
| len = 0; |
| for (i = 0; i < len; i++) |
| data[i] = i2c_recv(bus); |
| i2c_nack(bus); |
| i2c_end_transfer(bus); |
| return len; |
| } |
| |
| void smbus_write_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data, |
| int len) |
| { |
| int i; |
| |
| if (len > 32) |
| len = 32; |
| |
| i2c_start_transfer(bus, addr, 0); |
| i2c_send(bus, command); |
| i2c_send(bus, len); |
| for (i = 0; i < len; i++) |
| i2c_send(bus, data[i]); |
| i2c_end_transfer(bus); |
| } |