blob: 131cea9c6b8b15c42936b506601ec0f6db4b90e6 [file] [log] [blame]
/*
* ASPEED LTPI Controller
*
* Copyright (C) 2025 ASPEED Technology Inc.
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "qemu/osdep.h"
#include "qemu/log.h"
#include "migration/vmstate.h"
#include "hw/misc/aspeed_ltpi.h"
#define ASPEED_LTPI_CTRL_BASE 0x000
#define ASPEED_LTPI_PHY_BASE 0x200
#define ASPEED_LTPI_TOP_BASE 0x800
#define LTPI_CTRL_LINK_MNG 0x42
#define LTPI_PHY_MODE 0x0
static uint64_t aspeed_ltpi_top_read(void *opaque, hwaddr offset, unsigned size)
{
AspeedLTPIState *s = opaque;
uint32_t idx = offset >> 2;
return s->top_regs[idx];
}
static void aspeed_ltpi_top_write(void *opaque, hwaddr offset,
uint64_t val, unsigned size)
{
AspeedLTPIState *s = opaque;
uint32_t idx = offset >> 2;
switch (offset) {
default:
s->top_regs[idx] = (uint32_t)val;
break;
}
}
static const MemoryRegionOps aspeed_ltpi_top_ops = {
.read = aspeed_ltpi_top_read,
.write = aspeed_ltpi_top_write,
.endianness = DEVICE_LITTLE_ENDIAN,
.valid = {
.min_access_size = 1,
.max_access_size = 4,
},
};
static uint64_t aspeed_ltpi_phy_read(void *opaque, hwaddr offset, unsigned size)
{
AspeedLTPIState *s = opaque;
uint32_t idx = offset >> 2;
return s->phy_regs[idx];
}
static void aspeed_ltpi_phy_write(void *opaque, hwaddr offset,
uint64_t val, unsigned size)
{
AspeedLTPIState *s = opaque;
uint32_t idx = offset >> 2;
switch (offset) {
default:
s->phy_regs[idx] = (uint32_t)val;
break;
}
}
static const MemoryRegionOps aspeed_ltpi_phy_ops = {
.read = aspeed_ltpi_phy_read,
.write = aspeed_ltpi_phy_write,
.endianness = DEVICE_LITTLE_ENDIAN,
.valid = {
.min_access_size = 1,
.max_access_size = 4,
},
};
static uint64_t aspeed_ltpi_ctrl_read(void *opaque,
hwaddr offset, unsigned size)
{
AspeedLTPIState *s = opaque;
uint32_t idx = offset >> 2;
return s->ctrl_regs[idx];
}
static void aspeed_ltpi_ctrl_write(void *opaque, hwaddr offset,
uint64_t val, unsigned size)
{
AspeedLTPIState *s = opaque;
uint32_t idx = offset >> 2;
switch (offset) {
default:
s->ctrl_regs[idx] = (uint32_t)val;
break;
}
}
static const MemoryRegionOps aspeed_ltpi_ctrl_ops = {
.read = aspeed_ltpi_ctrl_read,
.write = aspeed_ltpi_ctrl_write,
.endianness = DEVICE_LITTLE_ENDIAN,
.valid = {
.min_access_size = 1,
.max_access_size = 4,
},
};
static void aspeed_ltpi_reset(DeviceState *dev)
{
AspeedLTPIState *s = ASPEED_LTPI(dev);
memset(s->ctrl_regs, 0, sizeof(s->ctrl_regs));
memset(s->phy_regs, 0, sizeof(s->phy_regs));
memset(s->top_regs, 0, sizeof(s->top_regs));
/* set default values */
s->ctrl_regs[LTPI_CTRL_LINK_MNG] = 0x11900007;
s->phy_regs[LTPI_PHY_MODE] = 0x2;
}
static const VMStateDescription vmstate_aspeed_ltpi = {
.name = TYPE_ASPEED_LTPI,
.version_id = 1,
.minimum_version_id = 1,
.fields = (VMStateField[]) {
VMSTATE_UINT32_ARRAY(ctrl_regs, AspeedLTPIState,
ASPEED_LTPI_CTRL_SIZE >> 2),
VMSTATE_UINT32_ARRAY(phy_regs, AspeedLTPIState,
ASPEED_LTPI_PHY_SIZE >> 2),
VMSTATE_UINT32_ARRAY(top_regs, AspeedLTPIState,
ASPEED_LTPI_TOP_SIZE >> 2),
VMSTATE_END_OF_LIST()
}
};
static void aspeed_ltpi_realize(DeviceState *dev, Error **errp)
{
AspeedLTPIState *s = ASPEED_LTPI(dev);
memory_region_init(&s->mmio, OBJECT(s), TYPE_ASPEED_LTPI,
ASPEED_LTPI_TOTAL_SIZE);
memory_region_init_io(&s->mmio_ctrl, OBJECT(s),
&aspeed_ltpi_ctrl_ops, s,
"aspeed-ltpi-ctrl", ASPEED_LTPI_CTRL_SIZE);
memory_region_init_io(&s->mmio_phy, OBJECT(s),
&aspeed_ltpi_phy_ops, s,
"aspeed-ltpi-phy", ASPEED_LTPI_PHY_SIZE);
memory_region_init_io(&s->mmio_top, OBJECT(s),
&aspeed_ltpi_top_ops, s,
"aspeed-ltpi-top", ASPEED_LTPI_TOP_SIZE);
memory_region_add_subregion(&s->mmio,
ASPEED_LTPI_CTRL_BASE, &s->mmio_ctrl);
memory_region_add_subregion(&s->mmio,
ASPEED_LTPI_PHY_BASE, &s->mmio_phy);
memory_region_add_subregion(&s->mmio,
ASPEED_LTPI_TOP_BASE, &s->mmio_top);
sysbus_init_mmio(SYS_BUS_DEVICE(s), &s->mmio);
}
static void aspeed_ltpi_class_init(ObjectClass *klass, const void *data)
{
DeviceClass *dc = DEVICE_CLASS(klass);
dc->realize = aspeed_ltpi_realize;
dc->vmsd = &vmstate_aspeed_ltpi;
device_class_set_legacy_reset(dc, aspeed_ltpi_reset);
}
static const TypeInfo aspeed_ltpi_info = {
.name = TYPE_ASPEED_LTPI,
.parent = TYPE_SYS_BUS_DEVICE,
.instance_size = sizeof(AspeedLTPIState),
.class_init = aspeed_ltpi_class_init,
};
static void aspeed_ltpi_register_types(void)
{
type_register_static(&aspeed_ltpi_info);
}
type_init(aspeed_ltpi_register_types);