blob: acee09d3ce65e7a2df702710bb2aa2b7404717c0 [file] [log] [blame]
Jack Palevich1cdef202009-05-22 12:06:27 -07001/*
2 * Android "Almost" C Compiler.
3 * This is a compiler for a small subset of the C language, intended for use
4 * in scripting environments where speed and memory footprint are important.
5 *
6 * This code is based upon the "unobfuscated" version of the
7 * Obfuscated Tiny C compiler, see the file LICENSE for details.
8 *
9 */
10
11#include <ctype.h>
12#include <dlfcn.h>
13#include <stdarg.h>
14#include <stdint.h>
15#include <stdio.h>
16#include <stdlib.h>
17#include <string.h>
18
19#if defined(__arm__)
20#include <unistd.h>
21#endif
22
23#include <acc/acc.h>
24
25
26typedef int (*MainPtr)(int, char**);
27// This is a separate function so it can easily be set by breakpoint in gdb.
28int run(MainPtr mainFunc, int argc, char** argv) {
29 return mainFunc(argc, argv);
30}
31
32int main(int argc, char** argv) {
33 const char* inFile = NULL;
34 bool printListing;
35 FILE* in = stdin;
36 int i;
37 for (i = 1; i < argc; i++) {
38 char* arg = argv[i];
39 if (arg[0] == '-') {
40 switch (arg[1]) {
41 case 'S':
42 printListing = true;
43 break;
44 default:
45 fprintf(stderr, "Unrecognized flag %s\n", arg);
46 return 3;
47 }
48 } else if (inFile == NULL) {
49 inFile = arg;
50 } else {
51 break;
52 }
53 }
54
55 if (! inFile) {
56 fprintf(stderr, "input file required\n");
57 return 2;
58 }
59
60 if (inFile) {
61 in = fopen(inFile, "r");
62 if (!in) {
63 fprintf(stderr, "Could not open input file %s\n", inFile);
64 return 1;
65 }
66 }
67
68 fseek(in, 0, SEEK_END);
69 size_t fileSize = (size_t) ftell(in);
70 rewind(in);
Jack Palevichb7c81e92009-06-04 19:56:13 -070071 ACCchar* text = new ACCchar[fileSize + 1];
Jack Palevich1cdef202009-05-22 12:06:27 -070072 size_t bytesRead = fread(text, 1, fileSize, in);
73 if (bytesRead != fileSize) {
74 fprintf(stderr, "Could not read all of file %s\n", inFile);
75 }
76
Jack Palevichb7c81e92009-06-04 19:56:13 -070077 text[fileSize] = '\0';
78
Jack Palevich1cdef202009-05-22 12:06:27 -070079 ACCscript* script = accCreateScript();
80
81 const ACCchar* scriptSource[] = {text};
82 accScriptSource(script, 1, scriptSource, NULL);
83 delete[] text;
84
85 accCompileScript(script);
Jack Palevichac0e95e2009-05-29 13:53:44 -070086 int result = accGetError(script);
Jack Palevich1cdef202009-05-22 12:06:27 -070087 MainPtr mainPointer = 0;
Jack Palevichac0e95e2009-05-29 13:53:44 -070088 if (result != 0) {
89 char buf[1024];
90 accGetScriptInfoLog(script, sizeof(buf), NULL, buf);
Jack Palevicheedf9d22009-06-04 16:23:40 -070091 fprintf(stderr, "%s", buf);
Jack Palevichac0e95e2009-05-29 13:53:44 -070092 goto exit;
93 }
Jack Palevich1cdef202009-05-22 12:06:27 -070094
Jack Palevicheedf9d22009-06-04 16:23:40 -070095 {
96 ACCsizei numPragmaStrings;
97 accGetPragmas(script, &numPragmaStrings, 0, NULL);
98 if (numPragmaStrings) {
99 char** strings = new char*[numPragmaStrings];
100 accGetPragmas(script, NULL, numPragmaStrings, strings);
101 for(ACCsizei i = 0; i < numPragmaStrings; i += 2) {
102 fprintf(stderr, "#pragma %s(%s)\n", strings[i], strings[i+1]);
103 }
104 delete[] strings;
105 }
106 }
107
Jack Palevich1cdef202009-05-22 12:06:27 -0700108 accGetScriptLabel(script, "main", (ACCvoid**) & mainPointer);
109
Jack Palevichac0e95e2009-05-29 13:53:44 -0700110 result = accGetError(script);
Jack Palevich1cdef202009-05-22 12:06:27 -0700111 if (result == ACC_NO_ERROR) {
112 fprintf(stderr, "Executing compiled code:\n");
113 int codeArgc = argc - i + 1;
114 char** codeArgv = argv + i - 1;
115 codeArgv[0] = (char*) (inFile ? inFile : "stdin");
116 result = run(mainPointer, codeArgc, codeArgv);
117 fprintf(stderr, "result: %d\n", result);
118 }
119
Jack Palevichac0e95e2009-05-29 13:53:44 -0700120exit:
121
Jack Palevich1cdef202009-05-22 12:06:27 -0700122 accDeleteScript(script);
123
124 return result;
125}