473 lines
14 KiB
C
473 lines
14 KiB
C
/*! \file ngbde_pgmem.c
|
|
*
|
|
* \brief PGMEM allocator.
|
|
*
|
|
* This module is used to allocate large physically contiguous memory
|
|
* blocks using the Linux kernel page allocator.
|
|
*
|
|
* The Linux page allocator can allocate contiguous memory up until a
|
|
* certain size, which depends on the kernel version and the CPU
|
|
* architecture.
|
|
*
|
|
* If a larger contiguous memory block is requested, then we need to
|
|
* allocate multiple blocks from the Linux page allocator and then
|
|
* check if which ones are contiguous.
|
|
*
|
|
* The smaller memory blocks from which the larger block is assembled
|
|
* are referred to as "chunks".
|
|
*
|
|
* The PGMEM allocator will continue to allocate chunks from the Linux
|
|
* page allocator, until a contiguous memory block of the requested
|
|
* size has been assembled, or until a predefined maximum number of
|
|
* chunks have been allocated. Obviously the process is also stopped
|
|
* if the Linux page allocator returns an error.
|
|
*
|
|
* A physically contiguous memory block assembled from smaller memory
|
|
* chunks are referred to as "cmblocks".
|
|
*
|
|
* The chance of success depends on the requested memory block size as
|
|
* well as the fragmentation level of the system memory, i.e. the
|
|
* sooner after system boot these memory block are requested, the more
|
|
* likely these requests are to succeed.
|
|
*/
|
|
/*
|
|
* $Copyright: Copyright 2018-2022 Broadcom. All rights reserved.
|
|
* The term 'Broadcom' refers to Broadcom Inc. and/or its subsidiaries.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* A copy of the GNU General Public License version 2 (GPLv2) can
|
|
* be found in the LICENSES folder.$
|
|
*/
|
|
|
|
#include <ngbde.h>
|
|
|
|
|
|
/*******************************************************************************
|
|
* Local definitions
|
|
******************************************************************************/
|
|
|
|
/*! Maximum size the kernel can allocate in a single allocation. */
|
|
#define MEM_CHUNK_SIZE_MAX (1 << (MAX_ORDER - 1 + PAGE_SHIFT))
|
|
|
|
/*! Default block size we wil request from the kernel. */
|
|
#define MEM_CHUNK_SIZE_DEFAULT (512 * ONE_KB)
|
|
|
|
/*! \cond */
|
|
static int pgmem_chunk_size = 0;
|
|
module_param(pgmem_chunk_size, int, S_IRUSR);
|
|
MODULE_PARM_DESC(pgmem_chunk_size,
|
|
"Memory chunk size in KB used by page allocator (default auto).");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int pgmem_debug = 0;
|
|
module_param(pgmem_debug, int, S_IRUSR | S_IWUSR);
|
|
MODULE_PARM_DESC(pgmem_debug,
|
|
"Enable page memory allocator debug output (default 0).");
|
|
/*! \endcond */
|
|
|
|
/*! Helper macro for debug trace output. */
|
|
#define PGMEM_TRACE(_s) \
|
|
do { \
|
|
if (pgmem_debug) { \
|
|
printk(_s); \
|
|
} \
|
|
} while (0)
|
|
|
|
/*!
|
|
* Chunk memory block descriptor.
|
|
*/
|
|
typedef struct cmblock_desc_s {
|
|
|
|
/*! Linked-list handle. */
|
|
struct list_head list;
|
|
|
|
/*! Requested cmblock size. */
|
|
unsigned long req_size;
|
|
|
|
/*! Memory chunk size. */
|
|
unsigned long chunk_size;
|
|
|
|
/*! Memory chunk size in alternate format (2^x). */
|
|
unsigned long chunk_order;
|
|
|
|
/*! Current cmblock size. */
|
|
unsigned long cmblk_size;
|
|
|
|
/*! Logical address of cmblock. */
|
|
unsigned long cmblk_begin;
|
|
|
|
/*! Logical end address of cmblock. */
|
|
unsigned long cmblk_end;
|
|
|
|
/*! Array of logical chunk addresses. */
|
|
unsigned long *chunk_ptr;
|
|
|
|
/*! Maximum number of chunks to allocate. */
|
|
int chunk_cnt_max;
|
|
|
|
/*! Current number of chunks allocated. */
|
|
int chunk_cnt;
|
|
|
|
} cmblock_desc_t;
|
|
|
|
static LIST_HEAD(cmblocks_list);
|
|
|
|
|
|
/*!
|
|
* \name Chunk tag mask.
|
|
* \anchor CT_xxx
|
|
*
|
|
* The lower two bits of the chunk address is used to tag the chunk
|
|
* with its current state.
|
|
*/
|
|
#define CT_MASK 0x3
|
|
|
|
/*! Chunk is untagged. */
|
|
#define CT_UNTAGGED 0
|
|
|
|
/*! Chunk was discarded. */
|
|
#define CT_DISCARDED 1
|
|
|
|
/*! Chunk is part of largest cmblock. */
|
|
#define CT_LARGEST 2
|
|
|
|
/*! Chunk is part of current cmblock. */
|
|
#define CT_CURRENT 3
|
|
|
|
/*! Set block as untagged. */
|
|
#define CTAG_SET(_a, _t) \
|
|
do { \
|
|
(_a) &= ~CT_MASK; \
|
|
(_a) |= _t; \
|
|
} while (0)
|
|
|
|
/*! Set block as untagged. */
|
|
#define CTAG_GET(_a) \
|
|
((_a) & CT_MASK)
|
|
|
|
|
|
/*******************************************************************************
|
|
* Private Functions
|
|
******************************************************************************/
|
|
|
|
/*!
|
|
* \brief Find largest contiguous memory block.
|
|
*
|
|
* Find largest contiguous memory block from a pool of memory chunks.
|
|
*
|
|
* Assembly stops if a cmblock of the requested cmblock size has been
|
|
* obtained.
|
|
*
|
|
* The lower two address bits of the memory chunks are encoded as a
|
|
* tag according to \ref CT_xxx.
|
|
*
|
|
* \param [in] cmbd cmblock descriptor.
|
|
*
|
|
* \return Always 0.
|
|
*/
|
|
static int
|
|
find_largest_cmblock(cmblock_desc_t *cmbd)
|
|
{
|
|
int i, j, chunks, found;
|
|
unsigned long b, e, a;
|
|
unsigned long *cptr;
|
|
|
|
/* Convenience variable */
|
|
chunks = cmbd->chunk_cnt;
|
|
cptr = cmbd->chunk_ptr;
|
|
|
|
/* Clear all chunk tags */
|
|
for (i = 0; i < chunks; i++) {
|
|
CTAG_SET(cptr[i], CT_UNTAGGED);
|
|
}
|
|
for (i = 0; i < chunks && cmbd->cmblk_size < cmbd->req_size; i++) {
|
|
/* First chunk must be an untagged chunk */
|
|
if (CTAG_GET(cptr[i]) == CT_UNTAGGED) {
|
|
/* Initial cmblock size is the chunk size */
|
|
b = cptr[i];
|
|
e = b + cmbd->chunk_size;
|
|
CTAG_SET(cptr[i], CT_CURRENT);
|
|
/* Loop looking for adjacent chunks */
|
|
do {
|
|
found = 0;
|
|
for (j = i + 1; j < chunks && (e - b) < cmbd->req_size; j++) {
|
|
a = cptr[j];
|
|
/* Check untagged chunks only */
|
|
if (CTAG_GET(a) == CT_UNTAGGED) {
|
|
if (a == (b - cmbd->chunk_size)) {
|
|
/* Found adjacent chunk below current cmblock */
|
|
CTAG_SET(cptr[j], CT_CURRENT);
|
|
b = a;
|
|
found = 1;
|
|
} else if (a == e) {
|
|
/* Found adjacent chunk above current cmblock */
|
|
CTAG_SET(cptr[j], CT_CURRENT);
|
|
e += cmbd->chunk_size;
|
|
found = 1;
|
|
}
|
|
}
|
|
}
|
|
} while (found);
|
|
/* Now check the size of the assembled memory block */
|
|
if ((e - b) > cmbd->cmblk_size) {
|
|
/* The current block is largest so far */
|
|
cmbd->cmblk_begin = b;
|
|
cmbd->cmblk_end = e;
|
|
cmbd->cmblk_size = e - b;
|
|
/* Re-tag current and previous largest cmblock */
|
|
for (j = 0; j < chunks; j++) {
|
|
if (CTAG_GET(cptr[j]) == CT_CURRENT) {
|
|
/* Tag current cmblock as the largest */
|
|
CTAG_SET(cptr[j], CT_LARGEST);
|
|
} else if (CTAG_GET(cptr[j]) == CT_LARGEST) {
|
|
/* Discard previous largest cmblock */
|
|
CTAG_SET(cptr[j], CT_DISCARDED);
|
|
}
|
|
}
|
|
} else {
|
|
/* Discard all chunks in current cmblock */
|
|
for (j = 0; j < chunks; j++) {
|
|
if (CTAG_GET(cptr[j]) == CT_CURRENT) {
|
|
CTAG_SET(cptr[j], CT_DISCARDED);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* \brief Allocate memory chunks and add them to the pool.
|
|
*
|
|
* Memory chunks are allocated using the kernel page allocator.
|
|
*
|
|
* \param [in] cmbd - cmblock descriptor.
|
|
* \param [in] chunks - Number of memory chunks to allocate.
|
|
*
|
|
* \return 0 if no errors, otherwise -1.
|
|
*/
|
|
static int
|
|
alloc_mem_chunks(cmblock_desc_t *cmbd, int chunks)
|
|
{
|
|
int i, start;
|
|
unsigned long addr;
|
|
|
|
if (cmbd->chunk_cnt + chunks > cmbd->chunk_cnt_max) {
|
|
printk("PGMEM: No more memory chunks\n");
|
|
return -1;
|
|
}
|
|
start = cmbd->chunk_cnt;
|
|
cmbd->chunk_cnt += chunks;
|
|
for (i = start; i < cmbd->chunk_cnt; i++) {
|
|
/* Get chunk from kernel allocator */
|
|
addr = __get_free_pages(GFP_KERNEL | GFP_DMA32, cmbd->chunk_order);
|
|
PGMEM_TRACE(".");
|
|
if (addr) {
|
|
cmbd->chunk_ptr[i] = addr;
|
|
} else {
|
|
printk("PGMEM: Page memory allocation failed\n");
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* \brief Allocate large physically contiguous memory block.
|
|
*
|
|
* If we cannot allocate a sufficiently large block of contiguous
|
|
* memory from the kernel, then we simply keep allocating smaller
|
|
* chunks until we can assemble a contiguous block of the desired
|
|
* size.
|
|
*
|
|
* When maximum amount of system memory has been allocated without the
|
|
* successful assembly of a contiguous memory block, the allocation
|
|
* function will return the largest contiguous block found so far. It
|
|
* is then up to the calling function to decide whether this amount is
|
|
* sufficient to proceed.
|
|
*
|
|
* \param [in] size Requested memory block size.
|
|
* \param [in] chunk_size Assemble cmblock from chunks of this size.
|
|
*
|
|
* \return Pointer to cmblock descriptor, or NULL if error.
|
|
*/
|
|
static cmblock_desc_t *
|
|
cmblock_alloc(size_t size, size_t chunk_size)
|
|
{
|
|
cmblock_desc_t *cmbd;
|
|
int i, chunk_ptr_size;
|
|
unsigned long page_addr;
|
|
struct sysinfo si;
|
|
|
|
/* Sanity check */
|
|
if (size == 0 || chunk_size == 0) {
|
|
return NULL;
|
|
}
|
|
|
|
/* Allocate an initialize memory cmblock descriptor */
|
|
if ((cmbd = kmalloc(sizeof(cmblock_desc_t), GFP_KERNEL)) == NULL) {
|
|
return NULL;
|
|
}
|
|
memset(cmbd, 0, sizeof(*cmbd));
|
|
cmbd->req_size = size;
|
|
cmbd->chunk_size = PAGE_ALIGN(chunk_size);
|
|
while ((PAGE_SIZE << cmbd->chunk_order) < cmbd->chunk_size) {
|
|
cmbd->chunk_order++;
|
|
}
|
|
|
|
/* Determine the maximum possible number of memory chunks */
|
|
si_meminfo(&si);
|
|
cmbd->chunk_cnt_max = (si.totalram << PAGE_SHIFT) / cmbd->chunk_size;
|
|
chunk_ptr_size = cmbd->chunk_cnt_max * sizeof(unsigned long);
|
|
|
|
/* Allocate an initialize memory chunk pool */
|
|
cmbd->chunk_ptr = kmalloc(chunk_ptr_size, GFP_KERNEL);
|
|
if (cmbd->chunk_ptr == NULL) {
|
|
kfree(cmbd);
|
|
return NULL;
|
|
}
|
|
memset(cmbd->chunk_ptr, 0, chunk_ptr_size);
|
|
|
|
/* Allocate minimum number of memory chunks */
|
|
(void)alloc_mem_chunks(cmbd, cmbd->req_size / cmbd->chunk_size);
|
|
|
|
/* Allocate more chunks until we have a complete cmblock */
|
|
do {
|
|
find_largest_cmblock(cmbd);
|
|
PGMEM_TRACE("o");
|
|
if (cmbd->cmblk_size >= cmbd->req_size) {
|
|
break;
|
|
}
|
|
} while (alloc_mem_chunks(cmbd, 8) == 0);
|
|
|
|
/* Reserve all pages in the cmblock and free unused chunks */
|
|
for (i = 0; i < cmbd->chunk_cnt; i++) {
|
|
if (CTAG_GET(cmbd->chunk_ptr[i]) == CT_LARGEST) {
|
|
CTAG_SET(cmbd->chunk_ptr[i], CT_UNTAGGED);
|
|
for (page_addr = cmbd->chunk_ptr[i];
|
|
page_addr < cmbd->chunk_ptr[i] + cmbd->chunk_size;
|
|
page_addr += PAGE_SIZE) {
|
|
SetPageReserved(virt_to_page((void *)page_addr));
|
|
}
|
|
} else if (cmbd->chunk_ptr[i]) {
|
|
CTAG_SET(cmbd->chunk_ptr[i], CT_UNTAGGED);
|
|
free_pages(cmbd->chunk_ptr[i], cmbd->chunk_order);
|
|
PGMEM_TRACE("x");
|
|
cmbd->chunk_ptr[i] = 0;
|
|
}
|
|
}
|
|
PGMEM_TRACE("O\n");
|
|
return cmbd;
|
|
}
|
|
|
|
/*!
|
|
* \brief Free cmblock and associated resources.
|
|
*
|
|
* Free all memory chunks and other associated resources associated
|
|
* with a contiguous memory block.
|
|
*
|
|
* See alse \ref cmblock_alloc.
|
|
*
|
|
* \param [in] cmbd Command block descriptor to free.
|
|
*
|
|
* \return Nothing.
|
|
*/
|
|
static void
|
|
cmblock_free(cmblock_desc_t *cmbd)
|
|
{
|
|
int i;
|
|
unsigned long page_addr;
|
|
|
|
if (cmbd->chunk_ptr) {
|
|
for (i = 0; i < cmbd->chunk_cnt; i++) {
|
|
if (cmbd->chunk_ptr[i]) {
|
|
for (page_addr = cmbd->chunk_ptr[i];
|
|
page_addr < cmbd->chunk_ptr[i] + cmbd->chunk_size;
|
|
page_addr += PAGE_SIZE) {
|
|
ClearPageReserved(virt_to_page((void *)page_addr));
|
|
}
|
|
free_pages(cmbd->chunk_ptr[i], cmbd->chunk_order);
|
|
PGMEM_TRACE("X");
|
|
}
|
|
}
|
|
kfree(cmbd->chunk_ptr);
|
|
kfree(cmbd);
|
|
}
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
* Public Functions
|
|
******************************************************************************/
|
|
|
|
void *
|
|
ngbde_pgmem_alloc(size_t size, gfp_t flags)
|
|
{
|
|
cmblock_desc_t *cmbd;
|
|
size_t chunk_size;
|
|
|
|
chunk_size = size;
|
|
|
|
if (pgmem_chunk_size > 0) {
|
|
chunk_size = pgmem_chunk_size * ONE_KB;
|
|
}
|
|
|
|
if (chunk_size > MEM_CHUNK_SIZE_MAX) {
|
|
chunk_size = MEM_CHUNK_SIZE_DEFAULT;
|
|
}
|
|
|
|
if (pgmem_debug) {
|
|
printk("PGMEM: Allocate %d MB in %d KB chunks\n",
|
|
(int)(size / ONE_MB), (int)(chunk_size / ONE_KB));
|
|
}
|
|
|
|
if ((cmbd = cmblock_alloc(size, chunk_size)) == NULL) {
|
|
return NULL;
|
|
}
|
|
if (cmbd->cmblk_size < size) {
|
|
/* If we didn't get the full size then forget it */
|
|
cmblock_free(cmbd);
|
|
return NULL;
|
|
}
|
|
list_add(&cmbd->list, &cmblocks_list);
|
|
return (void *)cmbd->cmblk_begin;
|
|
}
|
|
|
|
int
|
|
ngbde_pgmem_free(void *ptr)
|
|
{
|
|
struct list_head *pos;
|
|
|
|
list_for_each(pos, &cmblocks_list) {
|
|
cmblock_desc_t *cmbd = list_entry(pos, cmblock_desc_t, list);
|
|
if (ptr == (void *)cmbd->cmblk_begin) {
|
|
list_del(&cmbd->list);
|
|
cmblock_free(cmbd);
|
|
return 0;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void
|
|
ngbde_pgmem_free_all(void)
|
|
{
|
|
struct list_head *pos, *tmp;
|
|
|
|
list_for_each_safe(pos, tmp, &cmblocks_list) {
|
|
cmblock_desc_t *cmbd = list_entry(pos, cmblock_desc_t, list);
|
|
list_del(&cmbd->list);
|
|
cmblock_free(cmbd);
|
|
}
|
|
}
|