// SPDX-License-Identifier: GPL-2.0 /* Copyright (C) 2018-2025, Advanced Micro Devices, Inc. */ #include #include "ionic_queue.h" int ionic_queue_init(struct ionic_queue *q, struct device *dma_dev, int depth, size_t stride) { if (depth < 0 || depth > 0xffff) return -EINVAL; if (stride == 0 || stride > 0x10000) return -EINVAL; if (depth == 0) depth = 1; q->depth_log2 = order_base_2(depth + 1); q->stride_log2 = order_base_2(stride); if (q->depth_log2 + q->stride_log2 < PAGE_SHIFT) q->depth_log2 = PAGE_SHIFT - q->stride_log2; if (q->depth_log2 > 16 || q->stride_log2 > 16) return -EINVAL; q->size = BIT_ULL(q->depth_log2 + q->stride_log2); q->mask = BIT(q->depth_log2) - 1; q->ptr = dma_alloc_coherent(dma_dev, q->size, &q->dma, GFP_KERNEL); if (!q->ptr) return -ENOMEM; /* it will always be page aligned, but just to be sure... */ if (!PAGE_ALIGNED(q->ptr)) { dma_free_coherent(dma_dev, q->size, q->ptr, q->dma); return -ENOMEM; } q->prod = 0; q->cons = 0; q->dbell = 0; return 0; } void ionic_queue_destroy(struct ionic_queue *q, struct device *dma_dev) { dma_free_coherent(dma_dev, q->size, q->ptr, q->dma); }