(sorry for the delayed response, but I've been on PTO) > > 1. VFIO_GROUP_GET_DEVICE_FD > > > > User space knows by out-of-band means which device it is accessing > > and will call VFIO_GROUP_GET_DEVICE_FD passing a specific sysfs path > > to get the device information: > > > > fd = ioctl(group, VFIO_GROUP_GET_DEVICE_FD, > > "/sys/bus/platform/devices/ffe210000.usb")); > > FWIW, I'm in favor of whichever way works out cleaner in the code for > pre-pending "/sys/bus" or not. It sort of seems like it's unnecessary. > It's also a little inconsistent that the returned path doesn't > pre-pend /sys in the examples below. Ok. For the returned path in the examples I have the actual device tree path which is slightly different from the path in /sys. The device tree path is what user space would need to interpret /proc/device-tree. > > 2. VFIO_DEVICE_GET_INFO > > > > The number of regions corresponds to the regions defined > > in "reg" and "ranges" in the device tree. > > > > Two new flags are added to struct vfio_device_info: > > > > #define VFIO_DEVICE_FLAGS_PLATFORM (1 << ?) /* A platform bus device */ > > #define VFIO_DEVICE_FLAGS_DEVTREE (1 << ?) /* device tree info available */ > > > > It is possible that there could be platform bus devices > > that are not in the device tree, so we use 2 flags to > > allow for that. > > > > If just VFIO_DEVICE_FLAGS_PLATFORM is set, it means > > that there are regions and IRQs but no device tree info > > available. > > > > If just VFIO_DEVICE_FLAGS_DEVTREE is set, it means > > there is device tree info available. > > But it would be invalid to only have DEVTREE w/o PLATFORM for now, > right? Right. The way I stated it is incorrect. DEVTREE would never be set by itself. > > 3. VFIO_DEVICE_GET_REGION_INFO > > > > For platform devices with multiple regions, information > > is needed to correlate the regions with the device > > tree structure that drivers use to determine the meaning > > of device resources. > > > > The VFIO_DEVICE_GET_REGION_INFO is extended to provide > > device tree information. > > > > The following information is needed: > > -the device tree path to the node corresponding to the > > region > > -whether it corresponds to a "reg" or "ranges" property > > -there could be multiple sub-regions per "reg" or "ranges" and > > the sub-index within the reg/ranges is needed > > > > There are 5 new flags added to vfio_region_info : > > > > struct vfio_region_info { > > __u32 argsz; > > __u32 flags; > > #define VFIO_REGION_INFO_FLAG_CACHEABLE (1 << ?) > > #define VFIO_DEVTREE_REGION_INFO_FLAG_REG (1 << ?) > > #define VFIO_DEVTREE_REGION_INFO_FLAG_RANGE (1 << ?) > > #define VFIO_DEVTREE_REGION_INFO_FLAG_INDEX (1 << ?) > > #define VFIO_DEVTREE_REGION_INFO_FLAG_PATH (1 << ?) > > __u32 index; /* Region index */ > > __u32 resv; /* Reserved for alignment */ > > __u64 size; /* Region size (bytes) */ > > __u64 offset; /* Region offset from start of device fd */ > > }; > > > > VFIO_REGION_INFO_FLAG_CACHEABLE > > -if set indicates that the region must be mapped as cacheable > > > > VFIO_DEVTREE_REGION_INFO_FLAG_REG > > -if set indicates that the region corresponds to a "reg" property > > in the device tree representation of the device > > > > VFIO_DEVTREE_REGION_INFO_FLAG_RANGE > > -if set indicates that the region corresponds to a "ranges" property > > in the device tree representation of the device > > > > VFIO_DEVTREE_REGION_INFO_FLAG_INDEX > > -if set indicates that there is a dword aligned struct > > struct vfio_devtree_region_info_index appended to the > > end of vfio_region_info: > > > > struct vfio_devtree_region_info_index > > { > > u32 index; > > } > > > > A reg or ranges property may have multiple regsion. The index > > specifies the index within the "reg" or "ranges" > > that this region corresponds to. > > > > VFIO_DEVTREE_REGION_INFO_FLAG_PATH > > -if set indicates that there is a dword aligned struct > > struct vfio_devtree_info_path appended to the > > end of vfio_region_info: > > > > struct vfio_devtree_info_path > > { > > u32 len; > > u8 path[]; > > } > > > > The path is the full path to the corresponding device > > tree node. The len field specifies the length of the > > path string. > > > > If multiple flags are set that indicate that there is > > an appended struct, the order of the flags indicates > > the order of the structs. > > > > argsz is set by the kernel specifying the total size of > > struct vfio_region_info and all appended structs. > > > > Suggested usage: > > -call VFIO_DEVICE_GET_REGION_INFO with argsz = > > sizeof(struct vfio_region_info) > > -realloc the buffer > > -call VFIO_DEVICE_GET_REGION_INFO again, and the appended > > structs will be returned > > > > 4. VFIO_DEVICE_GET_IRQ_INFO > > > > For platform devices with multiple interrupts that > > correspond to different subnodes in the device tree, > > information is needed to correlate the interrupts > > to the the device tree structure. > > > > The VFIO_DEVICE_GET_REGION_INFO is extended to provide > > device tree information. > > > > 1 new flag is added to vfio_irq_info : > > > > struct vfio_irq_info { > > __u32 argsz; > > __u32 flags; > > #define VFIO_DEVTREE_IRQ_INFO_FLAG_PATH (1 << ?) > > __u32 index; /* IRQ index */ > > __u32 count; /* Number of IRQs within this index */ > > }; > > > > VFIO_DEVTREE_IRQ_INFO_FLAG_PATH > > -if set indicates that there is a dword aligned struct > > struct vfio_devtree_info_path appended to the > > end of vfio_irq_info : > > > > struct vfio_devtree_info_path > > { > > u32 len; > > u8 path[]; > > } > > > > The path is the full path to the corresponding device > > tree node. The len field specifies the length of the > > path string. > > > > argsz is set by the kernel specifying the total size of > > struct vfio_region_info and all appended structs. > > > > 5. EXAMPLE 1 > > > > Example, Freescale SATA controller: > > > > sata@220000 { > > compatible = "fsl,p2041-sata", "fsl,pq-sata-v2"; > > reg = <0x220000 0x1000>; > > interrupts = <0x44 0x2 0x0 0x0>; > > }; > > > > request to get device FD would look like: > > fd = ioctl(group, VFIO_GROUP_GET_DEVICE_FD, "/sys/bus/platform/devices/ffe220000.sata"); > > > > The VFIO_DEVICE_GET_INFO ioctl would return: > > -1 region > > -1 interrupts > > > > The VFIO_DEVICE_GET_REGION_INFO ioctl would return: > > -for index 0: > > offset=0, size=0x10000 -- allows mmap of physical 0xffe220000 > > flags = VFIO_DEVTREE_REGION_INFO_FLAG_REG | > > VFIO_DEVTREE_REGION_INFO_FLAG_PATH > > vfio_devtree_info_path > > len = 26 > > path = "/soc@ffe000000/sata@220000" > > > > The VFIO_DEVICE_GET_IRQ_INFO ioctl would return: > > -for index 0: > > flags = VFIO_IRQ_INFO_EVENTFD | > > VFIO_IRQ_INFO_MASKABLE | > > VFIO_DEVTREE_IRQ_INFO_FLAG_PATH > > vfio_devtree_info_path > > len = 26 > > path = "/soc@ffe000000/sata@220000" > > > > 6. EXAMPLE 2 > > > > Example, Freescale DMA engine (modified to illustrate): > > > > dma@101300 { > > cell-index = <0x1>; > > ranges = <0x0 0x101100 0x200>; > > reg = <0x101300 0x4>; > > compatible = "fsl,eloplus-dma"; > > #size-cells = <0x1>; > > #address-cells = <0x1>; > > fsl,liodn = <0xc6>; > > > > dma-channel@180 { > > interrupts = <0x23 0x2 0x0 0x0>; > > cell-index = <0x3>; > > reg = <0x180 0x80>; > > compatible = "fsl,eloplus-dma-channel"; > > }; > > > > dma-channel@100 { > > interrupts = <0x22 0x2 0x0 0x0>; > > cell-index = <0x2>; > > reg = <0x100 0x80>; > > compatible = "fsl,eloplus-dma-channel"; > > }; > > > > }; > > > > request to get device FD would look like: > > fd = ioctl(group, VFIO_GROUP_GET_DEVICE_FD, "/sys/bus/platform/devices/ffe101300.dma"); > > > > The VFIO_DEVICE_GET_INFO ioctl would return: > > -2 regions > > -2 interrupts > > > > The VFIO_DEVICE_GET_REGION_INFO ioctl would return: > > -for index 0: > > offset=0x100, size=0x200 -- allows mmap of physical 0xffe101100 > > flags = VFIO_DEVTREE_REGION_INFO_FLAG_RANGES | > > VFIO_DEVTREE_REGION_INFO_FLAG_PATH > > vfio_devtree_info_path > > len = 25 > > path = "/soc@ffe000000/dma@101300" > > > > -for index 1: > > offset=0x300, size=0x4 -- allows mmap of physical 0xffe101300 > > flags = VFIO_DEVTREE_REGION_INFO_FLAG_REG | > > VFIO_DEVTREE_REGION_INFO_FLAG_PATH > > vfio_devtree_info_path > > len = 25 > > path = "/soc@ffe000000/dma@101300" > > > > The VFIO_DEVICE_GET_IRQ_INFO ioctl would return: > > -for index 0: > > flags = VFIO_IRQ_INFO_EVENTFD | > > VFIO_IRQ_INFO_MASKABLE | > > VFIO_DEVTREE_IRQ_INFO_FLAG_PATH > > vfio_devtree_info_path > > len = 41 > > path = "/soc@ffe000000/dma@101300/dma-channel@180" > > > > -for index 0: > > flags = VFIO_IRQ_INFO_EVENTFD | > > VFIO_IRQ_INFO_MASKABLE | > > VFIO_DEVTREE_IRQ_INFO_FLAG_PATH > > vfio_devtree_info_path > > len = 41 > > path = "/soc@ffe000000/dma@101300/dma-channel@100" > > > Seems like it should work. My only API concern with this model of > appending structs is that a user needs to know the size of each struct > even if they don't otherwise care about it in order to step over it. In > some cases, like the path, the size is variable and the user needs to > look into it. The structs must also be strictly ordered based on the > order of the flags or all hope is lost. If we assign flags sequentially > there should be no case where the user needs to step over something that > they doesn't know the size of. Even so, we may still be ahead to define > the first word of each struct as the length (I'm guessing a byte might > be too limiting). It would sure make walking it easier. The 'path' structs already start with the length, so the only change would be to add a length to the vfio_devtree_region_info_index struct right? I guess will make it a u32. Stuart _______________________________________________ Virtualization mailing list Virtualization@xxxxxxxxxxxxxxxxxxxxxxxxxx https://lists.linuxfoundation.org/mailman/listinfo/virtualization