-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbindings2.c
226 lines (190 loc) · 5.65 KB
/
bindings2.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
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
/**
* Second try with GC to subsetphp
* Using semi-space copying collector
*
* @since 2015-10-11
*/
#include <stdio.h>
#include "php-src/Zend/zend.h"
#include <zend_types.h>
#include <zend_string.h>
#include <math.h>
#include "bindings2.h"
#include "semigc/alloc.h"
static int nr_of_free = 0;
extern double printd(double x) {
printf("%.9f\n", x);
return 0;
}
extern double prints(zend_string* str) {
printf("%s\n", str->val);
return 0;
}
// Not needed
extern double subsetphp_sqrt(double x) {
return sqrt(x);
}
/**
* Keep track of struct types, so the
* GC knows what pointers to follow.
*/
typedef struct _structtype {
int type_nr;
int nr_of_pointers; //
int* pointer_offsetts;
struct _structtype* next;
} structtype;
extern structtype* structtypes;
/* String functions */
/**
* Allocate memory for a string of length len
*
* @param size_t len
* @param int persistent Rest from Zend
* @return value
*/
extern zend_string* subsetphp_string_alloc(size_t len, int persistent)
{
//#define pemalloc(size, persistent) ((persistent)?__zend_malloc(size):emalloc(size))
/*
zend_always_inline static void * __zend_malloc(size_t len)
{
void *tmp = malloc(len);
if (tmp) {
return tmp;
}
fprintf(stderr, "Out of memory\n");
exit(1);
}
*/
// TODO: How to allocate this memory so that OCaml GC can collect it?
//zend_string *str = (zend_string *)pemalloc(ZEND_MM_ALIGNED_SIZE(_STR_HEADER_SIZE + len + 1), persistent);
// Cast from value to zend_string* ?
//zend_string *str = (zend_string*) caml_alloc(ZEND_MM_ALIGNED_SIZE(_STR_HEADER_SIZE + len + 1), 0);
zend_string *str = llvm_gc_allocate(ZEND_MM_ALIGNED_SIZE(_STR_HEADER_SIZE + len + 1));
//printf("sizeof(str1) = %d\n", sizeof(*str));
//printf("ZEND_MM_ALIGNED_SIZE(_STR_HEADER_SIZE etc) = %d\n", ZEND_MM_ALIGNED_SIZE(_STR_HEADER_SIZE + len + 1));
//zend_string *str = (zend_string *)caml_alloc((len + 1), 0);
GC_REFCOUNT(str) = 0;
#if 1
/* optimized single assignment */
GC_TYPE_INFO(str) = IS_STRING | ((persistent ? IS_STR_PERSISTENT : 0) << 8);
#else
GC_TYPE(ret) = IS_STRING;
GC_FLAGS(ret) = (persistent ? IS_STR_PERSISTENT : 0);
GC_INFO(ret) = 0;
#endif
str->h = 0;
str->len = len;
//value v = caml_alloc_custom(&subsetphp_zend_string, sizeof(zend_string *), 1, 1000);
return str;
}
/**
* Returns Val_unit if realloc happened
* Otherwise the pointer to the new value
*/
/*
extern value subsetphp_string_realloc(value v1, zend_string *str, size_t len) {
//printf("len = %d\n", len);
zend_string *new_str = realloc(str, ZEND_MM_ALIGNED_SIZE(_STR_HEADER_SIZE + len + 1));
if (new_str == str) {
// realloc happened, pointer not moved, using same value
return NULL;
}
else {
// no realloc, pointer moved
// str need to be freed
//free(str);
//v = caml_alloc_custom(&subsetphp_zend_string, sizeof(zend_string *), 1, 16);
GC_REFCOUNT(new_str) = 1;
GC_TYPE_INFO(new_str) = IS_STRING | ((1 ? IS_STR_PERSISTENT : 0) << 8);
new_str->h = 0;
new_str->len = len;
return v1;
}
}
*/
/**
* All alloc functions must be changed to play nicely with
* the OCaml GC
*
* @return value
*/
extern zend_string* subsetphp_string_init(const char *str, size_t len, int persistent)
{
zend_string* ret = subsetphp_string_alloc(len, persistent);
memcpy(ret->val, str, len);
ret->val[len] = '\0';
return ret;
}
/**
* Contcat two zend string and store in result
* Original function in zend_operators.c
*
* @return int
*/
extern zend_string* subsetphp_concat_function(zend_string* str1, zend_string* str2)
{
size_t str1_len = str1->len;
size_t str2_len = str2->len;
size_t result_len = str1_len + str2_len;
//printf("str1->len = %d, ", str1_len);
//printf("str2->len = %d\n", str2_len);
zend_string *zend_result = subsetphp_string_alloc(result_len, 1);
//result = subsetphp_string_init("", result_len, 1);
// TODO: What to do here?
if (str1_len > SIZE_MAX - str2_len) {
printf("subsetphp_concat_function: str1_len > SIZE_MAX - str2_len\n");
exit(1);
//zend_error_noreturn(E_ERROR, "String size overflow");
}
//printf("str1->len = %d, ", str1->len);
//printf("strlen(str1->val) = %d\n", strlen(str1->val));
memcpy(zend_result->val, str1->val, str1_len);
memcpy(zend_result->val + str1_len, str2->val, str2_len);
zend_result->len = result_len;
zend_result->val[result_len] = '\0';
return zend_result;
//printf("result = %s\n", zend_result->val);
}
/**
* Init the OCaml GC
*/
extern void subsetphp_gc_init() {
llvm_gc_initialize(100000);
}
extern void subsetphp_add_new_structtype(zend_string* name, int lenght, int* pointer_offsets) {
}
/**
* Implementation of dynamic size array
* Copied from http://stackoverflow.com/questions/3536153/c-dynamically-growing-array
*/
typedef struct {
/* Type header for GC info, 64 bit. Should never
* be accessed manually outside of GC. */
int32_t _i;
int32_t _j;
int8_t *array; /* i8_ptr used as pointer in LLVM */
int32_t used; /* size_t = i32_t */
int32_t size;
} array;
extern void initArray(array *a, size_t initialSize) {
//struct StackEntry *entry = llvm_gc_root_chain;
// a->array needs to be put on the shadow stack
// easier to just generate this code in llvm? or code it in llvm with 'shadow-stack' on?
a->array = (int8_t *)llvm_gc_allocate(initialSize * sizeof(int32_t));
a->used = 0;
a->size = initialSize;
}
extern void insertArray(array *a, int32_t element) {
if (a->used == a->size) {
a->size *= 2;
a->array = (int8_t *)llvm_gc_allocate(a->size * sizeof(int32_t));
}
a->array[a->used++] = element;
}
extern void freeArray(array *a) {
//free(a->array); // GC will free the array
a->array = NULL;
a->used = a->size = 0;
}