k3-cppi-desc-pool.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. // SPDX-License-Identifier: GPL-2.0
  2. /* TI K3 CPPI5 descriptors pool API
  3. *
  4. * Copyright (C) 2020 Texas Instruments Incorporated - http://www.ti.com
  5. */
  6. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  7. #include <linux/device.h>
  8. #include <linux/dma-mapping.h>
  9. #include <linux/err.h>
  10. #include <linux/genalloc.h>
  11. #include <linux/kernel.h>
  12. #include "k3-cppi-desc-pool.h"
  13. struct k3_cppi_desc_pool {
  14. struct device *dev;
  15. dma_addr_t dma_addr;
  16. void *cpumem; /* dma_alloc map */
  17. size_t desc_size;
  18. size_t mem_size;
  19. size_t num_desc;
  20. struct gen_pool *gen_pool;
  21. void **desc_infos;
  22. };
  23. void k3_cppi_desc_pool_destroy(struct k3_cppi_desc_pool *pool)
  24. {
  25. if (!pool)
  26. return;
  27. WARN(gen_pool_size(pool->gen_pool) != gen_pool_avail(pool->gen_pool),
  28. "k3_knav_desc_pool size %zu != avail %zu",
  29. gen_pool_size(pool->gen_pool),
  30. gen_pool_avail(pool->gen_pool));
  31. if (pool->cpumem)
  32. dma_free_coherent(pool->dev, pool->mem_size, pool->cpumem,
  33. pool->dma_addr);
  34. kfree(pool->desc_infos);
  35. gen_pool_destroy(pool->gen_pool); /* frees pool->name */
  36. kfree(pool);
  37. }
  38. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_destroy);
  39. struct k3_cppi_desc_pool *
  40. k3_cppi_desc_pool_create_name(struct device *dev, size_t size,
  41. size_t desc_size,
  42. const char *name)
  43. {
  44. struct k3_cppi_desc_pool *pool;
  45. const char *pool_name = NULL;
  46. int ret = -ENOMEM;
  47. pool = kzalloc_obj(*pool);
  48. if (!pool)
  49. return ERR_PTR(ret);
  50. pool->dev = dev;
  51. pool->desc_size = roundup_pow_of_two(desc_size);
  52. pool->num_desc = size;
  53. pool->mem_size = pool->num_desc * pool->desc_size;
  54. pool_name = kstrdup_const(name ? name : dev_name(pool->dev),
  55. GFP_KERNEL);
  56. if (!pool_name)
  57. goto gen_pool_create_fail;
  58. pool->gen_pool = gen_pool_create(ilog2(pool->desc_size), -1);
  59. if (!pool->gen_pool) {
  60. kfree_const(pool_name);
  61. goto gen_pool_create_fail;
  62. }
  63. pool->gen_pool->name = pool_name;
  64. pool->desc_infos = kzalloc_objs(*pool->desc_infos, pool->num_desc);
  65. if (!pool->desc_infos)
  66. goto gen_pool_desc_infos_alloc_fail;
  67. pool->cpumem = dma_alloc_coherent(pool->dev, pool->mem_size,
  68. &pool->dma_addr, GFP_KERNEL);
  69. if (!pool->cpumem)
  70. goto dma_alloc_fail;
  71. ret = gen_pool_add_virt(pool->gen_pool, (unsigned long)pool->cpumem,
  72. (phys_addr_t)pool->dma_addr, pool->mem_size,
  73. -1);
  74. if (ret < 0) {
  75. dev_err(pool->dev, "pool add failed %d\n", ret);
  76. goto gen_pool_add_virt_fail;
  77. }
  78. return pool;
  79. gen_pool_add_virt_fail:
  80. dma_free_coherent(pool->dev, pool->mem_size, pool->cpumem,
  81. pool->dma_addr);
  82. dma_alloc_fail:
  83. kfree(pool->desc_infos);
  84. gen_pool_desc_infos_alloc_fail:
  85. gen_pool_destroy(pool->gen_pool); /* frees pool->name */
  86. gen_pool_create_fail:
  87. kfree(pool);
  88. return ERR_PTR(ret);
  89. }
  90. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_create_name);
  91. dma_addr_t k3_cppi_desc_pool_virt2dma(struct k3_cppi_desc_pool *pool,
  92. void *addr)
  93. {
  94. return addr ? pool->dma_addr + (addr - pool->cpumem) : 0;
  95. }
  96. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_virt2dma);
  97. void *k3_cppi_desc_pool_dma2virt(struct k3_cppi_desc_pool *pool, dma_addr_t dma)
  98. {
  99. return dma ? pool->cpumem + (dma - pool->dma_addr) : NULL;
  100. }
  101. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_dma2virt);
  102. void *k3_cppi_desc_pool_alloc(struct k3_cppi_desc_pool *pool)
  103. {
  104. return (void *)gen_pool_alloc(pool->gen_pool, pool->desc_size);
  105. }
  106. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_alloc);
  107. void k3_cppi_desc_pool_free(struct k3_cppi_desc_pool *pool, void *addr)
  108. {
  109. gen_pool_free(pool->gen_pool, (unsigned long)addr, pool->desc_size);
  110. }
  111. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_free);
  112. size_t k3_cppi_desc_pool_avail(struct k3_cppi_desc_pool *pool)
  113. {
  114. return gen_pool_avail(pool->gen_pool) / pool->desc_size;
  115. }
  116. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_avail);
  117. size_t k3_cppi_desc_pool_desc_size(const struct k3_cppi_desc_pool *pool)
  118. {
  119. return pool->desc_size;
  120. }
  121. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_desc_size);
  122. void *k3_cppi_desc_pool_cpuaddr(const struct k3_cppi_desc_pool *pool)
  123. {
  124. return pool->cpumem;
  125. }
  126. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_cpuaddr);
  127. void k3_cppi_desc_pool_desc_info_set(struct k3_cppi_desc_pool *pool,
  128. int desc_idx, void *info)
  129. {
  130. pool->desc_infos[desc_idx] = info;
  131. }
  132. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_desc_info_set);
  133. void *k3_cppi_desc_pool_desc_info(const struct k3_cppi_desc_pool *pool,
  134. int desc_idx)
  135. {
  136. return pool->desc_infos[desc_idx];
  137. }
  138. EXPORT_SYMBOL_GPL(k3_cppi_desc_pool_desc_info);
  139. MODULE_LICENSE("GPL");
  140. MODULE_DESCRIPTION("TI K3 CPPI5 descriptors pool API");