Alien-TinyCCx
view release on metacpan or search on metacpan
src/tests/exsymtab/35-two-contexts-vtable-share.c view on Meta::CPAN
#define INCLUDE_MALLOC
#include "test_setup.h"
char first_code[] =
"void * malloc(int);\n"
"void free(void *);\n"
"struct point {\n"
" int (*squared_distance)(struct point * self);\n"
" int x;\n"
" int y;\n"
"};\n"
"int point_squared_distance(struct point * self) {\n"
" return self->x * self->x + self->y * self->y;\n"
"}\n"
"void * new_point(int x, int y) {\n"
" struct point * pt = malloc(sizeof(struct point));\n"
" pt->squared_distance = point_squared_distance;\n"
" pt->x = x;\n"
" pt->y = y;\n"
" return pt;\n"
"}\n"
;
char second_code[] =
"void * distance_func_ptr(struct point * pt) {\n"
" return pt->squared_distance;\n"
//" return pt->x;\n"
"}\n"
"int sq_distance_to_pt(struct point * pt) {\n"
" return pt->squared_distance(pt);\n"
//" return pt->x;\n"
"}\n"
;
int main(int argc, char **argv)
{
/* ---- Compile the first code string and setup the callback data ---- */
TCCState *s1 = tcc_new();
extended_symtab_p my_symtab;
setup_and_compile_s1(my_symtab, first_code);
SETUP_SECOND_CALLBACK_DATA();
/* ---- Allocate a point ---- */
void* (*allocate_ptr)(int, int) = tcc_get_symbol(s1, "new_point");
if (allocate_ptr == NULL) return 1;
void * point_p = allocate_ptr(6, 9);
if (point_p == NULL) {
fail("Unable to allocate point");
return 1;
}
else
pass("Allocated point");
/* ---- Check function address ---- */
int (*squared_distance_func)(void*) = tcc_get_symbol(s1, "point_squared_distance");
void * func_p = ((void **)point_p)[0];
is_p(func_p, squared_distance_func, "Symbol table's function points to relocated location");
/* ---- Make sure the function works correctly ---- */
is_i(squared_distance_func(point_p), 9*9+6*6, "Vtable function works correctly");
/* ---- Compile the second compiler context ---- */
if (func_p == squared_distance_func) {
TCCState * s_second = tcc_new();
setup_and_relocate_second_state(s_second, second_code);
/* Test pointer extraction method */
void* (*second_extraction_func)(void*) = tcc_get_symbol(s_second, "distance_func_ptr");
if (second_extraction_func == NULL) return 1;
is_p(second_extraction_func(point_p), squared_distance_func,
"address in vtable extracted in second context is correct");
int (*sq_dist_ptr)(void*) = tcc_get_symbol(s_second, "sq_distance_to_pt");
if (sq_dist_ptr == NULL) return 1;
is_i(sq_dist_ptr(point_p), 9*9+6*6, "Second context able to call function pointer from struct");
tcc_delete(s_second);
}
/* ---- clean up the memory ---- */
tcc_delete_extended_symbol_table(my_symtab);
tcc_delete(s1);
free(point_p);
pass("cleanup");
return done_testing();
}
( run in 0.541 second using v1.01-cache-2.11-cpan-71847e10f99 )