The response header lives in PF-controlled shared memory. Copy it
into a local struct once, then read cmd_res and output_size from the
snapshot so the PF cannot flip cmd_res or grow output_size between
checks.

Signed-off-by: Stanley.Yang <[email protected]>
---
 .../drm/amd/ras/ras_mgr/amdgpu_virt_ras_cmd.c | 29 +++++++++++++++----
 1 file changed, 23 insertions(+), 6 deletions(-)

diff --git a/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_virt_ras_cmd.c 
b/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_virt_ras_cmd.c
index 838eb91aef39..ebbf92a2bd94 100644
--- a/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_virt_ras_cmd.c
+++ b/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_virt_ras_cmd.c
@@ -92,6 +92,7 @@ static int amdgpu_virt_ras_remote_ioctl_cmd(struct 
ras_core_context *ras_core,
        struct amdgpu_virt_ras_cmd *virt_ras = ras_mgr->virt_ras_cmd;
        uint32_t mem_len = ALIGN(sizeof(*cmd) + output_size, 
AMDGPU_GPU_PAGE_SIZE);
        struct ras_cmd_ctx *rcmd;
+       struct ras_cmd_ctx hdr_snap;
        struct amdgpu_virt_shared_mem shared_mem = {0};
        int ret = 0;
 
@@ -108,15 +109,31 @@ static int amdgpu_virt_ras_remote_ioctl_cmd(struct 
ras_core_context *ras_core,
        ret = amdgpu_virt_send_remote_ras_cmd(ras_core->dev,
                                shared_mem.gpa, mem_len);
        if (!ret) {
-               if (rcmd->cmd_res) {
-                       ret = rcmd->cmd_res;
+               /*
+                * rcmd lives in shared memory the PF can mutate at any time.
+                * Snapshot the entire fixed-size response header into a local
+                * struct in one shot so every subsequent decision (cmd_res,
+                * output_size, version, etc.) operates on a stable copy. This
+                * defeats double-fetch / TOCTOU attacks where a malicious or
+                * buggy PF could flip cmd_res from SUCCESS to an error after
+                * our success branch, or enlarge output_size between the
+                * bounds check and the memcpy below to corrupt the caller's
+                * local output buffer.
+                */
+               memcpy(&hdr_snap, rcmd, sizeof(hdr_snap));
+               barrier();
+
+               if (hdr_snap.cmd_res) {
+                       ret = hdr_snap.cmd_res;
                        goto out;
                }
 
-               cmd->cmd_res = rcmd->cmd_res;
-               cmd->output_size = rcmd->output_size;
-               if (rcmd->output_size && (rcmd->output_size <= output_size) && 
output_data)
-                       memcpy(output_data, rcmd->output_buff_raw, 
rcmd->output_size);
+               cmd->cmd_res = hdr_snap.cmd_res;
+               cmd->output_size = hdr_snap.output_size;
+
+               if (hdr_snap.output_size && output_data &&
+                   hdr_snap.output_size <= output_size)
+                       memcpy(output_data, rcmd->output_buff_raw, 
hdr_snap.output_size);
        }
 
 out:
-- 
2.43.0

Reply via email to