| /* 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_file.h" |
| #include "goldfish_device.h" |
| #include "goldfish_vmem.h" |
| |
| enum { |
| SW_NAME_LEN = 0x00, |
| SW_NAME_PTR = 0x04, |
| SW_FLAGS = 0x08, |
| SW_STATE = 0x0c, |
| SW_INT_STATUS = 0x10, |
| SW_INT_ENABLE = 0x14, |
| |
| SW_FLAGS_OUTPUT = 1U << 0 |
| }; |
| |
| |
| struct switch_state { |
| struct goldfish_device dev; |
| char *name; |
| uint32_t state; |
| uint32_t state_changed : 1; |
| uint32_t int_enable : 1; |
| uint32_t (*writefn)(void *opaque, uint32_t state); |
| void *writeopaque; |
| }; |
| |
| #define GOLDFISH_SWITCH_SAVE_VERSION 1 |
| |
| static void goldfish_switch_save(QEMUFile* f, void* opaque) |
| { |
| struct switch_state* s = opaque; |
| |
| qemu_put_be32(f, s->state); |
| qemu_put_byte(f, s->state_changed); |
| qemu_put_byte(f, s->int_enable); |
| } |
| |
| static int goldfish_switch_load(QEMUFile* f, void* opaque, int version_id) |
| { |
| struct switch_state* s = opaque; |
| |
| if (version_id != GOLDFISH_SWITCH_SAVE_VERSION) |
| return -1; |
| |
| s->state = qemu_get_be32(f); |
| s->state_changed = qemu_get_byte(f); |
| s->int_enable = qemu_get_byte(f); |
| |
| return 0; |
| } |
| |
| static uint32_t goldfish_switch_read(void *opaque, target_phys_addr_t offset) |
| { |
| struct switch_state *s = (struct switch_state *)opaque; |
| |
| //printf("goldfish_switch_read %x %x\n", offset, size); |
| |
| switch (offset) { |
| case SW_NAME_LEN: |
| return strlen(s->name); |
| case SW_FLAGS: |
| return s->writefn ? SW_FLAGS_OUTPUT : 0; |
| case SW_STATE: |
| return s->state; |
| case SW_INT_STATUS: |
| if(s->state_changed && s->int_enable) { |
| s->state_changed = 0; |
| goldfish_device_set_irq(&s->dev, 0, 0); |
| return 1; |
| } |
| return 0; |
| default: |
| cpu_abort (cpu_single_env, "goldfish_switch_read: Bad offset %x\n", offset); |
| return 0; |
| } |
| } |
| |
| static void goldfish_switch_write(void *opaque, target_phys_addr_t offset, uint32_t value) |
| { |
| struct switch_state *s = (struct switch_state *)opaque; |
| |
| //printf("goldfish_switch_read %x %x %x\n", offset, value, size); |
| |
| switch(offset) { |
| case SW_NAME_PTR: |
| safe_memory_rw_debug(cpu_single_env, value, (void*)s->name, strlen(s->name), 1); |
| break; |
| |
| case SW_STATE: |
| if(s->writefn) { |
| uint32_t new_state; |
| new_state = s->writefn(s->writeopaque, value); |
| if(new_state != s->state) { |
| goldfish_switch_set_state(s, new_state); |
| } |
| } |
| else |
| cpu_abort (cpu_single_env, "goldfish_switch_write: write to SW_STATE on input\n"); |
| break; |
| |
| case SW_INT_ENABLE: |
| value &= 1; |
| if(s->state_changed && s->int_enable != value) |
| goldfish_device_set_irq(&s->dev, 0, value); |
| s->int_enable = value; |
| break; |
| |
| default: |
| cpu_abort (cpu_single_env, "goldfish_switch_write: Bad offset %x\n", offset); |
| } |
| } |
| |
| static CPUReadMemoryFunc *goldfish_switch_readfn[] = { |
| goldfish_switch_read, |
| goldfish_switch_read, |
| goldfish_switch_read |
| }; |
| |
| static CPUWriteMemoryFunc *goldfish_switch_writefn[] = { |
| goldfish_switch_write, |
| goldfish_switch_write, |
| goldfish_switch_write |
| }; |
| |
| void goldfish_switch_set_state(void *opaque, uint32_t state) |
| { |
| struct switch_state *s = opaque; |
| s->state_changed = 1; |
| s->state = state; |
| if(s->int_enable) |
| goldfish_device_set_irq(&s->dev, 0, 1); |
| } |
| |
| void *goldfish_switch_add(char *name, uint32_t (*writefn)(void *opaque, uint32_t state), void *writeopaque, int id) |
| { |
| int ret; |
| struct switch_state *s; |
| |
| s = qemu_mallocz(sizeof(*s)); |
| s->dev.name = "goldfish-switch"; |
| s->dev.id = id; |
| s->dev.size = 0x1000; |
| s->dev.irq_count = 1; |
| s->name = name; |
| s->writefn = writefn; |
| s->writeopaque = writeopaque; |
| |
| |
| ret = goldfish_device_add(&s->dev, goldfish_switch_readfn, goldfish_switch_writefn, s); |
| if(ret) { |
| qemu_free(s); |
| return NULL; |
| } |
| |
| register_savevm( "goldfish_switch", 0, GOLDFISH_SWITCH_SAVE_VERSION, |
| goldfish_switch_save, goldfish_switch_load, s); |
| |
| return s; |
| } |
| |