| The genalloc/genpool subsystem | 
 | ============================== | 
 |  | 
 | There are a number of memory-allocation subsystems in the kernel, each | 
 | aimed at a specific need.  Sometimes, however, a kernel developer needs to | 
 | implement a new allocator for a specific range of special-purpose memory; | 
 | often that memory is located on a device somewhere.  The author of the | 
 | driver for that device can certainly write a little allocator to get the | 
 | job done, but that is the way to fill the kernel with dozens of poorly | 
 | tested allocators.  Back in 2005, Jes Sorensen lifted one of those | 
 | allocators from the sym53c8xx_2 driver and posted_ it as a generic module | 
 | for the creation of ad hoc memory allocators.  This code was merged | 
 | for the 2.6.13 release; it has been modified considerably since then. | 
 |  | 
 | .. _posted: https://lwn.net/Articles/125842/ | 
 |  | 
 | Code using this allocator should include <linux/genalloc.h>.  The action | 
 | begins with the creation of a pool using one of: | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_create		 | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: devm_gen_pool_create | 
 |  | 
 | A call to :c:func:`gen_pool_create` will create a pool.  The granularity of | 
 | allocations is set with min_alloc_order; it is a log-base-2 number like | 
 | those used by the page allocator, but it refers to bytes rather than pages. | 
 | So, if min_alloc_order is passed as 3, then all allocations will be a | 
 | multiple of eight bytes.  Increasing min_alloc_order decreases the memory | 
 | required to track the memory in the pool.  The nid parameter specifies | 
 | which NUMA node should be used for the allocation of the housekeeping | 
 | structures; it can be -1 if the caller doesn't care. | 
 |  | 
 | The "managed" interface :c:func:`devm_gen_pool_create` ties the pool to a | 
 | specific device.  Among other things, it will automatically clean up the | 
 | pool when the given device is destroyed. | 
 |  | 
 | A pool is shut down with: | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_destroy | 
 |  | 
 | It's worth noting that, if there are still allocations outstanding from the | 
 | given pool, this function will take the rather extreme step of invoking | 
 | BUG(), crashing the entire system.  You have been warned. | 
 |  | 
 | A freshly created pool has no memory to allocate.  It is fairly useless in | 
 | that state, so one of the first orders of business is usually to add memory | 
 | to the pool.  That can be done with one of: | 
 |  | 
 | .. kernel-doc:: include/linux/genalloc.h | 
 |    :functions: gen_pool_add | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_add_virt | 
 |  | 
 | A call to :c:func:`gen_pool_add` will place the size bytes of memory | 
 | starting at addr (in the kernel's virtual address space) into the given | 
 | pool, once again using nid as the node ID for ancillary memory allocations. | 
 | The :c:func:`gen_pool_add_virt` variant associates an explicit physical | 
 | address with the memory; this is only necessary if the pool will be used | 
 | for DMA allocations. | 
 |  | 
 | The functions for allocating memory from the pool (and putting it back) | 
 | are: | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_alloc | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_dma_alloc | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_free | 
 |  | 
 | As one would expect, :c:func:`gen_pool_alloc` will allocate size< bytes | 
 | from the given pool.  The :c:func:`gen_pool_dma_alloc` variant allocates | 
 | memory for use with DMA operations, returning the associated physical | 
 | address in the space pointed to by dma.  This will only work if the memory | 
 | was added with :c:func:`gen_pool_add_virt`.  Note that this function | 
 | departs from the usual genpool pattern of using unsigned long values to | 
 | represent kernel addresses; it returns a void * instead. | 
 |  | 
 | That all seems relatively simple; indeed, some developers clearly found it | 
 | to be too simple.  After all, the interface above provides no control over | 
 | how the allocation functions choose which specific piece of memory to | 
 | return.  If that sort of control is needed, the following functions will be | 
 | of interest: | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_alloc_algo | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_set_algo | 
 |  | 
 | Allocations with :c:func:`gen_pool_alloc_algo` specify an algorithm to be | 
 | used to choose the memory to be allocated; the default algorithm can be set | 
 | with :c:func:`gen_pool_set_algo`.  The data value is passed to the | 
 | algorithm; most ignore it, but it is occasionally needed.  One can, | 
 | naturally, write a special-purpose algorithm, but there is a fair set | 
 | already available: | 
 |  | 
 | - gen_pool_first_fit is a simple first-fit allocator; this is the default | 
 |   algorithm if none other has been specified. | 
 |  | 
 | - gen_pool_first_fit_align forces the allocation to have a specific | 
 |   alignment (passed via data in a genpool_data_align structure). | 
 |  | 
 | - gen_pool_first_fit_order_align aligns the allocation to the order of the | 
 |   size.  A 60-byte allocation will thus be 64-byte aligned, for example. | 
 |  | 
 | - gen_pool_best_fit, as one would expect, is a simple best-fit allocator. | 
 |  | 
 | - gen_pool_fixed_alloc allocates at a specific offset (passed in a | 
 |   genpool_data_fixed structure via the data parameter) within the pool. | 
 |   If the indicated memory is not available the allocation fails. | 
 |  | 
 | There is a handful of other functions, mostly for purposes like querying | 
 | the space available in the pool or iterating through chunks of memory. | 
 | Most users, however, should not need much beyond what has been described | 
 | above.  With luck, wider awareness of this module will help to prevent the | 
 | writing of special-purpose memory allocators in the future. | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_virt_to_phys | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_for_each_chunk | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: addr_in_gen_pool | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_avail | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_size | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: gen_pool_get | 
 |  | 
 | .. kernel-doc:: lib/genalloc.c | 
 |    :functions: of_gen_pool_get |