[PATCH i-g-t 3/5] lib/intel_allocator_bst: Implement the allocator with a BST
Andrzej Turko
andrzej.turko at linux.intel.com
Mon Apr 19 16:49:02 UTC 2021
From: Andrzej Turko <andrzej.turko at intel.com>
The use of a binary search tree speeds up finding free blocks
of memory (compared to a list).
Signed-off-by: Andrzej Turko <andrzej.turko at intel.com>
---
lib/intel_allocator.c | 12 +
lib/intel_allocator.h | 1 +
lib/intel_allocator_bst.c | 740 ++++++++++++++++++++++++++++++++++++++
lib/meson.build | 1 +
4 files changed, 754 insertions(+)
create mode 100644 lib/intel_allocator_bst.c
diff --git a/lib/intel_allocator.c b/lib/intel_allocator.c
index 96f839d4b..b0d720115 100644
--- a/lib/intel_allocator.c
+++ b/lib/intel_allocator.c
@@ -65,6 +65,11 @@ struct intel_allocator *
intel_allocator_simple_create_full(int fd, uint64_t start, uint64_t end,
enum allocator_strategy strategy);
+struct intel_allocator *intel_allocator_bst_create(int fd);
+struct intel_allocator *
+intel_allocator_bst_create_full(int fd, uint64_t start, uint64_t end,
+ enum allocator_strategy strategy);
+
/*
* Instead of trying to find first empty handle just get new one. Assuming
* our counter is incremented 2^32 times per second (4GHz clock and handle
@@ -298,6 +303,13 @@ static struct intel_allocator *intel_allocator_create(int fd,
ial = intel_allocator_simple_create_full(fd, start, end,
allocator_strategy);
break;
+ case INTEL_ALLOCATOR_BST:
+ if (!start && !end)
+ ial = intel_allocator_bst_create(fd);
+ else
+ ial = intel_allocator_bst_create_full(fd, start, end,
+ allocator_strategy);
+ break;
default:
igt_assert_f(ial, "Allocator type %d not implemented\n",
allocator_type);
diff --git a/lib/intel_allocator.h b/lib/intel_allocator.h
index 7d9d01123..93ecbc3e4 100644
--- a/lib/intel_allocator.h
+++ b/lib/intel_allocator.h
@@ -211,6 +211,7 @@ void intel_allocator_print(uint64_t allocator_handle);
#define INTEL_ALLOCATOR_RELOC 1
#define INTEL_ALLOCATOR_RANDOM 2
#define INTEL_ALLOCATOR_SIMPLE 3
+#define INTEL_ALLOCATOR_BST 4
#define GEN8_GTT_ADDRESS_WIDTH 48
diff --git a/lib/intel_allocator_bst.c b/lib/intel_allocator_bst.c
new file mode 100644
index 000000000..611b36c06
--- /dev/null
+++ b/lib/intel_allocator_bst.c
@@ -0,0 +1,740 @@
+// SPDX-License-Identifier: MIT
+/*
+ * Copyright © 2021 Intel Corporation
+ */
+
+#include <stdlib.h>
+#include "igt.h"
+#include "igt_map.h"
+#include "igt_bst.h"
+#include "intel_allocator.h"
+
+#define BSTDBG
+#ifdef BSTDBG
+#define bst_assert igt_assert
+#define bst_assert_f igt_assert_f
+#else
+#define bst_assert(...) {}
+#define bst_assert_f(...) {}
+#endif
+
+//#define BSTVERB
+#ifdef BSTVERB
+#define bst_info igt_info
+#define bst_debug(...) fprintf(stderr, __VA_ARGS__)
+#else
+#define bst_info(...) {}
+#define bst_debug(...) {}
+#endif
+
+/* All addresses of allocated objects are
+ * strictly smaller than this value (upper end of
+ * the virtual address range cannot exceed it).
+ * This is to ensure there are no overflows
+ * and prevent the existance of a hole or
+ * block of size 2^64.
+ *
+ * Thanks to this -1 can be returned from alloc
+ * to indicate that allocator has failed to find
+ * enough space for the object.
+ *
+ * IALB_MAX_ADDR can be set to any positive
+ * integer smaller than 2^64.
+ */
+#define IALB_MAX_ADDR (1ull<<63)
+
+/* flags for the bst allocator */
+#define IALB_ALLOC_HIGH 1
+
+#define RESERVED 4096
+
+/* Avoid compilation warning */
+struct intel_allocator *intel_allocator_bst_create(int fd);
+struct intel_allocator *intel_allocator_bst_create_full(int fd, uint64_t start, uint64_t end, enum allocator_strategy strategy);
+
+struct intel_allocator_bst {
+ struct igt_map *objects;
+ struct igt_map *reserved;
+ struct igt_bst *by_size;
+ struct igt_bst *by_offset;
+
+ uint64_t start;
+ uint64_t end;
+
+ /* statistics */
+ uint64_t total_size;
+ uint64_t allocated_size;
+ uint64_t allocated_objects;
+ uint64_t reserved_size;
+ uint64_t reserved_areas;
+};
+
+struct intel_allocator_bst_vma_hole {
+
+ /* pointer to the node in the by_size bst */
+ void *size_ptr;
+
+ /* pointer to the node in the by_offset bst */
+ void *offset_ptr;
+
+ uint64_t size;
+ uint64_t offset;
+};
+
+struct intel_allocator_record {
+ uint32_t handle;
+ uint64_t offset;
+ uint64_t size;
+};
+
+/* 2^31 + 2^29 - 2^25 + 2^22 - 2^19 - 2^16 + 1 */
+#define GOLDEN_RATIO_PRIME_32 0x9e370001UL
+
+/* 2^63 + 2^61 - 2^57 + 2^54 - 2^51 - 2^18 + 1 */
+#define GOLDEN_RATIO_PRIME_64 0x9e37fffffffc0001ULL
+
+static inline uint32_t hash_handles(const void *val)
+{
+ uint32_t hash = *(uint32_t *) val;
+
+ hash = hash * GOLDEN_RATIO_PRIME_32;
+ return hash;
+}
+
+static int equal_handles(const void *a, const void *b)
+{
+ uint32_t *key1 = (uint32_t *) a, *key2 = (uint32_t *) b;
+
+ return *key1 == *key2;
+}
+
+static inline uint32_t hash_offsets(const void *val)
+{
+ uint64_t hash = *(uint64_t *) val;
+
+ hash = hash * GOLDEN_RATIO_PRIME_64;
+ /* High bits are more random, so use them. */
+ return hash >> 32;
+}
+
+static int equal_offsets(const void *a, const void *b)
+{
+ uint64_t *key1 = (uint64_t *) a, *key2 = (uint64_t *) b;
+
+ return *key1 == *key2;
+}
+
+static void map_entry_free_func(struct igt_map_entry *entry)
+{
+ free(entry->data);
+}
+
+static void holes_bst_validate(struct intel_allocator_bst *ialb)
+{
+ struct igt_bst *offset_bst, *size_bst;
+ struct intel_allocator_bst_vma_hole *hole;
+ void *nodeptr;
+ int by_offset_size, by_size_size;
+ uint64_t prev_offset;
+
+ offset_bst = ialb->by_offset;
+ size_bst = ialb->by_size;
+ by_offset_size = offset_bst->validate(offset_bst);
+ by_size_size = size_bst->validate(size_bst);
+ bst_assert(by_offset_size == by_size_size);
+
+ prev_offset = IALB_MAX_ADDR;
+ igt_bst_for_each_node_rev(offset_bst, nodeptr) {
+ hole = offset_bst->get_value(nodeptr);
+ bst_assert(hole);
+ /* Make sure the hole is stored under correct keys in
+ * the balanced search trees.
+ */
+ bst_assert(hole->offset == offset_bst->get_key(hole->offset_ptr));
+ bst_assert(hole->size == size_bst->get_key(hole->size_ptr));
+
+ bst_assert(hole->offset + hole->size > hole->offset);
+ /* Make sure no pair of holes is adjacent. */
+ bst_assert(prev_offset > hole->offset + hole->size);
+
+ prev_offset = hole->offset;
+ }
+}
+
+static void __intel_allocator_bst_alloc(struct intel_allocator_bst *ialb,
+ struct intel_allocator_bst_vma_hole *hole,
+ uint64_t offset, uint64_t size)
+{
+ struct igt_bst *offset_bst, *size_bst;
+ struct intel_allocator_bst_vma_hole *newhole;
+ uint64_t lower, higher;
+
+ bst_info("Allocate 0x%"PRIX64" bytes at 0x%"PRIX64"\n", size, offset);
+ bst_assert(hole->offset <= offset);
+ bst_assert(hole->offset + hole->size >= offset + size);
+
+ offset_bst = ialb->by_offset;
+ size_bst = ialb->by_size;
+
+ lower = offset - hole->offset;
+ higher = hole->size - size - lower;
+
+ if (lower > 0 && higher > 0) {
+ /* There are leftovers on both lower and higher addresses.
+ * Create a hole for higher addresses from scratch and
+ * one for lower addresses by modifying the existing hole.
+ */
+
+ newhole = malloc(sizeof(struct intel_allocator_bst_vma_hole));
+ igt_assert(newhole);
+
+ newhole->offset = offset + size;
+ newhole->size = higher;
+
+ newhole->offset_ptr = offset_bst->insert(offset_bst, newhole, newhole->offset);
+ newhole->size_ptr = size_bst->insert(size_bst, newhole, newhole->size);
+
+ hole->size = lower;
+ size_bst->erase(size_bst, hole->size_ptr);
+ hole->size_ptr = size_bst->insert(size_bst, hole, hole->size);
+
+ } else if (higher > 0) {
+ /* There is no free address space in this hole left
+ * below the allocated object. Only a portion of
+ * higher addresses is still free, so just adjust the
+ * existing hole accordingly (change offset and size).
+ */
+
+ hole->offset = offset + size;
+ hole->size = higher;
+
+ offset_bst->erase(offset_bst, hole->offset_ptr);
+ size_bst->erase(size_bst, hole->size_ptr);
+
+ hole->offset_ptr = offset_bst->insert(offset_bst, hole, hole->offset);
+ hole->size_ptr = size_bst->insert(size_bst, hole, hole->size);
+
+ } else if (lower > 0) {
+ /* The allocated block is aligned to the end of the
+ * hole and the lower addresses are still free,
+ * so shrink the existing hole (change size,
+ * but not the offset).
+ */
+
+ hole->size = lower;
+
+ size_bst->erase(size_bst, hole->size_ptr);
+ hole->size_ptr = size_bst->insert(size_bst, hole, hole->size);
+
+ } else {
+ /* There are no leftovers, just erase the hole. */
+
+ offset_bst->erase(offset_bst, hole->offset_ptr);
+ size_bst->erase(size_bst, hole->size_ptr);
+ free(hole);
+ }
+
+}
+
+static void __intel_allocator_bst_add_hole(struct intel_allocator_bst *ialb,
+ uint64_t offset, uint64_t size)
+{
+ struct igt_bst *offset_bst, *size_bst;
+ struct intel_allocator_bst_vma_hole *hole, *nxt, *prv;
+
+ offset_bst = ialb->by_offset;
+ size_bst = ialb->by_size;
+
+ prv = offset_bst->query_predecessor(offset_bst, offset);
+ if (prv && prv->offset + prv->size < offset)
+ prv = NULL;
+
+ nxt = offset_bst->query_successor(offset_bst, offset);
+ if (nxt && nxt->offset > offset + size)
+ nxt = NULL;
+
+ if (nxt && prv) {
+ /* There are holes adjacent to the new one
+ * both below and above it in the address space.
+ * Merge them all into a single hole:
+ * erase the upper and increase size of the
+ * lower one.
+ */
+ prv->size += size + nxt->size;
+
+ offset_bst->erase(offset_bst, nxt->offset_ptr);
+ size_bst->erase(size_bst, nxt->size_ptr);
+ free(nxt);
+
+ size_bst->erase(size_bst, prv->size_ptr);
+ prv->size_ptr = size_bst->insert(size_bst, prv, prv->size);
+ } else if (nxt) {
+ /* The only adjacent hole is above the new one,
+ * so increase its size and update its offset.
+ */
+ nxt->size += size;
+ nxt->offset = offset;
+
+ offset_bst->erase(offset_bst, nxt->offset_ptr);
+ size_bst->erase(size_bst, nxt->size_ptr);
+
+ nxt->offset_ptr = offset_bst->insert(offset_bst, nxt, nxt->offset);
+ nxt->size_ptr = size_bst->insert(size_bst, nxt, nxt->size);
+ } else if (prv) {
+ /* The only adjacent hole is below the new one,
+ * just increase its size (its offset does not
+ * change).
+ */
+ prv->size += size;
+
+ size_bst->erase(size_bst, prv->size_ptr);
+ prv->size_ptr = size_bst->insert(size_bst, prv, prv->size);
+ } else {
+ /* There are no holes neighbouring the new one,
+ * so just create a new hole.
+ */
+ hole = malloc(sizeof(struct intel_allocator_bst_vma_hole));
+ igt_assert(hole);
+
+ hole->offset = offset;
+ hole->size = size;
+ hole->offset_ptr = offset_bst->insert(offset_bst, hole, offset);
+ hole->size_ptr = size_bst->insert(size_bst, hole, size);
+ }
+
+}
+
+
+static void intel_allocator_bst_get_address_range(struct intel_allocator *ial,
+ uint64_t *startp, uint64_t *endp)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+
+ if (startp)
+ *startp = ialb->start;
+
+ if (endp)
+ *endp = ialb->end;
+}
+
+static uint64_t intel_allocator_bst_alloc(struct intel_allocator *ial, uint32_t handle,
+ uint64_t size, uint64_t alignment, enum allocator_strategy strategy)
+{
+ struct intel_allocator_bst *ialb =ial->priv;
+ struct igt_bst *bst;
+ struct intel_allocator_record *record;
+ struct intel_allocator_bst_vma_hole *hole;
+ uint64_t misalignment, offset;
+
+ bst_info("\nintel_allocator_bst_alloc(%d, 0x%"PRIX64", 0x%"PRIX64")\n", handle, size, alignment);
+
+ igt_assert(size > 0);
+ alignment = (alignment > 0 ? alignment : 1);
+
+ record = igt_map_search(ialb->objects, &handle);
+
+ if (record){
+ /* This block has already been allocated,
+ * check that it satifies the requirements. */
+
+ igt_assert(size == record->size);
+ igt_assert(record->offset % alignment == 0);
+
+ return record->offset;
+ }
+
+ bst = ialb->by_size;
+ /* Look for a slightly bigger hole to make sure
+ * the block can be aligned properly. */
+ hole = bst->query_successor(bst, size + alignment -1);
+
+ /* Look for a smaller hole, maybe proper alignment
+ * will still be possible. */
+ if (!hole)
+ hole = bst->query_successor(bst, size);
+
+ if (!hole) {
+ igt_warn("Failed to allocate: not enough memory\n");
+ return ALLOC_INVALID_ADDRESS;
+ }
+
+ if (strategy == ALLOC_STRATEGY_NONE)
+ strategy = ial->strategy;
+
+ switch (strategy) {
+ case ALLOC_STRATEGY_LOW_TO_HIGH:
+ misalignment = alignment - (hole->offset % alignment);
+ misalignment = misalignment==alignment ? 0 : misalignment;
+ offset = hole->offset + misalignment;
+
+ if (misalignment + size > hole->size) {
+ igt_warn("Failed to allocate: not enough memory\n");
+ return ALLOC_INVALID_ADDRESS;
+ }
+ break;
+ case ALLOC_STRATEGY_HIGH_TO_LOW:
+ offset = hole->offset + hole->size - size;
+ offset = offset - offset % alignment;
+
+ if (offset < hole->offset) {
+ igt_warn("Failed to allocate: not enough memory\n");
+ return ALLOC_INVALID_ADDRESS;
+ }
+ break;
+ default:
+ igt_assert_f(0, "Unsupported strategy.");
+ }
+
+ __intel_allocator_bst_alloc(ialb, hole, offset, size);
+
+ record = malloc(sizeof(struct intel_allocator_record));
+ igt_assert(record);
+
+ record->handle = handle;
+ record->offset = offset;
+ record->size = size;
+
+ igt_map_insert(ialb->objects, &record->handle, record);
+ ialb->allocated_objects++;
+ ialb->allocated_size+=size;
+
+ return offset;
+}
+
+static bool intel_allocator_bst_is_allocated(struct intel_allocator *ial, uint32_t handle,
+ uint64_t size, uint64_t offset)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+ struct intel_allocator_record *record;
+ bool same;
+
+ record = igt_map_search(ialb->objects, &handle);
+ if (record) {
+ same = (record->handle == handle && record->size == size &&
+ record->offset == offset);
+ } else {
+ same = false;
+ }
+
+ return same;
+}
+
+static bool intel_allocator_bst_free(struct intel_allocator *ial, uint32_t handle)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+ struct intel_allocator_record *record;
+
+ bst_info("\nintel_allocator_bst_free(%d)\n", handle);
+
+ record = igt_map_search(ialb->objects, &handle);
+
+ if (!record)
+ return false;
+
+ __intel_allocator_bst_add_hole(ialb, record->offset, record->size);
+ ialb->allocated_objects--;
+ ialb->allocated_size-=record->size;
+ igt_map_remove(ialb->objects, &handle, map_entry_free_func);
+
+ return true;
+}
+
+static bool intel_allocator_bst_reserve(struct intel_allocator *ial, uint32_t handle,
+ uint64_t start, uint64_t end)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+ struct intel_allocator_bst_vma_hole *hole;
+ struct intel_allocator_record *record;
+ struct igt_bst *offset_bst;
+
+ bst_info("\nintel_allocator_bst_reserve(%d, 0x%"PRIX64", 0x%"PRIX64")\n", handle, start, end);
+
+ igt_assert(start < end);
+ igt_assert(start >= ialb->start);
+ igt_assert(end <= ialb->end);
+
+ offset_bst = ialb->by_offset;
+ hole = offset_bst->query_predecessor(offset_bst, start);
+
+ if (!hole)
+ return false;
+
+ bst_assert(hole->offset <= start);
+
+ if (hole->offset + hole->size < end)
+ return false;
+
+ __intel_allocator_bst_alloc(ialb, hole, start, end - start);
+
+ record = malloc(sizeof(struct intel_allocator_bst_vma_hole));
+ igt_assert(record);
+ record->offset = start;
+ record->size = end - start;
+ record->handle = handle;
+ igt_map_insert(ialb->reserved, &record->offset, record);
+ ialb->reserved_areas++;
+ ialb->reserved_size += end - start;
+
+ return true;
+}
+
+static bool intel_allocator_bst_unreserve(struct intel_allocator *ial, uint32_t handle,
+ uint64_t start, uint64_t end)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+ struct intel_allocator_record *record;
+
+ bst_info("\nintel_allocator_bst_unreserve(%d, 0x%"PRIX64", 0x%"PRIX64")\n", handle, start, end);
+
+ record = igt_map_search(ialb->reserved, &start);
+
+ if (!record) {
+ igt_warn("Only a reserved block can be unreserved.\n");
+ return false;
+ }
+
+ if (record->size != end - start) {
+ igt_warn("Size of the block does not match passed value.\n");
+ return false;
+ }
+
+ if (record->handle != handle) {
+ igt_warn("Handle %u does not match the passed handle %u.\n",
+ record->handle, handle);
+ return false;
+ }
+
+ __intel_allocator_bst_add_hole(ialb, start, end - start);
+ ialb->reserved_areas--;
+ ialb->reserved_size -= record->size;
+ igt_map_remove(ialb->reserved, &start, map_entry_free_func);
+
+ return true;
+}
+
+static bool intel_allocator_bst_is_reserved(struct intel_allocator *ial,
+ uint64_t start, uint64_t end)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+ struct intel_allocator_record *record;
+
+ igt_assert(start < end);
+ record = igt_map_search(ialb->reserved, &start);
+
+ if (!record)
+ return false;
+
+ if (record->offset == start && record->size == end - start)
+ return true;
+
+ return false;
+}
+
+static void intel_allocator_bst_destroy(struct intel_allocator *ial)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+ struct igt_bst *size_bst, *offset_bst;
+ void *nodeptr;
+
+ size_bst = ialb->by_size;
+ offset_bst = ialb->by_offset;
+
+ igt_bst_for_each_node(offset_bst, nodeptr)
+ free(offset_bst->get_value(nodeptr));
+
+ size_bst->bst_free(size_bst);
+ offset_bst->bst_free(offset_bst);
+ free(offset_bst);
+ free(size_bst);
+
+ igt_map_destroy(ialb->objects, map_entry_free_func);
+ igt_map_destroy(ialb->reserved, map_entry_free_func);
+
+ free(ialb);
+ free(ial);
+}
+
+static bool intel_allocator_bst_is_empty(struct intel_allocator *ial)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+
+ return ialb->allocated_objects == 0 && ialb->reserved_areas == 0;
+}
+
+static void intel_allocator_bst_print(struct intel_allocator *ial, bool full)
+{
+ struct intel_allocator_bst *ialb = ial->priv;
+ struct intel_allocator_record *record;
+ struct intel_allocator_bst_vma_hole *hole;
+ struct igt_bst *offset_bst, *size_bst;
+ struct igt_map_entry *entry;
+ void *nodeptr;
+ uint64_t total_free;
+
+
+ igt_info("\nintel_allocator_bst <fd:%d> on "
+ "[0x%"PRIX64" : 0x%"PRIx64"]\n", ial->fd,
+ ialb->start, ialb->end);
+
+
+ total_free = 0;
+ if (full) {
+ offset_bst = ialb->by_offset;
+ size_bst = ialb->by_size;
+
+ igt_info(" holes by offset\n");
+
+ igt_bst_for_each_node(offset_bst, nodeptr) {
+ hole = offset_bst->get_value(nodeptr);
+
+ igt_info("0x%"PRIx64" -- hole of size %"PRIu64
+ " (0x%"PRIx64")"
+ " stored at 0x%"PRIx64
+ " by size and at 0x%"PRIx64
+ " by offset\n", hole->offset, hole->size,
+ hole->size, (uint64_t) hole->size_ptr,
+ (uint64_t) hole->offset_ptr);
+
+ total_free += hole->size;
+ }
+
+ igt_info(" holes by size\n");
+
+ igt_bst_for_each_node(size_bst, nodeptr) {
+ hole = offset_bst->get_value(nodeptr);
+
+ igt_info("0x%"PRIx64" -- hole of size %"PRIu64
+ " (0x%"PRIx64")"
+ " stored at 0x%"PRIx64
+ " by size and at 0x%"PRIx64
+ " by offset\n", hole->offset, hole->size,
+ hole->size, (uint64_t) hole->size_ptr,
+ (uint64_t) hole->offset_ptr);
+
+ }
+
+ igt_info(" allocated objects\n");
+
+ igt_map_foreach(ialb->objects, entry) {
+ record = entry->data;
+
+ igt_info("0x%"PRIx64" -- object of size %"PRIu64
+ " (0x%"PRIx64") with handle %"SCNu32"\n",
+ record->offset, record->size, record->size,
+ record->handle);
+
+ }
+
+ igt_info(" reserved objects\n");
+ igt_map_foreach(ialb->reserved, entry) {
+ record = entry->data;
+
+ igt_info("0x%"PRIx64" -- reserved area of size %"
+ PRIu64" (0x%"PRIx64") with handle %"SCNu32"\n",
+ record->offset, record->size, record->size,
+ record->handle);
+ }
+
+ } else {
+ offset_bst = ialb->by_offset;
+
+ igt_bst_for_each_node(offset_bst, nodeptr) {
+ hole = offset_bst->get_value(nodeptr);
+ total_free += hole->size;
+ }
+ }
+
+ igt_info("free space: %"PRIu64"B (0x%"PRIx64") (%.2f%% full)\n"
+ "allocated objects: %"PRIu64", size: %"PRIu64
+ " (%"PRIx64")\n"
+ "reserved objects: %"PRIu64", size: %"PRIu64
+ " (%"PRIx64")\n", total_free, total_free,
+ ((double) (ialb->total_size - total_free) /
+ (double) ialb->total_size) * 100.0,
+ ialb->allocated_objects, ialb->allocated_size,
+ ialb->allocated_size, ialb->reserved_areas,
+ ialb->reserved_areas, ialb->reserved_size);
+
+ holes_bst_validate(ialb);
+}
+
+static struct intel_allocator *__intel_allocator_bst_create(int fd, uint64_t start, uint64_t end, enum allocator_strategy strategy)
+{
+ struct intel_allocator *ial;
+ struct intel_allocator_bst *ialb;
+
+ ialb = malloc(sizeof(struct intel_allocator_bst));
+ igt_assert(ialb);
+
+ /* allocated object are stored by handles,
+ * which are 4 bytes long. */
+ ialb->objects = igt_map_create(hash_handles, equal_handles);
+ /* reserved object are stored by
+ * 8-byte-long offsets */
+ ialb->reserved = igt_map_create(hash_offsets, equal_offsets);
+
+ ialb->by_offset = igt_bst_avl_create();
+ ialb->by_size = igt_bst_avl_create();
+
+ ialb->start = start;
+ ialb->end = end;
+ igt_assert(ialb->end > ialb->start);
+ igt_assert(ialb->end <= IALB_MAX_ADDR);
+
+ ialb->total_size = ialb->end - ialb->start;
+ ialb->allocated_size = 0;
+ ialb->allocated_objects = 0;
+ ialb->reserved_size = 0;
+ ialb->reserved_areas = 0;
+
+ /* Whole address space is free,
+ * so add a corresponding hole. */
+ __intel_allocator_bst_add_hole(ialb, ialb->start, ialb->total_size);
+
+ ial = malloc(sizeof(struct intel_allocator));
+ igt_assert(ial);
+
+ ial->fd = fd;
+ ial->type = INTEL_ALLOCATOR_BST;
+ ial->strategy = strategy;
+ ial->refcount = 0;
+ ial->priv = ialb;
+ ial->get_address_range = intel_allocator_bst_get_address_range;
+ ial->alloc = intel_allocator_bst_alloc;
+ ial->is_allocated = intel_allocator_bst_is_allocated;
+ ial->free = intel_allocator_bst_free;
+ ial->reserve = intel_allocator_bst_reserve;
+ ial->unreserve = intel_allocator_bst_unreserve;
+ ial->is_reserved = intel_allocator_bst_is_reserved;
+ ial->destroy = intel_allocator_bst_destroy;
+ ial->is_empty = intel_allocator_bst_is_empty;
+ ial->print = intel_allocator_bst_print;
+
+
+ return ial;
+}
+
+struct intel_allocator *intel_allocator_bst_create(int fd)
+{
+ uint64_t gtt_size = gem_aperture_size(fd);
+
+ if (!gem_uses_full_ppgtt(fd))
+ gtt_size /= 2;
+ else
+ gtt_size -= RESERVED;
+
+ return __intel_allocator_bst_create(fd, 0, gtt_size, ALLOC_STRATEGY_HIGH_TO_LOW);
+}
+
+struct intel_allocator *intel_allocator_bst_create_full(int fd, uint64_t start, uint64_t end,
+ enum allocator_strategy strategy)
+{
+ uint64_t gtt_size = gem_aperture_size(fd);
+
+ igt_assert(end <= gtt_size);
+ if (!gem_uses_full_ppgtt(fd))
+ gtt_size /= 2;
+ igt_assert(end - start <= gtt_size);
+
+ return __intel_allocator_bst_create(fd, start, end, strategy);
+}
diff --git a/lib/meson.build b/lib/meson.build
index dddd171d9..f2811959e 100644
--- a/lib/meson.build
+++ b/lib/meson.build
@@ -36,6 +36,7 @@ lib_sources = [
'igt_x86.c',
'instdone.c',
'intel_allocator.c',
+ 'intel_allocator_bst.c',
'intel_allocator_msgchannel.c',
'intel_allocator_random.c',
'intel_allocator_reloc.c',
--
2.25.1
More information about the Intel-gfx-trybot
mailing list