aboutsummaryrefslogtreecommitdiff
path: root/src/mem.hff
blob: f3689ac96dd54712f8b504d0c8a2bcae9acdf552 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
import "libc.hff";

defmacro ALIGNUP(x,a) [ (((x) + ((a) - 1)) & -(a)) ]
def ARENA_SIZE = 4z * 1024;
extern fn xmalloc(n usize) *void;
extern fn xcalloc(n usize, m usize) *void;
extern fn xrealloc(p *void, n usize) *void;

struct ArenaRegion {
   prev *ArenaRegion,
   idx usize,
   siz usize,
   mem [0]u8,
}

struct Arena {
   r *ArenaRegion,

   fn addregion(a *Arena, siz usize) void {
      let r *ArenaRegion = xcalloc(sizeof ArenaRegion + siz, 1);
      r.siz = siz;
      r.prev = a.r;
      a.r = r;
   }

   fn allocf(a *void, n usize, align usize) *void {
      let a *Arena = a;
      if a.r {
         a.r.idx = ALIGNUP(a.r.idx, align);
      }
      if n > ARENA_SIZE {
         a->addregion(n);
         let p *void = a.r.mem;
         a->addregion(ARENA_SIZE);
         return p;
      } else if a.r == #null {
         a->addregion(ARENA_SIZE);
         return allocf(a, n, align);
      } else if n > a.r.siz - a.r.idx {
         a->addregion(ARENA_SIZE);
         return allocf(a, n, align);
      } else {
         let p = &a.r.mem[a.r.idx];
         a.r.idx += n;
         return p;
      }
   }

   fn destroy(a *Arena) void {
      for let r = a.r, next *ArenaRegion = #null; r; r = next {
         next = r.prev;
         free(r);
      }
   }
}

struct Allocator {
   a *void,
   allocf *fn(*void, usize, usize) *void,
   freef *fn(*void, *void) void,

   fn alloc(self *Allocator, n usize, align usize) *void {
      if self.allocf {
         let p = self.allocf(self.a, n, align);
         return p;
      }
      return #null;
   }

   fn free(self *Allocator, ptr *void) void {
      if self.freef {
         self.freef(self.a, ptr);
      }
   }
}

defmacro anew(a, T) [ ((a)->alloc(sizeof T, alignof T)) ]