2021-04-01 15:47:15 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
/*
|
|
|
|
* Handle device page faults
|
|
|
|
*
|
|
|
|
* Copyright (C) 2020 ARM Ltd.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/iommu.h>
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/sched/mm.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/workqueue.h>
|
|
|
|
|
2024-02-12 01:22:21 +00:00
|
|
|
#include "iommu-priv.h"
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
/*
|
|
|
|
* Return the fault parameter of a device if it exists. Otherwise, return NULL.
|
|
|
|
* On a successful return, the caller takes a reference of this parameter and
|
|
|
|
* should put it after use by calling iopf_put_dev_fault_param().
|
|
|
|
*/
|
|
|
|
static struct iommu_fault_param *iopf_get_dev_fault_param(struct device *dev)
|
|
|
|
{
|
|
|
|
struct dev_iommu *param = dev->iommu;
|
|
|
|
struct iommu_fault_param *fault_param;
|
|
|
|
|
|
|
|
rcu_read_lock();
|
|
|
|
fault_param = rcu_dereference(param->fault_param);
|
|
|
|
if (fault_param && !refcount_inc_not_zero(&fault_param->users))
|
|
|
|
fault_param = NULL;
|
|
|
|
rcu_read_unlock();
|
|
|
|
|
|
|
|
return fault_param;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Caller must hold a reference of the fault parameter. */
|
|
|
|
static void iopf_put_dev_fault_param(struct iommu_fault_param *fault_param)
|
|
|
|
{
|
|
|
|
if (refcount_dec_and_test(&fault_param->users))
|
|
|
|
kfree_rcu(fault_param, rcu);
|
|
|
|
}
|
|
|
|
|
2024-02-12 01:22:21 +00:00
|
|
|
void iopf_free_group(struct iopf_group *group)
|
2024-02-12 01:22:19 +00:00
|
|
|
{
|
|
|
|
struct iopf_fault *iopf, *next;
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:19 +00:00
|
|
|
list_for_each_entry_safe(iopf, next, &group->faults, list) {
|
|
|
|
if (!(iopf->fault.prm.flags & IOMMU_FAULT_PAGE_REQUEST_LAST_PAGE))
|
|
|
|
kfree(iopf);
|
|
|
|
}
|
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
/* Pair with iommu_report_device_fault(). */
|
|
|
|
iopf_put_dev_fault_param(group->fault_param);
|
2024-02-12 01:22:19 +00:00
|
|
|
kfree(group);
|
|
|
|
}
|
2024-02-12 01:22:21 +00:00
|
|
|
EXPORT_SYMBOL_GPL(iopf_free_group);
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:20 +00:00
|
|
|
static struct iommu_domain *get_domain_for_iopf(struct device *dev,
|
|
|
|
struct iommu_fault *fault)
|
|
|
|
{
|
|
|
|
struct iommu_domain *domain;
|
|
|
|
|
|
|
|
if (fault->prm.flags & IOMMU_FAULT_PAGE_REQUEST_PASID_VALID) {
|
|
|
|
domain = iommu_get_domain_for_dev_pasid(dev, fault->prm.pasid, 0);
|
|
|
|
if (IS_ERR(domain))
|
|
|
|
domain = NULL;
|
|
|
|
} else {
|
|
|
|
domain = iommu_get_domain_for_dev(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!domain || !domain->iopf_handler) {
|
|
|
|
dev_warn_ratelimited(dev,
|
|
|
|
"iopf (pasid %d) without domain attached or handler installed\n",
|
|
|
|
fault->prm.pasid);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return domain;
|
|
|
|
}
|
|
|
|
|
2021-04-01 15:47:15 +00:00
|
|
|
/**
|
2024-02-12 01:22:21 +00:00
|
|
|
* iommu_handle_iopf - IO Page Fault handler
|
2021-04-01 15:47:15 +00:00
|
|
|
* @fault: fault event
|
2024-02-12 01:22:22 +00:00
|
|
|
* @iopf_param: the fault parameter of the device.
|
2021-04-01 15:47:15 +00:00
|
|
|
*
|
|
|
|
* Add a fault to the device workqueue, to be handled by mm.
|
|
|
|
*
|
|
|
|
* This module doesn't handle PCI PASID Stop Marker; IOMMU drivers must discard
|
|
|
|
* them before reporting faults. A PASID Stop Marker (LRW = 0b100) doesn't
|
|
|
|
* expect a response. It may be generated when disabling a PASID (issuing a
|
|
|
|
* PASID stop request) by some PCI devices.
|
|
|
|
*
|
|
|
|
* The PASID stop request is issued by the device driver before unbind(). Once
|
|
|
|
* it completes, no page request is generated for this PASID anymore and
|
|
|
|
* outstanding ones have been pushed to the IOMMU (as per PCIe 4.0r1.0 - 6.20.1
|
|
|
|
* and 10.4.1.2 - Managing PASID TLP Prefix Usage). Some PCI devices will wait
|
|
|
|
* for all outstanding page requests to come back with a response before
|
|
|
|
* completing the PASID stop request. Others do not wait for page responses, and
|
|
|
|
* instead issue this Stop Marker that tells us when the PASID can be
|
|
|
|
* reallocated.
|
|
|
|
*
|
|
|
|
* It is safe to discard the Stop Marker because it is an optimization.
|
|
|
|
* a. Page requests, which are posted requests, have been flushed to the IOMMU
|
|
|
|
* when the stop request completes.
|
|
|
|
* b. The IOMMU driver flushes all fault queues on unbind() before freeing the
|
|
|
|
* PASID.
|
|
|
|
*
|
|
|
|
* So even though the Stop Marker might be issued by the device *after* the stop
|
|
|
|
* request completes, outstanding faults will have been dealt with by the time
|
|
|
|
* the PASID is freed.
|
|
|
|
*
|
2022-10-31 00:59:15 +00:00
|
|
|
* Any valid page fault will be eventually routed to an iommu domain and the
|
|
|
|
* page fault handler installed there will get called. The users of this
|
|
|
|
* handling framework should guarantee that the iommu domain could only be
|
|
|
|
* freed after the device has stopped generating page faults (or the iommu
|
|
|
|
* hardware has been set to block the page faults) and the pending page faults
|
|
|
|
* have been flushed.
|
|
|
|
*
|
2021-04-01 15:47:15 +00:00
|
|
|
* Return: 0 on success and <0 on error.
|
|
|
|
*/
|
2024-02-12 01:22:22 +00:00
|
|
|
static int iommu_handle_iopf(struct iommu_fault *fault,
|
|
|
|
struct iommu_fault_param *iopf_param)
|
2021-04-01 15:47:15 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct iopf_group *group;
|
2024-02-12 01:22:20 +00:00
|
|
|
struct iommu_domain *domain;
|
2021-04-01 15:47:15 +00:00
|
|
|
struct iopf_fault *iopf, *next;
|
2024-02-12 01:22:22 +00:00
|
|
|
struct device *dev = iopf_param->dev;
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:22 +00:00
|
|
|
lockdep_assert_held(&iopf_param->lock);
|
2021-04-01 15:47:15 +00:00
|
|
|
|
|
|
|
if (fault->type != IOMMU_FAULT_PAGE_REQ)
|
|
|
|
/* Not a recoverable page fault */
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
if (!(fault->prm.flags & IOMMU_FAULT_PAGE_REQUEST_LAST_PAGE)) {
|
|
|
|
iopf = kzalloc(sizeof(*iopf), GFP_KERNEL);
|
|
|
|
if (!iopf)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
iopf->fault = *fault;
|
|
|
|
|
|
|
|
/* Non-last request of a group. Postpone until the last one */
|
|
|
|
list_add(&iopf->list, &iopf_param->partial);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-02-12 01:22:20 +00:00
|
|
|
domain = get_domain_for_iopf(dev, fault);
|
|
|
|
if (!domain) {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto cleanup_partial;
|
|
|
|
}
|
|
|
|
|
2021-04-01 15:47:15 +00:00
|
|
|
group = kzalloc(sizeof(*group), GFP_KERNEL);
|
|
|
|
if (!group) {
|
|
|
|
/*
|
|
|
|
* The caller will send a response to the hardware. But we do
|
|
|
|
* need to clean up before leaving, otherwise partial faults
|
|
|
|
* will be stuck.
|
|
|
|
*/
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto cleanup_partial;
|
|
|
|
}
|
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
group->fault_param = iopf_param;
|
2021-04-01 15:47:15 +00:00
|
|
|
group->last_fault.fault = *fault;
|
|
|
|
INIT_LIST_HEAD(&group->faults);
|
2024-02-12 01:22:20 +00:00
|
|
|
group->domain = domain;
|
2021-04-01 15:47:15 +00:00
|
|
|
list_add(&group->last_fault.list, &group->faults);
|
|
|
|
|
|
|
|
/* See if we have partial faults for this group */
|
|
|
|
list_for_each_entry_safe(iopf, next, &iopf_param->partial, list) {
|
|
|
|
if (iopf->fault.prm.grpid == fault->prm.grpid)
|
|
|
|
/* Insert *before* the last fault */
|
|
|
|
list_move(&iopf->list, &group->faults);
|
|
|
|
}
|
|
|
|
|
2024-02-12 01:22:20 +00:00
|
|
|
mutex_unlock(&iopf_param->lock);
|
|
|
|
ret = domain->iopf_handler(group);
|
|
|
|
mutex_lock(&iopf_param->lock);
|
|
|
|
if (ret)
|
|
|
|
iopf_free_group(group);
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:20 +00:00
|
|
|
return ret;
|
2021-04-01 15:47:15 +00:00
|
|
|
cleanup_partial:
|
|
|
|
list_for_each_entry_safe(iopf, next, &iopf_param->partial, list) {
|
|
|
|
if (iopf->fault.prm.grpid == fault->prm.grpid) {
|
|
|
|
list_del(&iopf->list);
|
|
|
|
kfree(iopf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2024-02-12 01:22:21 +00:00
|
|
|
/**
|
|
|
|
* iommu_report_device_fault() - Report fault event to device driver
|
|
|
|
* @dev: the device
|
|
|
|
* @evt: fault event data
|
|
|
|
*
|
|
|
|
* Called by IOMMU drivers when a fault is detected, typically in a threaded IRQ
|
|
|
|
* handler. When this function fails and the fault is recoverable, it is the
|
|
|
|
* caller's responsibility to complete the fault.
|
|
|
|
*
|
|
|
|
* Return 0 on success, or an error.
|
|
|
|
*/
|
|
|
|
int iommu_report_device_fault(struct device *dev, struct iopf_fault *evt)
|
|
|
|
{
|
2024-02-12 01:22:23 +00:00
|
|
|
bool last_prq = evt->fault.type == IOMMU_FAULT_PAGE_REQ &&
|
|
|
|
(evt->fault.prm.flags & IOMMU_FAULT_PAGE_REQUEST_LAST_PAGE);
|
2024-02-12 01:22:22 +00:00
|
|
|
struct iommu_fault_param *fault_param;
|
2024-02-12 01:22:23 +00:00
|
|
|
struct iopf_fault *evt_pending;
|
|
|
|
int ret;
|
2024-02-12 01:22:21 +00:00
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
fault_param = iopf_get_dev_fault_param(dev);
|
|
|
|
if (!fault_param)
|
2024-02-12 01:22:22 +00:00
|
|
|
return -EINVAL;
|
2024-02-12 01:22:21 +00:00
|
|
|
|
2024-02-12 01:22:22 +00:00
|
|
|
mutex_lock(&fault_param->lock);
|
2024-02-12 01:22:23 +00:00
|
|
|
if (last_prq) {
|
2024-02-12 01:22:21 +00:00
|
|
|
evt_pending = kmemdup(evt, sizeof(struct iopf_fault),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!evt_pending) {
|
|
|
|
ret = -ENOMEM;
|
2024-02-12 01:22:23 +00:00
|
|
|
goto err_unlock;
|
2024-02-12 01:22:21 +00:00
|
|
|
}
|
2024-02-12 01:22:22 +00:00
|
|
|
list_add_tail(&evt_pending->list, &fault_param->faults);
|
2024-02-12 01:22:21 +00:00
|
|
|
}
|
|
|
|
|
2024-02-12 01:22:22 +00:00
|
|
|
ret = iommu_handle_iopf(&evt->fault, fault_param);
|
2024-02-12 01:22:23 +00:00
|
|
|
if (ret)
|
|
|
|
goto err_free;
|
|
|
|
|
|
|
|
mutex_unlock(&fault_param->lock);
|
|
|
|
/* The reference count of fault_param is now held by iopf_group. */
|
|
|
|
if (!last_prq)
|
|
|
|
iopf_put_dev_fault_param(fault_param);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
err_free:
|
|
|
|
if (last_prq) {
|
2024-02-12 01:22:21 +00:00
|
|
|
list_del(&evt_pending->list);
|
|
|
|
kfree(evt_pending);
|
|
|
|
}
|
2024-02-12 01:22:23 +00:00
|
|
|
err_unlock:
|
2024-02-12 01:22:22 +00:00
|
|
|
mutex_unlock(&fault_param->lock);
|
2024-02-12 01:22:23 +00:00
|
|
|
iopf_put_dev_fault_param(fault_param);
|
2024-02-12 01:22:22 +00:00
|
|
|
|
2024-02-12 01:22:21 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iommu_report_device_fault);
|
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
static int iommu_page_response(struct iopf_group *group,
|
|
|
|
struct iommu_page_response *msg)
|
2024-02-12 01:22:20 +00:00
|
|
|
{
|
2024-02-12 01:22:21 +00:00
|
|
|
bool needs_pasid;
|
|
|
|
int ret = -EINVAL;
|
|
|
|
struct iopf_fault *evt;
|
|
|
|
struct iommu_fault_page_request *prm;
|
2024-02-12 01:22:23 +00:00
|
|
|
struct device *dev = group->fault_param->dev;
|
2024-02-12 01:22:21 +00:00
|
|
|
const struct iommu_ops *ops = dev_iommu_ops(dev);
|
|
|
|
bool has_pasid = msg->flags & IOMMU_PAGE_RESP_PASID_VALID;
|
2024-02-12 01:22:23 +00:00
|
|
|
struct iommu_fault_param *fault_param = group->fault_param;
|
2024-02-12 01:22:21 +00:00
|
|
|
|
|
|
|
/* Only send response if there is a fault report pending */
|
2024-02-12 01:22:22 +00:00
|
|
|
mutex_lock(&fault_param->lock);
|
|
|
|
if (list_empty(&fault_param->faults)) {
|
2024-02-12 01:22:21 +00:00
|
|
|
dev_warn_ratelimited(dev, "no pending PRQ, drop response\n");
|
|
|
|
goto done_unlock;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Check if we have a matching page request pending to respond,
|
|
|
|
* otherwise return -EINVAL
|
|
|
|
*/
|
2024-02-12 01:22:22 +00:00
|
|
|
list_for_each_entry(evt, &fault_param->faults, list) {
|
2024-02-12 01:22:21 +00:00
|
|
|
prm = &evt->fault.prm;
|
|
|
|
if (prm->grpid != msg->grpid)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the PASID is required, the corresponding request is
|
|
|
|
* matched using the group ID, the PASID valid bit and the PASID
|
|
|
|
* value. Otherwise only the group ID matches request and
|
|
|
|
* response.
|
|
|
|
*/
|
|
|
|
needs_pasid = prm->flags & IOMMU_FAULT_PAGE_RESPONSE_NEEDS_PASID;
|
|
|
|
if (needs_pasid && (!has_pasid || msg->pasid != prm->pasid))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!needs_pasid && has_pasid) {
|
|
|
|
/* No big deal, just clear it. */
|
|
|
|
msg->flags &= ~IOMMU_PAGE_RESP_PASID_VALID;
|
|
|
|
msg->pasid = 0;
|
|
|
|
}
|
2024-02-12 01:22:20 +00:00
|
|
|
|
2024-02-12 01:22:21 +00:00
|
|
|
ret = ops->page_response(dev, evt, msg);
|
|
|
|
list_del(&evt->list);
|
|
|
|
kfree(evt);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
done_unlock:
|
2024-02-12 01:22:22 +00:00
|
|
|
mutex_unlock(&fault_param->lock);
|
2024-02-12 01:22:23 +00:00
|
|
|
|
2024-02-12 01:22:21 +00:00
|
|
|
return ret;
|
2024-02-12 01:22:20 +00:00
|
|
|
}
|
|
|
|
|
2021-04-01 15:47:15 +00:00
|
|
|
/**
|
|
|
|
* iopf_queue_flush_dev - Ensure that all queued faults have been processed
|
|
|
|
* @dev: the endpoint whose faults need to be flushed.
|
|
|
|
*
|
|
|
|
* The IOMMU driver calls this before releasing a PASID, to ensure that all
|
|
|
|
* pending faults for this PASID have been handled, and won't hit the address
|
|
|
|
* space of the next process that uses this PASID. The driver must make sure
|
|
|
|
* that no new fault is added to the queue. In particular it must flush its
|
|
|
|
* low-level queue before calling this function.
|
|
|
|
*
|
|
|
|
* Return: 0 on success and <0 on error.
|
|
|
|
*/
|
|
|
|
int iopf_queue_flush_dev(struct device *dev)
|
|
|
|
{
|
2024-02-12 01:22:16 +00:00
|
|
|
struct iommu_fault_param *iopf_param;
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
/*
|
|
|
|
* It's a driver bug to be here after iopf_queue_remove_device().
|
|
|
|
* Therefore, it's safe to dereference the fault parameter without
|
|
|
|
* holding the lock.
|
|
|
|
*/
|
|
|
|
iopf_param = rcu_dereference_check(dev->iommu->fault_param, true);
|
|
|
|
if (WARN_ON(!iopf_param))
|
2021-04-01 15:47:15 +00:00
|
|
|
return -ENODEV;
|
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
flush_workqueue(iopf_param->queue->wq);
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
return 0;
|
2021-04-01 15:47:15 +00:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iopf_queue_flush_dev);
|
|
|
|
|
2024-02-12 01:22:21 +00:00
|
|
|
/**
|
|
|
|
* iopf_group_response - Respond a group of page faults
|
|
|
|
* @group: the group of faults with the same group id
|
|
|
|
* @status: the response code
|
|
|
|
*
|
|
|
|
* Return 0 on success and <0 on error.
|
|
|
|
*/
|
|
|
|
int iopf_group_response(struct iopf_group *group,
|
|
|
|
enum iommu_page_response_code status)
|
|
|
|
{
|
|
|
|
struct iopf_fault *iopf = &group->last_fault;
|
|
|
|
struct iommu_page_response resp = {
|
|
|
|
.pasid = iopf->fault.prm.pasid,
|
|
|
|
.grpid = iopf->fault.prm.grpid,
|
|
|
|
.code = status,
|
|
|
|
};
|
|
|
|
|
|
|
|
if ((iopf->fault.prm.flags & IOMMU_FAULT_PAGE_REQUEST_PASID_VALID) &&
|
|
|
|
(iopf->fault.prm.flags & IOMMU_FAULT_PAGE_RESPONSE_NEEDS_PASID))
|
|
|
|
resp.flags = IOMMU_PAGE_RESP_PASID_VALID;
|
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
return iommu_page_response(group, &resp);
|
2024-02-12 01:22:21 +00:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iopf_group_response);
|
|
|
|
|
2021-04-01 15:47:15 +00:00
|
|
|
/**
|
|
|
|
* iopf_queue_discard_partial - Remove all pending partial fault
|
|
|
|
* @queue: the queue whose partial faults need to be discarded
|
|
|
|
*
|
|
|
|
* When the hardware queue overflows, last page faults in a group may have been
|
|
|
|
* lost and the IOMMU driver calls this to discard all partial faults. The
|
|
|
|
* driver shouldn't be adding new faults to this queue concurrently.
|
|
|
|
*
|
|
|
|
* Return: 0 on success and <0 on error.
|
|
|
|
*/
|
|
|
|
int iopf_queue_discard_partial(struct iopf_queue *queue)
|
|
|
|
{
|
|
|
|
struct iopf_fault *iopf, *next;
|
2024-02-12 01:22:16 +00:00
|
|
|
struct iommu_fault_param *iopf_param;
|
2021-04-01 15:47:15 +00:00
|
|
|
|
|
|
|
if (!queue)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
mutex_lock(&queue->lock);
|
|
|
|
list_for_each_entry(iopf_param, &queue->devices, queue_list) {
|
2024-02-12 01:22:22 +00:00
|
|
|
mutex_lock(&iopf_param->lock);
|
2021-04-01 15:47:15 +00:00
|
|
|
list_for_each_entry_safe(iopf, next, &iopf_param->partial,
|
|
|
|
list) {
|
|
|
|
list_del(&iopf->list);
|
|
|
|
kfree(iopf);
|
|
|
|
}
|
2024-02-12 01:22:22 +00:00
|
|
|
mutex_unlock(&iopf_param->lock);
|
2021-04-01 15:47:15 +00:00
|
|
|
}
|
|
|
|
mutex_unlock(&queue->lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iopf_queue_discard_partial);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* iopf_queue_add_device - Add producer to the fault queue
|
|
|
|
* @queue: IOPF queue
|
|
|
|
* @dev: device to add
|
|
|
|
*
|
|
|
|
* Return: 0 on success and <0 on error.
|
|
|
|
*/
|
|
|
|
int iopf_queue_add_device(struct iopf_queue *queue, struct device *dev)
|
|
|
|
{
|
2024-02-12 01:22:16 +00:00
|
|
|
int ret = 0;
|
2021-04-01 15:47:15 +00:00
|
|
|
struct dev_iommu *param = dev->iommu;
|
2024-02-12 01:22:16 +00:00
|
|
|
struct iommu_fault_param *fault_param;
|
2024-02-12 01:22:23 +00:00
|
|
|
const struct iommu_ops *ops = dev_iommu_ops(dev);
|
|
|
|
|
|
|
|
if (!ops->page_response)
|
|
|
|
return -ENODEV;
|
2021-04-01 15:47:15 +00:00
|
|
|
|
|
|
|
mutex_lock(&queue->lock);
|
|
|
|
mutex_lock(¶m->lock);
|
2024-02-12 01:22:23 +00:00
|
|
|
if (rcu_dereference_check(param->fault_param,
|
|
|
|
lockdep_is_held(¶m->lock))) {
|
2024-02-12 01:22:16 +00:00
|
|
|
ret = -EBUSY;
|
|
|
|
goto done_unlock;
|
2021-04-01 15:47:15 +00:00
|
|
|
}
|
2024-02-12 01:22:16 +00:00
|
|
|
|
|
|
|
fault_param = kzalloc(sizeof(*fault_param), GFP_KERNEL);
|
|
|
|
if (!fault_param) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto done_unlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_init(&fault_param->lock);
|
|
|
|
INIT_LIST_HEAD(&fault_param->faults);
|
|
|
|
INIT_LIST_HEAD(&fault_param->partial);
|
|
|
|
fault_param->dev = dev;
|
2024-02-12 01:22:23 +00:00
|
|
|
refcount_set(&fault_param->users, 1);
|
2024-02-12 01:22:16 +00:00
|
|
|
list_add(&fault_param->queue_list, &queue->devices);
|
|
|
|
fault_param->queue = queue;
|
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
rcu_assign_pointer(param->fault_param, fault_param);
|
2024-02-12 01:22:16 +00:00
|
|
|
|
|
|
|
done_unlock:
|
2021-04-01 15:47:15 +00:00
|
|
|
mutex_unlock(¶m->lock);
|
|
|
|
mutex_unlock(&queue->lock);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iopf_queue_add_device);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* iopf_queue_remove_device - Remove producer from fault queue
|
|
|
|
* @queue: IOPF queue
|
|
|
|
* @dev: device to remove
|
|
|
|
*
|
2024-02-12 01:22:24 +00:00
|
|
|
* Removing a device from an iopf_queue. It's recommended to follow these
|
|
|
|
* steps when removing a device:
|
2021-04-01 15:47:15 +00:00
|
|
|
*
|
2024-02-12 01:22:24 +00:00
|
|
|
* - Disable new PRI reception: Turn off PRI generation in the IOMMU hardware
|
|
|
|
* and flush any hardware page request queues. This should be done before
|
|
|
|
* calling into this helper.
|
|
|
|
* - Acknowledge all outstanding PRQs to the device: Respond to all outstanding
|
|
|
|
* page requests with IOMMU_PAGE_RESP_INVALID, indicating the device should
|
|
|
|
* not retry. This helper function handles this.
|
|
|
|
* - Disable PRI on the device: After calling this helper, the caller could
|
|
|
|
* then disable PRI on the device.
|
|
|
|
*
|
|
|
|
* Calling iopf_queue_remove_device() essentially disassociates the device.
|
|
|
|
* The fault_param might still exist, but iommu_page_response() will do
|
|
|
|
* nothing. The device fault parameter reference count has been properly
|
|
|
|
* passed from iommu_report_device_fault() to the fault handling work, and
|
|
|
|
* will eventually be released after iommu_page_response().
|
2021-04-01 15:47:15 +00:00
|
|
|
*/
|
2024-02-12 01:22:24 +00:00
|
|
|
void iopf_queue_remove_device(struct iopf_queue *queue, struct device *dev)
|
2021-04-01 15:47:15 +00:00
|
|
|
{
|
|
|
|
struct iopf_fault *iopf, *next;
|
2024-02-12 01:22:24 +00:00
|
|
|
struct iommu_page_response resp;
|
2021-04-01 15:47:15 +00:00
|
|
|
struct dev_iommu *param = dev->iommu;
|
2024-02-12 01:22:23 +00:00
|
|
|
struct iommu_fault_param *fault_param;
|
2024-02-12 01:22:24 +00:00
|
|
|
const struct iommu_ops *ops = dev_iommu_ops(dev);
|
2021-04-01 15:47:15 +00:00
|
|
|
|
|
|
|
mutex_lock(&queue->lock);
|
|
|
|
mutex_lock(¶m->lock);
|
2024-02-12 01:22:23 +00:00
|
|
|
fault_param = rcu_dereference_check(param->fault_param,
|
|
|
|
lockdep_is_held(¶m->lock));
|
2024-02-12 01:22:16 +00:00
|
|
|
|
2024-02-12 01:22:24 +00:00
|
|
|
if (WARN_ON(!fault_param || fault_param->queue != queue))
|
2024-02-12 01:22:16 +00:00
|
|
|
goto unlock;
|
|
|
|
|
2024-02-12 01:22:24 +00:00
|
|
|
mutex_lock(&fault_param->lock);
|
|
|
|
list_for_each_entry_safe(iopf, next, &fault_param->partial, list)
|
|
|
|
kfree(iopf);
|
2024-02-12 01:22:16 +00:00
|
|
|
|
2024-02-12 01:22:24 +00:00
|
|
|
list_for_each_entry_safe(iopf, next, &fault_param->faults, list) {
|
|
|
|
memset(&resp, 0, sizeof(struct iommu_page_response));
|
|
|
|
resp.pasid = iopf->fault.prm.pasid;
|
|
|
|
resp.grpid = iopf->fault.prm.grpid;
|
|
|
|
resp.code = IOMMU_PAGE_RESP_INVALID;
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:24 +00:00
|
|
|
if (iopf->fault.prm.flags & IOMMU_FAULT_PAGE_RESPONSE_NEEDS_PASID)
|
|
|
|
resp.flags = IOMMU_PAGE_RESP_PASID_VALID;
|
|
|
|
|
|
|
|
ops->page_response(dev, iopf, &resp);
|
|
|
|
list_del(&iopf->list);
|
2021-04-01 15:47:15 +00:00
|
|
|
kfree(iopf);
|
2024-02-12 01:22:24 +00:00
|
|
|
}
|
|
|
|
mutex_unlock(&fault_param->lock);
|
|
|
|
|
|
|
|
list_del(&fault_param->queue_list);
|
2021-04-01 15:47:15 +00:00
|
|
|
|
2024-02-12 01:22:23 +00:00
|
|
|
/* dec the ref owned by iopf_queue_add_device() */
|
|
|
|
rcu_assign_pointer(param->fault_param, NULL);
|
|
|
|
iopf_put_dev_fault_param(fault_param);
|
2024-02-12 01:22:16 +00:00
|
|
|
unlock:
|
|
|
|
mutex_unlock(¶m->lock);
|
|
|
|
mutex_unlock(&queue->lock);
|
2021-04-01 15:47:15 +00:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iopf_queue_remove_device);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* iopf_queue_alloc - Allocate and initialize a fault queue
|
|
|
|
* @name: a unique string identifying the queue (for workqueue)
|
|
|
|
*
|
|
|
|
* Return: the queue on success and NULL on error.
|
|
|
|
*/
|
|
|
|
struct iopf_queue *iopf_queue_alloc(const char *name)
|
|
|
|
{
|
|
|
|
struct iopf_queue *queue;
|
|
|
|
|
|
|
|
queue = kzalloc(sizeof(*queue), GFP_KERNEL);
|
|
|
|
if (!queue)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The WQ is unordered because the low-level handler enqueues faults by
|
|
|
|
* group. PRI requests within a group have to be ordered, but once
|
|
|
|
* that's dealt with, the high-level function can handle groups out of
|
|
|
|
* order.
|
|
|
|
*/
|
|
|
|
queue->wq = alloc_workqueue("iopf_queue/%s", WQ_UNBOUND, 0, name);
|
|
|
|
if (!queue->wq) {
|
|
|
|
kfree(queue);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
INIT_LIST_HEAD(&queue->devices);
|
|
|
|
mutex_init(&queue->lock);
|
|
|
|
|
|
|
|
return queue;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iopf_queue_alloc);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* iopf_queue_free - Free IOPF queue
|
|
|
|
* @queue: queue to free
|
|
|
|
*
|
|
|
|
* Counterpart to iopf_queue_alloc(). The driver must not be queuing faults or
|
|
|
|
* adding/removing devices on this queue anymore.
|
|
|
|
*/
|
|
|
|
void iopf_queue_free(struct iopf_queue *queue)
|
|
|
|
{
|
2024-02-12 01:22:16 +00:00
|
|
|
struct iommu_fault_param *iopf_param, *next;
|
2021-04-01 15:47:15 +00:00
|
|
|
|
|
|
|
if (!queue)
|
|
|
|
return;
|
|
|
|
|
|
|
|
list_for_each_entry_safe(iopf_param, next, &queue->devices, queue_list)
|
|
|
|
iopf_queue_remove_device(queue, iopf_param->dev);
|
|
|
|
|
|
|
|
destroy_workqueue(queue->wq);
|
|
|
|
kfree(queue);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iopf_queue_free);
|