.. index::
pair: free list allocator; design
.. _design-freelist:
Free list allocator
===================
.. mps:prefix:: design.mps.freelist
Introduction
------------
:mps:tag:`intro` This document describes the free list allocator for the
Memory Pool System.
:mps:tag:`readership` Any MPS developer.
Overview
--------
:mps:tag:`overview` The free list allocator is an "emergency" allocator. It
is intended for use as a fallback allocation strategy in low memory
situations, when memory is not available for the control structures
needed by other allocators. In these situations the free list allocator
ensures that memory is not lost, but with several disadvantages:
#. operations on the free list take time proportional to the number of
free blocks;
#. the data structures are stored in client memory and so are
vulnerable to corruption;
#. the data structures have poor locality (and thus potentially poor
cache performance).
When memory becomes available again to allocate control structures,
the free lists can be "flushed" back into the more efficient data
structures.
Requirements
------------
In addition to the generic land requirements (see design.mps.land_),
free lists must satisfy:
.. _design.mps.land: land
:mps:tag:`req.zero-overhead` Must have zero space overhead for the storage
of any set of free blocks, so that it can be used to manage memory
when no memory can be allocated for control structures.
Interface
---------
:mps:tag:`land` Free lists are an implementation of the *land* abstract data
type, so the interface consists of the generic functions for lands.
See design.mps.land_.
Types
.....
.. c:type:: struct FreelistStruct *Freelist
:mps:tag:`type.freelist` The type of free lists. A :c:type:`FreelistStruct` may be
embedded in another structure, or you can create it using
:c:func:`LandCreate()`.
External functions
..................
.. c:function:: LandClass FreelistLandClassGet(void)
:mps:tag:`function.class` The function :c:func:`FreelistLandClassGet()` returns
the free list class, a subclass of :c:type:`LandClass` suitable for passing
to :c:func:`LandCreate()` or :c:func:`LandInit()`.
Keyword arguments
.................
When initializing a free list, :c:func:`LandCreate()` and :c:func:`LandInit()`
take no keyword arguments. Pass ``mps_args_none``.
Implementation
--------------
:mps:tag:`impl.list` The isolated contiguous free address ranges are kept on
an address-ordered singly linked free list. (As in traditional
:c:func:`malloc()` implementations.)
:mps:tag:`impl.block` If the free address range is large enough to contain
an inline block descriptor consisting of two pointers, then the two
pointers stored are to the next free range in address order (or
``freelistEND`` if there are no more ranges), and to the limit of the
current free address range, in that order.
:mps:tag:`impl.grain` Otherwise, the free address range must be large enough
to contain a single pointer. The pointer stored is to the next free
range in address order, or ``freelistEND`` if there are no more
ranges.
:mps:tag:`impl.tag` Grains and blocks are distinguished by a one-bit tag in
the low bit of the first word (the one containing the pointer to the
next range). Grains have this bit set; blocks have this bit reset.
:mps:tag:`impl.invariant` The ranges stored in the free list are *isolated*:
no two ranges are adjacent or overlapping.
:mps:tag:`impl.merge` When a free address range is added to the free list,
it is merged with adjacent ranges so as to maintain
:mps:ref:`.impl.invariant`.
:mps:tag:`impl.rule.break` The use of ``freelistEND`` to mark the end of the
list violates the rule that exceptional values should not be used to
distinguish exeptional situations. This infraction allows the
implementation to meet :mps:ref:`.req.zero-overhead`. (There are other ways to
do this, such as using another tag to indicate the last block in the
list, but these would be more complicated.)
Testing
-------
:mps:tag:`test` The following testing will be performed on this module:
:mps:tag:`test.land` A generic test for land implementations. See
design.mps.land.test.
:mps:tag:`test.pool` Two pools (MVT_ and MVFF_) use free lists as a fallback
when low on memory. These are subject to testing in development, QA,
and are heavily exercised by customers.
.. _MVT: poolmvt
.. _MVFF: poolmvff
Opportunities for improvement
-----------------------------
:mps:tag:`improve.length` When iterating over the list, we could check that
the number of elements visited in the course of the iteration does not
exceed the recorded size of the list.
:mps:tag:`improve.maxsize` We could maintain the maximum size of any range
on the list, and use that to make an early exit from
:c:func:`freelistFindLargest()`. It's not clear that this would actually be
an improvement.