Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 1 | /* |
| 2 | * PCA9552 I2C LED blinker |
| 3 | * |
| 4 | * https://www.nxp.com/docs/en/application-note/AN264.pdf |
| 5 | * |
| 6 | * Copyright (c) 2017-2018, IBM Corporation. |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 7 | * Copyright (c) 2020 Philippe Mathieu-Daudé |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 8 | * |
| 9 | * This work is licensed under the terms of the GNU GPL, version 2 or |
| 10 | * later. See the COPYING file in the top-level directory. |
| 11 | */ |
| 12 | |
| 13 | #include "qemu/osdep.h" |
| 14 | #include "qemu/log.h" |
Markus Armbruster | 0b8fa32 | 2019-05-23 16:35:07 +0200 | [diff] [blame] | 15 | #include "qemu/module.h" |
Philippe Mathieu-Daudé | b989b89 | 2020-06-23 09:27:20 +0200 | [diff] [blame] | 16 | #include "qemu/bitops.h" |
Philippe Mathieu-Daudé | 2df252d | 2020-06-23 09:27:19 +0200 | [diff] [blame] | 17 | #include "hw/qdev-properties.h" |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 18 | #include "hw/misc/pca9552.h" |
| 19 | #include "hw/misc/pca9552_regs.h" |
Philippe Mathieu-Daudé | 586f495 | 2020-06-23 09:27:23 +0200 | [diff] [blame] | 20 | #include "hw/irq.h" |
Markus Armbruster | d645427 | 2019-08-12 07:23:45 +0200 | [diff] [blame] | 21 | #include "migration/vmstate.h" |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 22 | #include "qapi/error.h" |
| 23 | #include "qapi/visitor.h" |
Philippe Mathieu-Daudé | b989b89 | 2020-06-23 09:27:20 +0200 | [diff] [blame] | 24 | #include "trace.h" |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 25 | |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 26 | typedef struct PCA955xClass { |
| 27 | /*< private >*/ |
| 28 | I2CSlaveClass parent_class; |
| 29 | /*< public >*/ |
| 30 | |
| 31 | uint8_t pin_count; |
| 32 | uint8_t max_reg; |
| 33 | } PCA955xClass; |
| 34 | |
| 35 | #define PCA955X_CLASS(klass) \ |
| 36 | OBJECT_CLASS_CHECK(PCA955xClass, (klass), TYPE_PCA955X) |
| 37 | #define PCA955X_GET_CLASS(obj) \ |
| 38 | OBJECT_GET_CLASS(PCA955xClass, (obj), TYPE_PCA955X) |
| 39 | |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 40 | #define PCA9552_LED_ON 0x0 |
| 41 | #define PCA9552_LED_OFF 0x1 |
| 42 | #define PCA9552_LED_PWM0 0x2 |
| 43 | #define PCA9552_LED_PWM1 0x3 |
| 44 | |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 45 | static const char *led_state[] = {"on", "off", "pwm0", "pwm1"}; |
| 46 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 47 | static uint8_t pca955x_pin_get_config(PCA955xState *s, int pin) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 48 | { |
| 49 | uint8_t reg = PCA9552_LS0 + (pin / 4); |
| 50 | uint8_t shift = (pin % 4) << 1; |
| 51 | |
| 52 | return extract32(s->regs[reg], shift, 2); |
| 53 | } |
| 54 | |
Philippe Mathieu-Daudé | b989b89 | 2020-06-23 09:27:20 +0200 | [diff] [blame] | 55 | /* Return INPUT status (bit #N belongs to GPIO #N) */ |
| 56 | static uint16_t pca955x_pins_get_status(PCA955xState *s) |
| 57 | { |
| 58 | return (s->regs[PCA9552_INPUT1] << 8) | s->regs[PCA9552_INPUT0]; |
| 59 | } |
| 60 | |
| 61 | static void pca955x_display_pins_status(PCA955xState *s, |
| 62 | uint16_t previous_pins_status) |
| 63 | { |
| 64 | PCA955xClass *k = PCA955X_GET_CLASS(s); |
| 65 | uint16_t pins_status, pins_changed; |
| 66 | int i; |
| 67 | |
| 68 | pins_status = pca955x_pins_get_status(s); |
| 69 | pins_changed = previous_pins_status ^ pins_status; |
| 70 | if (!pins_changed) { |
| 71 | return; |
| 72 | } |
| 73 | if (trace_event_get_state_backends(TRACE_PCA955X_GPIO_STATUS)) { |
| 74 | char *buf = g_newa(char, k->pin_count + 1); |
| 75 | |
| 76 | for (i = 0; i < k->pin_count; i++) { |
| 77 | if (extract32(pins_status, i, 1)) { |
| 78 | buf[i] = '*'; |
| 79 | } else { |
| 80 | buf[i] = '.'; |
| 81 | } |
| 82 | } |
| 83 | buf[i] = '\0'; |
| 84 | trace_pca955x_gpio_status(s->description, buf); |
| 85 | } |
Philippe Mathieu-Daudé | d82ab29 | 2020-06-23 09:27:22 +0200 | [diff] [blame] | 86 | if (trace_event_get_state_backends(TRACE_PCA955X_GPIO_CHANGE)) { |
| 87 | for (i = 0; i < k->pin_count; i++) { |
| 88 | if (extract32(pins_changed, i, 1)) { |
| 89 | unsigned new_state = extract32(pins_status, i, 1); |
| 90 | |
| 91 | /* |
| 92 | * We display the state using the PCA logic ("active-high"). |
| 93 | * This is not the state of the LED, which signal might be |
| 94 | * wired "active-low" on the board. |
| 95 | */ |
| 96 | trace_pca955x_gpio_change(s->description, i, |
| 97 | !new_state, new_state); |
| 98 | } |
| 99 | } |
| 100 | } |
Philippe Mathieu-Daudé | b989b89 | 2020-06-23 09:27:20 +0200 | [diff] [blame] | 101 | } |
| 102 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 103 | static void pca955x_update_pin_input(PCA955xState *s) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 104 | { |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 105 | PCA955xClass *k = PCA955X_GET_CLASS(s); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 106 | int i; |
| 107 | |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 108 | for (i = 0; i < k->pin_count; i++) { |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 109 | uint8_t input_reg = PCA9552_INPUT0 + (i / 8); |
| 110 | uint8_t input_shift = (i % 8); |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 111 | uint8_t config = pca955x_pin_get_config(s, i); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 112 | |
| 113 | switch (config) { |
| 114 | case PCA9552_LED_ON: |
Philippe Mathieu-Daudé | 586f495 | 2020-06-23 09:27:23 +0200 | [diff] [blame] | 115 | qemu_set_irq(s->gpio[i], 1); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 116 | s->regs[input_reg] |= 1 << input_shift; |
| 117 | break; |
| 118 | case PCA9552_LED_OFF: |
Philippe Mathieu-Daudé | 586f495 | 2020-06-23 09:27:23 +0200 | [diff] [blame] | 119 | qemu_set_irq(s->gpio[i], 0); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 120 | s->regs[input_reg] &= ~(1 << input_shift); |
| 121 | break; |
| 122 | case PCA9552_LED_PWM0: |
| 123 | case PCA9552_LED_PWM1: |
| 124 | /* TODO */ |
| 125 | default: |
| 126 | break; |
| 127 | } |
| 128 | } |
| 129 | } |
| 130 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 131 | static uint8_t pca955x_read(PCA955xState *s, uint8_t reg) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 132 | { |
| 133 | switch (reg) { |
| 134 | case PCA9552_INPUT0: |
| 135 | case PCA9552_INPUT1: |
| 136 | case PCA9552_PSC0: |
| 137 | case PCA9552_PWM0: |
| 138 | case PCA9552_PSC1: |
| 139 | case PCA9552_PWM1: |
| 140 | case PCA9552_LS0: |
| 141 | case PCA9552_LS1: |
| 142 | case PCA9552_LS2: |
| 143 | case PCA9552_LS3: |
| 144 | return s->regs[reg]; |
| 145 | default: |
| 146 | qemu_log_mask(LOG_GUEST_ERROR, "%s: unexpected read to register %d\n", |
| 147 | __func__, reg); |
| 148 | return 0xFF; |
| 149 | } |
| 150 | } |
| 151 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 152 | static void pca955x_write(PCA955xState *s, uint8_t reg, uint8_t data) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 153 | { |
Philippe Mathieu-Daudé | b989b89 | 2020-06-23 09:27:20 +0200 | [diff] [blame] | 154 | uint16_t pins_status; |
| 155 | |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 156 | switch (reg) { |
| 157 | case PCA9552_PSC0: |
| 158 | case PCA9552_PWM0: |
| 159 | case PCA9552_PSC1: |
| 160 | case PCA9552_PWM1: |
| 161 | s->regs[reg] = data; |
| 162 | break; |
| 163 | |
| 164 | case PCA9552_LS0: |
| 165 | case PCA9552_LS1: |
| 166 | case PCA9552_LS2: |
| 167 | case PCA9552_LS3: |
Philippe Mathieu-Daudé | b989b89 | 2020-06-23 09:27:20 +0200 | [diff] [blame] | 168 | pins_status = pca955x_pins_get_status(s); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 169 | s->regs[reg] = data; |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 170 | pca955x_update_pin_input(s); |
Philippe Mathieu-Daudé | b989b89 | 2020-06-23 09:27:20 +0200 | [diff] [blame] | 171 | pca955x_display_pins_status(s, pins_status); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 172 | break; |
| 173 | |
| 174 | case PCA9552_INPUT0: |
| 175 | case PCA9552_INPUT1: |
| 176 | default: |
| 177 | qemu_log_mask(LOG_GUEST_ERROR, "%s: unexpected write to register %d\n", |
| 178 | __func__, reg); |
| 179 | } |
| 180 | } |
| 181 | |
| 182 | /* |
| 183 | * When Auto-Increment is on, the register address is incremented |
| 184 | * after each byte is sent to or received by the device. The index |
| 185 | * rollovers to 0 when the maximum register address is reached. |
| 186 | */ |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 187 | static void pca955x_autoinc(PCA955xState *s) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 188 | { |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 189 | PCA955xClass *k = PCA955X_GET_CLASS(s); |
| 190 | |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 191 | if (s->pointer != 0xFF && s->pointer & PCA9552_AUTOINC) { |
| 192 | uint8_t reg = s->pointer & 0xf; |
| 193 | |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 194 | reg = (reg + 1) % (k->max_reg + 1); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 195 | s->pointer = reg | PCA9552_AUTOINC; |
| 196 | } |
| 197 | } |
| 198 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 199 | static uint8_t pca955x_recv(I2CSlave *i2c) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 200 | { |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 201 | PCA955xState *s = PCA955X(i2c); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 202 | uint8_t ret; |
| 203 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 204 | ret = pca955x_read(s, s->pointer & 0xf); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 205 | |
| 206 | /* |
| 207 | * From the Specs: |
| 208 | * |
| 209 | * Important Note: When a Read sequence is initiated and the |
| 210 | * AI bit is set to Logic Level 1, the Read Sequence MUST |
| 211 | * start by a register different from 0. |
| 212 | * |
| 213 | * I don't know what should be done in this case, so throw an |
| 214 | * error. |
| 215 | */ |
| 216 | if (s->pointer == PCA9552_AUTOINC) { |
| 217 | qemu_log_mask(LOG_GUEST_ERROR, |
| 218 | "%s: Autoincrement read starting with register 0\n", |
| 219 | __func__); |
| 220 | } |
| 221 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 222 | pca955x_autoinc(s); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 223 | |
| 224 | return ret; |
| 225 | } |
| 226 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 227 | static int pca955x_send(I2CSlave *i2c, uint8_t data) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 228 | { |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 229 | PCA955xState *s = PCA955X(i2c); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 230 | |
| 231 | /* First byte sent by is the register address */ |
| 232 | if (s->len == 0) { |
| 233 | s->pointer = data; |
| 234 | s->len++; |
| 235 | } else { |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 236 | pca955x_write(s, s->pointer & 0xf, data); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 237 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 238 | pca955x_autoinc(s); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 239 | } |
| 240 | |
| 241 | return 0; |
| 242 | } |
| 243 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 244 | static int pca955x_event(I2CSlave *i2c, enum i2c_event event) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 245 | { |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 246 | PCA955xState *s = PCA955X(i2c); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 247 | |
| 248 | s->len = 0; |
| 249 | return 0; |
| 250 | } |
| 251 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 252 | static void pca955x_get_led(Object *obj, Visitor *v, const char *name, |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 253 | void *opaque, Error **errp) |
| 254 | { |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 255 | PCA955xClass *k = PCA955X_GET_CLASS(obj); |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 256 | PCA955xState *s = PCA955X(obj); |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 257 | int led, rc, reg; |
| 258 | uint8_t state; |
| 259 | |
| 260 | rc = sscanf(name, "led%2d", &led); |
| 261 | if (rc != 1) { |
| 262 | error_setg(errp, "%s: error reading %s", __func__, name); |
| 263 | return; |
| 264 | } |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 265 | if (led < 0 || led > k->pin_count) { |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 266 | error_setg(errp, "%s invalid led %s", __func__, name); |
| 267 | return; |
| 268 | } |
| 269 | /* |
| 270 | * Get the LSx register as the qom interface should expose the device |
| 271 | * state, not the modeled 'input line' behaviour which would come from |
| 272 | * reading the INPUTx reg |
| 273 | */ |
| 274 | reg = PCA9552_LS0 + led / 4; |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 275 | state = (pca955x_read(s, reg) >> (led % 8)) & 0x3; |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 276 | visit_type_str(v, name, (char **)&led_state[state], errp); |
| 277 | } |
| 278 | |
| 279 | /* |
| 280 | * Return an LED selector register value based on an existing one, with |
| 281 | * the appropriate 2-bit state value set for the given LED number (0-3). |
| 282 | */ |
| 283 | static inline uint8_t pca955x_ledsel(uint8_t oldval, int led_num, int state) |
| 284 | { |
| 285 | return (oldval & (~(0x3 << (led_num << 1)))) | |
| 286 | ((state & 0x3) << (led_num << 1)); |
| 287 | } |
| 288 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 289 | static void pca955x_set_led(Object *obj, Visitor *v, const char *name, |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 290 | void *opaque, Error **errp) |
| 291 | { |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 292 | PCA955xClass *k = PCA955X_GET_CLASS(obj); |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 293 | PCA955xState *s = PCA955X(obj); |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 294 | Error *local_err = NULL; |
| 295 | int led, rc, reg, val; |
| 296 | uint8_t state; |
| 297 | char *state_str; |
| 298 | |
| 299 | visit_type_str(v, name, &state_str, &local_err); |
| 300 | if (local_err) { |
| 301 | error_propagate(errp, local_err); |
| 302 | return; |
| 303 | } |
| 304 | rc = sscanf(name, "led%2d", &led); |
| 305 | if (rc != 1) { |
| 306 | error_setg(errp, "%s: error reading %s", __func__, name); |
| 307 | return; |
| 308 | } |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 309 | if (led < 0 || led > k->pin_count) { |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 310 | error_setg(errp, "%s invalid led %s", __func__, name); |
| 311 | return; |
| 312 | } |
| 313 | |
| 314 | for (state = 0; state < ARRAY_SIZE(led_state); state++) { |
| 315 | if (!strcmp(state_str, led_state[state])) { |
| 316 | break; |
| 317 | } |
| 318 | } |
| 319 | if (state >= ARRAY_SIZE(led_state)) { |
| 320 | error_setg(errp, "%s invalid led state %s", __func__, state_str); |
| 321 | return; |
| 322 | } |
| 323 | |
| 324 | reg = PCA9552_LS0 + led / 4; |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 325 | val = pca955x_read(s, reg); |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 326 | val = pca955x_ledsel(val, led % 4, state); |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 327 | pca955x_write(s, reg, val); |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 328 | } |
| 329 | |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 330 | static const VMStateDescription pca9552_vmstate = { |
| 331 | .name = "PCA9552", |
| 332 | .version_id = 0, |
| 333 | .minimum_version_id = 0, |
| 334 | .fields = (VMStateField[]) { |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 335 | VMSTATE_UINT8(len, PCA955xState), |
| 336 | VMSTATE_UINT8(pointer, PCA955xState), |
| 337 | VMSTATE_UINT8_ARRAY(regs, PCA955xState, PCA955X_NR_REGS), |
| 338 | VMSTATE_I2C_SLAVE(i2c, PCA955xState), |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 339 | VMSTATE_END_OF_LIST() |
| 340 | } |
| 341 | }; |
| 342 | |
| 343 | static void pca9552_reset(DeviceState *dev) |
| 344 | { |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 345 | PCA955xState *s = PCA955X(dev); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 346 | |
| 347 | s->regs[PCA9552_PSC0] = 0xFF; |
| 348 | s->regs[PCA9552_PWM0] = 0x80; |
| 349 | s->regs[PCA9552_PSC1] = 0xFF; |
| 350 | s->regs[PCA9552_PWM1] = 0x80; |
| 351 | s->regs[PCA9552_LS0] = 0x55; /* all OFF */ |
| 352 | s->regs[PCA9552_LS1] = 0x55; |
| 353 | s->regs[PCA9552_LS2] = 0x55; |
| 354 | s->regs[PCA9552_LS3] = 0x55; |
| 355 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 356 | pca955x_update_pin_input(s); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 357 | |
| 358 | s->pointer = 0xFF; |
| 359 | s->len = 0; |
| 360 | } |
| 361 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 362 | static void pca955x_initfn(Object *obj) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 363 | { |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 364 | PCA955xClass *k = PCA955X_GET_CLASS(obj); |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 365 | int led; |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 366 | |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 367 | assert(k->pin_count <= PCA955X_PIN_COUNT_MAX); |
| 368 | for (led = 0; led < k->pin_count; led++) { |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 369 | char *name; |
| 370 | |
| 371 | name = g_strdup_printf("led%d", led); |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 372 | object_property_add(obj, name, "bool", pca955x_get_led, pca955x_set_led, |
Markus Armbruster | d262312 | 2020-05-05 17:29:22 +0200 | [diff] [blame] | 373 | NULL, NULL); |
Joel Stanley | a90d8f8 | 2020-01-30 16:02:02 +0000 | [diff] [blame] | 374 | g_free(name); |
| 375 | } |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 376 | } |
| 377 | |
Philippe Mathieu-Daudé | 2df252d | 2020-06-23 09:27:19 +0200 | [diff] [blame] | 378 | static void pca955x_realize(DeviceState *dev, Error **errp) |
| 379 | { |
Philippe Mathieu-Daudé | 586f495 | 2020-06-23 09:27:23 +0200 | [diff] [blame] | 380 | PCA955xClass *k = PCA955X_GET_CLASS(dev); |
Philippe Mathieu-Daudé | 2df252d | 2020-06-23 09:27:19 +0200 | [diff] [blame] | 381 | PCA955xState *s = PCA955X(dev); |
| 382 | |
| 383 | if (!s->description) { |
| 384 | s->description = g_strdup("pca-unspecified"); |
| 385 | } |
Philippe Mathieu-Daudé | 586f495 | 2020-06-23 09:27:23 +0200 | [diff] [blame] | 386 | |
| 387 | qdev_init_gpio_out(dev, s->gpio, k->pin_count); |
Philippe Mathieu-Daudé | 2df252d | 2020-06-23 09:27:19 +0200 | [diff] [blame] | 388 | } |
| 389 | |
| 390 | static Property pca955x_properties[] = { |
| 391 | DEFINE_PROP_STRING("description", PCA955xState, description), |
| 392 | DEFINE_PROP_END_OF_LIST(), |
| 393 | }; |
| 394 | |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 395 | static void pca955x_class_init(ObjectClass *klass, void *data) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 396 | { |
Philippe Mathieu-Daudé | 2df252d | 2020-06-23 09:27:19 +0200 | [diff] [blame] | 397 | DeviceClass *dc = DEVICE_CLASS(klass); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 398 | I2CSlaveClass *k = I2C_SLAVE_CLASS(klass); |
| 399 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 400 | k->event = pca955x_event; |
| 401 | k->recv = pca955x_recv; |
| 402 | k->send = pca955x_send; |
Philippe Mathieu-Daudé | 2df252d | 2020-06-23 09:27:19 +0200 | [diff] [blame] | 403 | dc->realize = pca955x_realize; |
| 404 | device_class_set_props(dc, pca955x_properties); |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 405 | } |
| 406 | |
| 407 | static const TypeInfo pca955x_info = { |
| 408 | .name = TYPE_PCA955X, |
| 409 | .parent = TYPE_I2C_SLAVE, |
| 410 | .instance_init = pca955x_initfn, |
| 411 | .instance_size = sizeof(PCA955xState), |
| 412 | .class_init = pca955x_class_init, |
Philippe Mathieu-Daudé | fc1bff9 | 2020-06-29 09:47:04 +0200 | [diff] [blame^] | 413 | .class_size = sizeof(PCA955xClass), |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 414 | .abstract = true, |
| 415 | }; |
| 416 | |
| 417 | static void pca9552_class_init(ObjectClass *oc, void *data) |
| 418 | { |
| 419 | DeviceClass *dc = DEVICE_CLASS(oc); |
| 420 | PCA955xClass *pc = PCA955X_CLASS(oc); |
| 421 | |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 422 | dc->reset = pca9552_reset; |
| 423 | dc->vmsd = &pca9552_vmstate; |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 424 | pc->max_reg = PCA9552_LS3; |
| 425 | pc->pin_count = 16; |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 426 | } |
| 427 | |
| 428 | static const TypeInfo pca9552_info = { |
| 429 | .name = TYPE_PCA9552, |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 430 | .parent = TYPE_PCA955X, |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 431 | .class_init = pca9552_class_init, |
| 432 | }; |
| 433 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 434 | static void pca955x_register_types(void) |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 435 | { |
Philippe Mathieu-Daudé | 736132e | 2020-06-23 09:27:18 +0200 | [diff] [blame] | 436 | type_register_static(&pca955x_info); |
Cédric Le Goater | 5141d41 | 2018-06-08 13:15:32 +0100 | [diff] [blame] | 437 | type_register_static(&pca9552_info); |
| 438 | } |
| 439 | |
Philippe Mathieu-Daudé | ec17228 | 2020-06-23 09:27:17 +0200 | [diff] [blame] | 440 | type_init(pca955x_register_types) |