| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Helpers for IOMMU drivers implementing SVA
- */
- #include <linux/mmu_context.h>
- #include <linux/mmu_notifier.h>
- #include <linux/mutex.h>
- #include <linux/sched/mm.h>
- #include <linux/iommu.h>
- #include "iommu-priv.h"
- static DEFINE_MUTEX(iommu_sva_lock);
- static bool iommu_sva_present;
- static LIST_HEAD(iommu_sva_mms);
- static struct iommu_domain *iommu_sva_domain_alloc(struct device *dev,
- struct mm_struct *mm);
- /* Allocate a PASID for the mm within range (inclusive) */
- static struct iommu_mm_data *iommu_alloc_mm_data(struct mm_struct *mm, struct device *dev)
- {
- struct iommu_mm_data *iommu_mm;
- ioasid_t pasid;
- lockdep_assert_held(&iommu_sva_lock);
- if (!arch_pgtable_dma_compat(mm))
- return ERR_PTR(-EBUSY);
- iommu_mm = mm->iommu_mm;
- /* Is a PASID already associated with this mm? */
- if (iommu_mm) {
- if (iommu_mm->pasid >= dev->iommu->max_pasids)
- return ERR_PTR(-EOVERFLOW);
- return iommu_mm;
- }
- iommu_mm = kzalloc_obj(struct iommu_mm_data);
- if (!iommu_mm)
- return ERR_PTR(-ENOMEM);
- pasid = iommu_alloc_global_pasid(dev);
- if (pasid == IOMMU_PASID_INVALID) {
- kfree(iommu_mm);
- return ERR_PTR(-ENOSPC);
- }
- iommu_mm->pasid = pasid;
- iommu_mm->mm = mm;
- INIT_LIST_HEAD(&iommu_mm->sva_domains);
- /*
- * Make sure the write to mm->iommu_mm is not reordered in front of
- * initialization to iommu_mm fields. If it does, readers may see a
- * valid iommu_mm with uninitialized values.
- */
- smp_store_release(&mm->iommu_mm, iommu_mm);
- return iommu_mm;
- }
- /**
- * iommu_sva_bind_device() - Bind a process address space to a device
- * @dev: the device
- * @mm: the mm to bind, caller must hold a reference to mm_users
- *
- * Create a bond between device and address space, allowing the device to
- * access the mm using the PASID returned by iommu_sva_get_pasid(). If a
- * bond already exists between @device and @mm, an additional internal
- * reference is taken. Caller must call iommu_sva_unbind_device()
- * to release each reference.
- *
- * On error, returns an ERR_PTR value.
- */
- struct iommu_sva *iommu_sva_bind_device(struct device *dev, struct mm_struct *mm)
- {
- struct iommu_group *group = dev->iommu_group;
- struct iommu_attach_handle *attach_handle;
- struct iommu_mm_data *iommu_mm;
- struct iommu_domain *domain;
- struct iommu_sva *handle;
- int ret;
- if (!group)
- return ERR_PTR(-ENODEV);
- mutex_lock(&iommu_sva_lock);
- /* Allocate mm->pasid if necessary. */
- iommu_mm = iommu_alloc_mm_data(mm, dev);
- if (IS_ERR(iommu_mm)) {
- ret = PTR_ERR(iommu_mm);
- goto out_unlock;
- }
- /* A bond already exists, just take a reference`. */
- attach_handle = iommu_attach_handle_get(group, iommu_mm->pasid, IOMMU_DOMAIN_SVA);
- if (!IS_ERR(attach_handle)) {
- handle = container_of(attach_handle, struct iommu_sva, handle);
- if (attach_handle->domain->mm != mm) {
- ret = -EBUSY;
- goto out_unlock;
- }
- refcount_inc(&handle->users);
- mutex_unlock(&iommu_sva_lock);
- return handle;
- }
- if (PTR_ERR(attach_handle) != -ENOENT) {
- ret = PTR_ERR(attach_handle);
- goto out_unlock;
- }
- handle = kzalloc_obj(*handle);
- if (!handle) {
- ret = -ENOMEM;
- goto out_unlock;
- }
- /* Search for an existing domain. */
- list_for_each_entry(domain, &mm->iommu_mm->sva_domains, next) {
- ret = iommu_attach_device_pasid(domain, dev, iommu_mm->pasid,
- &handle->handle);
- if (!ret) {
- domain->users++;
- goto out;
- }
- }
- /* Allocate a new domain and set it on device pasid. */
- domain = iommu_sva_domain_alloc(dev, mm);
- if (IS_ERR(domain)) {
- ret = PTR_ERR(domain);
- goto out_free_handle;
- }
- ret = iommu_attach_device_pasid(domain, dev, iommu_mm->pasid,
- &handle->handle);
- if (ret)
- goto out_free_domain;
- domain->users = 1;
- if (list_empty(&iommu_mm->sva_domains)) {
- if (list_empty(&iommu_sva_mms))
- iommu_sva_present = true;
- list_add(&iommu_mm->mm_list_elm, &iommu_sva_mms);
- }
- list_add(&domain->next, &iommu_mm->sva_domains);
- out:
- refcount_set(&handle->users, 1);
- mutex_unlock(&iommu_sva_lock);
- handle->dev = dev;
- return handle;
- out_free_domain:
- iommu_domain_free(domain);
- out_free_handle:
- kfree(handle);
- out_unlock:
- mutex_unlock(&iommu_sva_lock);
- return ERR_PTR(ret);
- }
- EXPORT_SYMBOL_GPL(iommu_sva_bind_device);
- /**
- * iommu_sva_unbind_device() - Remove a bond created with iommu_sva_bind_device
- * @handle: the handle returned by iommu_sva_bind_device()
- *
- * Put reference to a bond between device and address space. The device should
- * not be issuing any more transaction for this PASID. All outstanding page
- * requests for this PASID must have been flushed to the IOMMU.
- */
- void iommu_sva_unbind_device(struct iommu_sva *handle)
- {
- struct iommu_domain *domain = handle->handle.domain;
- struct iommu_mm_data *iommu_mm = domain->mm->iommu_mm;
- struct device *dev = handle->dev;
- mutex_lock(&iommu_sva_lock);
- if (!refcount_dec_and_test(&handle->users)) {
- mutex_unlock(&iommu_sva_lock);
- return;
- }
- iommu_detach_device_pasid(domain, dev, iommu_mm->pasid);
- if (--domain->users == 0) {
- list_del(&domain->next);
- if (list_empty(&iommu_mm->sva_domains)) {
- list_del(&iommu_mm->mm_list_elm);
- if (list_empty(&iommu_sva_mms))
- iommu_sva_present = false;
- }
- iommu_domain_free(domain);
- }
- mutex_unlock(&iommu_sva_lock);
- kfree(handle);
- }
- EXPORT_SYMBOL_GPL(iommu_sva_unbind_device);
- u32 iommu_sva_get_pasid(struct iommu_sva *handle)
- {
- struct iommu_domain *domain = handle->handle.domain;
- return mm_get_enqcmd_pasid(domain->mm);
- }
- EXPORT_SYMBOL_GPL(iommu_sva_get_pasid);
- void mm_pasid_drop(struct mm_struct *mm)
- {
- struct iommu_mm_data *iommu_mm = mm->iommu_mm;
- if (!iommu_mm)
- return;
- iommu_free_global_pasid(iommu_mm->pasid);
- kfree(iommu_mm);
- }
- /*
- * I/O page fault handler for SVA
- */
- static enum iommu_page_response_code
- iommu_sva_handle_mm(struct iommu_fault *fault, struct mm_struct *mm)
- {
- vm_fault_t ret;
- struct vm_area_struct *vma;
- unsigned int access_flags = 0;
- unsigned int fault_flags = FAULT_FLAG_REMOTE;
- struct iommu_fault_page_request *prm = &fault->prm;
- enum iommu_page_response_code status = IOMMU_PAGE_RESP_INVALID;
- if (!(prm->flags & IOMMU_FAULT_PAGE_REQUEST_PASID_VALID))
- return status;
- if (!mmget_not_zero(mm))
- return status;
- mmap_read_lock(mm);
- vma = vma_lookup(mm, prm->addr);
- if (!vma)
- /* Unmapped area */
- goto out_put_mm;
- if (prm->perm & IOMMU_FAULT_PERM_READ)
- access_flags |= VM_READ;
- if (prm->perm & IOMMU_FAULT_PERM_WRITE) {
- access_flags |= VM_WRITE;
- fault_flags |= FAULT_FLAG_WRITE;
- }
- if (prm->perm & IOMMU_FAULT_PERM_EXEC) {
- access_flags |= VM_EXEC;
- fault_flags |= FAULT_FLAG_INSTRUCTION;
- }
- if (!(prm->perm & IOMMU_FAULT_PERM_PRIV))
- fault_flags |= FAULT_FLAG_USER;
- if (access_flags & ~vma->vm_flags)
- /* Access fault */
- goto out_put_mm;
- ret = handle_mm_fault(vma, prm->addr, fault_flags, NULL);
- status = ret & VM_FAULT_ERROR ? IOMMU_PAGE_RESP_INVALID :
- IOMMU_PAGE_RESP_SUCCESS;
- out_put_mm:
- mmap_read_unlock(mm);
- mmput(mm);
- return status;
- }
- static void iommu_sva_handle_iopf(struct work_struct *work)
- {
- struct iopf_fault *iopf;
- struct iopf_group *group;
- enum iommu_page_response_code status = IOMMU_PAGE_RESP_SUCCESS;
- group = container_of(work, struct iopf_group, work);
- list_for_each_entry(iopf, &group->faults, list) {
- /*
- * For the moment, errors are sticky: don't handle subsequent
- * faults in the group if there is an error.
- */
- if (status != IOMMU_PAGE_RESP_SUCCESS)
- break;
- status = iommu_sva_handle_mm(&iopf->fault,
- group->attach_handle->domain->mm);
- }
- iopf_group_response(group, status);
- iopf_free_group(group);
- }
- static int iommu_sva_iopf_handler(struct iopf_group *group)
- {
- struct iommu_fault_param *fault_param = group->fault_param;
- INIT_WORK(&group->work, iommu_sva_handle_iopf);
- if (!queue_work(fault_param->queue->wq, &group->work))
- return -EBUSY;
- return 0;
- }
- static struct iommu_domain *iommu_sva_domain_alloc(struct device *dev,
- struct mm_struct *mm)
- {
- const struct iommu_ops *ops = dev_iommu_ops(dev);
- struct iommu_domain *domain;
- if (!ops->domain_alloc_sva)
- return ERR_PTR(-EOPNOTSUPP);
- domain = ops->domain_alloc_sva(dev, mm);
- if (IS_ERR(domain))
- return domain;
- domain->type = IOMMU_DOMAIN_SVA;
- domain->cookie_type = IOMMU_COOKIE_SVA;
- mmgrab(mm);
- domain->mm = mm;
- domain->owner = ops;
- domain->iopf_handler = iommu_sva_iopf_handler;
- return domain;
- }
- void iommu_sva_invalidate_kva_range(unsigned long start, unsigned long end)
- {
- struct iommu_mm_data *iommu_mm;
- guard(mutex)(&iommu_sva_lock);
- if (!iommu_sva_present)
- return;
- list_for_each_entry(iommu_mm, &iommu_sva_mms, mm_list_elm)
- mmu_notifier_arch_invalidate_secondary_tlbs(iommu_mm->mm, start, end);
- }
|