blob: 0ae6e11bd2e818a8d1ae91660c24f72a7760cce1 [file] [log] [blame]
Dan Gohmanf17a25c2007-07-18 16:29:46 +00001/*===-- semispace.c - Simple semi-space copying garbage collector ---------===*\
2|*
3|* The LLVM Compiler Infrastructure
4|*
Chris Lattner3023be42007-12-29 22:59:10 +00005|* This file is distributed under the University of Illinois Open Source
6|* License. See LICENSE.TXT for details.
Dan Gohmanf17a25c2007-07-18 16:29:46 +00007|*
8|*===----------------------------------------------------------------------===*|
9|*
10|* This garbage collector is an extremely simple copying collector. It splits
11|* the managed region of memory into two pieces: the current space to allocate
12|* from, and the copying space. When the portion being allocated from fills up,
13|* a garbage collection cycle happens, which copies all live blocks to the other
14|* half of the managed space.
15|*
16\*===----------------------------------------------------------------------===*/
17
18#include "../GCInterface.h"
19#include <stdio.h>
20#include <stdlib.h>
21#include <string.h>
22
23/* AllocPtr - This points to the next byte that is available for allocation.
24 */
25static char *AllocPtr;
26
27/* AllocEnd - This points to the first byte not available for allocation. When
28 * AllocPtr passes this, we have run out of space.
29 */
30static char *AllocEnd;
31
32/* CurSpace/OtherSpace - These pointers point to the two regions of memory that
33 * we switch between. The unallocated portion of the CurSpace is known to be
34 * zero'd out, but the OtherSpace contains junk.
35 */
36static void *CurSpace, *OtherSpace;
37
38/* SpaceSize - The size of each space. */
39static unsigned SpaceSize;
40
41/* llvm_gc_initialize - Allocate the two spaces that we plan to switch between.
42 */
43void llvm_gc_initialize(unsigned InitialHeapSize) {
44 SpaceSize = InitialHeapSize/2;
45 CurSpace = AllocPtr = calloc(1, SpaceSize);
46 OtherSpace = malloc(SpaceSize);
47 AllocEnd = AllocPtr + SpaceSize;
48}
49
50/* We always want to inline the fast path, but never want to inline the slow
51 * path.
52 */
53void *llvm_gc_allocate(unsigned Size) __attribute__((always_inline));
54static void* llvm_gc_alloc_slow(unsigned Size) __attribute__((noinline));
55
56void *llvm_gc_allocate(unsigned Size) {
57 char *OldAP = AllocPtr;
58 char *NewEnd = OldAP+Size;
59 if (NewEnd > AllocEnd)
60 return llvm_gc_alloc_slow(Size);
61 AllocPtr = NewEnd;
62 return OldAP;
63}
64
65static void* llvm_gc_alloc_slow(unsigned Size) {
66 llvm_gc_collect();
67 if (AllocPtr+Size > AllocEnd) {
68 fprintf(stderr, "Garbage collector ran out of memory "
69 "allocating object of size: %d\n", Size);
70 exit(1);
71 }
72
73 return llvm_gc_allocate(Size);
74}
75
76
77static void process_pointer(void **Root, void *Meta) {
78 printf("process_root[0x%p] = 0x%p\n", (void*) Root, (void*) *Root);
79}
80
81void llvm_gc_collect() {
82 // Clear out the space we will be copying into.
83 // FIXME: This should do the copy, then clear out whatever space is left.
84 memset(OtherSpace, 0, SpaceSize);
85
86 printf("Garbage collecting!!\n");
87 llvm_cg_walk_gcroots(process_pointer);
88 abort();
89}
90
91/* We use no read/write barriers */
92void *llvm_gc_read(void *ObjPtr, void **FieldPtr) { return *FieldPtr; }
93void llvm_gc_write(void *V, void *ObjPtr, void **FieldPtr) { *FieldPtr = V; }
94
95
96/*===----------------------------------------------------------------------===**
97 * FIXME: This should be in a code-generator specific library, but for now this
98 * will work for all code generators.
99 */
Gordon Henriksen95fc9462008-01-24 05:16:36 +0000100typedef struct FrameMap FrameMap;
Gordon Henriksencc71a502008-01-07 01:30:53 +0000101struct FrameMap {
102 int32_t NumRoots; // Number of roots in stack frame.
103 int32_t NumMeta; // Number of metadata descriptors. May be < NumRoots.
104 void *Meta[]; // May be absent for roots without metadata.
105};
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000106
Gordon Henriksen95fc9462008-01-24 05:16:36 +0000107typedef struct StackEntry StackEntry;
Gordon Henriksencc71a502008-01-07 01:30:53 +0000108struct StackEntry {
Gordon Henriksen95fc9462008-01-24 05:16:36 +0000109 StackEntry *Next; // Caller's stack entry.
Gordon Henriksencc71a502008-01-07 01:30:53 +0000110 const FrameMap *Map; // Pointer to constant FrameMap.
111 void *Roots[]; // Stack roots (in-place array).
112};
113StackEntry *llvm_gc_root_chain;
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000114
115void llvm_cg_walk_gcroots(void (*FP)(void **Root, void *Meta)) {
Gordon Henriksen95fc9462008-01-24 05:16:36 +0000116 StackEntry *R;
117 for (R = llvm_gc_root_chain; R; R = R->Next) {
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000118 unsigned i, e;
Gordon Henriksen95fc9462008-01-24 05:16:36 +0000119 for (i = 0, e = R->Map->NumMeta; i != e; ++i)
Gordon Henriksencc71a502008-01-07 01:30:53 +0000120 FP(&R->Roots[i], R->Map->Meta[i]);
Gordon Henriksen95fc9462008-01-24 05:16:36 +0000121 for (e = R->Map->NumRoots; i != e; ++i)
Gordon Henriksencc71a502008-01-07 01:30:53 +0000122 FP(&R->Roots[i], NULL);
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000123 }
124}
125/* END FIXME! */
126
127