[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Qemu-devel] [PATCH v1 4/6] hw: M24Cxx I2C EEPROM device model
From: |
Peter Crosthwaite |
Subject: |
[Qemu-devel] [PATCH v1 4/6] hw: M24Cxx I2C EEPROM device model |
Date: |
Wed, 20 Feb 2013 15:29:57 +1000 |
Device model for the ST M24Cxx I2C EEPROM devices. Device can optionally be
backed onto a file for persistent storage (using -mtd-block).
Signed-off-by: Peter Crosthwaite <address@hidden>
---
default-configs/arm-softmmu.mak | 1 +
hw/Makefile.objs | 1 +
hw/m24cxx.c | 243 +++++++++++++++++++++++++++++++++++++++
3 files changed, 245 insertions(+), 0 deletions(-)
create mode 100644 hw/m24cxx.c
diff --git a/default-configs/arm-softmmu.mak b/default-configs/arm-softmmu.mak
index 2f1a5c9..9114382 100644
--- a/default-configs/arm-softmmu.mak
+++ b/default-configs/arm-softmmu.mak
@@ -26,6 +26,7 @@ CONFIG_SSI_M25P80=y
CONFIG_LAN9118=y
CONFIG_SMC91C111=y
CONFIG_DS1338=y
+CONFIG_M24CXX=y
CONFIG_PFLASH_CFI01=y
CONFIG_PFLASH_CFI02=y
diff --git a/hw/Makefile.objs b/hw/Makefile.objs
index 6b278cc..dc75c9f 100644
--- a/hw/Makefile.objs
+++ b/hw/Makefile.objs
@@ -178,6 +178,7 @@ common-obj-$(CONFIG_ADS7846) += ads7846.o
common-obj-$(CONFIG_MAX111X) += max111x.o
common-obj-$(CONFIG_DS1338) += ds1338.o
common-obj-y += i2c.o smbus.o smbus_eeprom.o
+common-obj-$(CONFIG_M24CXX) += m24cxx.o
common-obj-y += eeprom93xx.o
common-obj-y += scsi-disk.o cdrom.o hd-geometry.o block-common.o
common-obj-y += scsi-generic.o scsi-bus.o
diff --git a/hw/m24cxx.c b/hw/m24cxx.c
new file mode 100644
index 0000000..567b820
--- /dev/null
+++ b/hw/m24cxx.c
@@ -0,0 +1,243 @@
+/*
+ * ST m24Cxx I2C EEPROMs
+ *
+ * Copyright (c) 2012 Xilinx Inc.
+ * Copyright (c) 2012 Peter Crosthwaite <address@hidden>
+ *
+ * This program is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License as published by the
+ * Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+ * for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "i2c.h"
+#include "hw.h"
+#include "sysemu/blockdev.h"
+
+#ifndef M24CXX_DEBUG
+#define M24CXX_DEBUG 0
+#endif
+#define DB_PRINT(fmt, args...) do {\
+ if (M24CXX_DEBUG) {\
+ fprintf(stderr, "M24CXX: %s:" fmt, __func__, ## args);\
+ } \
+} while (0);
+
+typedef enum {
+ STOPPED,
+ ADDRESSING,
+ READING,
+ WRITING,
+} M24CXXXferState;
+
+const char *m24cxx_state_names[] = {
+ [STOPPED] = "STOPPED",
+ [ADDRESSING] = "ADDRESSING",
+ [READING] = "READING",
+ [WRITING] = "WRITING",
+};
+
+typedef struct {
+ I2CSlave i2c;
+ uint16_t cur_addr;
+ uint8_t state;
+
+ BlockDriverState *bdrv;
+ uint16_t size;
+
+ uint8_t *storage;
+} M24CXXState;
+
+#define TYPE_M24CXX "m24cxx"
+
+#define M24CXX(obj) \
+ OBJECT_CHECK(M24CXXState, (obj), TYPE_M24CXX)
+
+static void m24cxx_sync_complete(void *opaque, int ret)
+{
+ /* do nothing. Masters do not directly interact with the backing store,
+ * only the working copy so no mutexing required.
+ */
+}
+
+static void m24cxx_sync(I2CSlave *i2c)
+{
+ M24CXXState *s = M24CXX(i2c);
+ int64_t nb_sectors;
+ QEMUIOVector iov;
+
+ if (!s->bdrv) {
+ return;
+ }
+
+ /* the device is so small, just sync the whole thing */
+ nb_sectors = DIV_ROUND_UP(s->size, BDRV_SECTOR_SIZE);
+ qemu_iovec_init(&iov, 1);
+ qemu_iovec_add(&iov, s->storage, nb_sectors * BDRV_SECTOR_SIZE);
+ bdrv_aio_writev(s->bdrv, 0, &iov, nb_sectors, m24cxx_sync_complete, NULL);
+}
+
+static void m24cxx_reset(DeviceState *dev)
+{
+ M24CXXState *s = M24CXX(dev);
+
+ m24cxx_sync(I2C_SLAVE(s));
+ s->state = STOPPED;
+ s->cur_addr = 0;
+}
+
+static int m24cxx_recv(I2CSlave *i2c)
+{
+ M24CXXState *s = M24CXX(i2c);
+ int ret = 0;
+
+ if (s->state == READING) {
+ ret = s->storage[s->cur_addr++];
+ DB_PRINT("storage %x <-> %x\n", s->cur_addr-1, ret);
+ s->cur_addr %= s->size;
+ } else {
+ /* should be impossible even with a degenerate guest */
+ qemu_log_mask(LOG_GUEST_ERROR, "read from m24cxx not in read state");
+ }
+ DB_PRINT("data: %02x\n", ret);
+ return ret;
+}
+
+static int m24cxx_send(I2CSlave *i2c, uint8_t data)
+{
+ M24CXXState *s = M24CXX(i2c);
+
+ switch (s->state) {
+ case (ADDRESSING):
+ s->cur_addr &= ~0xFF;
+ s->cur_addr |= data;
+ DB_PRINT("setting address to %x\n", s->cur_addr);
+ s->state = WRITING;
+ return 0;
+ case (WRITING):
+ DB_PRINT("storage %x <-> %x\n", s->cur_addr, data);
+ s->storage[s->cur_addr++] = data;
+ s->cur_addr %= s->size;
+ return 0;
+ default:
+ DB_PRINT("write to m24cxx not in writable state\n");
+ qemu_log_mask(LOG_GUEST_ERROR, "write to m24cxx not in writable
state");
+ return 1;
+ }
+}
+
+static void m24cxx_event(I2CSlave *i2c, enum i2c_event event)
+{
+ M24CXXState *s = M24CXX(i2c);
+
+ switch (event) {
+ case I2C_START_SEND:
+ s->state = ADDRESSING;
+ break;
+ case I2C_START_RECV:
+ s->state = READING;
+ break;
+ case I2C_FINISH:
+ m24cxx_sync(i2c);
+ s->state = STOPPED;
+ break;
+ case I2C_NACK:
+ DB_PRINT("NACKED\n");
+ break;
+ }
+
+ DB_PRINT("transitioning to state %s\n", m24cxx_state_names[s->state]);
+}
+
+static void m24cxx_decode_address(I2CSlave *i2c, uint8_t address)
+{
+ M24CXXState *s = M24CXX(i2c);
+
+ s->cur_addr &= ~(0x0300);
+ s->cur_addr |= (address & ((s->size - 1) >> 8)) << 8;
+}
+
+static void m24cxx_realize(DeviceState *dev, Error **errp)
+{
+ M24CXXState *s = M24CXX(dev);
+ I2CSlave *i2c = I2C_SLAVE(dev);
+ DriveInfo *dinfo = drive_get_next(IF_MTD);
+
+ i2c->address_range = s->size >> 8 ? s->size >> 8 : 1;
+ s->storage = g_new0(uint8_t, DIV_ROUND_UP(s->size, BDRV_SECTOR_SIZE) *
+ BDRV_SECTOR_SIZE);
+
+ if (dinfo && dinfo->bdrv) {
+ s->bdrv = dinfo->bdrv;
+ /* FIXME: Move to late init */
+ if (bdrv_read(s->bdrv, 0, s->storage, DIV_ROUND_UP(s->size,
+ BDRV_SECTOR_SIZE))) {
+ error_setg(errp, "Failed to initialize I2C EEPROM!\n");
+ return;
+ }
+ } else {
+ memset(s->storage, 0xFF, s->size);
+ }
+}
+
+static void m24cxx_pre_save(void *opaque)
+{
+ m24cxx_sync((I2CSlave *)opaque);
+}
+
+static const VMStateDescription vmstate_m24cxx = {
+ .name = "m24cxx",
+ .version_id = 1,
+ .minimum_version_id = 1,
+ .minimum_version_id_old = 1,
+ .pre_save = m24cxx_pre_save,
+ .fields = (VMStateField[]) {
+ VMSTATE_I2C_SLAVE(i2c, M24CXXState),
+ VMSTATE_UINT8(state, M24CXXState),
+ VMSTATE_UINT16(cur_addr, M24CXXState),
+ VMSTATE_END_OF_LIST()
+ }
+};
+
+static Property m24cxx_properties[] = {
+ DEFINE_PROP_UINT16("size", M24CXXState, size, 1024),
+ DEFINE_PROP_END_OF_LIST(),
+};
+
+static void m24cxx_class_init(ObjectClass *klass, void *data)
+{
+ DeviceClass *dc = DEVICE_CLASS(klass);
+ I2CSlaveClass *k = I2C_SLAVE_CLASS(klass);
+
+ k->event = m24cxx_event;
+ k->recv = m24cxx_recv;
+ k->send = m24cxx_send;
+ k->decode_address = m24cxx_decode_address;
+
+ dc->realize = m24cxx_realize;
+ dc->reset = m24cxx_reset;
+ dc->vmsd = &vmstate_m24cxx;
+ dc->props = m24cxx_properties;
+}
+
+static TypeInfo m24cxx_info = {
+ .name = TYPE_M24CXX,
+ .parent = TYPE_I2C_SLAVE,
+ .instance_size = sizeof(M24CXXState),
+ .class_init = m24cxx_class_init,
+};
+
+static void m24cxx_register_types(void)
+{
+ type_register_static(&m24cxx_info);
+}
+
+type_init(m24cxx_register_types)
--
1.7.0.4
- [Qemu-devel] [PATCH v1 0/6] Xilinx Zynq I2C, Peter Crosthwaite, 2013/02/20
- [Qemu-devel] [PATCH v1 2/6] i2c: Add no_init version of i2c_create_slave, Peter Crosthwaite, 2013/02/20
- [Qemu-devel] [PATCH v1 3/6] cadence_i2c: first revision, Peter Crosthwaite, 2013/02/20
- [Qemu-devel] [PATCH v1 4/6] hw: M24Cxx I2C EEPROM device model,
Peter Crosthwaite <=
- [Qemu-devel] [PATCH v1 5/6] hw: pca9548: Device model, Peter Crosthwaite, 2013/02/20
- [Qemu-devel] [PATCH v1 6/6] xilinx_zynq: Add i2c components, Peter Crosthwaite, 2013/02/20