1 // SPDX-License-Identifier: GPL-2.0+
3 * Copyright (c) 2014 Google, Inc
4 * Written by Simon Glass <sjg@chromium.org>
10 #include <linux/libfdt.h>
14 struct sandbox_pci_emul_priv {
18 int sandbox_pci_get_emul(struct udevice *bus, pci_dev_t find_devfn,
19 struct udevice **containerp, struct udevice **emulp)
25 ret = pci_bus_find_devfn(bus, PCI_MASK_BUS(find_devfn), &dev);
27 debug("%s: Could not find emulator for dev %x\n", __func__,
34 * See commit 4345998ae9df,
35 * "pci: sandbox: Support dynamically binding device driver"
37 ret = uclass_get_device_by_phandle(UCLASS_PCI_EMUL, dev, "sandbox,emul",
39 if (ret && device_get_uclass_id(dev) != UCLASS_PCI_GENERIC)
42 return *emulp ? 0 : -ENODEV;
45 static int sandbox_pci_emul_post_probe(struct udevice *dev)
47 struct sandbox_pci_emul_priv *priv = dev->uclass->priv;
50 sandbox_set_enable_pci_map(true);
55 static int sandbox_pci_emul_pre_remove(struct udevice *dev)
57 struct sandbox_pci_emul_priv *priv = dev->uclass->priv;
60 sandbox_set_enable_pci_map(priv->dev_count > 0);
65 UCLASS_DRIVER(pci_emul) = {
66 .id = UCLASS_PCI_EMUL,
68 .post_probe = sandbox_pci_emul_post_probe,
69 .pre_remove = sandbox_pci_emul_pre_remove,
70 .priv_auto_alloc_size = sizeof(struct sandbox_pci_emul_priv),
74 * This uclass is a child of the pci bus. Its platdata is not defined here so
75 * is defined by its parent, UCLASS_PCI, which uses struct pci_child_platdata.
76 * See per_child_platdata_auto_alloc_size in UCLASS_DRIVER(pci).
78 UCLASS_DRIVER(pci_emul_parent) = {
79 .id = UCLASS_PCI_EMUL_PARENT,
80 .name = "pci_emul_parent",
81 .post_bind = dm_scan_fdt_dev,
84 static const struct udevice_id pci_emul_parent_ids[] = {
85 { .compatible = "sandbox,pci-emul-parent" },
89 U_BOOT_DRIVER(pci_emul_parent_drv) = {
90 .name = "pci_emul_parent_drv",
91 .id = UCLASS_PCI_EMUL_PARENT,
92 .of_match = pci_emul_parent_ids,