xref: /openbmc/qemu/hw/gpio/pcf8574.c (revision 28ae3179fc52d2e4d870b635c4a412aab99759e7)
14cbb1513SDmitriy Sharikhin /* SPDX-License-Identifier: GPL-2.0-only */
24cbb1513SDmitriy Sharikhin 
34cbb1513SDmitriy Sharikhin /*
44cbb1513SDmitriy Sharikhin  * NXP PCF8574 8-port I2C GPIO expansion chip.
54cbb1513SDmitriy Sharikhin  * Copyright (c) 2024 KNS Group (YADRO).
64cbb1513SDmitriy Sharikhin  * Written by Dmitrii Sharikhin <d.sharikhin@yadro.com>
74cbb1513SDmitriy Sharikhin  */
84cbb1513SDmitriy Sharikhin 
94cbb1513SDmitriy Sharikhin #include "qemu/osdep.h"
104cbb1513SDmitriy Sharikhin #include "hw/i2c/i2c.h"
114cbb1513SDmitriy Sharikhin #include "hw/gpio/pcf8574.h"
124cbb1513SDmitriy Sharikhin #include "hw/irq.h"
134cbb1513SDmitriy Sharikhin #include "migration/vmstate.h"
144cbb1513SDmitriy Sharikhin #include "qemu/log.h"
154cbb1513SDmitriy Sharikhin #include "qemu/module.h"
164cbb1513SDmitriy Sharikhin #include "qom/object.h"
174cbb1513SDmitriy Sharikhin 
184cbb1513SDmitriy Sharikhin /*
194cbb1513SDmitriy Sharikhin  * PCF8574 and compatible chips incorporate quasi-bidirectional
204cbb1513SDmitriy Sharikhin  * IO. Electrically it means that device sustain pull-up to line
214cbb1513SDmitriy Sharikhin  * unless IO port is configured as output _and_ driven low.
224cbb1513SDmitriy Sharikhin  *
234cbb1513SDmitriy Sharikhin  * IO access is implemented as simple I2C single-byte read
244cbb1513SDmitriy Sharikhin  * or write operation. So, to configure line to input user write 1
254cbb1513SDmitriy Sharikhin  * to corresponding bit. To configure line to output and drive it low
264cbb1513SDmitriy Sharikhin  * user write 0 to corresponding bit.
274cbb1513SDmitriy Sharikhin  *
284cbb1513SDmitriy Sharikhin  * In essence, user can think of quasi-bidirectional IO as
294cbb1513SDmitriy Sharikhin  * open-drain line, except presence of builtin rising edge acceleration
304cbb1513SDmitriy Sharikhin  * embedded in PCF8574 IC
314cbb1513SDmitriy Sharikhin  *
324cbb1513SDmitriy Sharikhin  * PCF8574 has interrupt request line, which is being pulled down when
334cbb1513SDmitriy Sharikhin  * port line state differs from last read. Port read operation clears
344cbb1513SDmitriy Sharikhin  * state and INT line returns to high state via pullup.
354cbb1513SDmitriy Sharikhin  */
364cbb1513SDmitriy Sharikhin 
374cbb1513SDmitriy Sharikhin OBJECT_DECLARE_SIMPLE_TYPE(PCF8574State, PCF8574)
384cbb1513SDmitriy Sharikhin 
394cbb1513SDmitriy Sharikhin #define PORTS_COUNT (8)
404cbb1513SDmitriy Sharikhin 
414cbb1513SDmitriy Sharikhin struct PCF8574State {
424cbb1513SDmitriy Sharikhin     I2CSlave parent_obj;
434cbb1513SDmitriy Sharikhin     uint8_t  lastrq;     /* Last requested state. If changed - assert irq */
444cbb1513SDmitriy Sharikhin     uint8_t  input;      /* external electrical line state */
454cbb1513SDmitriy Sharikhin     uint8_t  output;     /* Pull-up (1) or drive low (0) on bit */
464cbb1513SDmitriy Sharikhin     qemu_irq handler[PORTS_COUNT];
474cbb1513SDmitriy Sharikhin     qemu_irq intrq;      /* External irq request */
484cbb1513SDmitriy Sharikhin };
494cbb1513SDmitriy Sharikhin 
pcf8574_reset(DeviceState * dev)504cbb1513SDmitriy Sharikhin static void pcf8574_reset(DeviceState *dev)
514cbb1513SDmitriy Sharikhin {
524cbb1513SDmitriy Sharikhin     PCF8574State *s = PCF8574(dev);
534cbb1513SDmitriy Sharikhin     s->lastrq = MAKE_64BIT_MASK(0, PORTS_COUNT);
544cbb1513SDmitriy Sharikhin     s->input  = MAKE_64BIT_MASK(0, PORTS_COUNT);
554cbb1513SDmitriy Sharikhin     s->output = MAKE_64BIT_MASK(0, PORTS_COUNT);
564cbb1513SDmitriy Sharikhin }
574cbb1513SDmitriy Sharikhin 
pcf8574_line_state(PCF8574State * s)584cbb1513SDmitriy Sharikhin static inline uint8_t pcf8574_line_state(PCF8574State *s)
594cbb1513SDmitriy Sharikhin {
604cbb1513SDmitriy Sharikhin     /* we driving line low or external circuit does that */
614cbb1513SDmitriy Sharikhin     return s->input & s->output;
624cbb1513SDmitriy Sharikhin }
634cbb1513SDmitriy Sharikhin 
pcf8574_rx(I2CSlave * i2c)644cbb1513SDmitriy Sharikhin static uint8_t pcf8574_rx(I2CSlave *i2c)
654cbb1513SDmitriy Sharikhin {
664cbb1513SDmitriy Sharikhin     PCF8574State *s = PCF8574(i2c);
674cbb1513SDmitriy Sharikhin     uint8_t linestate = pcf8574_line_state(s);
684cbb1513SDmitriy Sharikhin     if (s->lastrq != linestate) {
694cbb1513SDmitriy Sharikhin         s->lastrq = linestate;
704cbb1513SDmitriy Sharikhin         if (s->intrq) {
714cbb1513SDmitriy Sharikhin             qemu_set_irq(s->intrq, 1);
724cbb1513SDmitriy Sharikhin         }
734cbb1513SDmitriy Sharikhin     }
744cbb1513SDmitriy Sharikhin     return linestate;
754cbb1513SDmitriy Sharikhin }
764cbb1513SDmitriy Sharikhin 
pcf8574_tx(I2CSlave * i2c,uint8_t data)774cbb1513SDmitriy Sharikhin static int pcf8574_tx(I2CSlave *i2c, uint8_t data)
784cbb1513SDmitriy Sharikhin {
794cbb1513SDmitriy Sharikhin     PCF8574State *s = PCF8574(i2c);
804cbb1513SDmitriy Sharikhin     uint8_t prev;
814cbb1513SDmitriy Sharikhin     uint8_t diff;
824cbb1513SDmitriy Sharikhin     uint8_t actual;
834cbb1513SDmitriy Sharikhin     int line = 0;
844cbb1513SDmitriy Sharikhin 
854cbb1513SDmitriy Sharikhin     prev = pcf8574_line_state(s);
864cbb1513SDmitriy Sharikhin     s->output = data;
874cbb1513SDmitriy Sharikhin     actual = pcf8574_line_state(s);
884cbb1513SDmitriy Sharikhin 
894cbb1513SDmitriy Sharikhin     for (diff = (actual ^ prev); diff; diff &= ~(1 << line)) {
904cbb1513SDmitriy Sharikhin         line = ctz32(diff);
914cbb1513SDmitriy Sharikhin         if (s->handler[line]) {
924cbb1513SDmitriy Sharikhin             qemu_set_irq(s->handler[line], (actual >> line) & 1);
934cbb1513SDmitriy Sharikhin         }
944cbb1513SDmitriy Sharikhin     }
954cbb1513SDmitriy Sharikhin 
964cbb1513SDmitriy Sharikhin     if (s->intrq) {
974cbb1513SDmitriy Sharikhin         qemu_set_irq(s->intrq, actual == s->lastrq);
984cbb1513SDmitriy Sharikhin     }
994cbb1513SDmitriy Sharikhin 
1004cbb1513SDmitriy Sharikhin     return 0;
1014cbb1513SDmitriy Sharikhin }
1024cbb1513SDmitriy Sharikhin 
1034cbb1513SDmitriy Sharikhin static const VMStateDescription vmstate_pcf8574 = {
1044cbb1513SDmitriy Sharikhin     .name               = "pcf8574",
1054cbb1513SDmitriy Sharikhin     .version_id         = 0,
1064cbb1513SDmitriy Sharikhin     .minimum_version_id = 0,
1074cbb1513SDmitriy Sharikhin     .fields = (VMStateField[]) {
1084cbb1513SDmitriy Sharikhin         VMSTATE_I2C_SLAVE(parent_obj, PCF8574State),
1094cbb1513SDmitriy Sharikhin         VMSTATE_UINT8(lastrq, PCF8574State),
1104cbb1513SDmitriy Sharikhin         VMSTATE_UINT8(input,  PCF8574State),
1114cbb1513SDmitriy Sharikhin         VMSTATE_UINT8(output, PCF8574State),
1124cbb1513SDmitriy Sharikhin         VMSTATE_END_OF_LIST()
1134cbb1513SDmitriy Sharikhin     }
1144cbb1513SDmitriy Sharikhin };
1154cbb1513SDmitriy Sharikhin 
pcf8574_gpio_set(void * opaque,int line,int level)1164cbb1513SDmitriy Sharikhin static void pcf8574_gpio_set(void *opaque, int line, int level)
1174cbb1513SDmitriy Sharikhin {
1184cbb1513SDmitriy Sharikhin     PCF8574State *s = (PCF8574State *) opaque;
1194cbb1513SDmitriy Sharikhin     assert(line >= 0 && line < ARRAY_SIZE(s->handler));
1204cbb1513SDmitriy Sharikhin 
1214cbb1513SDmitriy Sharikhin     if (level) {
1224cbb1513SDmitriy Sharikhin         s->input |=  (1 << line);
1234cbb1513SDmitriy Sharikhin     } else {
1244cbb1513SDmitriy Sharikhin         s->input &= ~(1 << line);
1254cbb1513SDmitriy Sharikhin     }
1264cbb1513SDmitriy Sharikhin 
1274cbb1513SDmitriy Sharikhin     if (pcf8574_line_state(s) != s->lastrq && s->intrq) {
1284cbb1513SDmitriy Sharikhin         qemu_set_irq(s->intrq, 0);
1294cbb1513SDmitriy Sharikhin     }
1304cbb1513SDmitriy Sharikhin }
1314cbb1513SDmitriy Sharikhin 
pcf8574_realize(DeviceState * dev,Error ** errp)1324cbb1513SDmitriy Sharikhin static void pcf8574_realize(DeviceState *dev, Error **errp)
1334cbb1513SDmitriy Sharikhin {
1344cbb1513SDmitriy Sharikhin     PCF8574State *s = PCF8574(dev);
1354cbb1513SDmitriy Sharikhin 
1364cbb1513SDmitriy Sharikhin     qdev_init_gpio_in(dev, pcf8574_gpio_set, ARRAY_SIZE(s->handler));
1374cbb1513SDmitriy Sharikhin     qdev_init_gpio_out(dev, s->handler, ARRAY_SIZE(s->handler));
1384cbb1513SDmitriy Sharikhin     qdev_init_gpio_out_named(dev, &s->intrq, "nINT", 1);
1394cbb1513SDmitriy Sharikhin }
1404cbb1513SDmitriy Sharikhin 
pcf8574_class_init(ObjectClass * klass,void * data)1414cbb1513SDmitriy Sharikhin static void pcf8574_class_init(ObjectClass *klass, void *data)
1424cbb1513SDmitriy Sharikhin {
1434cbb1513SDmitriy Sharikhin     DeviceClass   *dc = DEVICE_CLASS(klass);
1444cbb1513SDmitriy Sharikhin     I2CSlaveClass *k  = I2C_SLAVE_CLASS(klass);
1454cbb1513SDmitriy Sharikhin 
1464cbb1513SDmitriy Sharikhin     k->recv     = pcf8574_rx;
1474cbb1513SDmitriy Sharikhin     k->send     = pcf8574_tx;
1484cbb1513SDmitriy Sharikhin     dc->realize = pcf8574_realize;
149*e3d08143SPeter Maydell     device_class_set_legacy_reset(dc, pcf8574_reset);
1504cbb1513SDmitriy Sharikhin     dc->vmsd    = &vmstate_pcf8574;
1514cbb1513SDmitriy Sharikhin }
1524cbb1513SDmitriy Sharikhin 
1534cbb1513SDmitriy Sharikhin static const TypeInfo pcf8574_infos[] = {
1544cbb1513SDmitriy Sharikhin     {
1554cbb1513SDmitriy Sharikhin         .name          = TYPE_PCF8574,
1564cbb1513SDmitriy Sharikhin         .parent        = TYPE_I2C_SLAVE,
1574cbb1513SDmitriy Sharikhin         .instance_size = sizeof(PCF8574State),
1584cbb1513SDmitriy Sharikhin         .class_init    = pcf8574_class_init,
1594cbb1513SDmitriy Sharikhin     }
1604cbb1513SDmitriy Sharikhin };
1614cbb1513SDmitriy Sharikhin 
1624cbb1513SDmitriy Sharikhin DEFINE_TYPES(pcf8574_infos);
163