hw/gpio: introduce pcf8574 driver
NXP PCF8574 and compatible ICs are simple I2C GPIO expanders. PCF8574 incorporates quasi-bidirectional IO, and simple communication protocol, when IO read is I2C byte read, and IO write is I2C byte write. User can think of it as open-drain port, when line high state is input and line low state is output. Signed-off-by: Dmitrii Sharikhin <d.sharikhin@yadro.com> Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org> Message-ID: <f1552d822276e878d84c01eba2cf2c7c9ebdde00.camel@yadro.com> Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
This commit is contained in:
parent
2ea09fe85a
commit
4cbb1513a2
@ -2503,6 +2503,12 @@ S: Maintained
|
||||
F: hw/i2c/i2c_mux_pca954x.c
|
||||
F: include/hw/i2c/i2c_mux_pca954x.h
|
||||
|
||||
pcf8574
|
||||
M: Dmitrii Sharikhin <d.sharikhin@yadro.com>
|
||||
S: Maintained
|
||||
F: hw/gpio/pcf8574.c
|
||||
F: include/gpio/pcf8574.h
|
||||
|
||||
Generic Loader
|
||||
M: Alistair Francis <alistair@alistair23.me>
|
||||
S: Maintained
|
||||
|
@ -19,3 +19,7 @@ config SIFIVE_GPIO
|
||||
|
||||
config STM32L4X5_GPIO
|
||||
bool
|
||||
|
||||
config PCF8574
|
||||
bool
|
||||
depends on I2C
|
||||
|
@ -16,3 +16,4 @@ system_ss.add(when: 'CONFIG_RASPI', if_true: files(
|
||||
system_ss.add(when: 'CONFIG_STM32L4X5_SOC', if_true: files('stm32l4x5_gpio.c'))
|
||||
system_ss.add(when: 'CONFIG_ASPEED_SOC', if_true: files('aspeed_gpio.c'))
|
||||
system_ss.add(when: 'CONFIG_SIFIVE_GPIO', if_true: files('sifive_gpio.c'))
|
||||
system_ss.add(when: 'CONFIG_PCF8574', if_true: files('pcf8574.c'))
|
||||
|
162
hw/gpio/pcf8574.c
Normal file
162
hw/gpio/pcf8574.c
Normal file
@ -0,0 +1,162 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
|
||||
/*
|
||||
* NXP PCF8574 8-port I2C GPIO expansion chip.
|
||||
* Copyright (c) 2024 KNS Group (YADRO).
|
||||
* Written by Dmitrii Sharikhin <d.sharikhin@yadro.com>
|
||||
*/
|
||||
|
||||
#include "qemu/osdep.h"
|
||||
#include "hw/i2c/i2c.h"
|
||||
#include "hw/gpio/pcf8574.h"
|
||||
#include "hw/irq.h"
|
||||
#include "migration/vmstate.h"
|
||||
#include "qemu/log.h"
|
||||
#include "qemu/module.h"
|
||||
#include "qom/object.h"
|
||||
|
||||
/*
|
||||
* PCF8574 and compatible chips incorporate quasi-bidirectional
|
||||
* IO. Electrically it means that device sustain pull-up to line
|
||||
* unless IO port is configured as output _and_ driven low.
|
||||
*
|
||||
* IO access is implemented as simple I2C single-byte read
|
||||
* or write operation. So, to configure line to input user write 1
|
||||
* to corresponding bit. To configure line to output and drive it low
|
||||
* user write 0 to corresponding bit.
|
||||
*
|
||||
* In essence, user can think of quasi-bidirectional IO as
|
||||
* open-drain line, except presence of builtin rising edge acceleration
|
||||
* embedded in PCF8574 IC
|
||||
*
|
||||
* PCF8574 has interrupt request line, which is being pulled down when
|
||||
* port line state differs from last read. Port read operation clears
|
||||
* state and INT line returns to high state via pullup.
|
||||
*/
|
||||
|
||||
OBJECT_DECLARE_SIMPLE_TYPE(PCF8574State, PCF8574)
|
||||
|
||||
#define PORTS_COUNT (8)
|
||||
|
||||
struct PCF8574State {
|
||||
I2CSlave parent_obj;
|
||||
uint8_t lastrq; /* Last requested state. If changed - assert irq */
|
||||
uint8_t input; /* external electrical line state */
|
||||
uint8_t output; /* Pull-up (1) or drive low (0) on bit */
|
||||
qemu_irq handler[PORTS_COUNT];
|
||||
qemu_irq intrq; /* External irq request */
|
||||
};
|
||||
|
||||
static void pcf8574_reset(DeviceState *dev)
|
||||
{
|
||||
PCF8574State *s = PCF8574(dev);
|
||||
s->lastrq = MAKE_64BIT_MASK(0, PORTS_COUNT);
|
||||
s->input = MAKE_64BIT_MASK(0, PORTS_COUNT);
|
||||
s->output = MAKE_64BIT_MASK(0, PORTS_COUNT);
|
||||
}
|
||||
|
||||
static inline uint8_t pcf8574_line_state(PCF8574State *s)
|
||||
{
|
||||
/* we driving line low or external circuit does that */
|
||||
return s->input & s->output;
|
||||
}
|
||||
|
||||
static uint8_t pcf8574_rx(I2CSlave *i2c)
|
||||
{
|
||||
PCF8574State *s = PCF8574(i2c);
|
||||
uint8_t linestate = pcf8574_line_state(s);
|
||||
if (s->lastrq != linestate) {
|
||||
s->lastrq = linestate;
|
||||
if (s->intrq) {
|
||||
qemu_set_irq(s->intrq, 1);
|
||||
}
|
||||
}
|
||||
return linestate;
|
||||
}
|
||||
|
||||
static int pcf8574_tx(I2CSlave *i2c, uint8_t data)
|
||||
{
|
||||
PCF8574State *s = PCF8574(i2c);
|
||||
uint8_t prev;
|
||||
uint8_t diff;
|
||||
uint8_t actual;
|
||||
int line = 0;
|
||||
|
||||
prev = pcf8574_line_state(s);
|
||||
s->output = data;
|
||||
actual = pcf8574_line_state(s);
|
||||
|
||||
for (diff = (actual ^ prev); diff; diff &= ~(1 << line)) {
|
||||
line = ctz32(diff);
|
||||
if (s->handler[line]) {
|
||||
qemu_set_irq(s->handler[line], (actual >> line) & 1);
|
||||
}
|
||||
}
|
||||
|
||||
if (s->intrq) {
|
||||
qemu_set_irq(s->intrq, actual == s->lastrq);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_pcf8574 = {
|
||||
.name = "pcf8574",
|
||||
.version_id = 0,
|
||||
.minimum_version_id = 0,
|
||||
.fields = (VMStateField[]) {
|
||||
VMSTATE_I2C_SLAVE(parent_obj, PCF8574State),
|
||||
VMSTATE_UINT8(lastrq, PCF8574State),
|
||||
VMSTATE_UINT8(input, PCF8574State),
|
||||
VMSTATE_UINT8(output, PCF8574State),
|
||||
VMSTATE_END_OF_LIST()
|
||||
}
|
||||
};
|
||||
|
||||
static void pcf8574_gpio_set(void *opaque, int line, int level)
|
||||
{
|
||||
PCF8574State *s = (PCF8574State *) opaque;
|
||||
assert(line >= 0 && line < ARRAY_SIZE(s->handler));
|
||||
|
||||
if (level) {
|
||||
s->input |= (1 << line);
|
||||
} else {
|
||||
s->input &= ~(1 << line);
|
||||
}
|
||||
|
||||
if (pcf8574_line_state(s) != s->lastrq && s->intrq) {
|
||||
qemu_set_irq(s->intrq, 0);
|
||||
}
|
||||
}
|
||||
|
||||
static void pcf8574_realize(DeviceState *dev, Error **errp)
|
||||
{
|
||||
PCF8574State *s = PCF8574(dev);
|
||||
|
||||
qdev_init_gpio_in(dev, pcf8574_gpio_set, ARRAY_SIZE(s->handler));
|
||||
qdev_init_gpio_out(dev, s->handler, ARRAY_SIZE(s->handler));
|
||||
qdev_init_gpio_out_named(dev, &s->intrq, "nINT", 1);
|
||||
}
|
||||
|
||||
static void pcf8574_class_init(ObjectClass *klass, void *data)
|
||||
{
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
I2CSlaveClass *k = I2C_SLAVE_CLASS(klass);
|
||||
|
||||
k->recv = pcf8574_rx;
|
||||
k->send = pcf8574_tx;
|
||||
dc->realize = pcf8574_realize;
|
||||
dc->reset = pcf8574_reset;
|
||||
dc->vmsd = &vmstate_pcf8574;
|
||||
}
|
||||
|
||||
static const TypeInfo pcf8574_infos[] = {
|
||||
{
|
||||
.name = TYPE_PCF8574,
|
||||
.parent = TYPE_I2C_SLAVE,
|
||||
.instance_size = sizeof(PCF8574State),
|
||||
.class_init = pcf8574_class_init,
|
||||
}
|
||||
};
|
||||
|
||||
DEFINE_TYPES(pcf8574_infos);
|
15
include/hw/gpio/pcf8574.h
Normal file
15
include/hw/gpio/pcf8574.h
Normal file
@ -0,0 +1,15 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
|
||||
/*
|
||||
* NXP PCF8574 8-port I2C GPIO expansion chip.
|
||||
*
|
||||
* Copyright (c) 2024 KNS Group (YADRO).
|
||||
* Written by Dmitrii Sharikhin <d.sharikhin@yadro.com>
|
||||
*/
|
||||
|
||||
#ifndef _HW_GPIO_PCF8574
|
||||
#define _HW_GPIO_PCF8574
|
||||
|
||||
#define TYPE_PCF8574 "pcf8574"
|
||||
|
||||
#endif /* _HW_GPIO_PCF8574 */
|
Loading…
Reference in New Issue
Block a user