// SPDX-License-Identifier: GPL-2.0+ // Copyright 2017 IBM Corp. #include "ocxl_internal.h" struct id_range { struct list_head list; u32 start; u32 end; }; #ifdef DEBUG static void dump_list(struct list_head *head, char *type_str) { struct id_range *cur; pr_debug("%s ranges allocated:\n", type_str); list_for_each_entry(cur, head, list) { pr_debug("Range %d->%d\n", cur->start, cur->end); } } #endif static int range_alloc(struct list_head *head, u32 size, int max_id, char *type_str) { struct list_head *pos; struct id_range *cur, *new; int rc, last_end; new = kmalloc(sizeof(struct id_range), GFP_KERNEL); if (!new) return -ENOMEM; pos = head; last_end = -1; list_for_each_entry(cur, head, list) { if ((cur->start - last_end) > size) break; last_end = cur->end; pos = &cur->list; } new->start = last_end + 1; new->end = new->start + size - 1; if (new->end > max_id) { kfree(new); rc = -ENOSPC; } else { list_add(&new->list, pos); rc = new->start; } #ifdef DEBUG dump_list(head, type_str); #endif return rc; } static void range_free(struct list_head *head, u32 start, u32 size, char *type_str) { bool found = false; struct id_range *cur, *tmp; list_for_each_entry_safe(cur, tmp, head, list) { if (cur->start == start && cur->end == (start + size - 1)) { found = true; list_del(&cur->list); kfree(cur); break; } } WARN_ON(!found); #ifdef DEBUG dump_list(head, type_str); #endif } int ocxl_pasid_afu_alloc(struct ocxl_fn *fn, u32 size) { int max_pasid; if (fn->config.max_pasid_log < 0) return -ENOSPC; max_pasid = 1 << fn->config.max_pasid_log; return range_alloc(&fn->pasid_list, size, max_pasid, "afu pasid"); } void ocxl_pasid_afu_free(struct ocxl_fn *fn, u32 start, u32 size) { return range_free(&fn->pasid_list, start, size, "afu pasid"); } int ocxl_actag_afu_alloc(struct ocxl_fn *fn, u32 size) { int max_actag; max_actag = fn->actag_enabled; return range_alloc(&fn->actag_list, size, max_actag, "afu actag"); } void ocxl_actag_afu_free(struct ocxl_fn *fn, u32 start, u32 size) { return range_free(&fn->actag_list, start, size, "afu actag"); }