cregit-Linux how code gets into the kernel

Release 4.16 drivers/pci/endpoint/pci-epc-mem.c

// SPDX-License-Identifier: GPL-2.0
/**
 * PCI Endpoint *Controller* Address Space Management
 *
 * Copyright (C) 2017 Texas Instruments
 * Author: Kishon Vijay Abraham I <kishon@ti.com>
 */

#include <linux/io.h>
#include <linux/module.h>
#include <linux/slab.h>

#include <linux/pci-epc.h>

/**
 * pci_epc_mem_get_order() - determine the allocation order of a memory size
 * @mem: address space of the endpoint controller
 * @size: the size for which to get the order
 *
 * Reimplement get_order() for mem->page_size since the generic get_order
 * always gets order with a constant PAGE_SIZE.
 */

static int pci_epc_mem_get_order(struct pci_epc_mem *mem, size_t size) { int order; unsigned int page_shift = ilog2(mem->page_size); size--; size >>= page_shift; #if BITS_PER_LONG == 32 order = fls(size); #else order = fls64(size); #endif return order; }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I61100.00%2100.00%
Total61100.00%2100.00%

/** * __pci_epc_mem_init() - initialize the pci_epc_mem structure * @epc: the EPC device that invoked pci_epc_mem_init * @phys_base: the physical address of the base * @size: the size of the address space * @page_size: size of each page * * Invoke to initialize the pci_epc_mem structure used by the * endpoint functions to allocate mapped PCI address. */
int __pci_epc_mem_init(struct pci_epc *epc, phys_addr_t phys_base, size_t size, size_t page_size) { int ret; struct pci_epc_mem *mem; unsigned long *bitmap; unsigned int page_shift; int pages; int bitmap_size; if (page_size < PAGE_SIZE) page_size = PAGE_SIZE; page_shift = ilog2(page_size); pages = size >> page_shift; bitmap_size = BITS_TO_LONGS(pages) * sizeof(long); mem = kzalloc(sizeof(*mem), GFP_KERNEL); if (!mem) { ret = -ENOMEM; goto err; } bitmap = kzalloc(bitmap_size, GFP_KERNEL); if (!bitmap) { ret = -ENOMEM; goto err_mem; } mem->bitmap = bitmap; mem->phys_base = phys_base; mem->page_size = page_size; mem->pages = pages; mem->size = size; epc->mem = mem; return 0; err_mem: kfree(mem); err: return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I180100.00%2100.00%
Total180100.00%2100.00%

EXPORT_SYMBOL_GPL(__pci_epc_mem_init); /** * pci_epc_mem_exit() - cleanup the pci_epc_mem structure * @epc: the EPC device that invoked pci_epc_mem_exit * * Invoke to cleanup the pci_epc_mem structure allocated in * pci_epc_mem_init(). */
void pci_epc_mem_exit(struct pci_epc *epc) { struct pci_epc_mem *mem = epc->mem; epc->mem = NULL; kfree(mem->bitmap); kfree(mem); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I37100.00%1100.00%
Total37100.00%1100.00%

EXPORT_SYMBOL_GPL(pci_epc_mem_exit); /** * pci_epc_mem_alloc_addr() - allocate memory address from EPC addr space * @epc: the EPC device on which memory has to be allocated * @phys_addr: populate the allocated physical address here * @size: the size of the address space that has to be allocated * * Invoke to allocate memory address from the EPC address space. This * is usually done to map the remote RC address into the local system. */
void __iomem *pci_epc_mem_alloc_addr(struct pci_epc *epc, phys_addr_t *phys_addr, size_t size) { int pageno; void __iomem *virt_addr; struct pci_epc_mem *mem = epc->mem; unsigned int page_shift = ilog2(mem->page_size); int order; size = ALIGN(size, mem->page_size); order = pci_epc_mem_get_order(mem, size); pageno = bitmap_find_free_region(mem->bitmap, mem->pages, order); if (pageno < 0) return NULL; *phys_addr = mem->phys_base + (pageno << page_shift); virt_addr = ioremap(*phys_addr, size); if (!virt_addr) bitmap_release_region(mem->bitmap, pageno, order); return virt_addr; }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I136100.00%2100.00%
Total136100.00%2100.00%

EXPORT_SYMBOL_GPL(pci_epc_mem_alloc_addr); /** * pci_epc_mem_free_addr() - free the allocated memory address * @epc: the EPC device on which memory was allocated * @phys_addr: the allocated physical address * @virt_addr: virtual address of the allocated mem space * @size: the size of the allocated address space * * Invoke to free the memory allocated using pci_epc_mem_alloc_addr. */
void pci_epc_mem_free_addr(struct pci_epc *epc, phys_addr_t phys_addr, void __iomem *virt_addr, size_t size) { int pageno; struct pci_epc_mem *mem = epc->mem; unsigned int page_shift = ilog2(mem->page_size); int order; iounmap(virt_addr); pageno = (phys_addr - mem->phys_base) >> page_shift; size = ALIGN(size, mem->page_size); order = pci_epc_mem_get_order(mem, size); bitmap_release_region(mem->bitmap, pageno, order); }

Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I95100.00%2100.00%
Total95100.00%2100.00%

EXPORT_SYMBOL_GPL(pci_epc_mem_free_addr); MODULE_DESCRIPTION("PCI EPC Address Space Management"); MODULE_AUTHOR("Kishon Vijay Abraham I <kishon@ti.com>"); MODULE_LICENSE("GPL v2");

Overall Contributors

PersonTokensPropCommitsCommitProp
Kishon Vijay Abraham I56199.64%266.67%
Björn Helgaas20.36%133.33%
Total563100.00%3100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.