| /* |
| * i.MX8 PCIe PHY emulation |
| * |
| * Copyright (c) 2025 Bernhard Beschow <shentey@gmail.com> |
| * |
| * SPDX-License-Identifier: GPL-2.0-or-later |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "hw/pci-host/fsl_imx8m_phy.h" |
| #include "hw/resettable.h" |
| #include "migration/vmstate.h" |
| |
| #define CMN_REG075 0x1d4 |
| #define ANA_PLL_LOCK_DONE BIT(1) |
| #define ANA_PLL_AFC_DONE BIT(0) |
| |
| static uint64_t fsl_imx8m_pcie_phy_read(void *opaque, hwaddr offset, |
| unsigned size) |
| { |
| FslImx8mPciePhyState *s = opaque; |
| |
| if (offset == CMN_REG075) { |
| return s->data[offset] | ANA_PLL_LOCK_DONE | ANA_PLL_AFC_DONE; |
| } |
| |
| return s->data[offset]; |
| } |
| |
| static void fsl_imx8m_pcie_phy_write(void *opaque, hwaddr offset, |
| uint64_t value, unsigned size) |
| { |
| FslImx8mPciePhyState *s = opaque; |
| |
| s->data[offset] = value; |
| } |
| |
| static const MemoryRegionOps fsl_imx8m_pcie_phy_ops = { |
| .read = fsl_imx8m_pcie_phy_read, |
| .write = fsl_imx8m_pcie_phy_write, |
| .impl = { |
| .min_access_size = 1, |
| .max_access_size = 1, |
| }, |
| .valid = { |
| .min_access_size = 1, |
| .max_access_size = 8, |
| }, |
| .endianness = DEVICE_LITTLE_ENDIAN, |
| }; |
| |
| static void fsl_imx8m_pcie_phy_realize(DeviceState *dev, Error **errp) |
| { |
| FslImx8mPciePhyState *s = FSL_IMX8M_PCIE_PHY(dev); |
| |
| memory_region_init_io(&s->iomem, OBJECT(s), &fsl_imx8m_pcie_phy_ops, s, |
| TYPE_FSL_IMX8M_PCIE_PHY, ARRAY_SIZE(s->data)); |
| sysbus_init_mmio(SYS_BUS_DEVICE(s), &s->iomem); |
| } |
| |
| static void fsl_imx8m_pcie_phy_reset_hold(Object *obj, ResetType type) |
| { |
| FslImx8mPciePhyState *s = FSL_IMX8M_PCIE_PHY(obj); |
| |
| memset(s->data, 0, sizeof(s->data)); |
| } |
| |
| static const VMStateDescription fsl_imx8m_pcie_phy_vmstate = { |
| .name = "fsl-imx8m-pcie-phy", |
| .version_id = 1, |
| .minimum_version_id = 1, |
| .fields = (const VMStateField[]) { |
| VMSTATE_UINT8_ARRAY(data, FslImx8mPciePhyState, |
| FSL_IMX8M_PCIE_PHY_DATA_SIZE), |
| VMSTATE_END_OF_LIST() |
| } |
| }; |
| |
| static void fsl_imx8m_pcie_phy_class_init(ObjectClass *klass, const void *data) |
| { |
| DeviceClass *dc = DEVICE_CLASS(klass); |
| ResettableClass *rc = RESETTABLE_CLASS(klass); |
| |
| dc->realize = fsl_imx8m_pcie_phy_realize; |
| dc->vmsd = &fsl_imx8m_pcie_phy_vmstate; |
| rc->phases.hold = fsl_imx8m_pcie_phy_reset_hold; |
| } |
| |
| static const TypeInfo fsl_imx8m_pcie_phy_types[] = { |
| { |
| .name = TYPE_FSL_IMX8M_PCIE_PHY, |
| .parent = TYPE_SYS_BUS_DEVICE, |
| .instance_size = sizeof(FslImx8mPciePhyState), |
| .class_init = fsl_imx8m_pcie_phy_class_init, |
| } |
| }; |
| |
| DEFINE_TYPES(fsl_imx8m_pcie_phy_types) |