2024-09-07 22:42:27 +08:00
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2024, sakumisu
|
|
|
|
|
*
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*/
|
2024-08-15 17:51:54 +08:00
|
|
|
#include "chry_mempool.h"
|
|
|
|
|
|
2025-07-18 22:50:48 +08:00
|
|
|
/*****************************************************************************
|
|
|
|
|
* @brief init ringbuffer
|
|
|
|
|
*
|
|
|
|
|
* @param[in] rb ringbuffer instance
|
|
|
|
|
* @param[in] pool memory pool address
|
|
|
|
|
* @param[in] size memory size in byte,
|
|
|
|
|
* must be power of 2 !!!
|
|
|
|
|
*
|
|
|
|
|
* @retval int 0:Success -1:Error
|
|
|
|
|
*****************************************************************************/
|
|
|
|
|
static int __chry_ringbuffer_init(chry_mempool_ringbuffer_t *rb, void *pool, uint32_t size)
|
|
|
|
|
{
|
|
|
|
|
if (NULL == rb) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (NULL == pool) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((size < 2) || (size & (size - 1))) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rb->in = 0;
|
|
|
|
|
rb->out = 0;
|
|
|
|
|
rb->mask = size - 1;
|
|
|
|
|
rb->pool = pool;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
|
* @brief reset ringbuffer, clean all data,
|
|
|
|
|
* should be add lock in multithread
|
|
|
|
|
*
|
|
|
|
|
* @param[in] rb ringbuffer instance
|
|
|
|
|
*
|
|
|
|
|
*****************************************************************************/
|
|
|
|
|
static void __chry_ringbuffer_reset(chry_mempool_ringbuffer_t *rb)
|
|
|
|
|
{
|
|
|
|
|
rb->in = 0;
|
|
|
|
|
rb->out = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
|
* @brief write data to ringbuffer,
|
|
|
|
|
* should be add lock in multithread,
|
|
|
|
|
* in single write thread not need lock
|
|
|
|
|
*
|
|
|
|
|
* @param[in] rb ringbuffer instance
|
|
|
|
|
* @param[in] data data pointer
|
|
|
|
|
* @param[in] size size in byte
|
|
|
|
|
*
|
|
|
|
|
* @retval uint32_t actual write size in byte
|
|
|
|
|
*****************************************************************************/
|
|
|
|
|
static uint32_t __chry_ringbuffer_write(chry_mempool_ringbuffer_t *rb, void *data, uint32_t size)
|
|
|
|
|
{
|
|
|
|
|
uint32_t unused;
|
|
|
|
|
uint32_t offset;
|
|
|
|
|
uint32_t remain;
|
|
|
|
|
|
|
|
|
|
unused = (rb->mask + 1) - (rb->in - rb->out);
|
|
|
|
|
|
|
|
|
|
if (size > unused) {
|
|
|
|
|
size = unused;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
offset = rb->in & rb->mask;
|
|
|
|
|
|
|
|
|
|
remain = rb->mask + 1 - offset;
|
|
|
|
|
remain = remain > size ? size : remain;
|
|
|
|
|
|
|
|
|
|
memcpy(((uint8_t *)(rb->pool)) + offset, data, remain);
|
|
|
|
|
memcpy(rb->pool, (uint8_t *)data + remain, size - remain);
|
|
|
|
|
|
|
|
|
|
rb->in += size;
|
|
|
|
|
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
|
* @brief peek data from ringbuffer
|
|
|
|
|
* should be add lock in multithread,
|
|
|
|
|
* in single read thread not need lock
|
|
|
|
|
*
|
|
|
|
|
* @param[in] rb ringbuffer instance
|
|
|
|
|
* @param[in] data data pointer
|
|
|
|
|
* @param[in] size size in byte
|
|
|
|
|
*
|
|
|
|
|
* @retval uint32_t actual peek size in byte
|
|
|
|
|
*****************************************************************************/
|
|
|
|
|
static uint32_t __chry_ringbuffer_peek(chry_mempool_ringbuffer_t *rb, void *data, uint32_t size)
|
|
|
|
|
{
|
|
|
|
|
uint32_t used;
|
|
|
|
|
uint32_t offset;
|
|
|
|
|
uint32_t remain;
|
|
|
|
|
|
|
|
|
|
used = rb->in - rb->out;
|
|
|
|
|
if (size > used) {
|
|
|
|
|
size = used;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
offset = rb->out & rb->mask;
|
|
|
|
|
|
|
|
|
|
remain = rb->mask + 1 - offset;
|
|
|
|
|
remain = remain > size ? size : remain;
|
|
|
|
|
|
|
|
|
|
memcpy(data, ((uint8_t *)(rb->pool)) + offset, remain);
|
|
|
|
|
memcpy((uint8_t *)data + remain, rb->pool, size - remain);
|
|
|
|
|
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
|
* @brief read data from ringbuffer
|
|
|
|
|
* should be add lock in multithread,
|
|
|
|
|
* in single read thread not need lock
|
|
|
|
|
*
|
|
|
|
|
* @param[in] rb ringbuffer instance
|
|
|
|
|
* @param[in] data data pointer
|
|
|
|
|
* @param[in] size size in byte
|
|
|
|
|
*
|
|
|
|
|
* @retval uint32_t actual read size in byte
|
|
|
|
|
*****************************************************************************/
|
|
|
|
|
static uint32_t __chry_ringbuffer_read(chry_mempool_ringbuffer_t *rb, void *data, uint32_t size)
|
|
|
|
|
{
|
|
|
|
|
size = __chry_ringbuffer_peek(rb, data, size);
|
|
|
|
|
rb->out += size;
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
2024-08-15 17:51:54 +08:00
|
|
|
int chry_mempool_create(struct chry_mempool *pool, void *block, uint32_t block_size, uint32_t block_count)
|
|
|
|
|
{
|
2025-03-04 17:12:41 +08:00
|
|
|
uintptr_t *item;
|
2024-08-15 17:51:54 +08:00
|
|
|
|
2025-03-04 17:12:41 +08:00
|
|
|
if (block_count > CONFIG_CHRY_MEMPOOL_MAX_BLOCK_COUNT) {
|
2024-11-26 20:57:15 +08:00
|
|
|
return -1;
|
|
|
|
|
}
|
2024-08-15 17:51:54 +08:00
|
|
|
|
2026-02-05 20:38:06 +08:00
|
|
|
if (block_size % 4) {
|
2024-09-07 22:42:27 +08:00
|
|
|
return -1;
|
|
|
|
|
}
|
2024-08-15 17:51:54 +08:00
|
|
|
|
2025-07-18 22:50:48 +08:00
|
|
|
if (__chry_ringbuffer_init(&pool->in, pool->in_buf, sizeof(uintptr_t) * block_count) == -1) {
|
2024-11-26 20:57:15 +08:00
|
|
|
return -1;
|
2024-08-15 17:51:54 +08:00
|
|
|
}
|
2024-11-26 20:57:15 +08:00
|
|
|
|
2025-07-18 22:50:48 +08:00
|
|
|
if (__chry_ringbuffer_init(&pool->out, pool->out_buf, sizeof(uintptr_t) * block_count) == -1) {
|
2024-08-15 17:51:54 +08:00
|
|
|
return -1;
|
|
|
|
|
}
|
2024-11-26 20:57:15 +08:00
|
|
|
|
2024-11-28 18:23:01 +08:00
|
|
|
pool->out_sem = chry_mempool_osal_sem_create(block_count);
|
2024-11-26 20:57:15 +08:00
|
|
|
if (pool->out_sem == NULL) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-28 18:23:01 +08:00
|
|
|
pool->block = block;
|
|
|
|
|
pool->block_size = block_size;
|
|
|
|
|
pool->block_count = block_count;
|
|
|
|
|
|
2025-03-04 17:12:41 +08:00
|
|
|
for (uint32_t i = 0; i < pool->block_count; i++) {
|
|
|
|
|
item = (uintptr_t *)((uint8_t *)pool->block + i * pool->block_size);
|
|
|
|
|
chry_mempool_free(pool, item);
|
2024-11-26 20:57:15 +08:00
|
|
|
}
|
|
|
|
|
|
2024-08-15 17:51:54 +08:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void chry_mempool_delete(struct chry_mempool *pool)
|
|
|
|
|
{
|
2025-07-18 22:50:48 +08:00
|
|
|
__chry_ringbuffer_reset(&pool->in);
|
|
|
|
|
__chry_ringbuffer_reset(&pool->out);
|
2026-02-05 20:38:06 +08:00
|
|
|
chry_mempool_osal_sem_delete(pool->out_sem);
|
2024-08-15 17:51:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uintptr_t *chry_mempool_alloc(struct chry_mempool *pool)
|
|
|
|
|
{
|
|
|
|
|
uintptr_t *addr;
|
2024-11-13 22:42:15 +08:00
|
|
|
uint32_t len;
|
2024-08-15 17:51:54 +08:00
|
|
|
|
2025-07-18 22:50:48 +08:00
|
|
|
len = __chry_ringbuffer_read(&pool->in, (uintptr_t *)&addr, sizeof(uintptr_t));
|
2024-11-13 22:42:15 +08:00
|
|
|
if (len == 0) {
|
2024-08-15 17:51:54 +08:00
|
|
|
return NULL;
|
|
|
|
|
} else {
|
|
|
|
|
return addr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int chry_mempool_free(struct chry_mempool *pool, uintptr_t *item)
|
|
|
|
|
{
|
|
|
|
|
uintptr_t addr;
|
|
|
|
|
|
|
|
|
|
addr = (uintptr_t)item;
|
2025-07-18 22:50:48 +08:00
|
|
|
return __chry_ringbuffer_write(&pool->in, &addr, sizeof(uintptr_t));
|
2024-08-15 17:51:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int chry_mempool_send(struct chry_mempool *pool, uintptr_t *item)
|
|
|
|
|
{
|
2024-11-26 20:57:15 +08:00
|
|
|
uintptr_t addr;
|
|
|
|
|
|
|
|
|
|
addr = (uintptr_t)item;
|
2025-07-18 22:50:48 +08:00
|
|
|
__chry_ringbuffer_write(&pool->out, &addr, sizeof(uintptr_t));
|
2024-11-28 18:23:01 +08:00
|
|
|
return chry_mempool_osal_sem_give(pool->out_sem);
|
2024-08-15 17:51:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int chry_mempool_recv(struct chry_mempool *pool, uintptr_t **item, uint32_t timeout)
|
|
|
|
|
{
|
2024-11-26 20:57:15 +08:00
|
|
|
uint32_t len;
|
|
|
|
|
int ret;
|
|
|
|
|
|
2024-11-28 18:23:01 +08:00
|
|
|
ret = chry_mempool_osal_sem_take(pool->out_sem, timeout);
|
2024-11-26 20:57:15 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-18 22:50:48 +08:00
|
|
|
len = __chry_ringbuffer_read(&pool->out, (uintptr_t *)item, sizeof(uintptr_t));
|
2024-11-26 20:57:15 +08:00
|
|
|
if (len == 0) {
|
|
|
|
|
return -1;
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2024-11-28 18:23:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void chry_mempool_reset(struct chry_mempool *pool)
|
|
|
|
|
{
|
2025-03-04 17:12:41 +08:00
|
|
|
uintptr_t *item;
|
2024-11-28 18:23:01 +08:00
|
|
|
|
2025-07-18 22:50:48 +08:00
|
|
|
__chry_ringbuffer_reset(&pool->in);
|
|
|
|
|
__chry_ringbuffer_reset(&pool->out);
|
2024-11-28 18:23:01 +08:00
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < pool->block_count; i++) {
|
2025-03-04 17:12:41 +08:00
|
|
|
item = (uintptr_t *)((uint8_t *)pool->block + i * pool->block_size);
|
|
|
|
|
chry_mempool_free(pool, item);
|
2024-11-28 18:23:01 +08:00
|
|
|
}
|
2024-08-15 17:51:54 +08:00
|
|
|
}
|