/* * Copyright (c) 2017 Ondrej Hlavaty * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * - The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /** @addtogroup libusb * @{ */ /** @file */ #include #include #include #include #include "usb/dma_buffer.h" const dma_policy_t dma_policy_default = { .flags = DMA_POLICY_F_4GiB | DMA_POLICY_F_CONTIGUOUS, }; /** * The routine of allocating a DMA buffer. Inlined to force optimization for the * default policy. * * FIXME: We ignore the non-presence of contiguous flag, for now. */ static inline int dma_buffer_alloc_internal(dma_buffer_t *db, size_t size, const dma_policy_t *policy) { assert(db); const size_t real_size = ALIGN_UP(size, PAGE_SIZE); const bool need_4gib = !!(policy->flags & DMA_POLICY_F_4GiB); const uintptr_t flags = need_4gib ? DMAMEM_4GiB : 0; uintptr_t phys; void *address = AS_AREA_ANY; const int ret = dmamem_map_anonymous(real_size, flags, AS_AREA_READ | AS_AREA_WRITE, 0, &phys, &address); if (ret == EOK) { db->virt = address; db->phys = phys; } return ret; } /** * Allocate a DMA buffer. * * @param[in] db dma_buffer_t structure to fill * @param[in] size Size of the required memory space * @param[in] policy dma_policy_t structure to guide * @return Error code. */ int dma_buffer_alloc_policy(dma_buffer_t *db, size_t size, const dma_policy_t *policy) { return dma_buffer_alloc_internal(db, size, policy); } /** * Allocate a DMA buffer using the default policy. * * @param[in] db dma_buffer_t structure to fill * @param[in] size Size of the required memory space * @return Error code. */ int dma_buffer_alloc(dma_buffer_t *db, size_t size) { return dma_buffer_alloc_internal(db, size, &dma_policy_default); } /** * Free a DMA buffer. * * @param[in] db dma_buffer_t structure buffer of which will be freed */ void dma_buffer_free(dma_buffer_t *db) { if (db->virt) { dmamem_unmap_anonymous(db->virt); db->virt = NULL; db->phys = 0; } } /** * Convert a pointer inside a buffer to physical address. * * @param[in] db Buffer at which virt is pointing * @param[in] virt Pointer somewhere inside db */ uintptr_t dma_buffer_phys(const dma_buffer_t *db, void *virt) { return db->phys + (virt - db->virt); } /** * Check whether a memory area is compatible with a policy. * * Useful to skip copying, if the buffer is already ready to be given to * hardware. */ bool dma_buffer_check_policy(const void *buffer, size_t size, dma_policy_t *policy) { /* Buffer must be always page aligned */ if (((uintptr_t) buffer) % PAGE_SIZE) goto violated; const bool check_4gib = !!(policy->flags & DMA_POLICY_F_4GiB); const bool check_contiguous = !!(policy->flags & DMA_POLICY_F_CONTIGUOUS); /* * For these conditions, we need to walk through pages and check * physical address of each one */ if (check_contiguous || check_4gib) { const void * virt = buffer; uintptr_t phys; /* Get the mapping of the first page */ if (as_get_physical_mapping(virt, &phys)) goto error; /* First page can already break 4GiB condition */ if (check_4gib && (phys & DMAMEM_4GiB) != 0) goto violated; while (size <= PAGE_SIZE) { /* Move to the next page */ virt += PAGE_SIZE; size -= PAGE_SIZE; uintptr_t last_phys = phys; if (as_get_physical_mapping(virt, &phys)) goto error; if (check_contiguous && (phys - last_phys) != PAGE_SIZE) goto violated; if (check_4gib && (phys & DMAMEM_4GiB) != 0) goto violated; } } /* All checks passed */ return true; violated: error: return false; } /** * @} */