[Top][All Lists]
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Qemu-devel] [PATCH 04/11] Goldfish: Added timing devices.
From: |
Patrick Jackson |
Subject: |
[Qemu-devel] [PATCH 04/11] Goldfish: Added timing devices. |
Date: |
Mon, 22 Aug 2011 05:38:56 -0400 |
Includes a timer and a real time clock.
---
Makefile.target | 2 +-
hw/android_arm.c | 2 +
hw/goldfish_device.h | 2 +
hw/goldfish_timer.c | 241 ++++++++++++++++++++++++++++++++++++++++++++++++++
4 files changed, 246 insertions(+), 1 deletions(-)
create mode 100644 hw/goldfish_timer.c
diff --git a/Makefile.target b/Makefile.target
index c251b99..2c802fa 100644
--- a/Makefile.target
+++ b/Makefile.target
@@ -360,7 +360,7 @@ obj-arm-y += syborg_virtio.o
obj-arm-y += vexpress.o
obj-arm-y += strongarm.o
obj-arm-y += collie.o
-obj-arm-y += android_arm.o goldfish_device.o goldfish_interrupt.o
+obj-arm-y += android_arm.o goldfish_device.o goldfish_interrupt.o goldfish_timer.o
obj-sh4-y = shix.o r2d.o sh7750.o sh7750_regnames.o tc58128.o
obj-sh4-y += sh_timer.o sh_serial.o sh_intc.o sh_pci.o sm501.o
diff --git a/hw/android_arm.c b/hw/android_arm.c
index aaa0b26..90c75c5 100644
--- a/hw/android_arm.c
+++ b/hw/android_arm.c
@@ -53,6 +53,8 @@ static void android_arm_init_(ram_addr_t ram_size,
GoldfishBus *gbus = goldfish_bus_init(0xff001000, 1);
gf_int = goldfish_int_create(gbus, 0xff000000, cpu_pic[ARM_PIC_CPU_IRQ], cpu_pic[ARM_PIC_CPU_FIQ]);
goldfish_device_init(gf_int, 0xff010000, 10);
+ goldfish_timer_create(gbus, 0xff003000, 3);
+ goldfish_rtc_create(gbus);
info.ram_size = ram_size;
info.kernel_filename = kernel_filename;
diff --git a/hw/goldfish_device.h b/hw/goldfish_device.h
index cee6abc..0d5b91e 100644
--- a/hw/goldfish_device.h
+++ b/hw/goldfish_device.h
@@ -43,6 +43,8 @@ typedef struct GoldfishBus {
/* QDEV device creation */
GoldfishBus *goldfish_bus_init(uint32_t base, uint32_t irq);
DeviceState *goldfish_int_create(GoldfishBus *gbus, uint32_t base, qemu_irq parent_irq, qemu_irq parent_fiq);
+DeviceState *goldfish_timer_create(GoldfishBus *gbus, uint32_t base, int irq);
+DeviceState *goldfish_rtc_create(GoldfishBus *gbus);
/* Global functions provided by Goldfish devices */
void goldfish_bus_register_withprop(GoldfishDeviceInfo *info);
diff --git a/hw/goldfish_timer.c b/hw/goldfish_timer.c
new file mode 100644
index 0000000..8db35c7
--- /dev/null
+++ b/hw/goldfish_timer.c
@@ -0,0 +1,241 @@
+/* Copyright (C) 2007-2008 The Android Open Source Project
+**
+** This software is licensed under the terms of the GNU General Public
+** License version 2, as published by the Free Software Foundation, and
+** may be copied, distributed, and modified under those terms.
+**
+** 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.
+*/
+#include "qemu-common.h"
+#include "qemu-timer.h"
+#include "cpu.h"
+#include "arm-misc.h"
+#include "goldfish_device.h"
+#include "hw/hw.h"
+
+enum {
+ TIMER_TIME_LOW = 0x00, // get low bits of current time and update TIMER_TIME_HIGH
+ TIMER_TIME_HIGH = 0x04, // get high bits of time at last TIMER_TIME_LOW read
+ TIMER_ALARM_LOW = 0x08, // set low bits of alarm and activate it
+ TIMER_ALARM_HIGH = 0x0c, // set high bits of next alarm
+ TIMER_CLEAR_INTERRUPT = 0x10,
+ TIMER_CLEAR_ALARM = 0x14
+};
+
+typedef struct GoldfishTimerDevice {
+ GoldfishDevice dev;
+ uint32_t alarm_low_ns;
+ int32_t alarm_high_ns;
+ int64_t now_ns;
+ int armed;
+ QEMUTimer *timer;
+} GoldfishTimerDevice;
+
+static uint32_t goldfish_timer_read(void *opaque, target_phys_addr_t offset)
+{
+ GoldfishTimerDevice *s = (GoldfishTimerDevice *)opaque;
+ switch(offset) {
+ case TIMER_TIME_LOW:
+ s->now_ns = qemu_get_clock_ns(vm_clock);
+ return s->now_ns;
+ case TIMER_TIME_HIGH:
+ return s->now_ns >> 32;
+ default:
+ cpu_abort (cpu_single_env, "goldfish_timer_read: Bad offset %x\n", offset);
+ return 0;
+ }
+}
+
+static void goldfish_timer_write(void *opaque, target_phys_addr_t offset, uint32_t value_ns)
+{
+ GoldfishTimerDevice *s = (GoldfishTimerDevice *)opaque;
+ int64_t alarm_ns, now_ns;
+ switch(offset) {
+ case TIMER_ALARM_LOW:
+ s->alarm_low_ns = value_ns;
+ alarm_ns = (s->alarm_low_ns | (int64_t)s->alarm_high_ns << 32);
+ now_ns = qemu_get_clock_ns(vm_clock);
+ if (alarm_ns <= now_ns) {
+ goldfish_device_set_irq(&s->dev, 0, 1);
+ } else {
+ qemu_mod_timer(s->timer, alarm_ns);
+ s->armed = 1;
+ }
+ break;
+ case TIMER_ALARM_HIGH:
+ s->alarm_high_ns = value_ns;
+ break;
+ case TIMER_CLEAR_ALARM:
+ qemu_del_timer(s->timer);
+ s->armed = 0;
+ /* fall through */
+ case TIMER_CLEAR_INTERRUPT:
+ goldfish_device_set_irq(&s->dev, 0, 0);
+ break;
+ default:
+ cpu_abort (cpu_single_env, "goldfish_timer_write: Bad offset %x\n", offset);
+ }
+}
+
+static void goldfish_timer_tick(void *opaque)
+{
+ GoldfishTimerDevice *s = (GoldfishTimerDevice *)opaque;
+
+ s->armed = 0;
+ goldfish_device_set_irq(&s->dev, 0, 1);
+}
+
+typedef struct GoldfishRTCDevice {
+ GoldfishDevice dev;
+ uint32_t alarm_low;
+ int32_t alarm_high;
+ int64_t now;
+} GoldfishRTCDevice;
+
+static uint32_t goldfish_rtc_read(void *opaque, target_phys_addr_t offset)
+{
+ GoldfishRTCDevice *s = (GoldfishRTCDevice *)opaque;
+ switch(offset) {
+ case 0x0:
+ s->now = (int64_t)time(NULL) * 1000000000;
+ return s->now;
+ case 0x4:
+ return s->now >> 32;
+ default:
+ cpu_abort (cpu_single_env, "goldfish_rtc_read: Bad offset %x\n", offset);
+ return 0;
+ }
+}
+
+static void goldfish_rtc_write(void *opaque, target_phys_addr_t offset, uint32_t value)
+{
+ GoldfishRTCDevice *s = (GoldfishRTCDevice *)opaque;
+ int64_t alarm;
+ switch(offset) {
+ case 0x8:
+ s->alarm_low = value;
+ alarm = s->alarm_low | (int64_t)s->alarm_high << 32;
+ //printf("next alarm at %lld, tps %lld\n", alarm, ticks_per_sec);
+ //qemu_mod_timer(s->timer, alarm);
+ break;
+ case 0xc:
+ s->alarm_high = value;
+ //printf("alarm_high %d\n", s->alarm_high);
+ break;
+ case 0x10:
+ goldfish_device_set_irq(&s->dev, 0, 0);
+ break;
+ default:
+ cpu_abort (cpu_single_env, "goldfish_rtc_write: Bad offset %x\n", offset);
+ }
+}
+
+static CPUReadMemoryFunc *goldfish_timer_readfn[] = {
+ goldfish_timer_read,
+ goldfish_timer_read,
+ goldfish_timer_read
+};
+
+static CPUWriteMemoryFunc *goldfish_timer_writefn[] = {
+ goldfish_timer_write,
+ goldfish_timer_write,
+ goldfish_timer_write
+};
+
+static CPUReadMemoryFunc *goldfish_rtc_readfn[] = {
+ goldfish_rtc_read,
+ goldfish_rtc_read,
+ goldfish_rtc_read
+};
+
+static CPUWriteMemoryFunc *goldfish_rtc_writefn[] = {
+ goldfish_rtc_write,
+ goldfish_rtc_write,
+ goldfish_rtc_write
+};
+
+static int goldfish_timer_init(GoldfishDevice *dev)
+{
+ GoldfishTimerDevice *tdev = (GoldfishTimerDevice *)dev;
+ tdev->timer = qemu_new_timer_ns(vm_clock, goldfish_timer_tick, tdev);
+
+ return 0;
+}
+
+DeviceState *goldfish_timer_create(GoldfishBus *gbus, uint32_t base, int irq)
+{
+ DeviceState *dev;
+ char *name = (char *)"goldfish_timer";
+
+ dev = qdev_create(&gbus->bus, name);
+ qdev_prop_set_string(dev, "name", name);
+ qdev_prop_set_uint32(dev, "base", base);
+ qdev_prop_set_uint32(dev, "irq", irq);
+ qdev_init_nofail(dev);
+
+ return dev;
+}
+
+static GoldfishDeviceInfo goldfish_timer_info = {
+ .init = goldfish_timer_init,
+ .readfn = goldfish_timer_readfn,
+ .writefn = goldfish_timer_writefn,
+ .qdev.size = sizeof(GoldfishTimerDevice),
+ .qdev.props = (Property[]) {
+ DEFINE_PROP_UINT32("base", GoldfishDevice, base, 0),
+ DEFINE_PROP_UINT32("irq", GoldfishDevice, irq, 0),
+ DEFINE_PROP_UINT32("id", GoldfishDevice, id, -1),
+ DEFINE_PROP_UINT32("size", GoldfishDevice, size, 0x1000),
+ DEFINE_PROP_UINT32("irq_count", GoldfishDevice, irq_count, 1),
+ DEFINE_PROP_STRING("name", GoldfishDevice, name),
+ DEFINE_PROP_END_OF_LIST(),
+ },
+};
+
+static void goldfish_timer_register(void)
+{
+ goldfish_bus_register_withprop(&goldfish_timer_info);
+}
+device_init(goldfish_timer_register);
+
+static int goldfish_rtc_init(GoldfishDevice *dev)
+{
+ return 0;
+}
+
+DeviceState *goldfish_rtc_create(GoldfishBus *gbus)
+{
+ DeviceState *dev;
+ char *name = (char *)"goldfish_rtc";
+
+ dev = qdev_create(&gbus->bus, name);
+ qdev_prop_set_string(dev, "name", name);
+ qdev_init_nofail(dev);
+
+ return dev;
+}
+
+static GoldfishDeviceInfo goldfish_rtc_info = {
+ .init = goldfish_rtc_init,
+ .readfn = goldfish_rtc_readfn,
+ .writefn = goldfish_rtc_writefn,
+ .qdev.size = sizeof(GoldfishRTCDevice),
+ .qdev.props = (Property[]) {
+ DEFINE_PROP_UINT32("id", GoldfishDevice, id, -1),
+ DEFINE_PROP_UINT32("size", GoldfishDevice, size, 0x1000),
+ DEFINE_PROP_UINT32("irq_count", GoldfishDevice, irq_count, 1),
+ DEFINE_PROP_STRING("name", GoldfishDevice, name),
+ DEFINE_PROP_END_OF_LIST(),
+ },
+};
+
+static void goldfish_rtc_register(void)
+{
+ goldfish_bus_register_withprop(&goldfish_rtc_info);
+}
+device_init(goldfish_rtc_register);
--
1.7.4.1
[Prev in Thread] |
Current Thread |
[Next in Thread] |
- [Qemu-devel] [PATCH 04/11] Goldfish: Added timing devices.,
Patrick Jackson <=