1 | /* Copyright (c) 2004, 2010, Oracle and/or its affiliates. |
2 | Copyright (c) 2012, 2014, Monty Program Ab |
3 | |
4 | This program is free software; you can redistribute it and/or modify |
5 | it under the terms of the GNU General Public License as published by |
6 | the Free Software Foundation; version 2 of the License. |
7 | |
8 | This program is distributed in the hope that it will be useful, |
9 | but WITHOUT ANY WARRANTY; without even the implied warranty of |
10 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
11 | GNU General Public License for more details. |
12 | |
13 | You should have received a copy of the GNU General Public License |
14 | along with this program; if not, write to the Free Software |
15 | Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ |
16 | |
17 | /* |
18 | Originally written by Magnus Svensson |
19 | */ |
20 | |
21 | /* |
22 | Converts a SQL file into a C file that can be compiled and linked |
23 | into other programs |
24 | */ |
25 | |
26 | #include <my_config.h> |
27 | #include <stdarg.h> |
28 | #include <stdlib.h> |
29 | #include <stdio.h> |
30 | |
31 | /* |
32 | This is an internal tool used during the build process only, |
33 | - do not make a library just for this, |
34 | which would make the Makefiles and the server link |
35 | more complex than necessary, |
36 | - do not duplicate the code either. |
37 | so just add the sql_bootstrap.cc code as is. |
38 | */ |
39 | #include "../sql/sql_bootstrap.cc" |
40 | |
41 | FILE *in, *out; |
42 | |
43 | static void die(const char *fmt, ...) |
44 | { |
45 | va_list args; |
46 | |
47 | /* Print the error message */ |
48 | fprintf(stderr, "FATAL ERROR: " ); |
49 | if (fmt) |
50 | { |
51 | va_start(args, fmt); |
52 | vfprintf(stderr, fmt, args); |
53 | va_end(args); |
54 | } |
55 | else |
56 | fprintf(stderr, "unknown error" ); |
57 | fprintf(stderr, "\n" ); |
58 | fflush(stderr); |
59 | |
60 | /* Close any open files */ |
61 | if (in) |
62 | fclose(in); |
63 | if (out) |
64 | fclose(out); |
65 | |
66 | exit(1); |
67 | } |
68 | |
69 | char *fgets_fn(char *buffer, size_t size, fgets_input_t input, int *error) |
70 | { |
71 | char *line= fgets(buffer, (int)size, (FILE*) input); |
72 | if (error) |
73 | *error= (line == NULL) ? ferror((FILE*)input) : 0; |
74 | return line; |
75 | } |
76 | |
77 | static void print_query(FILE *out, const char *query) |
78 | { |
79 | const char *ptr= query; |
80 | int column= 0; |
81 | |
82 | fprintf(out, "\"" ); |
83 | while (*ptr) |
84 | { |
85 | switch(*ptr) |
86 | { |
87 | case '\n': |
88 | /* |
89 | Preserve the \n character in the query text, |
90 | and wrap to the next line, tabulated. |
91 | */ |
92 | fprintf(out, "\\n\"\n \"" ); |
93 | column= 2; |
94 | break; |
95 | case '\r': |
96 | /* Skipped */ |
97 | break; |
98 | case '\"': |
99 | fprintf(out, "\\\"" ); |
100 | column++; |
101 | break; |
102 | case '\\': |
103 | fprintf(out, "\\\\" ); |
104 | break; |
105 | default: |
106 | putc(*ptr, out); |
107 | column++; |
108 | break; |
109 | } |
110 | ptr++; |
111 | } |
112 | fprintf(out, "\\n\",\n" ); |
113 | } |
114 | |
115 | int main(int argc, char *argv[]) |
116 | { |
117 | char query[MAX_BOOTSTRAP_QUERY_SIZE]; |
118 | char* struct_name= argv[1]; |
119 | char* infile_name= argv[2]; |
120 | char* outfile_name= argv[3]; |
121 | int rc; |
122 | int query_length; |
123 | int error= 0; |
124 | char *err_ptr; |
125 | |
126 | if (argc != 4) |
127 | die("Usage: comp_sql <struct_name> <sql_filename> <c_filename>" ); |
128 | |
129 | /* Open input and output file */ |
130 | if (!(in= fopen(infile_name, "r" ))) |
131 | die("Failed to open SQL file '%s'" , infile_name); |
132 | if (!(out= fopen(outfile_name, "w" ))) |
133 | die("Failed to open output file '%s'" , outfile_name); |
134 | |
135 | fprintf(out, "/*\n" ); |
136 | fprintf(out, " Do not edit this file, it is automatically generated from:\n" ); |
137 | fprintf(out, " <%s>\n" , infile_name); |
138 | fprintf(out, "*/\n" ); |
139 | fprintf(out, "#include <stdlib.h>\n" ); /* NULL */ |
140 | fprintf(out, "const char* %s[]={\n" , struct_name); |
141 | |
142 | for ( ; ; ) |
143 | { |
144 | rc= read_bootstrap_query(query, &query_length, |
145 | (fgets_input_t) in, fgets_fn, &error); |
146 | |
147 | if (rc == READ_BOOTSTRAP_EOF) |
148 | break; |
149 | |
150 | if (rc != READ_BOOTSTRAP_SUCCESS) |
151 | { |
152 | /* Get the most recent query text for reference. */ |
153 | err_ptr= query + (query_length <= MAX_BOOTSTRAP_ERROR_LEN ? |
154 | 0 : (query_length - MAX_BOOTSTRAP_ERROR_LEN)); |
155 | switch (rc) |
156 | { |
157 | case READ_BOOTSTRAP_ERROR: |
158 | die("Failed to read the bootstrap input file. Return code (%d).\n" |
159 | "Last query: '%s'\n" , error, err_ptr); |
160 | break; |
161 | |
162 | case READ_BOOTSTRAP_QUERY_SIZE: |
163 | die("Failed to read the boostrap input file. Query size exceeded %d bytes.\n" |
164 | "Last query: '%s'.\n" , MAX_BOOTSTRAP_LINE_SIZE, err_ptr); |
165 | break; |
166 | |
167 | default: |
168 | die("Failed to read the boostrap input file. Unknown error.\n" ); |
169 | break; |
170 | } |
171 | } |
172 | |
173 | print_query(out, query); |
174 | } |
175 | |
176 | fprintf(out, "NULL\n};\n" ); |
177 | |
178 | fclose(in); |
179 | fclose(out); |
180 | |
181 | exit(0); |
182 | |
183 | } |
184 | |
185 | |