diff options
author | Antonio Nino Diaz <antonio.ninodiaz@arm.com> | 2018-11-08 09:21:48 +0000 |
---|---|---|
committer | Antonio Nino Diaz <antonio.ninodiaz@arm.com> | 2018-12-11 15:04:24 +0000 |
commit | 07c13a30d275d5310d71ca5130726ccf2c1cbcc8 (patch) | |
tree | 00b583c396f78a651018771aca122861e79ba469 /lib/sprt/sprt_queue.c | |
parent | 56ae97924dc80fe1f6fea4896b118d0ca3ea8814 (diff) | |
download | platform_external_arm-trusted-firmware-07c13a30d275d5310d71ca5130726ccf2c1cbcc8.tar.gz platform_external_arm-trusted-firmware-07c13a30d275d5310d71ca5130726ccf2c1cbcc8.tar.bz2 platform_external_arm-trusted-firmware-07c13a30d275d5310d71ca5130726ccf2c1cbcc8.zip |
SPM: Introduce SPRT C host library
Change-Id: If57ec9cc0791f49d9ade83dff9d24ef9047963a8
Co-authored-by: Jean-Paul Etienne <jean-paul.etienne@arm.com>
Signed-off-by: Antonio Nino Diaz <antonio.ninodiaz@arm.com>
Diffstat (limited to 'lib/sprt/sprt_queue.c')
-rw-r--r-- | lib/sprt/sprt_queue.c | 104 |
1 files changed, 104 insertions, 0 deletions
diff --git a/lib/sprt/sprt_queue.c b/lib/sprt/sprt_queue.c new file mode 100644 index 000000000..2bd4139ea --- /dev/null +++ b/lib/sprt/sprt_queue.c @@ -0,0 +1,104 @@ +/* + * Copyright (c) 2018, Arm Limited. All rights reserved. + * + * SPDX-License-Identifier: BSD-3-Clause + */ + +#include <assert.h> +#include <errno.h> +#include <stdint.h> +#include <string.h> + +#include "sprt_queue.h" + +void sprt_queue_init(void *queue_base, uint32_t entry_num, uint32_t entry_size) +{ + assert(queue_base != NULL); + assert(entry_size > 0U); + assert(entry_num > 0U); + + struct sprt_queue *queue = (struct sprt_queue *)queue_base; + + queue->entry_num = entry_num; + queue->entry_size = entry_size; + queue->idx_write = 0U; + queue->idx_read = 0U; + + memset(queue->data, 0, entry_num * entry_size); +} + +int sprt_queue_is_empty(void *queue_base) +{ + assert(queue_base != NULL); + + struct sprt_queue *queue = (struct sprt_queue *)queue_base; + + return (queue->idx_write == queue->idx_read); +} + +int sprt_queue_is_full(void *queue_base) +{ + assert(queue_base != NULL); + + struct sprt_queue *queue = (struct sprt_queue *)queue_base; + + uint32_t idx_next_write = (queue->idx_write + 1) % queue->entry_num; + + return (idx_next_write == queue->idx_read); +} + +int sprt_queue_push(void *queue_base, const void *entry) +{ + assert(entry != NULL); + assert(queue_base != NULL); + + if (sprt_queue_is_full(queue_base) != 0) { + return -ENOMEM; + } + + struct sprt_queue *queue = (struct sprt_queue *)queue_base; + + uint8_t *dst_entry = &queue->data[queue->entry_size * queue->idx_write]; + + memcpy(dst_entry, entry, queue->entry_size); + + /* + * Make sure that the message data is visible before increasing the + * counter of available messages. + */ + __asm__ volatile("dmb st" ::: "memory"); + + queue->idx_write = (queue->idx_write + 1) % queue->entry_num; + + __asm__ volatile("dmb st" ::: "memory"); + + return 0; +} + +int sprt_queue_pop(void *queue_base, void *entry) +{ + assert(entry != NULL); + assert(queue_base != NULL); + + if (sprt_queue_is_empty(queue_base) != 0) { + return -ENOENT; + } + + struct sprt_queue *queue = (struct sprt_queue *)queue_base; + + uint8_t *src_entry = &queue->data[queue->entry_size * queue->idx_read]; + + memcpy(entry, src_entry, queue->entry_size); + + /* + * Make sure that the message data is visible before increasing the + * counter of read messages. + */ + __asm__ volatile("dmb st" ::: "memory"); + + queue->idx_read = (queue->idx_read + 1) % queue->entry_num; + + __asm__ volatile("dmb st" ::: "memory"); + + return 0; +} |