parrotcode: Memory Internals | |
Contents | Documentation |
docs/memory_internals.pod - Memory Internals
This document tries to explain the internals of the Parrot memory subsystem, and the data structures related to memory management and garbage collection.
All allocated items are managed in memory pools. Memory pools hold collections of similar items, items of the same data type and usually of the same size. These can be roughly divided into two kinds: pools holding fixed sized items and pools holding variable sized items.
A pool is organized as a linked list of big chunks, called Arenas. Each Arena holds and manages a set number of items of a single size and shape.
Garbage collector. A garbage collector is a system that finds memory objects that are not being used by the system, and frees them automatically. Garbage collection is broken into two distinct phases: the mark phase that searches for unreferenced objects, and the sweep phase that frees all dead objects.
Dead object detection, a deprecated term that refers to the mark phase of the garbage collector.
By scanning through all the interpreter's registers, stacks and the processor stack, all objects that are detected by the garbage collector are marked as alive. Objects in all pools not alive at the end of the mark phase are considered dead and are collected.
See docs/pdds/pdd09_gc.pod for details about the garbage collector system.
A overall layout of the interpreter's memory management looks like so:
typedef struct Interp {
...
struct Arenas *arena_base;
...
} Interp;
All object-like things that get allocated during the execution of parrot bytecode are managed from the arena_base
member of the interpreter structure.
struct Arenas
holds pointers to a variety of different kinds of managed memory. A simplification looks similar to this:
typedef struct Arenas {
struct Memory_Pool *memory_pool;
...
struct Small_Object_Pool * header_pool;
struct Small_Object_Pool ** sized_pools;
} Arenas;
memory_pool
and header_pool
are variable and fixed sized pool pointers respectively. These are just two examples, there are other Memory_Pool
s and Small_Object_Pool
s in the Parrot system. Pools of type struct Memory_Pool
are for variable-size objects, such as constant string buffers. Pools of type struct Small_Object_Pool
are for fixed-size objects such as headers or PMCs.
Fixed-size items are either objects by themselves, like a PMC
, or are header structures of variable-sized objects like STRING
. The general object of this kind is a buffer-like object, which consists of a Buffer
or a PObj
header at the beginning connected to a variable-sized memory block for data.
Buffer-like objects of the same size are maintained in the list of sized_pools
, which manage objects of the same size in one slot.
Every garbage collector must supply 4 function pointers: an initialization routine, a de-initialization routine, a function to initialize a new memory pool, and a function to initiate a garbage collection run. Every pool also requires the garbage collector to supply 4 additional function pointers: a function to allocate a new arena for the pool, a function to find more items (possibly through running the garbage collector, or allocating objects otherwise), a function to add an unused item to the free list, and a function to retreive a new free item from the pool.
Every pool maintains a list of unused items within the pool, called the free_list
When there is need for a new object it is taken off the free list, and when it stops being used, it will be put back on the free list again. GC mark determines which objects are not being used, and GC sweep adds those unused items back to the free list.
If the free list is empty then a GC run could be initiated. This may be able to find some dead objects and free them up for immediate reuse. However, if there are no objects that can be immediately recycled, a new arena will be allocated.
Arenas, which contain the fixed-sized items used by Parrot, are never returned to the operating system, even if they are empty. In the future, this behavior may change, however. Variable-sized buffers may be returned when they are no longer being used.
struct parrot_object_t {
UnionVal cache;
unsigned flags;
...
} PObj;
This does not totally reflect the current implementation, but is the spirit of the abstraction of current objects. The UnionVal cache
field is a union
that contains a variety of pointer and data configurations. The flags field may contain a series of flags which indicate the type, status, configuration, and special requirements of each item. Buffers, PMC
s, and PObj
s all have these basic fields in common, although they also contain a variety of other data fields, depending on type.
Only three flags need to be checked during a DOD run: PObj_live_FLAG
, PObj_on_free_list_FLAG
, and PObj_is_special_PMC_FLAG
. Normally these flags are stored in PObj->flags
, meaning that each PMC must be accessed during the DOD run.
An alternative approach is to store the DOD-Flags together somewhere, such as in the individual arenas, as a packed array of bits. This approach, called cardmarking, should be indicated by defining the preprocessor variable ARENA_DOD_FLAGS
to 1.
pobj.h provides macros to facilitate referencing individual object flags: DOD_flag_SET
, DOD_flag_CLEAR
and DOD_flag_TEST
. They make up a portable way of manipulating the DOD-relevant object flags.
Variable-sized items do not exist by themselves, they are always preceeded by a buffer structure that contains information about them. These buffer structures are described above, and the UnionVal cache
item typically points to the memory block that contains the data. The variable-sized data items are managed in two different pools: the memory_pool
, which contains a general mish-mash of data types, and the constant_string_pool
which contains immutable string buffers used by programs running on Parrot.
Here, different memory allocation schemes jump in:
A memory_pool
gets allocated in big blocks, namely a Memory_Block
. When some part is needed, e.g. to store a string, this part is carved out of the memory block, until this block is used up. If no more space is available in this memory block, a special copying garbage collection (GC) is started. This copies all living items of all used memory blocks into one new block, which holds thereafter only used items tightly packed together.
The old memory blocks, containing sparse unused parts and used parts already copied to the new place, are then unused altogether and get free()
ed thereafter. When GC doesn't provide enough free space needed for a new item, a new block is added to the memory pool.
This also implies that buffers are moved around during their life. Users of these buffers are therefore not allowed to hold pointers to buffers over pieces of code that might trigger a GC run, like Parrot_allocate()
or string_compare()
.
An alternative to the above is to use a memory allocator, which is as fast as the above and does reuse free()
ed items in a memory-conserving way. Actually, the malloc implementations in some systems' libc efficiently provide this, such as the glibc malloc based on Doug Lea's allocator.
Using this allocator, all variable sized items are just allocated via a plain malloc()
call, or resized with realloc()
, and after they lose their existence (ie when DOD detects that the managing buffer header is no longer in use) they are free()
ed. That's all. The underlying allocator collects these pieces, coalesces them if possible to form bigger pieces, and then puts them on free lists, sorted by size. Eventually, when a new allocation request arrives, it may give them back to Parrot.
So here, the variable length memory_pool
is unused. You can consider this pool to live inside the allocator. Buffers allocated this way don't move around, except when reallocated, of course.
The Configure.pl option --gc
allows one to use either method.
Both implementations have the same problem to solve: STRING
s that get copied, or parts of strings as the result of a substr() call, do not allocate new space in the memory pool to hold this string or part of string. They just use a new_string_header()
and set up a pointer (strstart
) pointing somewhere into the original string and remember the used length there in bufused
.
This is all OK, as long as the original and the lazy copy of the string are not modified. So that's well-known and called COW (copy on write). We don't have to make a copy of the buffer until one of the copies tries to modify it. In practice, strings often get copied from place to place but do not often get modified, so this optimization saves us a lot of time.
Now, during GC (or freeing buffers) the problem arises: to whom does this string belong? You shouldn't copy the same string to different places, thus rendering COW to a noop, nor are you allowed to free this same string more then once. Both allocation schemes therefore use a part of the allocated string to do this bookkeeping.
Copying GC uses a Buffer_Tail
after the end of the actual variable length string and marks such COW strings with TAIL_moved
and stores the new address in the buffer header, so other users of this string can be updated to reuse the same string (RT#47764 one or all other users?).
The malloc()
/free()
approach stores a refcount at bufstart
. During DOD all dead users increment the refcount, living users set it to an huge value. When freeing the buffer, the string is only freed if the refcount reaches zero.
+--------+
+------------------<---| Arenas |<-----------+
| +--------+-->--+ |
| | |
| +------+ +-----------+ | +=============+
| | S0 |<---| Registers |<--)--| Interpreter |
| +------+ +-----------+ | +=============+
| +---| S1 | |
| | +------+ +----------+
+-------+ | |
| Blk 1 |--)-->+----------+ +--------------+ +---------+
+-------+ | | Buffer 1 | | OnestringAno | | Block 1 |
| Blk 2 | | +----------+ | therstring.. | +---------+
+-------+ | | Buffer 2 | | ..String... |<--| Block 2 |
| . | | +----------+ +--------------+ +---------+
+-------+ | | ... | ^ ^ | ... |
Small Obj | +----------+ | | +---------+
Pool +-->| Buffer N |--------+----+ Memory Pool
+----------+
Buffer Memory Block
Now consider that the Interpreter shall be a PMC
living in Buffer X
of the underlying interpreter and is currently running perldoc docs/memory_internals.pod, and then redo this figure, with all the blanks filled in.
smallobject.[ch], headers.c, resources.[ch], res_lea.c, dod.c, string.[ch], pobj.h. Other garbage collector implementations may use separate files as well.
To minimize this bugs section - please patch this document and keep it up to date - thanks.
Leopold Tötsch lt@toetsch.at
0.1.1 June 2008
|