2012-08-10 14:42:32 +08:00
|
|
|
/*
|
|
|
|
* DMA device simulation in PKUnity SoC
|
|
|
|
*
|
|
|
|
* Copyright (C) 2010-2012 Guan Xuetao
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation, or any later version.
|
|
|
|
* See the COPYING file in the top-level directory.
|
|
|
|
*/
|
2016-01-27 02:17:01 +08:00
|
|
|
#include "qemu/osdep.h"
|
2013-02-04 22:40:22 +08:00
|
|
|
#include "hw/hw.h"
|
|
|
|
#include "hw/sysbus.h"
|
2012-08-10 14:42:32 +08:00
|
|
|
|
|
|
|
#undef DEBUG_PUV3
|
2013-02-06 00:06:20 +08:00
|
|
|
#include "hw/unicore32/puv3.h"
|
2012-08-10 14:42:32 +08:00
|
|
|
|
|
|
|
#define PUV3_DMA_CH_NR (6)
|
|
|
|
#define PUV3_DMA_CH_MASK (0xff)
|
|
|
|
#define PUV3_DMA_CH(offset) ((offset) >> 8)
|
|
|
|
|
2013-07-26 22:04:39 +08:00
|
|
|
#define TYPE_PUV3_DMA "puv3_dma"
|
|
|
|
#define PUV3_DMA(obj) OBJECT_CHECK(PUV3DMAState, (obj), TYPE_PUV3_DMA)
|
|
|
|
|
|
|
|
typedef struct PUV3DMAState {
|
|
|
|
SysBusDevice parent_obj;
|
|
|
|
|
2012-08-10 14:42:32 +08:00
|
|
|
MemoryRegion iomem;
|
|
|
|
uint32_t reg_CFG[PUV3_DMA_CH_NR];
|
|
|
|
} PUV3DMAState;
|
|
|
|
|
2012-10-23 18:30:10 +08:00
|
|
|
static uint64_t puv3_dma_read(void *opaque, hwaddr offset,
|
2012-08-10 14:42:32 +08:00
|
|
|
unsigned size)
|
|
|
|
{
|
|
|
|
PUV3DMAState *s = opaque;
|
|
|
|
uint32_t ret = 0;
|
|
|
|
|
|
|
|
assert(PUV3_DMA_CH(offset) < PUV3_DMA_CH_NR);
|
|
|
|
|
|
|
|
switch (offset & PUV3_DMA_CH_MASK) {
|
|
|
|
case 0x10:
|
|
|
|
ret = s->reg_CFG[PUV3_DMA_CH(offset)];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DPRINTF("Bad offset 0x%x\n", offset);
|
|
|
|
}
|
|
|
|
DPRINTF("offset 0x%x, value 0x%x\n", offset, ret);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-10-23 18:30:10 +08:00
|
|
|
static void puv3_dma_write(void *opaque, hwaddr offset,
|
2012-08-10 14:42:32 +08:00
|
|
|
uint64_t value, unsigned size)
|
|
|
|
{
|
|
|
|
PUV3DMAState *s = opaque;
|
|
|
|
|
|
|
|
assert(PUV3_DMA_CH(offset) < PUV3_DMA_CH_NR);
|
|
|
|
|
|
|
|
switch (offset & PUV3_DMA_CH_MASK) {
|
|
|
|
case 0x10:
|
|
|
|
s->reg_CFG[PUV3_DMA_CH(offset)] = value;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DPRINTF("Bad offset 0x%x\n", offset);
|
|
|
|
}
|
|
|
|
DPRINTF("offset 0x%x, value 0x%x\n", offset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const MemoryRegionOps puv3_dma_ops = {
|
|
|
|
.read = puv3_dma_read,
|
|
|
|
.write = puv3_dma_write,
|
|
|
|
.impl = {
|
|
|
|
.min_access_size = 4,
|
|
|
|
.max_access_size = 4,
|
|
|
|
},
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
|
|
|
};
|
|
|
|
|
2018-12-13 21:47:58 +08:00
|
|
|
static void puv3_dma_realize(DeviceState *dev, Error **errp)
|
2012-08-10 14:42:32 +08:00
|
|
|
{
|
2013-07-26 22:04:39 +08:00
|
|
|
PUV3DMAState *s = PUV3_DMA(dev);
|
2012-08-10 14:42:32 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < PUV3_DMA_CH_NR; i++) {
|
|
|
|
s->reg_CFG[i] = 0x0;
|
|
|
|
}
|
|
|
|
|
2013-06-07 09:25:08 +08:00
|
|
|
memory_region_init_io(&s->iomem, OBJECT(s), &puv3_dma_ops, s, "puv3_dma",
|
2012-08-10 14:42:32 +08:00
|
|
|
PUV3_REGS_OFFSET);
|
2018-12-13 21:47:58 +08:00
|
|
|
sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->iomem);
|
2012-08-10 14:42:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void puv3_dma_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
2018-12-13 21:47:58 +08:00
|
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
2012-08-10 14:42:32 +08:00
|
|
|
|
2018-12-13 21:47:58 +08:00
|
|
|
dc->realize = puv3_dma_realize;
|
2012-08-10 14:42:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static const TypeInfo puv3_dma_info = {
|
2013-07-26 22:04:39 +08:00
|
|
|
.name = TYPE_PUV3_DMA,
|
2012-08-10 14:42:32 +08:00
|
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
|
|
.instance_size = sizeof(PUV3DMAState),
|
|
|
|
.class_init = puv3_dma_class_init,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void puv3_dma_register_type(void)
|
|
|
|
{
|
|
|
|
type_register_static(&puv3_dma_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
type_init(puv3_dma_register_type)
|