[PATCH v8 06/18] iommu: iommu_get_group_resv_regions
Tomasz Nowicki
tnowicki at caviumnetworks.com
Tue Jan 17 04:14:31 PST 2017
On 11.01.2017 10:41, Eric Auger wrote:
> Introduce iommu_get_group_resv_regions whose role consists in
> enumerating all devices from the group and collecting their
> reserved regions. The list is sorted and overlaps between
> regions of the same type are handled by merging the regions.
>
> Signed-off-by: Eric Auger <eric.auger at redhat.com>
Reviewed-by: Tomasz Nowicki <tomasz.nowicki at caviumnetworks.com>
Thanks,
Tomasz
>
> ---
> v6 -> v7:
> - avoid merge of regions of different type
>
> v3 -> v4:
> - take the iommu_group lock in iommu_get_group_resv_regions
> - the list now is sorted and overlaps are checked
>
> NOTE:
> - we do not move list elements from device to group list since
> the iommu_put_resv_regions() could not be called.
> - at the moment I did not introduce any iommu_put_group_resv_regions
> since it simply consists in voiding/freeing the list
> ---
> drivers/iommu/iommu.c | 98 +++++++++++++++++++++++++++++++++++++++++++++++++++
> include/linux/iommu.h | 8 +++++
> 2 files changed, 106 insertions(+)
>
> diff --git a/drivers/iommu/iommu.c b/drivers/iommu/iommu.c
> index 41c1906..640056b 100644
> --- a/drivers/iommu/iommu.c
> +++ b/drivers/iommu/iommu.c
> @@ -133,6 +133,104 @@ static ssize_t iommu_group_show_name(struct iommu_group *group, char *buf)
> return sprintf(buf, "%s\n", group->name);
> }
>
> +/**
> + * iommu_insert_resv_region - Insert a new region in the
> + * list of reserved regions.
> + * @new: new region to insert
> + * @regions: list of regions
> + *
> + * The new element is sorted by address with respect to the other
> + * regions of the same type. In case it overlaps with another
> + * region of the same type, regions are merged. In case it
> + * overlaps with another region of different type, regions are
> + * not merged.
> + */
> +static int iommu_insert_resv_region(struct iommu_resv_region *new,
> + struct list_head *regions)
> +{
> + struct iommu_resv_region *region;
> + phys_addr_t start = new->start;
> + phys_addr_t end = new->start + new->length - 1;
> + struct list_head *pos = regions->next;
> +
> + while (pos != regions) {
> + struct iommu_resv_region *entry =
> + list_entry(pos, struct iommu_resv_region, list);
> + phys_addr_t a = entry->start;
> + phys_addr_t b = entry->start + entry->length - 1;
> + int type = entry->type;
> +
> + if (end < a) {
> + goto insert;
> + } else if (start > b) {
> + pos = pos->next;
> + } else if ((start >= a) && (end <= b)) {
> + if (new->type == type)
> + goto done;
> + else
> + pos = pos->next;
> + } else {
> + if (new->type == type) {
> + phys_addr_t new_start = min(a, start);
> + phys_addr_t new_end = max(b, end);
> +
> + list_del(&entry->list);
> + entry->start = new_start;
> + entry->length = new_end - new_start + 1;
> + iommu_insert_resv_region(entry, regions);
> + } else {
> + pos = pos->next;
> + }
> + }
> + }
> +insert:
> + region = iommu_alloc_resv_region(new->start, new->length,
> + new->prot, new->type);
> + if (!region)
> + return -ENOMEM;
> +
> + list_add_tail(®ion->list, pos);
> +done:
> + return 0;
> +}
> +
> +static int
> +iommu_insert_device_resv_regions(struct list_head *dev_resv_regions,
> + struct list_head *group_resv_regions)
> +{
> + struct iommu_resv_region *entry;
> + int ret;
> +
> + list_for_each_entry(entry, dev_resv_regions, list) {
> + ret = iommu_insert_resv_region(entry, group_resv_regions);
> + if (ret)
> + break;
> + }
> + return ret;
> +}
> +
> +int iommu_get_group_resv_regions(struct iommu_group *group,
> + struct list_head *head)
> +{
> + struct iommu_device *device;
> + int ret = 0;
> +
> + mutex_lock(&group->mutex);
> + list_for_each_entry(device, &group->devices, list) {
> + struct list_head dev_resv_regions;
> +
> + INIT_LIST_HEAD(&dev_resv_regions);
> + iommu_get_resv_regions(device->dev, &dev_resv_regions);
> + ret = iommu_insert_device_resv_regions(&dev_resv_regions, head);
> + iommu_put_resv_regions(device->dev, &dev_resv_regions);
> + if (ret)
> + break;
> + }
> + mutex_unlock(&group->mutex);
> + return ret;
> +}
> +EXPORT_SYMBOL_GPL(iommu_get_group_resv_regions);
> +
> static IOMMU_GROUP_ATTR(name, S_IRUGO, iommu_group_show_name, NULL);
>
> static void iommu_group_release(struct kobject *kobj)
> diff --git a/include/linux/iommu.h b/include/linux/iommu.h
> index f6bb55d3..bec3730 100644
> --- a/include/linux/iommu.h
> +++ b/include/linux/iommu.h
> @@ -246,6 +246,8 @@ extern void iommu_set_fault_handler(struct iommu_domain *domain,
> extern int iommu_request_dm_for_dev(struct device *dev);
> extern struct iommu_resv_region *
> iommu_alloc_resv_region(phys_addr_t start, size_t length, int prot, int type);
> +extern int iommu_get_group_resv_regions(struct iommu_group *group,
> + struct list_head *head);
>
> extern int iommu_attach_group(struct iommu_domain *domain,
> struct iommu_group *group);
> @@ -463,6 +465,12 @@ static inline void iommu_put_resv_regions(struct device *dev,
> {
> }
>
> +static inline int iommu_get_group_resv_regions(struct iommu_group *group,
> + struct list_head *head)
> +{
> + return -ENODEV;
> +}
> +
> static inline int iommu_request_dm_for_dev(struct device *dev)
> {
> return -ENODEV;
>
More information about the linux-arm-kernel
mailing list