Commit 1ea96673c611cdc05ea32ac81f40d9f864e18507
1 parent
fd1eb2ea
SMBus qdev conversion
Signed-off-by: Paul Brook <paul@codesourcery.com>
Showing
5 changed files
with
78 additions
and
51 deletions
hw/mips_malta.c
@@ -911,7 +911,11 @@ void mips_malta_init (ram_addr_t ram_size, | @@ -911,7 +911,11 @@ void mips_malta_init (ram_addr_t ram_size, | ||
911 | eeprom_buf = qemu_mallocz(8 * 256); /* XXX: make this persistent */ | 911 | eeprom_buf = qemu_mallocz(8 * 256); /* XXX: make this persistent */ |
912 | for (i = 0; i < 8; i++) { | 912 | for (i = 0; i < 8; i++) { |
913 | /* TODO: Populate SPD eeprom data. */ | 913 | /* TODO: Populate SPD eeprom data. */ |
914 | - smbus_eeprom_device_init(smbus, 0x50 + i, eeprom_buf + (i * 256)); | 914 | + DeviceState *eeprom; |
915 | + eeprom = qdev_create(smbus, "smbus-eeprom"); | ||
916 | + qdev_set_prop_int(eeprom, "address", 0x50 + i); | ||
917 | + qdev_set_prop_ptr(eeprom, "data", eeprom_buf + (i * 256)); | ||
918 | + qdev_init(eeprom); | ||
915 | } | 919 | } |
916 | pit = pit_init(0x40, i8259[0]); | 920 | pit = pit_init(0x40, i8259[0]); |
917 | DMA_init(0); | 921 | DMA_init(0); |
hw/pc.c
@@ -1109,7 +1109,11 @@ static void pc_init1(ram_addr_t ram_size, | @@ -1109,7 +1109,11 @@ static void pc_init1(ram_addr_t ram_size, | ||
1109 | /* TODO: Populate SPD eeprom data. */ | 1109 | /* TODO: Populate SPD eeprom data. */ |
1110 | smbus = piix4_pm_init(pci_bus, piix3_devfn + 3, 0xb100, i8259[9]); | 1110 | smbus = piix4_pm_init(pci_bus, piix3_devfn + 3, 0xb100, i8259[9]); |
1111 | for (i = 0; i < 8; i++) { | 1111 | for (i = 0; i < 8; i++) { |
1112 | - smbus_eeprom_device_init(smbus, 0x50 + i, eeprom_buf + (i * 256)); | 1112 | + DeviceState *eeprom; |
1113 | + eeprom = qdev_create(smbus, "smbus-eeprom"); | ||
1114 | + qdev_set_prop_int(eeprom, "address", 0x50 + i); | ||
1115 | + qdev_set_prop_ptr(eeprom, "data", eeprom_buf + (i * 256)); | ||
1116 | + qdev_init(eeprom); | ||
1113 | } | 1117 | } |
1114 | } | 1118 | } |
1115 | 1119 |
hw/smbus.c
@@ -37,33 +37,38 @@ enum { | @@ -37,33 +37,38 @@ enum { | ||
37 | 37 | ||
38 | static void smbus_do_quick_cmd(SMBusDevice *dev, int recv) | 38 | static void smbus_do_quick_cmd(SMBusDevice *dev, int recv) |
39 | { | 39 | { |
40 | + SMBusDeviceInfo *t = container_of(dev->i2c.info, SMBusDeviceInfo, i2c); | ||
41 | + | ||
40 | DPRINTF("Quick Command %d\n", recv); | 42 | DPRINTF("Quick Command %d\n", recv); |
41 | - if (dev->quick_cmd) | ||
42 | - dev->quick_cmd(dev, recv); | 43 | + if (t->quick_cmd) |
44 | + t->quick_cmd(dev, recv); | ||
43 | } | 45 | } |
44 | 46 | ||
45 | static void smbus_do_write(SMBusDevice *dev) | 47 | static void smbus_do_write(SMBusDevice *dev) |
46 | { | 48 | { |
49 | + SMBusDeviceInfo *t = container_of(dev->i2c.info, SMBusDeviceInfo, i2c); | ||
50 | + | ||
47 | if (dev->data_len == 0) { | 51 | if (dev->data_len == 0) { |
48 | smbus_do_quick_cmd(dev, 0); | 52 | smbus_do_quick_cmd(dev, 0); |
49 | } else if (dev->data_len == 1) { | 53 | } else if (dev->data_len == 1) { |
50 | DPRINTF("Send Byte\n"); | 54 | DPRINTF("Send Byte\n"); |
51 | - if (dev->send_byte) { | ||
52 | - dev->send_byte(dev, dev->data_buf[0]); | 55 | + if (t->send_byte) { |
56 | + t->send_byte(dev, dev->data_buf[0]); | ||
53 | } | 57 | } |
54 | } else { | 58 | } else { |
55 | dev->command = dev->data_buf[0]; | 59 | dev->command = dev->data_buf[0]; |
56 | DPRINTF("Command %d len %d\n", dev->command, dev->data_len - 1); | 60 | DPRINTF("Command %d len %d\n", dev->command, dev->data_len - 1); |
57 | - if (dev->write_data) { | ||
58 | - dev->write_data(dev, dev->command, dev->data_buf + 1, | ||
59 | - dev->data_len - 1); | 61 | + if (t->write_data) { |
62 | + t->write_data(dev, dev->command, dev->data_buf + 1, | ||
63 | + dev->data_len - 1); | ||
60 | } | 64 | } |
61 | } | 65 | } |
62 | } | 66 | } |
63 | 67 | ||
64 | static void smbus_i2c_event(i2c_slave *s, enum i2c_event event) | 68 | static void smbus_i2c_event(i2c_slave *s, enum i2c_event event) |
65 | { | 69 | { |
66 | - SMBusDevice *dev = (SMBusDevice *)s; | 70 | + SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, s); |
71 | + | ||
67 | switch (event) { | 72 | switch (event) { |
68 | case I2C_START_SEND: | 73 | case I2C_START_SEND: |
69 | switch (dev->mode) { | 74 | switch (dev->mode) { |
@@ -145,13 +150,14 @@ static void smbus_i2c_event(i2c_slave *s, enum i2c_event event) | @@ -145,13 +150,14 @@ static void smbus_i2c_event(i2c_slave *s, enum i2c_event event) | ||
145 | 150 | ||
146 | static int smbus_i2c_recv(i2c_slave *s) | 151 | static int smbus_i2c_recv(i2c_slave *s) |
147 | { | 152 | { |
148 | - SMBusDevice *dev = (SMBusDevice *)s; | 153 | + SMBusDeviceInfo *t = container_of(s->info, SMBusDeviceInfo, i2c); |
154 | + SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, s); | ||
149 | int ret; | 155 | int ret; |
150 | 156 | ||
151 | switch (dev->mode) { | 157 | switch (dev->mode) { |
152 | case SMBUS_RECV_BYTE: | 158 | case SMBUS_RECV_BYTE: |
153 | - if (dev->receive_byte) { | ||
154 | - ret = dev->receive_byte(dev); | 159 | + if (t->receive_byte) { |
160 | + ret = t->receive_byte(dev); | ||
155 | } else { | 161 | } else { |
156 | ret = 0; | 162 | ret = 0; |
157 | } | 163 | } |
@@ -159,8 +165,8 @@ static int smbus_i2c_recv(i2c_slave *s) | @@ -159,8 +165,8 @@ static int smbus_i2c_recv(i2c_slave *s) | ||
159 | dev->mode = SMBUS_DONE; | 165 | dev->mode = SMBUS_DONE; |
160 | break; | 166 | break; |
161 | case SMBUS_READ_DATA: | 167 | case SMBUS_READ_DATA: |
162 | - if (dev->read_data) { | ||
163 | - ret = dev->read_data(dev, dev->command, dev->data_len); | 168 | + if (t->read_data) { |
169 | + ret = t->read_data(dev, dev->command, dev->data_len); | ||
164 | dev->data_len++; | 170 | dev->data_len++; |
165 | } else { | 171 | } else { |
166 | ret = 0; | 172 | ret = 0; |
@@ -178,7 +184,8 @@ static int smbus_i2c_recv(i2c_slave *s) | @@ -178,7 +184,8 @@ static int smbus_i2c_recv(i2c_slave *s) | ||
178 | 184 | ||
179 | static int smbus_i2c_send(i2c_slave *s, uint8_t data) | 185 | static int smbus_i2c_send(i2c_slave *s, uint8_t data) |
180 | { | 186 | { |
181 | - SMBusDevice *dev = (SMBusDevice *)s; | 187 | + SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, s); |
188 | + | ||
182 | switch (dev->mode) { | 189 | switch (dev->mode) { |
183 | case SMBUS_WRITE_DATA: | 190 | case SMBUS_WRITE_DATA: |
184 | DPRINTF("Write data %02x\n", data); | 191 | DPRINTF("Write data %02x\n", data); |
@@ -191,19 +198,22 @@ static int smbus_i2c_send(i2c_slave *s, uint8_t data) | @@ -191,19 +198,22 @@ static int smbus_i2c_send(i2c_slave *s, uint8_t data) | ||
191 | return 0; | 198 | return 0; |
192 | } | 199 | } |
193 | 200 | ||
194 | -SMBusDevice *smbus_device_init(i2c_bus *bus, int address, int size) | 201 | +static void smbus_device_init(i2c_slave *i2c) |
195 | { | 202 | { |
196 | - SMBusDevice *dev; | 203 | + SMBusDeviceInfo *t = container_of(i2c->info, SMBusDeviceInfo, i2c); |
204 | + SMBusDevice *dev = FROM_I2C_SLAVE(SMBusDevice, i2c); | ||
197 | 205 | ||
198 | - if (size < sizeof(SMBusDevice)) | ||
199 | - hw_error("SMBus struct too small"); | ||
200 | - | ||
201 | - dev = (SMBusDevice *)i2c_slave_init(bus, address, size); | ||
202 | - dev->i2c.event = smbus_i2c_event; | ||
203 | - dev->i2c.recv = smbus_i2c_recv; | ||
204 | - dev->i2c.send = smbus_i2c_send; | 206 | + t->init(dev); |
207 | +} | ||
205 | 208 | ||
206 | - return dev; | 209 | +void smbus_register_device(const char *name, int size, SMBusDeviceInfo *info) |
210 | +{ | ||
211 | + assert(size >= sizeof(SMBusDevice)); | ||
212 | + info->i2c.init = smbus_device_init; | ||
213 | + info->i2c.event = smbus_i2c_event; | ||
214 | + info->i2c.recv = smbus_i2c_recv; | ||
215 | + info->i2c.send = smbus_i2c_send; | ||
216 | + i2c_register_slave(name, size, &info->i2c); | ||
207 | } | 217 | } |
208 | 218 | ||
209 | /* Master device commands. */ | 219 | /* Master device commands. */ |
hw/smbus.h
@@ -28,7 +28,16 @@ struct SMBusDevice { | @@ -28,7 +28,16 @@ struct SMBusDevice { | ||
28 | /* The SMBus protocol is implemented on top of I2C. */ | 28 | /* The SMBus protocol is implemented on top of I2C. */ |
29 | i2c_slave i2c; | 29 | i2c_slave i2c; |
30 | 30 | ||
31 | - /* Callbacks set by the device. */ | 31 | + /* Remaining fields for internal use only. */ |
32 | + int mode; | ||
33 | + int data_len; | ||
34 | + uint8_t data_buf[34]; /* command + len + 32 bytes of data. */ | ||
35 | + uint8_t command; | ||
36 | +}; | ||
37 | + | ||
38 | +typedef struct { | ||
39 | + I2CSlaveInfo i2c; | ||
40 | + void (*init)(SMBusDevice *dev); | ||
32 | void (*quick_cmd)(SMBusDevice *dev, uint8_t read); | 41 | void (*quick_cmd)(SMBusDevice *dev, uint8_t read); |
33 | void (*send_byte)(SMBusDevice *dev, uint8_t val); | 42 | void (*send_byte)(SMBusDevice *dev, uint8_t val); |
34 | uint8_t (*receive_byte)(SMBusDevice *dev); | 43 | uint8_t (*receive_byte)(SMBusDevice *dev); |
@@ -42,16 +51,9 @@ struct SMBusDevice { | @@ -42,16 +51,9 @@ struct SMBusDevice { | ||
42 | byte at a time. The device is responsible for adding the length | 51 | byte at a time. The device is responsible for adding the length |
43 | byte on block reads. */ | 52 | byte on block reads. */ |
44 | uint8_t (*read_data)(SMBusDevice *dev, uint8_t cmd, int n); | 53 | uint8_t (*read_data)(SMBusDevice *dev, uint8_t cmd, int n); |
54 | +} SMBusDeviceInfo; | ||
45 | 55 | ||
46 | - /* Remaining fields for internal use only. */ | ||
47 | - int mode; | ||
48 | - int data_len; | ||
49 | - uint8_t data_buf[34]; /* command + len + 32 bytes of data. */ | ||
50 | - uint8_t command; | ||
51 | -}; | ||
52 | - | ||
53 | -/* Create a slave device. */ | ||
54 | -SMBusDevice *smbus_device_init(i2c_bus *bus, int address, int size); | 56 | +void smbus_register_device(const char *name, int size, SMBusDeviceInfo *info); |
55 | 57 | ||
56 | /* Master device commands. */ | 58 | /* Master device commands. */ |
57 | void smbus_quick_command(i2c_bus *bus, int addr, int read); | 59 | void smbus_quick_command(i2c_bus *bus, int addr, int read); |
@@ -64,6 +66,3 @@ void smbus_write_word(i2c_bus *bus, int addr, uint8_t command, uint16_t data); | @@ -64,6 +66,3 @@ void smbus_write_word(i2c_bus *bus, int addr, uint8_t command, uint16_t data); | ||
64 | int smbus_read_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data); | 66 | int smbus_read_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data); |
65 | void smbus_write_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data, | 67 | void smbus_write_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data, |
66 | int len); | 68 | int len); |
67 | - | ||
68 | -/* smbus_eeprom.c */ | ||
69 | -void smbus_eeprom_device_init(i2c_bus *bus, uint8_t addr, uint8_t *buf); |
hw/smbus_eeprom.c
@@ -29,7 +29,7 @@ | @@ -29,7 +29,7 @@ | ||
29 | //#define DEBUG | 29 | //#define DEBUG |
30 | 30 | ||
31 | typedef struct SMBusEEPROMDevice { | 31 | typedef struct SMBusEEPROMDevice { |
32 | - SMBusDevice dev; | 32 | + SMBusDevice smbusdev; |
33 | uint8_t *data; | 33 | uint8_t *data; |
34 | uint8_t offset; | 34 | uint8_t offset; |
35 | } SMBusEEPROMDevice; | 35 | } SMBusEEPROMDevice; |
@@ -95,18 +95,28 @@ static uint8_t eeprom_read_data(SMBusDevice *dev, uint8_t cmd, int n) | @@ -95,18 +95,28 @@ static uint8_t eeprom_read_data(SMBusDevice *dev, uint8_t cmd, int n) | ||
95 | return eeprom_receive_byte(dev); | 95 | return eeprom_receive_byte(dev); |
96 | } | 96 | } |
97 | 97 | ||
98 | -void smbus_eeprom_device_init(i2c_bus *bus, uint8_t addr, uint8_t *buf) | 98 | +static void smbus_eeprom_init(SMBusDevice *dev) |
99 | { | 99 | { |
100 | - SMBusEEPROMDevice *eeprom; | 100 | + SMBusEEPROMDevice *eeprom = (SMBusEEPROMDevice *)dev; |
101 | 101 | ||
102 | - eeprom = (SMBusEEPROMDevice *)smbus_device_init(bus, addr, | ||
103 | - sizeof(SMBusEEPROMDevice)); | ||
104 | - | ||
105 | - eeprom->dev.quick_cmd = eeprom_quick_cmd; | ||
106 | - eeprom->dev.send_byte = eeprom_send_byte; | ||
107 | - eeprom->dev.receive_byte = eeprom_receive_byte; | ||
108 | - eeprom->dev.write_data = eeprom_write_data; | ||
109 | - eeprom->dev.read_data = eeprom_read_data; | ||
110 | - eeprom->data = buf; | 102 | + /* FIXME: Should be a blob rather than a ptr. */ |
103 | + eeprom->data = qdev_get_prop_ptr(&dev->i2c.qdev, "data"); | ||
111 | eeprom->offset = 0; | 104 | eeprom->offset = 0; |
112 | } | 105 | } |
106 | + | ||
107 | +static SMBusDeviceInfo smbus_eeprom_info = { | ||
108 | + .init = smbus_eeprom_init, | ||
109 | + .quick_cmd = eeprom_quick_cmd, | ||
110 | + .send_byte = eeprom_send_byte, | ||
111 | + .receive_byte = eeprom_receive_byte, | ||
112 | + .write_data = eeprom_write_data, | ||
113 | + .read_data = eeprom_read_data | ||
114 | +}; | ||
115 | + | ||
116 | +static void smbus_eeprom_register_devices(void) | ||
117 | +{ | ||
118 | + smbus_register_device("smbus-eeprom", sizeof(SMBusEEPROMDevice), | ||
119 | + &smbus_eeprom_info); | ||
120 | +} | ||
121 | + | ||
122 | +device_init(smbus_eeprom_register_devices) |