-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnstr.c
69 lines (60 loc) · 1.8 KB
/
nstr.c
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
#include "nstr.h"
#include <unistd.h>
#include <string.h>
#include <sys/mman.h>
#define ALIGNUP(addr, align) (((addr)+((align)-1))&-(align))
static size_t page_size;
void
nstr_init(struct nstr_block *block)
{
page_size = (size_t)sysconf(_SC_PAGESIZE);
block->size = page_size;
block->data = mmap(NULL, (size_t)block->size, PROT_READ|PROT_WRITE, MAP_ANONYMOUS|MAP_PRIVATE, -1, 0);
block->offset = 0;
#ifdef NSTR_STATS
block->waste = 0;
block->total = 0;
#endif
}
struct nstr *
nstr_alloc(struct nstr_block *block, nstrlen n)
{
struct nstr *result;
size_t size = sizeof(struct nstr) + ALIGNUP(n, sizeof(struct nstr));
if (size > block->size - block->offset) {
#ifdef NSTR_STATS
block->waste += block->size - block->offset;
#endif
block->size = ALIGNUP(size, page_size);
block->data = mmap(NULL, block->size, PROT_READ|PROT_WRITE, MAP_ANONYMOUS|MAP_PRIVATE, -1, 0);
block->offset = 0;
}
result = (struct nstr *)__builtin_assume_aligned(block->data + block->offset, sizeof(struct nstr));
block->offset += size;
#ifdef NSTR_STATS
block->total += size;
block->waste += n % sizeof(struct nstr);
#endif
return result;
}
struct nstr *
nstr_dup(struct nstr_block *block, nstrlen n, char *str)
{
struct nstr *ns = nstr_alloc(block, n);
memcpy(&ns->str, str, n);
ns->n = (uint16_t)n;
return ns;
}
#ifdef NSTR_STATS
#include <stdio.h>
void
nstr_print_stats(struct nstr_block *block)
{
size_t alloced = block->total + block->size - block->offset;
size_t real_waste = block->waste + block->size - block->offset;
float waste_percent = (100.0f*real_waste)/alloced;
fprintf(stderr, "nstr stats: s=%lu a=%lu t=%lu w=%lu (%f%%)\n",
block->size, alloced, block->total,
real_waste, waste_percent);
}
#endif