1 #include <stdlib.h>
2 #include <stdint.h>
3 #include <errno.h>
4 #include "libc.h"
5 #include "pthread_impl.h"
6 #include "malloc_impl.h"
7
lock(volatile int * lk)8 static inline void lock(volatile int *lk)
9 {
10 if (libc.threads_minus_1)
11 while(a_swap(lk, 1)) __wait(lk, lk+1, 1, 1);
12 }
13
unlock(volatile int * lk)14 static inline void unlock(volatile int *lk)
15 {
16 if (lk[0]) {
17 a_store(lk, 0);
18 if (lk[1]) __wake(lk, 1, 1);
19 }
20 }
21
__memalign(size_t align,size_t len)22 void *__memalign(size_t align, size_t len)
23 {
24 unsigned char *mem, *new;
25
26 if ((align & -align) != align) {
27 errno = EINVAL;
28 return 0;
29 }
30
31 if (len > SIZE_MAX - align || __malloc_replaced) {
32 errno = ENOMEM;
33 return 0;
34 }
35
36 if (align <= SIZE_ALIGN)
37 return malloc(len);
38
39 if (!(mem = malloc(len + align-1)))
40 return 0;
41
42 new = (void *)((uintptr_t)mem + align-1 & -align);
43 if (new == mem) return mem;
44
45 struct chunk *c = MEM_TO_CHUNK(mem);
46 struct chunk *n = MEM_TO_CHUNK(new);
47
48 if (g_enable_check) {
49 int status = delete_node(mem);
50 if (status != 0) {
51 get_free_trace(mem);
52 a_crash();
53 }
54 }
55
56 if (IS_MMAPPED(c)) {
57 /* Apply difference between aligned and original
58 * address to the "extra" field of mmapped chunk. */
59 n->psize = c->psize + (new-mem);
60 n->csize = c->csize - (new-mem);
61 if (g_enable_check) {
62 insert_node(CHUNK_TO_MEM(n), CHUNK_SIZE(n));
63 }
64 return new;
65 }
66
67 struct chunk *t = NEXT_CHUNK(c);
68
69 /* Split the allocated chunk into two chunks. The aligned part
70 * that will be used has the size in its footer reduced by the
71 * difference between the aligned and original addresses, and
72 * the resulting size copied to its header. A new header and
73 * footer are written for the split-off part to be freed. */
74 lock(g_mem_lock);
75 n->psize = c->csize = C_INUSE | (new-mem);
76 n->csize = t->psize -= new-mem;
77 calculate_checksum(c, n);
78 calculate_checksum(NULL, t);
79 unlock(g_mem_lock);
80 if (g_enable_check) {
81 insert_node(CHUNK_TO_MEM(n), CHUNK_SIZE(n));
82 }
83
84 __bin_chunk(c);
85 return new;
86 }
87
88 weak_alias(__memalign, memalign);
89