usb_mempool.h 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. /*
  2. * Copyright (c) 2022, sakumisu
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef USB_MEMPOOL_H
  7. #define USB_MEMPOOL_H
  8. #include "usb_osal.h"
  9. #include "usb_ringbuffer.h"
  10. #ifndef CONFIG_USB_MEMPOOL_MAX_BLOCK_COUNT
  11. #define CONFIG_USB_MEMPOOL_MAX_BLOCK_COUNT 16
  12. #endif
  13. struct usb_mempool {
  14. usb_ringbuffer_t in;
  15. usb_ringbuffer_t out;
  16. usb_osal_sem_t out_sem;
  17. void *block;
  18. uint32_t block_size;
  19. uint32_t block_count;
  20. uint8_t in_buf[sizeof(uintptr_t) * CONFIG_USB_MEMPOOL_MAX_BLOCK_COUNT];
  21. uint8_t out_buf[sizeof(uintptr_t) * CONFIG_USB_MEMPOOL_MAX_BLOCK_COUNT];
  22. };
  23. #define usb_mempool_osal_sem_create(max_count) usb_osal_sem_create_counting(max_count)
  24. #define usb_mempool_osal_sem_delete(sem) usb_osal_sem_delete(sem)
  25. #define usb_mempool_osal_sem_take(sem, timeout) usb_osal_sem_take(sem, timeout)
  26. #define usb_mempool_osal_sem_give(sem) usb_osal_sem_give(sem)
  27. #ifdef __cplusplus
  28. extern "C" {
  29. #endif
  30. static inline int usb_mempool_create(struct usb_mempool *pool, void *block, uint32_t block_size, uint32_t block_count)
  31. {
  32. uintptr_t *item;
  33. uint32_t addr;
  34. if (block_count > CONFIG_USB_MEMPOOL_MAX_BLOCK_COUNT) {
  35. return -1;
  36. }
  37. if (block_size % 4) {
  38. return -1;
  39. }
  40. if (usb_ringbuffer_init(&pool->in, pool->in_buf, sizeof(uintptr_t) * block_count) == -1) {
  41. return -1;
  42. }
  43. if (usb_ringbuffer_init(&pool->out, pool->out_buf, sizeof(uintptr_t) * block_count) == -1) {
  44. return -1;
  45. }
  46. pool->out_sem = usb_mempool_osal_sem_create(block_count);
  47. if (pool->out_sem == NULL) {
  48. return -1;
  49. }
  50. pool->block = block;
  51. pool->block_size = block_size;
  52. pool->block_count = block_count;
  53. for (uint32_t i = 0; i < pool->block_count; i++) {
  54. item = (uintptr_t *)((uint8_t *)pool->block + i * pool->block_size);
  55. addr = (uintptr_t)item;
  56. usb_ringbuffer_write(&pool->in, &addr, sizeof(uintptr_t));
  57. }
  58. return 0;
  59. }
  60. static inline void usb_mempool_delete(struct usb_mempool *pool)
  61. {
  62. usb_ringbuffer_reset(&pool->in);
  63. usb_ringbuffer_reset(&pool->out);
  64. usb_mempool_osal_sem_delete(pool->out_sem);
  65. }
  66. static inline uintptr_t *usb_mempool_alloc(struct usb_mempool *pool)
  67. {
  68. uintptr_t *addr;
  69. uint32_t len;
  70. len = usb_ringbuffer_read(&pool->in, (uintptr_t *)&addr, sizeof(uintptr_t));
  71. if (len == 0) {
  72. return NULL;
  73. } else {
  74. return addr;
  75. }
  76. }
  77. static inline int usb_mempool_free(struct usb_mempool *pool, uintptr_t *item)
  78. {
  79. uintptr_t addr;
  80. addr = (uintptr_t)item;
  81. return usb_ringbuffer_write(&pool->in, &addr, sizeof(uintptr_t));
  82. }
  83. static inline int usb_mempool_send(struct usb_mempool *pool, uintptr_t *item)
  84. {
  85. uintptr_t addr;
  86. addr = (uintptr_t)item;
  87. usb_ringbuffer_write(&pool->out, &addr, sizeof(uintptr_t));
  88. return usb_mempool_osal_sem_give(pool->out_sem);
  89. }
  90. static inline int usb_mempool_recv(struct usb_mempool *pool, uintptr_t **item, uint32_t timeout)
  91. {
  92. uint32_t len;
  93. int ret;
  94. ret = usb_mempool_osal_sem_take(pool->out_sem, timeout);
  95. if (ret < 0) {
  96. return -1;
  97. }
  98. len = usb_ringbuffer_read(&pool->out, (uintptr_t *)item, sizeof(uintptr_t));
  99. if (len == 0) {
  100. return -1;
  101. } else {
  102. return 0;
  103. }
  104. }
  105. static inline void usb_mempool_reset(struct usb_mempool *pool)
  106. {
  107. uintptr_t *item;
  108. usb_ringbuffer_reset(&pool->in);
  109. usb_ringbuffer_reset(&pool->out);
  110. for (uint32_t i = 0; i < pool->block_count; i++) {
  111. item = (uintptr_t *)((uint8_t *)pool->block + i * pool->block_size);
  112. usb_mempool_free(pool, item);
  113. }
  114. }
  115. #ifdef __cplusplus
  116. }
  117. #endif
  118. #endif