/* 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"
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;
offset -= s->dev.base;
//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;
offset -= s->dev.base;
//printf("goldfish_switch_read %x %x %x\n", offset, value, size);
switch(offset) {
case SW_NAME_PTR:
pmemcpy(value, s->name, strlen(s->name));
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;
}