On 2/9/2023 6:36 AM, Chengwen Feng wrote:
This patch supports rte_memarea_alloc() and rte_memarea_free() API.
Signed-off-by: Chengwen Feng <fengcheng...@huawei.com>
Reviewed-by: Dongdong Liu <liudongdo...@huawei.com>
Acked-by: Morten Brørup <m...@smartsharesystems.com>
---
General note: this patchset could benefit from a bit more comments. I
don't suggest commenting every line, but at least more comments denoting
various logical steps (like you have in `rte_memarea_free`) would be
nice to have.
#include <rte_common.h>
#include <rte_log.h>
@@ -88,6 +90,8 @@ memarea_alloc_area(const struct rte_memarea_param *init)
init->numa_socket);
else if (init->source == RTE_MEMAREA_SOURCE_LIBC)
ptr = memarea_alloc_from_libc(init->total_sz);
+ else if (init->source == RTE_MEMAREA_SOURCE_MEMAREA)
+ ptr = rte_memarea_alloc(init->src_ma, init->total_sz);
Why `if` not `switch`?
return ptr;
}
@@ -99,6 +103,8 @@ memarea_free_area(const struct rte_memarea_param *init, void
*ptr)
rte_free(ptr);
else if (init->source == RTE_MEMAREA_SOURCE_LIBC)
free(ptr);
+ else if (init->source == RTE_MEMAREA_SOURCE_MEMAREA)
+ rte_memarea_free(init->src_ma, ptr);
Similarly here: why `if` not `switch`?
+static inline void
+memarea_add_node(struct rte_memarea *ma, struct memarea_objhdr *hdr, size_t
alloc_sz)
+{
+#ifdef RTE_LIBRTE_MEMAREA_DEBUG
+ struct memarea_objtlr *cur_tlr;
+#endif
+ struct memarea_objhdr *new_hdr;
+
+#ifdef RTE_LIBRTE_MEMAREA_DEBUG
+ cur_tlr = RTE_PTR_ADD(hdr, sizeof(struct memarea_objhdr) + alloc_sz);
+ cur_tlr->cookie = MEMAREA_OBJECT_TRAILER_COOKIE;
+ new_hdr = RTE_PTR_ADD(cur_tlr, sizeof(struct memarea_objtlr));
+ new_hdr->cookie = MEMAREA_OBJECT_HEADER_AVAILABLE_COOKIE;
+#else
+ new_hdr = RTE_PTR_ADD(hdr, sizeof(struct memarea_objhdr) + alloc_sz);
+#endif
+ TAILQ_INSERT_AFTER(&ma->obj_list, hdr, new_hdr, obj_next);
+ TAILQ_INSERT_AFTER(&ma->avail_list, hdr, new_hdr, avail_next);
+}
It seems to me that this function isn't "adding" node but rather is
splitting the `hdr` into two nodes. This is nitpicking, but I feel like
this part could be clearer semantically (splitting the function, adding
comments, some other way...).
+
+void *
+rte_memarea_alloc(struct rte_memarea *ma, size_t size)
+{
+ size_t align_sz = RTE_ALIGN(size, MEMAREA_OBJECT_SIZE_ALIGN);
+ struct memarea_objhdr *hdr;
+ size_t avail_sz;
+ void *ptr = NULL;
+
+ if (unlikely(ma == NULL || size == 0 || align_sz < size))
+ return ptr;
It would be nice if API also set rte_errno to indicate what kind of
error has happened.
+
+ memarea_lock(ma);
+ TAILQ_FOREACH(hdr, &ma->avail_list, avail_next) {
+ memarea_check_cookie(ma, hdr, 0);
+ avail_sz = MEMAREA_OBJECT_GET_SIZE(hdr);
+ if (avail_sz < align_sz)
+ continue;
+ if (memarea_whether_add_node(avail_sz, align_sz))
+ memarea_add_node(ma, hdr, align_sz);
I didn't get this at first, which means it needs comments :)
Specifically, it seems to me that we're only "adding" a node when we can
comfortably split it. So, in addition to comments documenting the above,
perhaps the above functions should also be called differently? Like
`memarea_can_split()` and `memarea_split`? IMO it'd communicate the
intent better (unless I misunderstood the intent, that is!).
+ TAILQ_REMOVE(&ma->avail_list, hdr, avail_next);
+ MEMAREA_OBJECT_MARK_ALLOCATED(hdr);
+#ifdef RTE_LIBRTE_MEMAREA_DEBUG
+ hdr->cookie = MEMAREA_OBJECT_HEADER_ALLOCATED_COOKIE;
+#endif
+ ptr = RTE_PTR_ADD(hdr, sizeof(struct memarea_objhdr));
+ break;
+ }
+ memarea_unlock(ma);
+
+ return ptr;
It seems that it's possible to reach the end of the loop and return NULL
as `ptr`, without any error. I would suggest setting rte_errno to
`ENOMEM` initially, and clearing it when we find a suitable element.
+}
+
+static inline void
+memarea_merge_node(struct rte_memarea *ma, struct memarea_objhdr *curr,
+ struct memarea_objhdr *next)
+{
+#ifdef RTE_LIBRTE_MEMAREA_DEBUG
+ struct memarea_objtlr *tlr;
+#endif
+ RTE_SET_USED(curr);
+ TAILQ_REMOVE(&ma->obj_list, next, obj_next);
+ TAILQ_REMOVE(&ma->avail_list, next, avail_next);
+#ifdef RTE_LIBRTE_MEMAREA_DEBUG
+ next->cookie = 0;
+ tlr = RTE_PTR_SUB(next, sizeof(struct memarea_objtlr));
+ tlr->cookie = 0;
+#endif
+}
+
+void
+rte_memarea_free(struct rte_memarea *ma, void *ptr)
+{
+ struct memarea_objhdr *hdr, *prev, *next;
+
+ if (unlikely(ma == NULL || ptr == NULL))
+ return;
+
+ hdr = RTE_PTR_SUB(ptr, sizeof(struct memarea_objhdr));
+ if (unlikely(!MEMAREA_OBJECT_IS_ALLOCATED(hdr))) {
Here and above - I question the value of using `unlikely` here. Are
there any numbers to prove these are useful?
+ RTE_MEMAREA_LOG(ERR, "detect invalid object in %s!",
ma->init.name);
+ return;
+ }
+ memarea_check_cookie(ma, hdr, 1);
+
+ memarea_lock(ma);
+
+ /** 1st: add to avail list. */
+#ifdef RTE_LIBRTE_MEMAREA_DEBUG
+ hdr->cookie = MEMAREA_OBJECT_HEADER_AVAILABLE_COOKIE;
+#endif
+ TAILQ_INSERT_HEAD(&ma->avail_list, hdr, avail_next);
+
+ /** 2nd: merge if previous object is avail. */
+ prev = TAILQ_PREV(hdr, memarea_objhdr_list, obj_next);
+ if (prev != NULL && !MEMAREA_OBJECT_IS_ALLOCATED(prev)) {
+ memarea_check_cookie(ma, prev, 0);
+ memarea_merge_node(ma, prev, hdr);
+ hdr = prev;
+ }
+
+ /** 3rd: merge if next object is avail. */
+ next = TAILQ_NEXT(hdr, obj_next);
+ if (next != NULL && !MEMAREA_OBJECT_IS_ALLOCATED(next)) {
+ memarea_check_cookie(ma, next, 0);
+ memarea_merge_node(ma, hdr, next);
+ }
+
+ memarea_unlock(ma);
+}
This function is an example of how I would like to see other functions:
good comments to denote logical blocks, clear and concise.
--
Thanks,
Anatoly