@@ -85,4 +85,8 @@ typedef struct HostIOMMUDeviceIOMMUFDCaps {
uint64_t hw_caps;
VendorCaps vendor_caps;
} HostIOMMUDeviceIOMMUFDCaps;
+
+int hiod_iommufd_get_common_cap(HostIOMMUDevice *hiod,
+ HostIOMMUDeviceIOMMUFDCaps *caps,
+ int cap, Error **errp);
#endif
@@ -355,3 +355,43 @@ static const TypeInfo types[] = {
};
DEFINE_TYPES(types)
+
+static int hiod_iommufd_get_vtd_cap(HostIOMMUDevice *hiod,
+ struct iommu_hw_info_vtd *vtd,
+ int cap, Error **errp)
+{
+ /* TODO: Check vtd->cap_reg/ecap_reg for capability */
+ error_setg(errp, "%s: unsupported capability %x", hiod->name, cap);
+ return -EINVAL;
+}
+
+static int hiod_iommufd_get_vendor_cap(HostIOMMUDevice *hiod,
+ HostIOMMUDeviceIOMMUFDCaps *caps,
+ int cap, Error **errp)
+{
+ enum iommu_hw_info_type type = caps->type;
+
+ switch (type) {
+ case IOMMU_HW_INFO_TYPE_INTEL_VTD:
+ return hiod_iommufd_get_vtd_cap(hiod, &caps->vendor_caps.vtd,
+ cap, errp);
+ case IOMMU_HW_INFO_TYPE_ARM_SMMUV3:
+ case IOMMU_HW_INFO_TYPE_NONE:
+ break;
+ }
+
+ error_setg(errp, "%s: unsupported capability type %x", hiod->name, type);
+ return -EINVAL;
+}
+
+int hiod_iommufd_get_common_cap(HostIOMMUDevice *hiod,
+ HostIOMMUDeviceIOMMUFDCaps *caps,
+ int cap, Error **errp)
+{
+ switch (cap) {
+ case HOST_IOMMU_DEVICE_CAP_IOMMU_TYPE:
+ return caps->type;
+ default:
+ return hiod_iommufd_get_vendor_cap(hiod, caps, cap, errp);
+ }
+}
@@ -833,6 +833,21 @@ static bool hiod_iommufd_vfio_realize(HostIOMMUDevice *hiod, void *opaque,
return true;
}
+static int hiod_iommufd_vfio_get_cap(HostIOMMUDevice *hiod, int cap,
+ Error **errp)
+{
+ VFIODevice *vdev = hiod->agent;
+ HostIOMMUDeviceIOMMUFDCaps *caps = &vdev->caps;
+
+ /* VFIO has its own way to get aw_bits which may be different from VDPA */
+ switch (cap) {
+ case HOST_IOMMU_DEVICE_CAP_AW_BITS:
+ return vfio_device_get_aw_bits(hiod->agent);
+ default:
+ return hiod_iommufd_get_common_cap(hiod, caps, cap, errp);
+ }
+}
+
static GList *
hiod_iommufd_vfio_get_iova_ranges(HostIOMMUDevice *hiod)
{
@@ -857,6 +872,7 @@ static void hiod_iommufd_vfio_class_init(ObjectClass *oc, void *data)
HostIOMMUDeviceClass *hiodc = HOST_IOMMU_DEVICE_CLASS(oc);
hiodc->realize = hiod_iommufd_vfio_realize;
+ hiodc->get_cap = hiod_iommufd_vfio_get_cap;
hiodc->get_iova_ranges = hiod_iommufd_vfio_get_iova_ranges;
hiodc->get_page_size_mask = hiod_iommufd_vfio_get_page_size_mask;
};
Now we have saved a copy of host iommu capabilities in VFIODevice, implemented hiod_iommufd_vfio_get_cap() by querying the caps copy in sub-class. This overrides .get_cap() implementation hiod_iommufd_vfio_get_cap() in TYPE_HOST_IOMMU_DEVICE_IOMMUFD parent class. Vendor caps are checked for a specific capability, e.g., for vtd, checking code will be in hiod_iommufd_get_vtd_cap(). This also fixes an issue that calling vfio_device_get_aw_bits() in TYPE_HOST_IOMMU_DEVICE_IOMMUFD parent class .get_cap(). Signed-off-by: Zhenzhong Duan <zhenzhong.duan@intel.com> --- include/system/iommufd.h | 4 ++++ backends/iommufd.c | 40 ++++++++++++++++++++++++++++++++++++++++ hw/vfio/iommufd.c | 16 ++++++++++++++++ 3 files changed, 60 insertions(+)