qemu-devel
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

[Qemu-devel] [PATCH v2 4/4] Add qtest support for i.MX I2C device emulat


From: Jean-Christophe DUBOIS
Subject: [Qemu-devel] [PATCH v2 4/4] Add qtest support for i.MX I2C device emulation.
Date: Sat, 4 May 2013 16:09:14 +0200

This is using a ds1338 RTC chip on the i2c bus. This RTC
chip is nop present on the real board

Signed-off-by: Jean-Christophe DUBOIS <address@hidden>
---
 tests/Makefile         |   3 +
 tests/ds1338-test.c    |  64 ++++++++++++++
 tests/libqos/i2c-imx.c | 224 +++++++++++++++++++++++++++++++++++++++++++++++++
 tests/libqos/i2c.h     |   3 +
 4 files changed, 294 insertions(+)
 create mode 100644 tests/ds1338-test.c
 create mode 100644 tests/libqos/i2c-imx.c

diff --git a/tests/Makefile b/tests/Makefile
index bf41d10..5f7a0e0 100644
--- a/tests/Makefile
+++ b/tests/Makefile
@@ -64,6 +64,7 @@ gcov-files-x86_64-y = $(subst 
i386-softmmu/,x86_64-softmmu/,$(gcov-files-i386-y)
 gcov-files-sparc-y += hw/m48t59.c
 gcov-files-sparc64-y += hw/m48t59.c
 check-qtest-arm-y = tests/tmp105-test$(EXESUF)
+check-qtest-arm-y += tests/ds1338-test$(EXESUF)
 gcov-files-arm-y += hw/tmp105.c
 
 GENERATED_HEADERS += tests/test-qapi-types.h tests/test-qapi-visit.h 
tests/test-qmp-commands.h
@@ -123,12 +124,14 @@ libqos-obj-y += tests/libqos/i2c.o
 libqos-pc-obj-y = $(libqos-obj-y) tests/libqos/pci-pc.o 
tests/libqos/fw_cfg-pc.o
 libqos-pc-obj-y += tests/libqos/malloc-pc.o
 libqos-omap-obj-y = $(libqos-obj-y) tests/libqos/i2c-omap.o
+libqos-imx-obj-y = $(libqos-obj-y) tests/libqos/i2c-imx.o
 
 tests/rtc-test$(EXESUF): tests/rtc-test.o
 tests/m48t59-test$(EXESUF): tests/m48t59-test.o
 tests/fdc-test$(EXESUF): tests/fdc-test.o
 tests/hd-geo-test$(EXESUF): tests/hd-geo-test.o
 tests/tmp105-test$(EXESUF): tests/tmp105-test.o $(libqos-omap-obj-y)
+tests/ds1338-test$(EXESUF): tests/ds1338-test.o $(libqos-imx-obj-y)
 tests/i440fx-test$(EXESUF): tests/i440fx-test.o $(libqos-pc-obj-y)
 tests/fw_cfg-test$(EXESUF): tests/fw_cfg-test.o $(libqos-pc-obj-y)
 
diff --git a/tests/ds1338-test.c b/tests/ds1338-test.c
new file mode 100644
index 0000000..3e3fa0b
--- /dev/null
+++ b/tests/ds1338-test.c
@@ -0,0 +1,64 @@
+/*
+ * QTest testcase for the DS1338 RTC
+ *
+ * Copyright (c) 2013 Jean-Christophe Dubois
+ *
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
+ * See the COPYING file in the top-level directory.
+ */
+#include "libqtest.h"
+#include "libqos/i2c.h"
+
+#include <glib.h>
+
+#define IMX25_I2C_0_BASE 0x43F80000
+
+#define DS1338_ADDR 0x68
+
+static I2CAdapter *i2c;
+static uint8_t addr;
+
+#define bcd2bin(x)        (((x) & 0x0f) + ((x) >> 4) * 10)
+
+static void send_and_receive(void)
+{
+    uint8_t cmd[1];
+    uint8_t resp[7];
+    time_t now = time(NULL);
+    struct tm *tm_ptr = localtime(&now);
+
+    /* reset the index in the RTC memory */
+    cmd[0] = 0;
+    i2c_send(i2c, addr, cmd, 1);
+
+    /* retrieve the date */
+    i2c_recv(i2c, addr, resp, 7);
+
+    /* check retreived time againt local time */
+    g_assert_cmpuint(bcd2bin(resp[4]), == , tm_ptr->tm_mday);
+    g_assert_cmpuint(bcd2bin(resp[5]), == , 1 + tm_ptr->tm_mon);
+    g_assert_cmpuint(2000 + bcd2bin(resp[6]), == , 1900 + tm_ptr->tm_year);
+}
+
+int main(int argc, char **argv)
+{
+    QTestState *s = NULL;
+    int ret;
+
+    g_test_init(&argc, &argv, NULL);
+
+    s = qtest_start("-display none -machine imx25_3ds");
+    i2c = imx_i2c_create(IMX25_I2C_0_BASE);
+    addr = DS1338_ADDR;
+
+    qtest_add_func("/ds1338/tx-rx", send_and_receive);
+
+    ret = g_test_run();
+
+    if (s) {
+        qtest_quit(s);
+    }
+    g_free(i2c);
+
+    return ret;
+}
diff --git a/tests/libqos/i2c-imx.c b/tests/libqos/i2c-imx.c
new file mode 100644
index 0000000..da7316f
--- /dev/null
+++ b/tests/libqos/i2c-imx.c
@@ -0,0 +1,224 @@
+/*
+ * QTest i.MX I2C driver
+ *
+ * Copyright (c) 2013 Jean-Christophe Dubois
+ *
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
+ * See the COPYING file in the top-level directory.
+ */
+#include "libqos/i2c.h"
+
+#include <glib.h>
+#include <string.h>
+
+#include "qemu/osdep.h"
+#include "qemu/bswap.h"
+#include "libqtest.h"
+
+enum IMXI2CRegisters {
+    IMX_I2C_IADR = 0x00,
+    IMX_I2C_IFDR = 0x04,
+    IMX_I2C_I2CR = 0x08,
+    IMX_I2C_I2SR = 0x0c,
+    IMX_I2C_I2DR = 0x10,
+};
+
+enum IMXI2CCRBits {
+    IMX_I2C_I2CR_IEN  = 1 << 7,
+    IMX_I2C_I2CR_IIEN = 1 << 6,
+    IMX_I2C_I2CR_MSTA = 1 << 5,
+    IMX_I2C_I2CR_MTX  = 1 << 4,
+    IMX_I2C_I2CR_TXAK = 1 << 3,
+    IMX_I2C_I2CR_RSTA = 1 << 2,
+};
+
+enum IMXI2CSRBits {
+    IMX_I2C_I2SR_ICF  = 1 << 7,
+    IMX_I2C_I2SR_IAAF = 1 << 6,
+    IMX_I2C_I2SR_IBB  = 1 << 5,
+    IMX_I2C_I2SR_IAL  = 1 << 4,
+    IMX_I2C_I2SR_SRW  = 1 << 2,
+    IMX_I2C_I2SR_IIF  = 1 << 1,
+    IMX_I2C_I2SR_RXAK = 1 << 0,
+};
+
+enum IMXI2CDirection {
+    IMX_I2C_READ,
+    IMX_I2C_WRITE,
+};
+
+typedef struct IMXI2C {
+    I2CAdapter parent;
+
+    uint64_t addr;
+} IMXI2C;
+
+
+static void imx_i2c_set_slave_addr(IMXI2C *s, uint8_t addr,
+                                   enum IMXI2CDirection direction)
+{
+    writeb(s->addr + IMX_I2C_I2DR, (addr << 1) |
+           (direction == IMX_I2C_READ ? 1 : 0));
+}
+
+static void imx_i2c_send(I2CAdapter *i2c, uint8_t addr,
+                         const uint8_t *buf, uint16_t len)
+{
+    IMXI2C *s = (IMXI2C *)i2c;
+    uint8_t data;
+    uint8_t status;
+    uint16_t size = 0;
+
+    if (!len) {
+        return;
+    }
+
+    /* set the bus for write */
+    data = IMX_I2C_I2CR_IEN |
+           IMX_I2C_I2CR_IIEN |
+           IMX_I2C_I2CR_MSTA |
+           IMX_I2C_I2CR_MTX |
+           IMX_I2C_I2CR_TXAK;
+
+    writeb(s->addr + IMX_I2C_I2CR, data);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IBB) != 0);
+
+    /* set the slave address */
+    imx_i2c_set_slave_addr(s, addr, IMX_I2C_WRITE);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IIF) != 0);
+    g_assert((status & IMX_I2C_I2SR_RXAK) == 0);
+
+    /* ack the interrupt */
+    writeb(s->addr + IMX_I2C_I2SR, 0);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IIF) == 0);
+
+    while (size < len) {
+        /* check we are still busy */
+        status = readb(s->addr + IMX_I2C_I2SR);
+        g_assert((status & IMX_I2C_I2SR_IBB) != 0);
+
+        /* write the data */
+        writeb(s->addr + IMX_I2C_I2DR, buf[size]);
+        status = readb(s->addr + IMX_I2C_I2SR);
+        g_assert((status & IMX_I2C_I2SR_IIF) != 0);
+        g_assert((status & IMX_I2C_I2SR_RXAK) == 0);
+
+        /* ack the interrupt */
+        writeb(s->addr + IMX_I2C_I2SR, 0);
+        status = readb(s->addr + IMX_I2C_I2SR);
+        g_assert((status & IMX_I2C_I2SR_IIF) == 0);
+
+        size++;
+    }
+
+    /* release the bus */
+    data &= ~(IMX_I2C_I2CR_MSTA | IMX_I2C_I2CR_MTX);
+    writeb(s->addr + IMX_I2C_I2CR, data);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IBB) == 0);
+}
+
+static void imx_i2c_recv(I2CAdapter *i2c, uint8_t addr,
+                         uint8_t *buf, uint16_t len)
+{
+    IMXI2C *s = (IMXI2C *)i2c;
+    uint8_t data;
+    uint8_t status;
+    uint16_t size = 0;
+
+    if (!len) {
+        return;
+    }
+
+    /* set the bus for write */
+    data = IMX_I2C_I2CR_IEN |
+           IMX_I2C_I2CR_IIEN |
+           IMX_I2C_I2CR_MSTA |
+           IMX_I2C_I2CR_MTX |
+           IMX_I2C_I2CR_TXAK;
+
+    writeb(s->addr + IMX_I2C_I2CR, data);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IBB) != 0);
+
+    /* set the slave address */
+    imx_i2c_set_slave_addr(s, addr, IMX_I2C_READ);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IIF) != 0);
+    g_assert((status & IMX_I2C_I2SR_RXAK) == 0);
+
+    /* ack the interrupt */
+    writeb(s->addr + IMX_I2C_I2SR, 0);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IIF) == 0);
+
+    /* set the bus for read */
+    data &= ~IMX_I2C_I2CR_MTX;
+    /* if only one byte don't ack */
+    if (len != 1) {
+        data &= ~IMX_I2C_I2CR_TXAK;
+    }
+    writeb(s->addr + IMX_I2C_I2CR, data);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IBB) != 0);
+
+    /* dummy read */
+    readb(s->addr + IMX_I2C_I2DR);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IIF) != 0);
+
+    /* ack the interrupt */
+    writeb(s->addr + IMX_I2C_I2SR, 0);
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IIF) == 0);
+
+    while (size < len) {
+        /* check we are still busy */
+        status = readb(s->addr + IMX_I2C_I2SR);
+        g_assert((status & IMX_I2C_I2SR_IBB) != 0);
+
+        if (size == (len - 1)) {
+            /* stop the read transaction */
+            data &= ~(IMX_I2C_I2CR_MSTA | IMX_I2C_I2CR_MTX);
+        } else {
+            /* ack the data read */
+            data |= IMX_I2C_I2CR_TXAK;
+        }
+        writeb(s->addr + IMX_I2C_I2CR, data);
+
+        /* read the data */
+        buf[size] = readb(s->addr + IMX_I2C_I2DR);
+
+        if (size != (len - 1)) {
+            status = readb(s->addr + IMX_I2C_I2SR);
+            g_assert((status & IMX_I2C_I2SR_IIF) != 0);
+
+            /* ack the interrupt */
+            writeb(s->addr + IMX_I2C_I2SR, 0);
+        }
+
+        status = readb(s->addr + IMX_I2C_I2SR);
+        g_assert((status & IMX_I2C_I2SR_IIF) == 0);
+
+        size++;
+    }
+
+    status = readb(s->addr + IMX_I2C_I2SR);
+    g_assert((status & IMX_I2C_I2SR_IBB) == 0);
+}
+
+I2CAdapter *imx_i2c_create(uint64_t addr)
+{
+    IMXI2C *s = g_malloc0(sizeof(*s));
+    I2CAdapter *i2c = (I2CAdapter *)s;
+
+    s->addr = addr;
+
+    i2c->send = imx_i2c_send;
+    i2c->recv = imx_i2c_recv;
+
+    return i2c;
+}
diff --git a/tests/libqos/i2c.h b/tests/libqos/i2c.h
index 1ce9af4..c21f1dc 100644
--- a/tests/libqos/i2c.h
+++ b/tests/libqos/i2c.h
@@ -27,4 +27,7 @@ void i2c_recv(I2CAdapter *i2c, uint8_t addr,
 /* libi2c-omap.c */
 I2CAdapter *omap_i2c_create(uint64_t addr);
 
+/* libi2c-imx.c */
+I2CAdapter *imx_i2c_create(uint64_t addr);
+
 #endif
-- 
1.8.1.2




reply via email to

[Prev in Thread] Current Thread [Next in Thread]