mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-14 01:34:43 +08:00
cc6711b0bf
Expose an 'override_only' helper macro (i.e. PCI_DRIVER_OVERRIDE_DEVICE_VFIO) for VFIO PCI sub system and add the required code to prefix its matching entries with "vfio_" in modules.alias file. It allows VFIO device drivers to include match entries in the modules.alias file produced by kbuild that are not used for normal driver autoprobing and module autoloading. Drivers using these match entries can be connected to the PCI device manually, by userspace, using the existing driver_override sysfs. For example the resulting modules.alias may have: alias pci:v000015B3d00001021sv*sd*bc*sc*i* mlx5_core alias vfio_pci:v000015B3d00001021sv*sd*bc*sc*i* mlx5_vfio_pci alias vfio_pci:v*d*sv*sd*bc*sc*i* vfio_pci In this example mlx5_core and mlx5_vfio_pci match to the same PCI device. The kernel will autoload and autobind to mlx5_core but the kernel and udev mechanisms will ignore mlx5_vfio_pci. When userspace wants to change a device to the VFIO subsystem it can implement a generic algorithm: 1) Identify the sysfs path to the device: /sys/devices/pci0000:00/0000:00:01.0/0000:01:00.0 2) Get the modalias string from the kernel: $ cat /sys/bus/pci/devices/0000:01:00.0/modalias pci:v000015B3d00001021sv000015B3sd00000001bc02sc00i00 3) Prefix it with vfio_: vfio_pci:v000015B3d00001021sv000015B3sd00000001bc02sc00i00 4) Search modules.alias for the above string and select the entry that has the fewest *'s: alias vfio_pci:v000015B3d00001021sv*sd*bc*sc*i* mlx5_vfio_pci 5) modprobe the matched module name: $ modprobe mlx5_vfio_pci 6) cat the matched module name to driver_override: echo mlx5_vfio_pci > /sys/bus/pci/devices/0000:01:00.0/driver_override 7) unbind device from original module echo 0000:01:00.0 > /sys/bus/pci/devices/0000:01:00.0/driver/unbind 8) probe PCI drivers (or explicitly bind to mlx5_vfio_pci) echo 0000:01:00.0 > /sys/bus/pci/drivers_probe The algorithm is independent of bus type. In future the other buses with VFIO device drivers, like platform and ACPI, can use this algorithm as well. This patch is the infrastructure to provide the information in the modules.alias to userspace. Convert the only VFIO pci_driver which results in one new line in the modules.alias: alias vfio_pci:v*d*sv*sd*bc*sc*i* vfio_pci Later series introduce additional HW specific VFIO PCI drivers, such as mlx5_vfio_pci. Signed-off-by: Max Gurtovoy <mgurtovoy@nvidia.com> Signed-off-by: Jason Gunthorpe <jgg@nvidia.com> Acked-by: Bjorn Helgaas <bhelgaas@google.com> # for pci.h Signed-off-by: Yishai Hadas <yishaih@nvidia.com> Link: https://lore.kernel.org/r/20210826103912.128972-11-yishaih@nvidia.com Signed-off-by: Alex Williamson <alex.williamson@redhat.com>
264 lines
7.3 KiB
C
264 lines
7.3 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
#include <linux/kbuild.h>
|
|
#include <linux/mod_devicetable.h>
|
|
|
|
#define DEVID(devid) DEFINE(SIZE_##devid, sizeof(struct devid))
|
|
#define DEVID_FIELD(devid, field) \
|
|
DEFINE(OFF_##devid##_##field, offsetof(struct devid, field))
|
|
|
|
int main(void)
|
|
{
|
|
DEVID(usb_device_id);
|
|
DEVID_FIELD(usb_device_id, match_flags);
|
|
DEVID_FIELD(usb_device_id, idVendor);
|
|
DEVID_FIELD(usb_device_id, idProduct);
|
|
DEVID_FIELD(usb_device_id, bcdDevice_lo);
|
|
DEVID_FIELD(usb_device_id, bcdDevice_hi);
|
|
DEVID_FIELD(usb_device_id, bDeviceClass);
|
|
DEVID_FIELD(usb_device_id, bDeviceSubClass);
|
|
DEVID_FIELD(usb_device_id, bDeviceProtocol);
|
|
DEVID_FIELD(usb_device_id, bInterfaceClass);
|
|
DEVID_FIELD(usb_device_id, bInterfaceSubClass);
|
|
DEVID_FIELD(usb_device_id, bInterfaceProtocol);
|
|
DEVID_FIELD(usb_device_id, bInterfaceNumber);
|
|
|
|
DEVID(hid_device_id);
|
|
DEVID_FIELD(hid_device_id, bus);
|
|
DEVID_FIELD(hid_device_id, group);
|
|
DEVID_FIELD(hid_device_id, vendor);
|
|
DEVID_FIELD(hid_device_id, product);
|
|
|
|
DEVID(ieee1394_device_id);
|
|
DEVID_FIELD(ieee1394_device_id, match_flags);
|
|
DEVID_FIELD(ieee1394_device_id, vendor_id);
|
|
DEVID_FIELD(ieee1394_device_id, model_id);
|
|
DEVID_FIELD(ieee1394_device_id, specifier_id);
|
|
DEVID_FIELD(ieee1394_device_id, version);
|
|
|
|
DEVID(pci_device_id);
|
|
DEVID_FIELD(pci_device_id, vendor);
|
|
DEVID_FIELD(pci_device_id, device);
|
|
DEVID_FIELD(pci_device_id, subvendor);
|
|
DEVID_FIELD(pci_device_id, subdevice);
|
|
DEVID_FIELD(pci_device_id, class);
|
|
DEVID_FIELD(pci_device_id, class_mask);
|
|
DEVID_FIELD(pci_device_id, override_only);
|
|
|
|
DEVID(ccw_device_id);
|
|
DEVID_FIELD(ccw_device_id, match_flags);
|
|
DEVID_FIELD(ccw_device_id, cu_type);
|
|
DEVID_FIELD(ccw_device_id, cu_model);
|
|
DEVID_FIELD(ccw_device_id, dev_type);
|
|
DEVID_FIELD(ccw_device_id, dev_model);
|
|
|
|
DEVID(ap_device_id);
|
|
DEVID_FIELD(ap_device_id, dev_type);
|
|
|
|
DEVID(css_device_id);
|
|
DEVID_FIELD(css_device_id, type);
|
|
|
|
DEVID(serio_device_id);
|
|
DEVID_FIELD(serio_device_id, type);
|
|
DEVID_FIELD(serio_device_id, proto);
|
|
DEVID_FIELD(serio_device_id, id);
|
|
DEVID_FIELD(serio_device_id, extra);
|
|
|
|
DEVID(acpi_device_id);
|
|
DEVID_FIELD(acpi_device_id, id);
|
|
DEVID_FIELD(acpi_device_id, cls);
|
|
DEVID_FIELD(acpi_device_id, cls_msk);
|
|
|
|
DEVID(pnp_device_id);
|
|
DEVID_FIELD(pnp_device_id, id);
|
|
|
|
DEVID(pnp_card_device_id);
|
|
DEVID_FIELD(pnp_card_device_id, devs);
|
|
|
|
DEVID(pcmcia_device_id);
|
|
DEVID_FIELD(pcmcia_device_id, match_flags);
|
|
DEVID_FIELD(pcmcia_device_id, manf_id);
|
|
DEVID_FIELD(pcmcia_device_id, card_id);
|
|
DEVID_FIELD(pcmcia_device_id, func_id);
|
|
DEVID_FIELD(pcmcia_device_id, function);
|
|
DEVID_FIELD(pcmcia_device_id, device_no);
|
|
DEVID_FIELD(pcmcia_device_id, prod_id_hash);
|
|
|
|
DEVID(of_device_id);
|
|
DEVID_FIELD(of_device_id, name);
|
|
DEVID_FIELD(of_device_id, type);
|
|
DEVID_FIELD(of_device_id, compatible);
|
|
|
|
DEVID(vio_device_id);
|
|
DEVID_FIELD(vio_device_id, type);
|
|
DEVID_FIELD(vio_device_id, compat);
|
|
|
|
DEVID(input_device_id);
|
|
DEVID_FIELD(input_device_id, flags);
|
|
DEVID_FIELD(input_device_id, bustype);
|
|
DEVID_FIELD(input_device_id, vendor);
|
|
DEVID_FIELD(input_device_id, product);
|
|
DEVID_FIELD(input_device_id, version);
|
|
DEVID_FIELD(input_device_id, evbit);
|
|
DEVID_FIELD(input_device_id, keybit);
|
|
DEVID_FIELD(input_device_id, relbit);
|
|
DEVID_FIELD(input_device_id, absbit);
|
|
DEVID_FIELD(input_device_id, mscbit);
|
|
DEVID_FIELD(input_device_id, ledbit);
|
|
DEVID_FIELD(input_device_id, sndbit);
|
|
DEVID_FIELD(input_device_id, ffbit);
|
|
DEVID_FIELD(input_device_id, swbit);
|
|
|
|
DEVID(eisa_device_id);
|
|
DEVID_FIELD(eisa_device_id, sig);
|
|
|
|
DEVID(parisc_device_id);
|
|
DEVID_FIELD(parisc_device_id, hw_type);
|
|
DEVID_FIELD(parisc_device_id, hversion);
|
|
DEVID_FIELD(parisc_device_id, hversion_rev);
|
|
DEVID_FIELD(parisc_device_id, sversion);
|
|
|
|
DEVID(sdio_device_id);
|
|
DEVID_FIELD(sdio_device_id, class);
|
|
DEVID_FIELD(sdio_device_id, vendor);
|
|
DEVID_FIELD(sdio_device_id, device);
|
|
|
|
DEVID(ssb_device_id);
|
|
DEVID_FIELD(ssb_device_id, vendor);
|
|
DEVID_FIELD(ssb_device_id, coreid);
|
|
DEVID_FIELD(ssb_device_id, revision);
|
|
|
|
DEVID(bcma_device_id);
|
|
DEVID_FIELD(bcma_device_id, manuf);
|
|
DEVID_FIELD(bcma_device_id, id);
|
|
DEVID_FIELD(bcma_device_id, rev);
|
|
DEVID_FIELD(bcma_device_id, class);
|
|
|
|
DEVID(virtio_device_id);
|
|
DEVID_FIELD(virtio_device_id, device);
|
|
DEVID_FIELD(virtio_device_id, vendor);
|
|
|
|
DEVID(hv_vmbus_device_id);
|
|
DEVID_FIELD(hv_vmbus_device_id, guid);
|
|
|
|
DEVID(rpmsg_device_id);
|
|
DEVID_FIELD(rpmsg_device_id, name);
|
|
|
|
DEVID(i2c_device_id);
|
|
DEVID_FIELD(i2c_device_id, name);
|
|
|
|
DEVID(i3c_device_id);
|
|
DEVID_FIELD(i3c_device_id, match_flags);
|
|
DEVID_FIELD(i3c_device_id, dcr);
|
|
DEVID_FIELD(i3c_device_id, manuf_id);
|
|
DEVID_FIELD(i3c_device_id, part_id);
|
|
DEVID_FIELD(i3c_device_id, extra_info);
|
|
|
|
DEVID(spi_device_id);
|
|
DEVID_FIELD(spi_device_id, name);
|
|
|
|
DEVID(dmi_system_id);
|
|
DEVID_FIELD(dmi_system_id, matches);
|
|
|
|
DEVID(platform_device_id);
|
|
DEVID_FIELD(platform_device_id, name);
|
|
|
|
DEVID(mdio_device_id);
|
|
DEVID_FIELD(mdio_device_id, phy_id);
|
|
DEVID_FIELD(mdio_device_id, phy_id_mask);
|
|
|
|
DEVID(zorro_device_id);
|
|
DEVID_FIELD(zorro_device_id, id);
|
|
|
|
DEVID(isapnp_device_id);
|
|
DEVID_FIELD(isapnp_device_id, vendor);
|
|
DEVID_FIELD(isapnp_device_id, function);
|
|
|
|
DEVID(ipack_device_id);
|
|
DEVID_FIELD(ipack_device_id, format);
|
|
DEVID_FIELD(ipack_device_id, vendor);
|
|
DEVID_FIELD(ipack_device_id, device);
|
|
|
|
DEVID(amba_id);
|
|
DEVID_FIELD(amba_id, id);
|
|
DEVID_FIELD(amba_id, mask);
|
|
|
|
DEVID(mips_cdmm_device_id);
|
|
DEVID_FIELD(mips_cdmm_device_id, type);
|
|
|
|
DEVID(x86_cpu_id);
|
|
DEVID_FIELD(x86_cpu_id, feature);
|
|
DEVID_FIELD(x86_cpu_id, family);
|
|
DEVID_FIELD(x86_cpu_id, model);
|
|
DEVID_FIELD(x86_cpu_id, vendor);
|
|
|
|
DEVID(cpu_feature);
|
|
DEVID_FIELD(cpu_feature, feature);
|
|
|
|
DEVID(mei_cl_device_id);
|
|
DEVID_FIELD(mei_cl_device_id, name);
|
|
DEVID_FIELD(mei_cl_device_id, uuid);
|
|
DEVID_FIELD(mei_cl_device_id, version);
|
|
|
|
DEVID(rio_device_id);
|
|
DEVID_FIELD(rio_device_id, did);
|
|
DEVID_FIELD(rio_device_id, vid);
|
|
DEVID_FIELD(rio_device_id, asm_did);
|
|
DEVID_FIELD(rio_device_id, asm_vid);
|
|
|
|
DEVID(ulpi_device_id);
|
|
DEVID_FIELD(ulpi_device_id, vendor);
|
|
DEVID_FIELD(ulpi_device_id, product);
|
|
|
|
DEVID(hda_device_id);
|
|
DEVID_FIELD(hda_device_id, vendor_id);
|
|
DEVID_FIELD(hda_device_id, rev_id);
|
|
DEVID_FIELD(hda_device_id, api_version);
|
|
|
|
DEVID(sdw_device_id);
|
|
DEVID_FIELD(sdw_device_id, mfg_id);
|
|
DEVID_FIELD(sdw_device_id, part_id);
|
|
DEVID_FIELD(sdw_device_id, sdw_version);
|
|
DEVID_FIELD(sdw_device_id, class_id);
|
|
|
|
DEVID(fsl_mc_device_id);
|
|
DEVID_FIELD(fsl_mc_device_id, vendor);
|
|
DEVID_FIELD(fsl_mc_device_id, obj_type);
|
|
|
|
DEVID(tb_service_id);
|
|
DEVID_FIELD(tb_service_id, match_flags);
|
|
DEVID_FIELD(tb_service_id, protocol_key);
|
|
DEVID_FIELD(tb_service_id, protocol_id);
|
|
DEVID_FIELD(tb_service_id, protocol_version);
|
|
DEVID_FIELD(tb_service_id, protocol_revision);
|
|
|
|
DEVID(typec_device_id);
|
|
DEVID_FIELD(typec_device_id, svid);
|
|
DEVID_FIELD(typec_device_id, mode);
|
|
|
|
DEVID(tee_client_device_id);
|
|
DEVID_FIELD(tee_client_device_id, uuid);
|
|
|
|
DEVID(wmi_device_id);
|
|
DEVID_FIELD(wmi_device_id, guid_string);
|
|
|
|
DEVID(mhi_device_id);
|
|
DEVID_FIELD(mhi_device_id, chan);
|
|
|
|
DEVID(auxiliary_device_id);
|
|
DEVID_FIELD(auxiliary_device_id, name);
|
|
|
|
DEVID(ssam_device_id);
|
|
DEVID_FIELD(ssam_device_id, match_flags);
|
|
DEVID_FIELD(ssam_device_id, domain);
|
|
DEVID_FIELD(ssam_device_id, category);
|
|
DEVID_FIELD(ssam_device_id, target);
|
|
DEVID_FIELD(ssam_device_id, instance);
|
|
DEVID_FIELD(ssam_device_id, function);
|
|
|
|
DEVID(dfl_device_id);
|
|
DEVID_FIELD(dfl_device_id, type);
|
|
DEVID_FIELD(dfl_device_id, feature_id);
|
|
|
|
return 0;
|
|
}
|