提交 26322ab5 编写于 作者: D David Woodhouse

iommu/vt-d: Fix NULL pointer dereference in page request error case

Dan Carpenter pointed out an error path which could lead to us
dereferencing the 'svm' pointer after we know it to be NULL because the
PASID lookup failed. Fix that, and make it less likely to happen again.

Fixes: a222a7f0 ('iommu/vt-d: Implement page request handling')
Signed-off-by: NDavid Woodhouse <David.Woodhouse@intel.com>
上级 5cec7537
...@@ -510,7 +510,7 @@ static irqreturn_t prq_event_thread(int irq, void *d) ...@@ -510,7 +510,7 @@ static irqreturn_t prq_event_thread(int irq, void *d)
pr_err("%s: Page request for invalid PASID %d: %08llx %08llx\n", pr_err("%s: Page request for invalid PASID %d: %08llx %08llx\n",
iommu->name, req->pasid, ((unsigned long long *)req)[0], iommu->name, req->pasid, ((unsigned long long *)req)[0],
((unsigned long long *)req)[1]); ((unsigned long long *)req)[1]);
goto bad_req; goto no_pasid;
} }
} }
...@@ -552,7 +552,11 @@ static irqreturn_t prq_event_thread(int irq, void *d) ...@@ -552,7 +552,11 @@ static irqreturn_t prq_event_thread(int irq, void *d)
(req->wr_req << 1) | (req->exe_req); (req->wr_req << 1) | (req->exe_req);
sdev->ops->fault_cb(sdev->dev, req->pasid, req->addr, req->private, rwxp, result); sdev->ops->fault_cb(sdev->dev, req->pasid, req->addr, req->private, rwxp, result);
} }
/* We get here in the error case where the PASID lookup failed,
and these can be NULL. Do not use them below this point! */
sdev = NULL;
svm = NULL;
no_pasid:
if (req->lpig) { if (req->lpig) {
/* Page Group Response */ /* Page Group Response */
resp.low = QI_PGRP_PASID(req->pasid) | resp.low = QI_PGRP_PASID(req->pasid) |
...@@ -562,7 +566,7 @@ static irqreturn_t prq_event_thread(int irq, void *d) ...@@ -562,7 +566,7 @@ static irqreturn_t prq_event_thread(int irq, void *d)
resp.high = QI_PGRP_IDX(req->prg_index) | resp.high = QI_PGRP_IDX(req->prg_index) |
QI_PGRP_PRIV(req->private) | QI_PGRP_RESP_CODE(result); QI_PGRP_PRIV(req->private) | QI_PGRP_RESP_CODE(result);
qi_submit_sync(&resp, svm->iommu); qi_submit_sync(&resp, iommu);
} else if (req->srr) { } else if (req->srr) {
/* Page Stream Response */ /* Page Stream Response */
resp.low = QI_PSTRM_IDX(req->prg_index) | resp.low = QI_PSTRM_IDX(req->prg_index) |
...@@ -571,7 +575,7 @@ static irqreturn_t prq_event_thread(int irq, void *d) ...@@ -571,7 +575,7 @@ static irqreturn_t prq_event_thread(int irq, void *d)
resp.high = QI_PSTRM_ADDR(address) | QI_PSTRM_DEVFN(req->devfn) | resp.high = QI_PSTRM_ADDR(address) | QI_PSTRM_DEVFN(req->devfn) |
QI_PSTRM_RESP_CODE(result); QI_PSTRM_RESP_CODE(result);
qi_submit_sync(&resp, svm->iommu); qi_submit_sync(&resp, iommu);
} }
head = (head + sizeof(*req)) & PRQ_RING_MASK; head = (head + sizeof(*req)) & PRQ_RING_MASK;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册