mirror of
https://github.com/qemu/qemu.git
synced 2024-11-27 05:43:47 +08:00
09d12c81ec
In real hardware, the APB and AHB PNP data tables can be accessed with byte and halfword reads as well as word reads. Our implementation currently only handles word reads. Add support for the 8 and 16 bit accesses. Note that we only need to handle aligned accesses -- unaligned accesses should continue to trap, as happens on hardware. Resolves: https://gitlab.com/qemu-project/qemu/-/issues/1132 Signed-off-by: Peter Maydell <peter.maydell@linaro.org> Reviewed-by: Frederic Konrad <fkonrad@amd.com> Message-Id: <20220802131925.3380923-1-peter.maydell@linaro.org> Tested-by: Tomasz Martyniak <gitlab.com/tom4r> Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
304 lines
10 KiB
C
304 lines
10 KiB
C
/*
|
|
* GRLIB AHB APB PNP
|
|
*
|
|
* Copyright (C) 2019 AdaCore
|
|
*
|
|
* Developed by :
|
|
* Frederic Konrad <frederic.konrad@adacore.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* 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.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "qemu/osdep.h"
|
|
#include "qemu/log.h"
|
|
#include "hw/sysbus.h"
|
|
#include "hw/misc/grlib_ahb_apb_pnp.h"
|
|
#include "trace.h"
|
|
|
|
#define GRLIB_PNP_VENDOR_SHIFT (24)
|
|
#define GRLIB_PNP_VENDOR_SIZE (8)
|
|
#define GRLIB_PNP_DEV_SHIFT (12)
|
|
#define GRLIB_PNP_DEV_SIZE (12)
|
|
#define GRLIB_PNP_VER_SHIFT (5)
|
|
#define GRLIB_PNP_VER_SIZE (5)
|
|
#define GRLIB_PNP_IRQ_SHIFT (0)
|
|
#define GRLIB_PNP_IRQ_SIZE (5)
|
|
#define GRLIB_PNP_ADDR_SHIFT (20)
|
|
#define GRLIB_PNP_ADDR_SIZE (12)
|
|
#define GRLIB_PNP_MASK_SHIFT (4)
|
|
#define GRLIB_PNP_MASK_SIZE (12)
|
|
|
|
#define GRLIB_AHB_DEV_ADDR_SHIFT (20)
|
|
#define GRLIB_AHB_DEV_ADDR_SIZE (12)
|
|
#define GRLIB_AHB_ENTRY_SIZE (0x20)
|
|
#define GRLIB_AHB_MAX_DEV (64)
|
|
#define GRLIB_AHB_SLAVE_OFFSET (0x800)
|
|
|
|
#define GRLIB_APB_DEV_ADDR_SHIFT (8)
|
|
#define GRLIB_APB_DEV_ADDR_SIZE (12)
|
|
#define GRLIB_APB_ENTRY_SIZE (0x08)
|
|
#define GRLIB_APB_MAX_DEV (512)
|
|
|
|
#define GRLIB_PNP_MAX_REGS (0x1000)
|
|
|
|
typedef struct AHBPnp {
|
|
SysBusDevice parent_obj;
|
|
MemoryRegion iomem;
|
|
|
|
uint32_t regs[GRLIB_PNP_MAX_REGS >> 2];
|
|
uint8_t master_count;
|
|
uint8_t slave_count;
|
|
} AHBPnp;
|
|
|
|
void grlib_ahb_pnp_add_entry(AHBPnp *dev, uint32_t address, uint32_t mask,
|
|
uint8_t vendor, uint16_t device, int slave,
|
|
int type)
|
|
{
|
|
unsigned int reg_start;
|
|
|
|
/*
|
|
* AHB entries look like this:
|
|
*
|
|
* 31 -------- 23 -------- 11 ----- 9 -------- 4 --- 0
|
|
* | VENDOR ID | DEVICE ID | IRQ ? | VERSION | IRQ |
|
|
* --------------------------------------------------
|
|
* | USER |
|
|
* --------------------------------------------------
|
|
* | USER |
|
|
* --------------------------------------------------
|
|
* | USER |
|
|
* --------------------------------------------------
|
|
* | USER |
|
|
* --------------------------------------------------
|
|
* 31 ----------- 20 --- 15 ----------------- 3 ---- 0
|
|
* | ADDR[31..12] | 00PC | MASK | TYPE |
|
|
* --------------------------------------------------
|
|
* 31 ----------- 20 --- 15 ----------------- 3 ---- 0
|
|
* | ADDR[31..12] | 00PC | MASK | TYPE |
|
|
* --------------------------------------------------
|
|
* 31 ----------- 20 --- 15 ----------------- 3 ---- 0
|
|
* | ADDR[31..12] | 00PC | MASK | TYPE |
|
|
* --------------------------------------------------
|
|
* 31 ----------- 20 --- 15 ----------------- 3 ---- 0
|
|
* | ADDR[31..12] | 00PC | MASK | TYPE |
|
|
* --------------------------------------------------
|
|
*/
|
|
|
|
if (slave) {
|
|
assert(dev->slave_count < GRLIB_AHB_MAX_DEV);
|
|
reg_start = (GRLIB_AHB_SLAVE_OFFSET
|
|
+ (dev->slave_count * GRLIB_AHB_ENTRY_SIZE)) >> 2;
|
|
dev->slave_count++;
|
|
} else {
|
|
assert(dev->master_count < GRLIB_AHB_MAX_DEV);
|
|
reg_start = (dev->master_count * GRLIB_AHB_ENTRY_SIZE) >> 2;
|
|
dev->master_count++;
|
|
}
|
|
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_VENDOR_SHIFT,
|
|
GRLIB_PNP_VENDOR_SIZE,
|
|
vendor);
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_DEV_SHIFT,
|
|
GRLIB_PNP_DEV_SIZE,
|
|
device);
|
|
reg_start += 4;
|
|
/* AHB Memory Space */
|
|
dev->regs[reg_start] = type;
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_ADDR_SHIFT,
|
|
GRLIB_PNP_ADDR_SIZE,
|
|
extract32(address,
|
|
GRLIB_AHB_DEV_ADDR_SHIFT,
|
|
GRLIB_AHB_DEV_ADDR_SIZE));
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_MASK_SHIFT,
|
|
GRLIB_PNP_MASK_SIZE,
|
|
mask);
|
|
}
|
|
|
|
static uint64_t grlib_ahb_pnp_read(void *opaque, hwaddr offset, unsigned size)
|
|
{
|
|
AHBPnp *ahb_pnp = GRLIB_AHB_PNP(opaque);
|
|
uint32_t val;
|
|
|
|
val = ahb_pnp->regs[offset >> 2];
|
|
val = extract32(val, (4 - (offset & 3) - size) * 8, size * 8);
|
|
trace_grlib_ahb_pnp_read(offset, size, val);
|
|
|
|
return val;
|
|
}
|
|
|
|
static void grlib_ahb_pnp_write(void *opaque, hwaddr addr,
|
|
uint64_t val, unsigned size)
|
|
{
|
|
qemu_log_mask(LOG_UNIMP, "%s not implemented\n", __func__);
|
|
}
|
|
|
|
static const MemoryRegionOps grlib_ahb_pnp_ops = {
|
|
.read = grlib_ahb_pnp_read,
|
|
.write = grlib_ahb_pnp_write,
|
|
.endianness = DEVICE_BIG_ENDIAN,
|
|
.impl = {
|
|
.min_access_size = 1,
|
|
.max_access_size = 4,
|
|
},
|
|
};
|
|
|
|
static void grlib_ahb_pnp_realize(DeviceState *dev, Error **errp)
|
|
{
|
|
AHBPnp *ahb_pnp = GRLIB_AHB_PNP(dev);
|
|
SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
|
|
|
|
memory_region_init_io(&ahb_pnp->iomem, OBJECT(dev), &grlib_ahb_pnp_ops,
|
|
ahb_pnp, TYPE_GRLIB_AHB_PNP, GRLIB_PNP_MAX_REGS);
|
|
sysbus_init_mmio(sbd, &ahb_pnp->iomem);
|
|
}
|
|
|
|
static void grlib_ahb_pnp_class_init(ObjectClass *klass, void *data)
|
|
{
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
|
|
|
dc->realize = grlib_ahb_pnp_realize;
|
|
}
|
|
|
|
static const TypeInfo grlib_ahb_pnp_info = {
|
|
.name = TYPE_GRLIB_AHB_PNP,
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
.instance_size = sizeof(AHBPnp),
|
|
.class_init = grlib_ahb_pnp_class_init,
|
|
};
|
|
|
|
/* APBPnp */
|
|
|
|
typedef struct APBPnp {
|
|
SysBusDevice parent_obj;
|
|
MemoryRegion iomem;
|
|
|
|
uint32_t regs[GRLIB_PNP_MAX_REGS >> 2];
|
|
uint32_t entry_count;
|
|
} APBPnp;
|
|
|
|
void grlib_apb_pnp_add_entry(APBPnp *dev, uint32_t address, uint32_t mask,
|
|
uint8_t vendor, uint16_t device, uint8_t version,
|
|
uint8_t irq, int type)
|
|
{
|
|
unsigned int reg_start;
|
|
|
|
/*
|
|
* APB entries look like this:
|
|
*
|
|
* 31 -------- 23 -------- 11 ----- 9 ------- 4 --- 0
|
|
* | VENDOR ID | DEVICE ID | IRQ ? | VERSION | IRQ |
|
|
*
|
|
* 31 ---------- 20 --- 15 ----------------- 3 ---- 0
|
|
* | ADDR[20..8] | 0000 | MASK | TYPE |
|
|
*/
|
|
|
|
assert(dev->entry_count < GRLIB_APB_MAX_DEV);
|
|
reg_start = (dev->entry_count * GRLIB_APB_ENTRY_SIZE) >> 2;
|
|
dev->entry_count++;
|
|
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_VENDOR_SHIFT,
|
|
GRLIB_PNP_VENDOR_SIZE,
|
|
vendor);
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_DEV_SHIFT,
|
|
GRLIB_PNP_DEV_SIZE,
|
|
device);
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_VER_SHIFT,
|
|
GRLIB_PNP_VER_SIZE,
|
|
version);
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_IRQ_SHIFT,
|
|
GRLIB_PNP_IRQ_SIZE,
|
|
irq);
|
|
reg_start += 1;
|
|
dev->regs[reg_start] = type;
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_ADDR_SHIFT,
|
|
GRLIB_PNP_ADDR_SIZE,
|
|
extract32(address,
|
|
GRLIB_APB_DEV_ADDR_SHIFT,
|
|
GRLIB_APB_DEV_ADDR_SIZE));
|
|
dev->regs[reg_start] = deposit32(dev->regs[reg_start],
|
|
GRLIB_PNP_MASK_SHIFT,
|
|
GRLIB_PNP_MASK_SIZE,
|
|
mask);
|
|
}
|
|
|
|
static uint64_t grlib_apb_pnp_read(void *opaque, hwaddr offset, unsigned size)
|
|
{
|
|
APBPnp *apb_pnp = GRLIB_APB_PNP(opaque);
|
|
uint32_t val;
|
|
|
|
val = apb_pnp->regs[offset >> 2];
|
|
val = extract32(val, (4 - (offset & 3) - size) * 8, size * 8);
|
|
trace_grlib_apb_pnp_read(offset, size, val);
|
|
|
|
return val;
|
|
}
|
|
|
|
static void grlib_apb_pnp_write(void *opaque, hwaddr addr,
|
|
uint64_t val, unsigned size)
|
|
{
|
|
qemu_log_mask(LOG_UNIMP, "%s not implemented\n", __func__);
|
|
}
|
|
|
|
static const MemoryRegionOps grlib_apb_pnp_ops = {
|
|
.read = grlib_apb_pnp_read,
|
|
.write = grlib_apb_pnp_write,
|
|
.endianness = DEVICE_BIG_ENDIAN,
|
|
.impl = {
|
|
.min_access_size = 1,
|
|
.max_access_size = 4,
|
|
},
|
|
};
|
|
|
|
static void grlib_apb_pnp_realize(DeviceState *dev, Error **errp)
|
|
{
|
|
APBPnp *apb_pnp = GRLIB_APB_PNP(dev);
|
|
SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
|
|
|
|
memory_region_init_io(&apb_pnp->iomem, OBJECT(dev), &grlib_apb_pnp_ops,
|
|
apb_pnp, TYPE_GRLIB_APB_PNP, GRLIB_PNP_MAX_REGS);
|
|
sysbus_init_mmio(sbd, &apb_pnp->iomem);
|
|
}
|
|
|
|
static void grlib_apb_pnp_class_init(ObjectClass *klass, void *data)
|
|
{
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
|
|
|
dc->realize = grlib_apb_pnp_realize;
|
|
}
|
|
|
|
static const TypeInfo grlib_apb_pnp_info = {
|
|
.name = TYPE_GRLIB_APB_PNP,
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
.instance_size = sizeof(APBPnp),
|
|
.class_init = grlib_apb_pnp_class_init,
|
|
};
|
|
|
|
static void grlib_ahb_apb_pnp_register_types(void)
|
|
{
|
|
type_register_static(&grlib_ahb_pnp_info);
|
|
type_register_static(&grlib_apb_pnp_info);
|
|
}
|
|
|
|
type_init(grlib_ahb_apb_pnp_register_types)
|