Commit 13dc15a3 authored by John Allen's avatar John Allen Committed by Herbert Xu

crypto: ccp - Use kzalloc for sev ioctl interfaces to prevent kernel memory leak

For some sev ioctl interfaces, input may be passed that is less than or
equal to SEV_FW_BLOB_MAX_SIZE, but larger than the data that PSP
firmware returns. In this case, kmalloc will allocate memory that is the
size of the input rather than the size of the data. Since PSP firmware
doesn't fully overwrite the buffer, the sev ioctl interfaces with the
issue may return uninitialized slab memory.

Currently, all of the ioctl interfaces in the ccp driver are safe, but
to prevent future problems, change all ioctl interfaces that allocate
memory with kmalloc to use kzalloc and memset the data buffer to zero
in sev_ioctl_do_platform_status.

Fixes: 38103671 ("crypto: ccp: Use the stack and common buffer for status commands")
Fixes: e7990356 ("crypto: ccp: Implement SEV_PEK_CSR ioctl command")
Fixes: 76a2b524 ("crypto: ccp: Implement SEV_PDH_CERT_EXPORT ioctl command")
Fixes: d6112ea0 ("crypto: ccp - introduce SEV_GET_ID2 command")
Cc: stable@vger.kernel.org
Reported-by: default avatarAndy Nguyen <theflow@google.com>
Suggested-by: default avatarDavid Rientjes <rientjes@google.com>
Suggested-by: default avatarPeter Gonda <pgonda@google.com>
Signed-off-by: default avatarJohn Allen <john.allen@amd.com>
Reviewed-by: default avatarPeter Gonda <pgonda@google.com>
Acked-by: default avatarDavid Rientjes <rientjes@google.com>
Signed-off-by: default avatarHerbert Xu <herbert@gondor.apana.org.au>
parent f2906aa8
...@@ -577,6 +577,8 @@ static int sev_ioctl_do_platform_status(struct sev_issue_cmd *argp) ...@@ -577,6 +577,8 @@ static int sev_ioctl_do_platform_status(struct sev_issue_cmd *argp)
struct sev_user_data_status data; struct sev_user_data_status data;
int ret; int ret;
memset(&data, 0, sizeof(data));
ret = __sev_do_cmd_locked(SEV_CMD_PLATFORM_STATUS, &data, &argp->error); ret = __sev_do_cmd_locked(SEV_CMD_PLATFORM_STATUS, &data, &argp->error);
if (ret) if (ret)
return ret; return ret;
...@@ -630,7 +632,7 @@ static int sev_ioctl_do_pek_csr(struct sev_issue_cmd *argp, bool writable) ...@@ -630,7 +632,7 @@ static int sev_ioctl_do_pek_csr(struct sev_issue_cmd *argp, bool writable)
if (input.length > SEV_FW_BLOB_MAX_SIZE) if (input.length > SEV_FW_BLOB_MAX_SIZE)
return -EFAULT; return -EFAULT;
blob = kmalloc(input.length, GFP_KERNEL); blob = kzalloc(input.length, GFP_KERNEL);
if (!blob) if (!blob)
return -ENOMEM; return -ENOMEM;
...@@ -854,7 +856,7 @@ static int sev_ioctl_do_get_id2(struct sev_issue_cmd *argp) ...@@ -854,7 +856,7 @@ static int sev_ioctl_do_get_id2(struct sev_issue_cmd *argp)
input_address = (void __user *)input.address; input_address = (void __user *)input.address;
if (input.address && input.length) { if (input.address && input.length) {
id_blob = kmalloc(input.length, GFP_KERNEL); id_blob = kzalloc(input.length, GFP_KERNEL);
if (!id_blob) if (!id_blob)
return -ENOMEM; return -ENOMEM;
...@@ -973,14 +975,14 @@ static int sev_ioctl_do_pdh_export(struct sev_issue_cmd *argp, bool writable) ...@@ -973,14 +975,14 @@ static int sev_ioctl_do_pdh_export(struct sev_issue_cmd *argp, bool writable)
if (input.cert_chain_len > SEV_FW_BLOB_MAX_SIZE) if (input.cert_chain_len > SEV_FW_BLOB_MAX_SIZE)
return -EFAULT; return -EFAULT;
pdh_blob = kmalloc(input.pdh_cert_len, GFP_KERNEL); pdh_blob = kzalloc(input.pdh_cert_len, GFP_KERNEL);
if (!pdh_blob) if (!pdh_blob)
return -ENOMEM; return -ENOMEM;
data.pdh_cert_address = __psp_pa(pdh_blob); data.pdh_cert_address = __psp_pa(pdh_blob);
data.pdh_cert_len = input.pdh_cert_len; data.pdh_cert_len = input.pdh_cert_len;
cert_blob = kmalloc(input.cert_chain_len, GFP_KERNEL); cert_blob = kzalloc(input.cert_chain_len, GFP_KERNEL);
if (!cert_blob) { if (!cert_blob) {
ret = -ENOMEM; ret = -ENOMEM;
goto e_free_pdh; goto e_free_pdh;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment