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
|
#include "scripting.h"
void handle_tcc_error(void* opaque, const char* msg)
{
TraceLog(LOG_ERROR, msg);
}
void initScripting(SldjScripting* scripting)
{
*scripting = (SldjScripting){
.state = NULL,
.lineScanner = NULL,
.fileBuf = NULL,
.fileSize = 0
};
}
void compileScript(SldjScripting* scripting, const char filePath[SLDJ_NAMEMAX])
{
// Get file size.
struct stat fileState;
if (stat(filePath, &fileState) < 0)
{
TraceLog(LOG_ERROR, strerror(errno));
return;
}
scripting->fileSize = fileState.st_size;
// Load file information into a buffer.
if (scripting->fileBuf == NULL)
{
scripting->fileBuf = (char*)malloc(scripting->fileSize);
} else {
scripting->fileBuf = (char*)realloc(scripting->fileBuf, scripting->fileSize);
}
if (scripting->fileBuf == NULL)
{
TraceLog(LOG_ERROR, strerror(errno));
return;
}
FILE* file = fopen(filePath, "r");
if (file == NULL)
{
TraceLog(LOG_ERROR, strerror(errno));
return;
}
fread(scripting->fileBuf, scripting->fileSize, 1, file);
scripting->fileBuf[scripting->fileSize - 1] = '\0';
fclose(file);
// Setup compiler.
if (scripting->state != NULL)
{
tcc_delete(scripting->state);
}
scripting->state = tcc_new();
if (scripting->state == NULL)
{
TraceLog(LOG_ERROR, "Could not open tcc state");
return;
}
tcc_set_error_func(scripting->state, stderr, handle_tcc_error);
tcc_set_output_type(scripting->state, TCC_OUTPUT_MEMORY);
// Compile.
if (tcc_compile_string(scripting->state, scripting->fileBuf) < 0)
{
return;
}
if (tcc_relocate(scripting->state) < 0)
{
return;
}
scripting->lineScanner = tcc_get_symbol(scripting->state, "lineScanner");
}
void closeScripting(SldjScripting* scripting)
{
if (scripting->state != NULL)
{
tcc_delete(scripting->state);
}
if (scripting->fileBuf != NULL)
{
free(scripting->fileBuf);
}
}
|