|
@@ -0,0 +1,98 @@
|
|
|
+/*
|
|
|
+ * 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, 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)
|