// SPDX-License-Identifier: GPL-2.0 /* * dwc3-haps.c - Synopsys HAPS PCI Specific glue layer * * Copyright (C) 2018 Synopsys, Inc. * * Authors: Thinh Nguyen <thinhn@synopsys.com>, * John Youn <johnyoun@synopsys.com> */ #include <linux/kernel.h> #include <linux/module.h> #include <linux/slab.h> #include <linux/pci.h> #include <linux/platform_device.h> #include <linux/property.h> /** * struct dwc3_haps - Driver private structure * @dwc3: child dwc3 platform_device * @pci: our link to PCI bus */ struct dwc3_haps { struct platform_device *dwc3; struct pci_dev *pci; }; static const struct property_entry initial_properties[] = { PROPERTY_ENTRY_BOOL("snps,usb3_lpm_capable"), PROPERTY_ENTRY_BOOL("snps,has-lpm-erratum"), PROPERTY_ENTRY_BOOL("snps,dis_enblslpm_quirk"), PROPERTY_ENTRY_BOOL("linux,sysdev_is_parent"), { }, }; static const struct software_node dwc3_haps_swnode = { .properties = initial_properties, }; static int dwc3_haps_probe(struct pci_dev *pci, const struct pci_device_id *id) { struct dwc3_haps *dwc; struct device *dev = &pci->dev; struct resource res[2]; int ret; ret = pcim_enable_device(pci); if (ret) { dev_err(dev, "failed to enable pci device\n"); return -ENODEV; } pci_set_master(pci); dwc = devm_kzalloc(dev, sizeof(*dwc), GFP_KERNEL); if (!dwc) return -ENOMEM; dwc->dwc3 = platform_device_alloc("dwc3", PLATFORM_DEVID_AUTO); if (!dwc->dwc3) return -ENOMEM; memset(res, 0x00, sizeof(struct resource) * ARRAY_SIZE(res)); res[0].start = pci_resource_start(pci, 0); res[0].end = pci_resource_end(pci, 0); res[0].name = "dwc_usb3"; res[0].flags = IORESOURCE_MEM; res[1].start = pci->irq; res[1].name = "dwc_usb3"; res[1].flags = IORESOURCE_IRQ; ret = platform_device_add_resources(dwc->dwc3, res, ARRAY_SIZE(res)); if (ret) { dev_err(dev, "couldn't add resources to dwc3 device\n"); goto err; } dwc->pci = pci; dwc->dwc3->dev.parent = dev; ret = device_add_software_node(&dwc->dwc3->dev, &dwc3_haps_swnode); if (ret) goto err; ret = platform_device_add(dwc->dwc3); if (ret) { dev_err(dev, "failed to register dwc3 device\n"); goto err; } pci_set_drvdata(pci, dwc); return 0; err: device_remove_software_node(&dwc->dwc3->dev); platform_device_put(dwc->dwc3); return ret; } static void dwc3_haps_remove(struct pci_dev *pci) { struct dwc3_haps *dwc = pci_get_drvdata(pci); device_remove_software_node(&dwc->dwc3->dev); platform_device_unregister(dwc->dwc3); } static const struct pci_device_id dwc3_haps_id_table[] = { { PCI_DEVICE(PCI_VENDOR_ID_SYNOPSYS, PCI_DEVICE_ID_SYNOPSYS_HAPSUSB3), /* * i.MX6QP and i.MX7D platform use a PCIe controller with the * same VID and PID as this USB controller. The system may * incorrectly match this driver to that PCIe controller. To * workaround this, specifically use class type USB to prevent * incorrect driver matching. */ .class = (PCI_CLASS_SERIAL_USB << 8), .class_mask = 0xffff00, }, { PCI_DEVICE(PCI_VENDOR_ID_SYNOPSYS, PCI_DEVICE_ID_SYNOPSYS_HAPSUSB3_AXI), }, { PCI_DEVICE(PCI_VENDOR_ID_SYNOPSYS, PCI_DEVICE_ID_SYNOPSYS_HAPSUSB31), }, { } /* Terminating Entry */ }; MODULE_DEVICE_TABLE(pci, dwc3_haps_id_table); static struct pci_driver dwc3_haps_driver = { .name = "dwc3-haps", .id_table = dwc3_haps_id_table, .probe = dwc3_haps_probe, .remove = dwc3_haps_remove, }; MODULE_AUTHOR("Thinh Nguyen <thinhn@synopsys.com>"); MODULE_LICENSE("GPL v2"); MODULE_DESCRIPTION("Synopsys HAPS PCI Glue Layer"); module_pci_driver(dwc3_haps_driver);