zephyr/subsys/sip_svc/sip_svc_id_mgr.c
Mahesh Rao dc9dc3d044 subsystem: sip_svc: Add ARM SiP SVC subsystem
Introduce a new SiP SVC subsystem to provide ARM Silicon Provider based
supervisory call services. SiP SVC service provides the capability to
send SMC/HVC call from kernel running at EL1 to hypervisor/
secure monitor firmware running at EL2/EL3 and also added SiP SVC shell
commands to exercise the service.

Signed-off-by: Mahesh Rao <mahesh.rao@intel.com>
2023-05-09 08:46:50 -04:00

283 lines
5.4 KiB
C

/*
* Copyright (c) 2022-2023, Intel Corporation.
*
* SPDX-License-Identifier: Apache-2.0
*
* Arm SiP services service ID manager and ID mapping table force
* clients and transactions.
*/
#include <zephyr/kernel.h>
#include <zephyr/drivers/sip_svc/sip_svc_proto.h>
#include "sip_svc_id_mgr.h"
/**
* Create a id key pool using size variable 0..size-1, where we can
* track the allocated id.
*/
struct sip_svc_id_pool *sip_svc_id_mgr_create(uint32_t size)
{
struct sip_svc_id_pool *id_pool = NULL;
uint32_t mask_size = 0;
uint32_t i = 0;
if (size == SIP_SVC_ID_INVALID) {
return NULL;
}
/* Allocate memory for ID pool */
id_pool = k_malloc(sizeof(struct sip_svc_id_pool));
if (!id_pool) {
return NULL;
}
id_pool->size = size;
id_pool->id_list = k_malloc(size * sizeof(uint32_t));
if (!id_pool->id_list) {
k_free(id_pool);
return NULL;
}
mask_size = size / sizeof(uint32_t);
if (size % sizeof(uint32_t)) {
mask_size++;
}
id_pool->id_mask = k_malloc(mask_size * sizeof(uint32_t));
if (!id_pool->id_mask) {
k_free(id_pool->id_list);
k_free(id_pool);
return NULL;
}
/* Initialize ID */
for (i = 0; i < size; i++) {
id_pool->id_list[i] = i;
}
/* ID pool is full during initialization */
id_pool->head = 0;
id_pool->tail = size - 1;
/* Initialize ID in use flags */
for (i = 0; i < mask_size; i++) {
id_pool->id_mask[i] = 0;
}
return id_pool;
}
/* Delete a created id pool*/
void sip_svc_id_mgr_delete(struct sip_svc_id_pool *id_pool)
{
if (id_pool) {
k_free(id_pool->id_mask);
k_free(id_pool->id_list);
k_free(id_pool);
}
}
/* Retrieve an id from the id pool*/
uint32_t sip_svc_id_mgr_alloc(struct sip_svc_id_pool *id_pool)
{
uint32_t id;
uint32_t row;
uint32_t col;
if (!id_pool) {
return SIP_SVC_ID_INVALID;
}
if (id_pool->head == SIP_SVC_ID_INVALID) {
return SIP_SVC_ID_INVALID;
}
id = id_pool->id_list[id_pool->head];
/* Calculate the mask bit position in id mask array*/
row = id / sizeof(uint32_t);
col = id % sizeof(uint32_t);
id_pool->id_mask[row] |= (1 << col);
if (id_pool->head == id_pool->tail) {
id_pool->head = SIP_SVC_ID_INVALID;
id_pool->tail = SIP_SVC_ID_INVALID;
} else {
id_pool->head++;
if (id_pool->head == id_pool->size) {
id_pool->head = 0;
}
}
return id;
}
/* Free the id that was allocated from the id_pool*/
void sip_svc_id_mgr_free(struct sip_svc_id_pool *id_pool, uint32_t id)
{
uint32_t row;
uint32_t col;
if (!id_pool) {
return;
}
if (id >= id_pool->size) {
return;
}
row = id / sizeof(uint32_t);
col = id % sizeof(uint32_t);
/* check if the id was allocated earlier*/
if (!(id_pool->id_mask[row] & (1 << col))) {
return;
}
/* Clear the id mask bit */
id_pool->id_mask[row] &= ~(1 << col);
if (id_pool->head == SIP_SVC_ID_INVALID) {
id_pool->head = 0;
id_pool->tail = 0;
} else {
id_pool->tail++;
if (id_pool->tail == id_pool->size) {
id_pool->tail = 0;
}
if (id_pool->head == id_pool->tail) {
return;
}
}
id_pool->id_list[id_pool->tail] = id;
}
/* Allocate database to store values */
struct sip_svc_id_map *sip_svc_id_map_create(uint32_t size)
{
struct sip_svc_id_map *id_map = NULL;
uint32_t items_size = (sizeof(struct sip_svc_id_map_item)) * size;
int i;
id_map = k_malloc(sizeof(struct sip_svc_id_map));
if (!id_map) {
return NULL;
}
id_map->items = k_malloc(items_size);
if (!id_map->items) {
k_free(id_map);
return NULL;
}
id_map->size = size;
for (i = 0; i < size; i++) {
id_map->items[i].id = SIP_SVC_ID_INVALID;
id_map->items[i].arg1 = NULL;
id_map->items[i].arg2 = NULL;
id_map->items[i].arg3 = NULL;
id_map->items[i].arg4 = NULL;
id_map->items[i].arg5 = NULL;
id_map->items[i].arg6 = NULL;
}
return id_map;
}
/* Delete a created database */
void sip_svc_id_map_delete(struct sip_svc_id_map *id_map)
{
if (id_map) {
k_free(id_map->items);
k_free(id_map);
}
}
/* Retrieve index from from database with key(id)*/
static int sip_svc_id_map_get_idx(struct sip_svc_id_map *id_map, uint32_t id)
{
int i;
if (!id_map) {
return -EINVAL;
}
for (i = 0; i < id_map->size; i++) {
if (id_map->items[i].id == id) {
return i;
}
}
return -EINVAL;
}
/* Insert an entry into database with key as id*/
int sip_svc_id_map_insert_item(struct sip_svc_id_map *id_map, uint32_t id, void *arg1, void *arg2,
void *arg3, void *arg4, void *arg5, void *arg6)
{
int i;
if (!id_map) {
return -EINVAL;
}
i = sip_svc_id_map_get_idx(id_map, SIP_SVC_ID_INVALID);
if (i < 0) {
return -EINVAL;
}
id_map->items[i].id = id;
id_map->items[i].arg1 = arg1;
id_map->items[i].arg2 = arg2;
id_map->items[i].arg3 = arg3;
id_map->items[i].arg4 = arg4;
id_map->items[i].arg5 = arg5;
id_map->items[i].arg6 = arg6;
return 0;
}
/* Remove an entry with key id */
int sip_svc_id_map_remove_item(struct sip_svc_id_map *id_map, uint32_t id)
{
int i;
if (!id_map) {
return -EINVAL;
}
i = sip_svc_id_map_get_idx(id_map, id);
if (i < 0) {
return -EINVAL;
}
id_map->items[i].id = SIP_SVC_ID_INVALID;
id_map->items[i].arg1 = NULL;
id_map->items[i].arg2 = NULL;
id_map->items[i].arg3 = NULL;
id_map->items[i].arg4 = NULL;
id_map->items[i].arg5 = NULL;
id_map->items[i].arg6 = NULL;
return 0;
}
/* Query an entry from database using key id*/
struct sip_svc_id_map_item *sip_svc_id_map_query_item(struct sip_svc_id_map *id_map, uint32_t id)
{
int i;
if (!id_map) {
return NULL;
}
i = sip_svc_id_map_get_idx(id_map, id);
if (i < 0) {
return NULL;
}
return &id_map->items[i];
}