2
0

fsl_imx8m_phy.c 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. /*
  2. * i.MX8 PCIe PHY emulation
  3. *
  4. * Copyright (c) 2025 Bernhard Beschow <shentey@gmail.com>
  5. *
  6. * SPDX-License-Identifier: GPL-2.0-or-later
  7. */
  8. #include "qemu/osdep.h"
  9. #include "hw/pci-host/fsl_imx8m_phy.h"
  10. #include "hw/resettable.h"
  11. #include "migration/vmstate.h"
  12. #define CMN_REG075 0x1d4
  13. #define ANA_PLL_LOCK_DONE BIT(1)
  14. #define ANA_PLL_AFC_DONE BIT(0)
  15. static uint64_t fsl_imx8m_pcie_phy_read(void *opaque, hwaddr offset,
  16. unsigned size)
  17. {
  18. FslImx8mPciePhyState *s = opaque;
  19. if (offset == CMN_REG075) {
  20. return s->data[offset] | ANA_PLL_LOCK_DONE | ANA_PLL_AFC_DONE;
  21. }
  22. return s->data[offset];
  23. }
  24. static void fsl_imx8m_pcie_phy_write(void *opaque, hwaddr offset,
  25. uint64_t value, unsigned size)
  26. {
  27. FslImx8mPciePhyState *s = opaque;
  28. s->data[offset] = value;
  29. }
  30. static const MemoryRegionOps fsl_imx8m_pcie_phy_ops = {
  31. .read = fsl_imx8m_pcie_phy_read,
  32. .write = fsl_imx8m_pcie_phy_write,
  33. .impl = {
  34. .min_access_size = 1,
  35. .max_access_size = 1,
  36. },
  37. .valid = {
  38. .min_access_size = 1,
  39. .max_access_size = 8,
  40. },
  41. .endianness = DEVICE_LITTLE_ENDIAN,
  42. };
  43. static void fsl_imx8m_pcie_phy_realize(DeviceState *dev, Error **errp)
  44. {
  45. FslImx8mPciePhyState *s = FSL_IMX8M_PCIE_PHY(dev);
  46. memory_region_init_io(&s->iomem, OBJECT(s), &fsl_imx8m_pcie_phy_ops, s,
  47. TYPE_FSL_IMX8M_PCIE_PHY, ARRAY_SIZE(s->data));
  48. sysbus_init_mmio(SYS_BUS_DEVICE(s), &s->iomem);
  49. }
  50. static void fsl_imx8m_pcie_phy_reset_hold(Object *obj, ResetType type)
  51. {
  52. FslImx8mPciePhyState *s = FSL_IMX8M_PCIE_PHY(obj);
  53. memset(s->data, 0, sizeof(s->data));
  54. }
  55. static const VMStateDescription fsl_imx8m_pcie_phy_vmstate = {
  56. .name = "fsl-imx8m-pcie-phy",
  57. .version_id = 1,
  58. .minimum_version_id = 1,
  59. .fields = (const VMStateField[]) {
  60. VMSTATE_UINT8_ARRAY(data, FslImx8mPciePhyState,
  61. FSL_IMX8M_PCIE_PHY_DATA_SIZE),
  62. VMSTATE_END_OF_LIST()
  63. }
  64. };
  65. static void fsl_imx8m_pcie_phy_class_init(ObjectClass *klass, void *data)
  66. {
  67. DeviceClass *dc = DEVICE_CLASS(klass);
  68. ResettableClass *rc = RESETTABLE_CLASS(klass);
  69. dc->realize = fsl_imx8m_pcie_phy_realize;
  70. dc->vmsd = &fsl_imx8m_pcie_phy_vmstate;
  71. rc->phases.hold = fsl_imx8m_pcie_phy_reset_hold;
  72. }
  73. static const TypeInfo fsl_imx8m_pcie_phy_types[] = {
  74. {
  75. .name = TYPE_FSL_IMX8M_PCIE_PHY,
  76. .parent = TYPE_SYS_BUS_DEVICE,
  77. .instance_size = sizeof(FslImx8mPciePhyState),
  78. .class_init = fsl_imx8m_pcie_phy_class_init,
  79. }
  80. };
  81. DEFINE_TYPES(fsl_imx8m_pcie_phy_types)