-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathutil.c
157 lines (151 loc) · 3.97 KB
/
util.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
#include "citrine.h"
char* np;
int errstack;
/**
* Determines the size of the specified file.
*/
int fsize(char* filename) {
int size;
FILE* fh;
fh = fopen(filename, "rb");
if(fh != NULL){
if( fseek(fh, 0, SEEK_END) ){
fclose(fh);
return -1;
}
size = ftell(fh);
fclose(fh);
return size;
}
return -1;
}
/**
* Export AST
*/
void ctr_internal_export_tree(ctr_tnode* ti) {
ctr_size i;
ctr_tlistitem* li;
ctr_tnode* t;
li = ti->nodes;
t = li->node;
while(1) {
printf( "%d;%d;", t->type, t->modifier );
if (t->value != NULL) {
printf("%lu;", (unsigned long) t->vlen);
for(i = 0; i < t->vlen; i++) {
putchar( t->value[i] );
}
} else {
printf("0;");
}
putchar(';');
if (t->nodes) {
putchar('[');
ctr_internal_export_tree(t);
putchar(']');
}
putchar(';');
if (!li->next) break;
li = li->next;
t = li->node;
}
}
/**
* @internal
* Internal plugin loader.
* Tries to locate a plugin by its name.
*
* In Citrine a plugin is loaded automatically as soon as a symbol
* has been enountered that cannot be found in global scope.
*
* Before raising an error Citrine will attempt to load a plugin
* to meet the dependency.
*
* To install a plugin, copy the plugin folder to the mods folder
* in the working directory of your script.
*
* So, for instance, to install the 'Coffee Percolator Plugin',
* we copy the libctrpercolator.so in folder mods/percolator/,
* the resulting path is: mods/percolator/libctrpercolator.so.
*
* General path format:
*
* mods/X/libctrX.so
*
* where X is the name of the object the plugin offers in lowercase.
*
* On loading, the plugin will get a chance to add its objects to the world
* through a constructor function.
*/
#ifdef REPLACE_PLUGIN_SYSTEM
#else
typedef void* (*plugin_init_func)();
void* ctr_internal_plugin_find(ctr_object* key) {
ctr_object* modNameObject = ctr_internal_cast2string(key);
void* handle;
char pathNameMod[1024];
char* modName;
char* modNameLow;
plugin_init_func init_plugin;
char* realPathModName = NULL;
modName = ctr_heap_allocate_cstring( modNameObject );
modNameLow = modName;
for ( ; *modNameLow; ++modNameLow) *modNameLow = tolower(*modNameLow);
snprintf(pathNameMod, 1024, "mods/%s/libctr%s.so", modName, modName);
ctr_heap_free( modName );
realPathModName = realpath(pathNameMod, NULL);
if (access(realPathModName, F_OK) == -1) return NULL;
handle = dlopen(realPathModName, RTLD_NOW);
free(realPathModName);
if ( !handle ) {
printf("%s\n",CTR_ERR_FOPEN);
printf("%s\n",dlerror());
exit(1);
}
/* the begin() function will add the missing object to the world */
*(void**)(&init_plugin) = dlsym( handle, "begin" );
if ( !init_plugin ) {
printf("%s\n",CTR_ERR_FOPEN);
printf("%s\n",dlerror());
exit(1);
}
(void) init_plugin();
return handle;
}
#endif
/**
* @internal
*
* Causes the program flow to switch to error mode.
* This function takes a pointer to a message string and
* an integer error code (errno). It will create a string object
* containing the specified message and the integrated error message
* derived from the specified error code. This error string will
* be assigned to the CtrStdFlow variable which will cause
* program execution to go into error mode, i.e. all remaining
* instructions will be ignored until either the error is dealt with
* by a catch clause or the program ends in which case an error will
* produced to stderr.
*/
ctr_object* ctr_error( char* message, int error_code ) {
char* errstr;
errstr = ctr_heap_allocate( sizeof(char) * 200 );
snprintf( errstr, 200, message, strerror( error_code ) );
CtrStdFlow = ctr_build_string_from_cstring( errstr );
ctr_heap_free( errstr );
CtrStdFlow->info.sticky = 1;
errstack = 0;
return CtrStdFlow;
}
/**
* @internal
*
* Prints a message to the error stream.
*/
void ctr_print_error( char* error, int code ) {
fwrite( error, sizeof(char), strlen(error), stderr );
fwrite( "\n", sizeof(char), 1, stderr );
if ( code > -1 ) {
exit(code);
}
}