/* * i.MX8 PCIe PHY emulation * * Copyright (c) 2025 Bernhard Beschow * * 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, 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)