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 | 911 | eeprom_buf = qemu_mallocz(8 * 256); /* XXX: make this persistent */ |
912 | 912 | for (i = 0; i < 8; i++) { |
913 | 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 | 920 | pit = pit_init(0x40, i8259[0]); |
917 | 921 | DMA_init(0); | ... | ... |
hw/pc.c
... | ... | @@ -1109,7 +1109,11 @@ static void pc_init1(ram_addr_t ram_size, |
1109 | 1109 | /* TODO: Populate SPD eeprom data. */ |
1110 | 1110 | smbus = piix4_pm_init(pci_bus, piix3_devfn + 3, 0xb100, i8259[9]); |
1111 | 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 | 37 | |
38 | 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 | 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 | 47 | static void smbus_do_write(SMBusDevice *dev) |
46 | 48 | { |
49 | + SMBusDeviceInfo *t = container_of(dev->i2c.info, SMBusDeviceInfo, i2c); | |
50 | + | |
47 | 51 | if (dev->data_len == 0) { |
48 | 52 | smbus_do_quick_cmd(dev, 0); |
49 | 53 | } else if (dev->data_len == 1) { |
50 | 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 | 58 | } else { |
55 | 59 | dev->command = dev->data_buf[0]; |
56 | 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 | 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 | 72 | switch (event) { |
68 | 73 | case I2C_START_SEND: |
69 | 74 | switch (dev->mode) { |
... | ... | @@ -145,13 +150,14 @@ static void smbus_i2c_event(i2c_slave *s, enum i2c_event event) |
145 | 150 | |
146 | 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 | 155 | int ret; |
150 | 156 | |
151 | 157 | switch (dev->mode) { |
152 | 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 | 161 | } else { |
156 | 162 | ret = 0; |
157 | 163 | } |
... | ... | @@ -159,8 +165,8 @@ static int smbus_i2c_recv(i2c_slave *s) |
159 | 165 | dev->mode = SMBUS_DONE; |
160 | 166 | break; |
161 | 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 | 170 | dev->data_len++; |
165 | 171 | } else { |
166 | 172 | ret = 0; |
... | ... | @@ -178,7 +184,8 @@ static int smbus_i2c_recv(i2c_slave *s) |
178 | 184 | |
179 | 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 | 189 | switch (dev->mode) { |
183 | 190 | case SMBUS_WRITE_DATA: |
184 | 191 | DPRINTF("Write data %02x\n", data); |
... | ... | @@ -191,19 +198,22 @@ static int smbus_i2c_send(i2c_slave *s, uint8_t data) |
191 | 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 | 219 | /* Master device commands. */ | ... | ... |
hw/smbus.h
... | ... | @@ -28,7 +28,16 @@ struct SMBusDevice { |
28 | 28 | /* The SMBus protocol is implemented on top of I2C. */ |
29 | 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 | 41 | void (*quick_cmd)(SMBusDevice *dev, uint8_t read); |
33 | 42 | void (*send_byte)(SMBusDevice *dev, uint8_t val); |
34 | 43 | uint8_t (*receive_byte)(SMBusDevice *dev); |
... | ... | @@ -42,16 +51,9 @@ struct SMBusDevice { |
42 | 51 | byte at a time. The device is responsible for adding the length |
43 | 52 | byte on block reads. */ |
44 | 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 | 58 | /* Master device commands. */ |
57 | 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 | 66 | int smbus_read_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data); |
65 | 67 | void smbus_write_block(i2c_bus *bus, int addr, uint8_t command, uint8_t *data, |
66 | 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 | 29 | //#define DEBUG |
30 | 30 | |
31 | 31 | typedef struct SMBusEEPROMDevice { |
32 | - SMBusDevice dev; | |
32 | + SMBusDevice smbusdev; | |
33 | 33 | uint8_t *data; |
34 | 34 | uint8_t offset; |
35 | 35 | } SMBusEEPROMDevice; |
... | ... | @@ -95,18 +95,28 @@ static uint8_t eeprom_read_data(SMBusDevice *dev, uint8_t cmd, int n) |
95 | 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 | 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) | ... | ... |