Commit f1383348 authored by Deepak Kumar Singh's avatar Deepak Kumar Singh Committed by Bjorn Andersson

soc: qcom: smem: validate fields of shared structures

Structures in shared memory that can be modified by remote
processors may have untrusted values, they should be validated
before use.

Adding proper validation before using fields of shared
structures.
Signed-off-by: default avatarDeepak Kumar Singh <quic_deesin@quicinc.com>
Signed-off-by: default avatarBjorn Andersson <bjorn.andersson@linaro.org>
Link: https://lore.kernel.org/r/1646147913-15791-2-git-send-email-quic_deesin@quicinc.com
parent 20bb6c9d
...@@ -367,13 +367,18 @@ static int qcom_smem_alloc_private(struct qcom_smem *smem, ...@@ -367,13 +367,18 @@ static int qcom_smem_alloc_private(struct qcom_smem *smem,
struct smem_partition_header *phdr; struct smem_partition_header *phdr;
size_t alloc_size; size_t alloc_size;
void *cached; void *cached;
void *p_end;
phdr = (struct smem_partition_header __force *)part->virt_base; phdr = (struct smem_partition_header __force *)part->virt_base;
p_end = (void *)phdr + part->size;
hdr = phdr_to_first_uncached_entry(phdr); hdr = phdr_to_first_uncached_entry(phdr);
end = phdr_to_last_uncached_entry(phdr); end = phdr_to_last_uncached_entry(phdr);
cached = phdr_to_last_cached_entry(phdr); cached = phdr_to_last_cached_entry(phdr);
if (WARN_ON((void *)end > p_end || cached > p_end))
return -EINVAL;
while (hdr < end) { while (hdr < end) {
if (hdr->canary != SMEM_PRIVATE_CANARY) if (hdr->canary != SMEM_PRIVATE_CANARY)
goto bad_canary; goto bad_canary;
...@@ -383,6 +388,9 @@ static int qcom_smem_alloc_private(struct qcom_smem *smem, ...@@ -383,6 +388,9 @@ static int qcom_smem_alloc_private(struct qcom_smem *smem,
hdr = uncached_entry_next(hdr); hdr = uncached_entry_next(hdr);
} }
if (WARN_ON((void *)hdr > p_end))
return -EINVAL;
/* Check that we don't grow into the cached region */ /* Check that we don't grow into the cached region */
alloc_size = sizeof(*hdr) + ALIGN(size, 8); alloc_size = sizeof(*hdr) + ALIGN(size, 8);
if ((void *)hdr + alloc_size > cached) { if ((void *)hdr + alloc_size > cached) {
...@@ -501,6 +509,8 @@ static void *qcom_smem_get_global(struct qcom_smem *smem, ...@@ -501,6 +509,8 @@ static void *qcom_smem_get_global(struct qcom_smem *smem,
struct smem_header *header; struct smem_header *header;
struct smem_region *region; struct smem_region *region;
struct smem_global_entry *entry; struct smem_global_entry *entry;
u64 entry_offset;
u32 e_size;
u32 aux_base; u32 aux_base;
unsigned i; unsigned i;
...@@ -515,9 +525,16 @@ static void *qcom_smem_get_global(struct qcom_smem *smem, ...@@ -515,9 +525,16 @@ static void *qcom_smem_get_global(struct qcom_smem *smem,
region = &smem->regions[i]; region = &smem->regions[i];
if ((u32)region->aux_base == aux_base || !aux_base) { if ((u32)region->aux_base == aux_base || !aux_base) {
e_size = le32_to_cpu(entry->size);
entry_offset = le32_to_cpu(entry->offset);
if (WARN_ON(e_size + entry_offset > region->size))
return ERR_PTR(-EINVAL);
if (size != NULL) if (size != NULL)
*size = le32_to_cpu(entry->size); *size = e_size;
return region->virt_base + le32_to_cpu(entry->offset);
return region->virt_base + entry_offset;
} }
} }
...@@ -531,8 +548,12 @@ static void *qcom_smem_get_private(struct qcom_smem *smem, ...@@ -531,8 +548,12 @@ static void *qcom_smem_get_private(struct qcom_smem *smem,
{ {
struct smem_private_entry *e, *end; struct smem_private_entry *e, *end;
struct smem_partition_header *phdr; struct smem_partition_header *phdr;
void *item_ptr, *p_end;
u32 padding_data;
u32 e_size;
phdr = (struct smem_partition_header __force *)part->virt_base; phdr = (struct smem_partition_header __force *)part->virt_base;
p_end = (void *)phdr + part->size;
e = phdr_to_first_uncached_entry(phdr); e = phdr_to_first_uncached_entry(phdr);
end = phdr_to_last_uncached_entry(phdr); end = phdr_to_last_uncached_entry(phdr);
...@@ -542,36 +563,65 @@ static void *qcom_smem_get_private(struct qcom_smem *smem, ...@@ -542,36 +563,65 @@ static void *qcom_smem_get_private(struct qcom_smem *smem,
goto invalid_canary; goto invalid_canary;
if (le16_to_cpu(e->item) == item) { if (le16_to_cpu(e->item) == item) {
if (size != NULL) if (size != NULL) {
*size = le32_to_cpu(e->size) - e_size = le32_to_cpu(e->size);
le16_to_cpu(e->padding_data); padding_data = le16_to_cpu(e->padding_data);
return uncached_entry_to_item(e); if (WARN_ON(e_size > part->size || padding_data > e_size))
return ERR_PTR(-EINVAL);
*size = e_size - padding_data;
}
item_ptr = uncached_entry_to_item(e);
if (WARN_ON(item_ptr > p_end))
return ERR_PTR(-EINVAL);
return item_ptr;
} }
e = uncached_entry_next(e); e = uncached_entry_next(e);
} }
if (WARN_ON((void *)e > p_end))
return ERR_PTR(-EINVAL);
/* Item was not found in the uncached list, search the cached list */ /* Item was not found in the uncached list, search the cached list */
e = phdr_to_first_cached_entry(phdr, part->cacheline); e = phdr_to_first_cached_entry(phdr, part->cacheline);
end = phdr_to_last_cached_entry(phdr); end = phdr_to_last_cached_entry(phdr);
if (WARN_ON((void *)e < (void *)phdr || (void *)end > p_end))
return ERR_PTR(-EINVAL);
while (e > end) { while (e > end) {
if (e->canary != SMEM_PRIVATE_CANARY) if (e->canary != SMEM_PRIVATE_CANARY)
goto invalid_canary; goto invalid_canary;
if (le16_to_cpu(e->item) == item) { if (le16_to_cpu(e->item) == item) {
if (size != NULL) if (size != NULL) {
*size = le32_to_cpu(e->size) - e_size = le32_to_cpu(e->size);
le16_to_cpu(e->padding_data); padding_data = le16_to_cpu(e->padding_data);
if (WARN_ON(e_size > part->size || padding_data > e_size))
return ERR_PTR(-EINVAL);
*size = e_size - padding_data;
}
item_ptr = cached_entry_to_item(e);
if (WARN_ON(item_ptr < (void *)phdr))
return ERR_PTR(-EINVAL);
return cached_entry_to_item(e); return item_ptr;
} }
e = cached_entry_next(e, part->cacheline); e = cached_entry_next(e, part->cacheline);
} }
if (WARN_ON((void *)e < (void *)phdr))
return ERR_PTR(-EINVAL);
return ERR_PTR(-ENOENT); return ERR_PTR(-ENOENT);
invalid_canary: invalid_canary:
...@@ -648,14 +698,23 @@ int qcom_smem_get_free_space(unsigned host) ...@@ -648,14 +698,23 @@ int qcom_smem_get_free_space(unsigned host)
phdr = part->virt_base; phdr = part->virt_base;
ret = le32_to_cpu(phdr->offset_free_cached) - ret = le32_to_cpu(phdr->offset_free_cached) -
le32_to_cpu(phdr->offset_free_uncached); le32_to_cpu(phdr->offset_free_uncached);
if (ret > le32_to_cpu(part->size))
return -EINVAL;
} else if (__smem->global_partition.virt_base) { } else if (__smem->global_partition.virt_base) {
part = &__smem->global_partition; part = &__smem->global_partition;
phdr = part->virt_base; phdr = part->virt_base;
ret = le32_to_cpu(phdr->offset_free_cached) - ret = le32_to_cpu(phdr->offset_free_cached) -
le32_to_cpu(phdr->offset_free_uncached); le32_to_cpu(phdr->offset_free_uncached);
if (ret > le32_to_cpu(part->size))
return -EINVAL;
} else { } else {
header = __smem->regions[0].virt_base; header = __smem->regions[0].virt_base;
ret = le32_to_cpu(header->available); ret = le32_to_cpu(header->available);
if (ret > __smem->regions[0].size)
return -EINVAL;
} }
return ret; return ret;
......
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