xref: /aosp_15_r20/external/bc/gen/strgen.c (revision 5a6e848804d15c18a0125914844ee4eb0bda4fcf)
1*5a6e8488SAndroid Build Coastguard Worker /*
2*5a6e8488SAndroid Build Coastguard Worker  * *****************************************************************************
3*5a6e8488SAndroid Build Coastguard Worker  *
4*5a6e8488SAndroid Build Coastguard Worker  * SPDX-License-Identifier: BSD-2-Clause
5*5a6e8488SAndroid Build Coastguard Worker  *
6*5a6e8488SAndroid Build Coastguard Worker  * Copyright (c) 2018-2024 Gavin D. Howard and contributors.
7*5a6e8488SAndroid Build Coastguard Worker  *
8*5a6e8488SAndroid Build Coastguard Worker  * Redistribution and use in source and binary forms, with or without
9*5a6e8488SAndroid Build Coastguard Worker  * modification, are permitted provided that the following conditions are met:
10*5a6e8488SAndroid Build Coastguard Worker  *
11*5a6e8488SAndroid Build Coastguard Worker  * * Redistributions of source code must retain the above copyright notice, this
12*5a6e8488SAndroid Build Coastguard Worker  *   list of conditions and the following disclaimer.
13*5a6e8488SAndroid Build Coastguard Worker  *
14*5a6e8488SAndroid Build Coastguard Worker  * * Redistributions in binary form must reproduce the above copyright notice,
15*5a6e8488SAndroid Build Coastguard Worker  *   this list of conditions and the following disclaimer in the documentation
16*5a6e8488SAndroid Build Coastguard Worker  *   and/or other materials provided with the distribution.
17*5a6e8488SAndroid Build Coastguard Worker  *
18*5a6e8488SAndroid Build Coastguard Worker  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19*5a6e8488SAndroid Build Coastguard Worker  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20*5a6e8488SAndroid Build Coastguard Worker  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21*5a6e8488SAndroid Build Coastguard Worker  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
22*5a6e8488SAndroid Build Coastguard Worker  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23*5a6e8488SAndroid Build Coastguard Worker  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24*5a6e8488SAndroid Build Coastguard Worker  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25*5a6e8488SAndroid Build Coastguard Worker  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26*5a6e8488SAndroid Build Coastguard Worker  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27*5a6e8488SAndroid Build Coastguard Worker  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28*5a6e8488SAndroid Build Coastguard Worker  * POSSIBILITY OF SUCH DAMAGE.
29*5a6e8488SAndroid Build Coastguard Worker  *
30*5a6e8488SAndroid Build Coastguard Worker  * *****************************************************************************
31*5a6e8488SAndroid Build Coastguard Worker  *
32*5a6e8488SAndroid Build Coastguard Worker  * Generates a const array from a bc script.
33*5a6e8488SAndroid Build Coastguard Worker  *
34*5a6e8488SAndroid Build Coastguard Worker  */
35*5a6e8488SAndroid Build Coastguard Worker 
36*5a6e8488SAndroid Build Coastguard Worker #include <assert.h>
37*5a6e8488SAndroid Build Coastguard Worker #include <stdbool.h>
38*5a6e8488SAndroid Build Coastguard Worker #include <stdio.h>
39*5a6e8488SAndroid Build Coastguard Worker #include <stdlib.h>
40*5a6e8488SAndroid Build Coastguard Worker #include <string.h>
41*5a6e8488SAndroid Build Coastguard Worker 
42*5a6e8488SAndroid Build Coastguard Worker #include <errno.h>
43*5a6e8488SAndroid Build Coastguard Worker 
44*5a6e8488SAndroid Build Coastguard Worker #include <fcntl.h>
45*5a6e8488SAndroid Build Coastguard Worker #include <sys/stat.h>
46*5a6e8488SAndroid Build Coastguard Worker 
47*5a6e8488SAndroid Build Coastguard Worker #ifndef _WIN32
48*5a6e8488SAndroid Build Coastguard Worker #include <unistd.h>
49*5a6e8488SAndroid Build Coastguard Worker #endif // _WIN32
50*5a6e8488SAndroid Build Coastguard Worker 
51*5a6e8488SAndroid Build Coastguard Worker // For some reason, Windows can't have this header.
52*5a6e8488SAndroid Build Coastguard Worker #ifndef _WIN32
53*5a6e8488SAndroid Build Coastguard Worker #include <libgen.h>
54*5a6e8488SAndroid Build Coastguard Worker #endif // _WIN32
55*5a6e8488SAndroid Build Coastguard Worker 
56*5a6e8488SAndroid Build Coastguard Worker // This pulls in cross-platform stuff.
57*5a6e8488SAndroid Build Coastguard Worker #include <status.h>
58*5a6e8488SAndroid Build Coastguard Worker 
59*5a6e8488SAndroid Build Coastguard Worker // clang-format off
60*5a6e8488SAndroid Build Coastguard Worker 
61*5a6e8488SAndroid Build Coastguard Worker // The usage help.
62*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_usage =
63*5a6e8488SAndroid Build Coastguard Worker 	"usage: %s input output exclude name [label [define [remove_tabs]]]\n";
64*5a6e8488SAndroid Build Coastguard Worker 
65*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_ex_start = "{{ A H N HN }}";
66*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_ex_end = "{{ end }}";
67*5a6e8488SAndroid Build Coastguard Worker 
68*5a6e8488SAndroid Build Coastguard Worker // This is exactly what it looks like. It just slaps a simple license header on
69*5a6e8488SAndroid Build Coastguard Worker // the generated C source file.
70*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_header =
71*5a6e8488SAndroid Build Coastguard Worker 	"// Copyright (c) 2018-2024 Gavin D. Howard and contributors.\n"
72*5a6e8488SAndroid Build Coastguard Worker 	"// Licensed under the 2-clause BSD license.\n"
73*5a6e8488SAndroid Build Coastguard Worker 	"// *** AUTOMATICALLY GENERATED FROM %s. DO NOT MODIFY. ***\n\n";
74*5a6e8488SAndroid Build Coastguard Worker // clang-format on
75*5a6e8488SAndroid Build Coastguard Worker 
76*5a6e8488SAndroid Build Coastguard Worker // These are just format strings used to generate the C source.
77*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_label = "const char *%s = \"%s\";\n\n";
78*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_label_extern = "extern const char *%s;\n\n";
79*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_ifdef = "#if %s\n";
80*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_endif = "#endif // %s\n";
81*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_name = "const char %s[] = {\n";
82*5a6e8488SAndroid Build Coastguard Worker static const char* const bc_gen_name_extern = "extern const char %s[];\n\n";
83*5a6e8488SAndroid Build Coastguard Worker 
84*5a6e8488SAndroid Build Coastguard Worker // Error codes. We can't use 0 because these are used as exit statuses, and 0
85*5a6e8488SAndroid Build Coastguard Worker // as an exit status is not an error.
86*5a6e8488SAndroid Build Coastguard Worker #define IO_ERR (1)
87*5a6e8488SAndroid Build Coastguard Worker #define INVALID_INPUT_FILE (2)
88*5a6e8488SAndroid Build Coastguard Worker #define INVALID_PARAMS (3)
89*5a6e8488SAndroid Build Coastguard Worker 
90*5a6e8488SAndroid Build Coastguard Worker // This is the max width to print characters to the screen. This is to ensure
91*5a6e8488SAndroid Build Coastguard Worker // that lines don't go much over 80 characters.
92*5a6e8488SAndroid Build Coastguard Worker #define MAX_WIDTH (72)
93*5a6e8488SAndroid Build Coastguard Worker 
94*5a6e8488SAndroid Build Coastguard Worker /**
95*5a6e8488SAndroid Build Coastguard Worker  * Open a file. This function is to smooth over differences between POSIX and
96*5a6e8488SAndroid Build Coastguard Worker  * Windows.
97*5a6e8488SAndroid Build Coastguard Worker  * @param f         A pointer to the FILE pointer that will be initialized.
98*5a6e8488SAndroid Build Coastguard Worker  * @param filename  The name of the file.
99*5a6e8488SAndroid Build Coastguard Worker  * @param mode      The mode to open the file in.
100*5a6e8488SAndroid Build Coastguard Worker  */
101*5a6e8488SAndroid Build Coastguard Worker static void
open_file(FILE ** f,const char * filename,const char * mode)102*5a6e8488SAndroid Build Coastguard Worker open_file(FILE** f, const char* filename, const char* mode)
103*5a6e8488SAndroid Build Coastguard Worker {
104*5a6e8488SAndroid Build Coastguard Worker #ifndef _WIN32
105*5a6e8488SAndroid Build Coastguard Worker 
106*5a6e8488SAndroid Build Coastguard Worker 	*f = fopen(filename, mode);
107*5a6e8488SAndroid Build Coastguard Worker 
108*5a6e8488SAndroid Build Coastguard Worker #else // _WIN32
109*5a6e8488SAndroid Build Coastguard Worker 
110*5a6e8488SAndroid Build Coastguard Worker 	// We want the file pointer to be NULL on failure, but fopen_s() is not
111*5a6e8488SAndroid Build Coastguard Worker 	// guaranteed to set it.
112*5a6e8488SAndroid Build Coastguard Worker 	*f = NULL;
113*5a6e8488SAndroid Build Coastguard Worker 	fopen_s(f, filename, mode);
114*5a6e8488SAndroid Build Coastguard Worker 
115*5a6e8488SAndroid Build Coastguard Worker #endif // _WIN32
116*5a6e8488SAndroid Build Coastguard Worker }
117*5a6e8488SAndroid Build Coastguard Worker 
118*5a6e8488SAndroid Build Coastguard Worker /**
119*5a6e8488SAndroid Build Coastguard Worker  * A portability file open function. This is copied from src/read.c. Make sure
120*5a6e8488SAndroid Build Coastguard Worker  * to update that if this changes.
121*5a6e8488SAndroid Build Coastguard Worker  * @param path  The path to the file to open.
122*5a6e8488SAndroid Build Coastguard Worker  * @param mode  The mode to open in.
123*5a6e8488SAndroid Build Coastguard Worker  */
124*5a6e8488SAndroid Build Coastguard Worker static int
bc_read_open(const char * path,int mode)125*5a6e8488SAndroid Build Coastguard Worker bc_read_open(const char* path, int mode)
126*5a6e8488SAndroid Build Coastguard Worker {
127*5a6e8488SAndroid Build Coastguard Worker 	int fd;
128*5a6e8488SAndroid Build Coastguard Worker 
129*5a6e8488SAndroid Build Coastguard Worker #ifndef _WIN32
130*5a6e8488SAndroid Build Coastguard Worker 	fd = open(path, mode);
131*5a6e8488SAndroid Build Coastguard Worker #else // _WIN32
132*5a6e8488SAndroid Build Coastguard Worker 	fd = -1;
133*5a6e8488SAndroid Build Coastguard Worker 	open(&fd, path, mode);
134*5a6e8488SAndroid Build Coastguard Worker #endif
135*5a6e8488SAndroid Build Coastguard Worker 
136*5a6e8488SAndroid Build Coastguard Worker 	return fd;
137*5a6e8488SAndroid Build Coastguard Worker }
138*5a6e8488SAndroid Build Coastguard Worker 
139*5a6e8488SAndroid Build Coastguard Worker /**
140*5a6e8488SAndroid Build Coastguard Worker  * Reads a file and returns the file as a string. This has been copied from
141*5a6e8488SAndroid Build Coastguard Worker  * src/read.c. Make sure to change that if this changes.
142*5a6e8488SAndroid Build Coastguard Worker  * @param path  The path to the file.
143*5a6e8488SAndroid Build Coastguard Worker  * @return      The contents of the file as a string.
144*5a6e8488SAndroid Build Coastguard Worker  */
145*5a6e8488SAndroid Build Coastguard Worker static char*
bc_read_file(const char * path)146*5a6e8488SAndroid Build Coastguard Worker bc_read_file(const char* path)
147*5a6e8488SAndroid Build Coastguard Worker {
148*5a6e8488SAndroid Build Coastguard Worker 	int e = IO_ERR;
149*5a6e8488SAndroid Build Coastguard Worker 	size_t size, to_read;
150*5a6e8488SAndroid Build Coastguard Worker 	struct stat pstat;
151*5a6e8488SAndroid Build Coastguard Worker 	int fd;
152*5a6e8488SAndroid Build Coastguard Worker 	char* buf;
153*5a6e8488SAndroid Build Coastguard Worker 	char* buf2;
154*5a6e8488SAndroid Build Coastguard Worker 
155*5a6e8488SAndroid Build Coastguard Worker 	// This has been copied from src/read.c. Make sure to change that if this
156*5a6e8488SAndroid Build Coastguard Worker 	// changes.
157*5a6e8488SAndroid Build Coastguard Worker 
158*5a6e8488SAndroid Build Coastguard Worker 	assert(path != NULL);
159*5a6e8488SAndroid Build Coastguard Worker 
160*5a6e8488SAndroid Build Coastguard Worker #if BC_DEBUG
161*5a6e8488SAndroid Build Coastguard Worker 	// Need this to quiet MSan.
162*5a6e8488SAndroid Build Coastguard Worker 	// NOLINTNEXTLINE
163*5a6e8488SAndroid Build Coastguard Worker 	memset(&pstat, 0, sizeof(struct stat));
164*5a6e8488SAndroid Build Coastguard Worker #endif // BC_DEBUG
165*5a6e8488SAndroid Build Coastguard Worker 
166*5a6e8488SAndroid Build Coastguard Worker 	fd = bc_read_open(path, O_RDONLY);
167*5a6e8488SAndroid Build Coastguard Worker 
168*5a6e8488SAndroid Build Coastguard Worker 	// If we can't read a file, we just barf.
169*5a6e8488SAndroid Build Coastguard Worker 	if (BC_ERR(fd < 0))
170*5a6e8488SAndroid Build Coastguard Worker 	{
171*5a6e8488SAndroid Build Coastguard Worker 		fprintf(stderr, "Could not open file: %s\n", path);
172*5a6e8488SAndroid Build Coastguard Worker 		exit(INVALID_INPUT_FILE);
173*5a6e8488SAndroid Build Coastguard Worker 	}
174*5a6e8488SAndroid Build Coastguard Worker 
175*5a6e8488SAndroid Build Coastguard Worker 	// The reason we call fstat is to eliminate TOCTOU race conditions. This
176*5a6e8488SAndroid Build Coastguard Worker 	// way, we have an open file, so it's not going anywhere.
177*5a6e8488SAndroid Build Coastguard Worker 	if (BC_ERR(fstat(fd, &pstat) == -1))
178*5a6e8488SAndroid Build Coastguard Worker 	{
179*5a6e8488SAndroid Build Coastguard Worker 		fprintf(stderr, "Could not stat file: %s\n", path);
180*5a6e8488SAndroid Build Coastguard Worker 		exit(INVALID_INPUT_FILE);
181*5a6e8488SAndroid Build Coastguard Worker 	}
182*5a6e8488SAndroid Build Coastguard Worker 
183*5a6e8488SAndroid Build Coastguard Worker 	// Make sure it's not a directory.
184*5a6e8488SAndroid Build Coastguard Worker 	if (BC_ERR(S_ISDIR(pstat.st_mode)))
185*5a6e8488SAndroid Build Coastguard Worker 	{
186*5a6e8488SAndroid Build Coastguard Worker 		fprintf(stderr, "Path is directory: %s\n", path);
187*5a6e8488SAndroid Build Coastguard Worker 		exit(INVALID_INPUT_FILE);
188*5a6e8488SAndroid Build Coastguard Worker 	}
189*5a6e8488SAndroid Build Coastguard Worker 
190*5a6e8488SAndroid Build Coastguard Worker 	// Get the size of the file and allocate that much.
191*5a6e8488SAndroid Build Coastguard Worker 	size = (size_t) pstat.st_size;
192*5a6e8488SAndroid Build Coastguard Worker 	buf = (char*) malloc(size + 1);
193*5a6e8488SAndroid Build Coastguard Worker 	if (buf == NULL)
194*5a6e8488SAndroid Build Coastguard Worker 	{
195*5a6e8488SAndroid Build Coastguard Worker 		fprintf(stderr, "Could not malloc\n");
196*5a6e8488SAndroid Build Coastguard Worker 		exit(INVALID_INPUT_FILE);
197*5a6e8488SAndroid Build Coastguard Worker 	}
198*5a6e8488SAndroid Build Coastguard Worker 	buf2 = buf;
199*5a6e8488SAndroid Build Coastguard Worker 	to_read = size;
200*5a6e8488SAndroid Build Coastguard Worker 
201*5a6e8488SAndroid Build Coastguard Worker 	do
202*5a6e8488SAndroid Build Coastguard Worker 	{
203*5a6e8488SAndroid Build Coastguard Worker 		// Read the file. We just bail if a signal interrupts. This is so that
204*5a6e8488SAndroid Build Coastguard Worker 		// users can interrupt the reading of big files if they want.
205*5a6e8488SAndroid Build Coastguard Worker 		ssize_t r = read(fd, buf2, to_read);
206*5a6e8488SAndroid Build Coastguard Worker 		if (BC_ERR(r < 0)) exit(e);
207*5a6e8488SAndroid Build Coastguard Worker 		to_read -= (size_t) r;
208*5a6e8488SAndroid Build Coastguard Worker 		buf2 += (size_t) r;
209*5a6e8488SAndroid Build Coastguard Worker 	}
210*5a6e8488SAndroid Build Coastguard Worker 	while (to_read);
211*5a6e8488SAndroid Build Coastguard Worker 
212*5a6e8488SAndroid Build Coastguard Worker 	// Got to have a nul byte.
213*5a6e8488SAndroid Build Coastguard Worker 	buf[size] = '\0';
214*5a6e8488SAndroid Build Coastguard Worker 
215*5a6e8488SAndroid Build Coastguard Worker 	close(fd);
216*5a6e8488SAndroid Build Coastguard Worker 
217*5a6e8488SAndroid Build Coastguard Worker 	return buf;
218*5a6e8488SAndroid Build Coastguard Worker }
219*5a6e8488SAndroid Build Coastguard Worker 
220*5a6e8488SAndroid Build Coastguard Worker /**
221*5a6e8488SAndroid Build Coastguard Worker  * Outputs a label, which is a string literal that the code can use as a name
222*5a6e8488SAndroid Build Coastguard Worker  * for the file that is being turned into a string. This is important for the
223*5a6e8488SAndroid Build Coastguard Worker  * math libraries because the parse and lex code expects a filename. The label
224*5a6e8488SAndroid Build Coastguard Worker  * becomes the filename for the purposes of lexing and parsing.
225*5a6e8488SAndroid Build Coastguard Worker  *
226*5a6e8488SAndroid Build Coastguard Worker  * The label is generated from bc_gen_label (above). It has the form:
227*5a6e8488SAndroid Build Coastguard Worker  *
228*5a6e8488SAndroid Build Coastguard Worker  * const char *<label_name> = <label>;
229*5a6e8488SAndroid Build Coastguard Worker  *
230*5a6e8488SAndroid Build Coastguard Worker  * This function is also needed to smooth out differences between POSIX and
231*5a6e8488SAndroid Build Coastguard Worker  * Windows, specifically, the fact that Windows uses backslashes for filenames
232*5a6e8488SAndroid Build Coastguard Worker  * and that backslashes have to be escaped in a string literal.
233*5a6e8488SAndroid Build Coastguard Worker  *
234*5a6e8488SAndroid Build Coastguard Worker  * @param out    The file to output to.
235*5a6e8488SAndroid Build Coastguard Worker  * @param label  The label name.
236*5a6e8488SAndroid Build Coastguard Worker  * @param name   The actual label text, which is a filename.
237*5a6e8488SAndroid Build Coastguard Worker  * @return       Positive if no error, negative on error, just like *printf().
238*5a6e8488SAndroid Build Coastguard Worker  */
239*5a6e8488SAndroid Build Coastguard Worker static int
output_label(FILE * out,const char * label,const char * name)240*5a6e8488SAndroid Build Coastguard Worker output_label(FILE* out, const char* label, const char* name)
241*5a6e8488SAndroid Build Coastguard Worker {
242*5a6e8488SAndroid Build Coastguard Worker #ifndef _WIN32
243*5a6e8488SAndroid Build Coastguard Worker 
244*5a6e8488SAndroid Build Coastguard Worker 	return fprintf(out, bc_gen_label, label, name);
245*5a6e8488SAndroid Build Coastguard Worker 
246*5a6e8488SAndroid Build Coastguard Worker #else // _WIN32
247*5a6e8488SAndroid Build Coastguard Worker 
248*5a6e8488SAndroid Build Coastguard Worker 	size_t i, count = 0, len = strlen(name);
249*5a6e8488SAndroid Build Coastguard Worker 	char* buf;
250*5a6e8488SAndroid Build Coastguard Worker 	int ret;
251*5a6e8488SAndroid Build Coastguard Worker 
252*5a6e8488SAndroid Build Coastguard Worker 	// This loop counts how many backslashes there are in the label.
253*5a6e8488SAndroid Build Coastguard Worker 	for (i = 0; i < len; ++i)
254*5a6e8488SAndroid Build Coastguard Worker 	{
255*5a6e8488SAndroid Build Coastguard Worker 		count += (name[i] == '\\');
256*5a6e8488SAndroid Build Coastguard Worker 	}
257*5a6e8488SAndroid Build Coastguard Worker 
258*5a6e8488SAndroid Build Coastguard Worker 	buf = (char*) malloc(len + 1 + count);
259*5a6e8488SAndroid Build Coastguard Worker 	if (buf == NULL) return -1;
260*5a6e8488SAndroid Build Coastguard Worker 
261*5a6e8488SAndroid Build Coastguard Worker 	count = 0;
262*5a6e8488SAndroid Build Coastguard Worker 
263*5a6e8488SAndroid Build Coastguard Worker 	// This loop is the meat of the Windows version. What it does is copy the
264*5a6e8488SAndroid Build Coastguard Worker 	// label byte-for-byte, unless it encounters a backslash, in which case, it
265*5a6e8488SAndroid Build Coastguard Worker 	// copies the backslash twice to have it escaped properly in the string
266*5a6e8488SAndroid Build Coastguard Worker 	// literal.
267*5a6e8488SAndroid Build Coastguard Worker 	for (i = 0; i < len; ++i)
268*5a6e8488SAndroid Build Coastguard Worker 	{
269*5a6e8488SAndroid Build Coastguard Worker 		buf[i + count] = name[i];
270*5a6e8488SAndroid Build Coastguard Worker 
271*5a6e8488SAndroid Build Coastguard Worker 		if (name[i] == '\\')
272*5a6e8488SAndroid Build Coastguard Worker 		{
273*5a6e8488SAndroid Build Coastguard Worker 			count += 1;
274*5a6e8488SAndroid Build Coastguard Worker 			buf[i + count] = name[i];
275*5a6e8488SAndroid Build Coastguard Worker 		}
276*5a6e8488SAndroid Build Coastguard Worker 	}
277*5a6e8488SAndroid Build Coastguard Worker 
278*5a6e8488SAndroid Build Coastguard Worker 	buf[i + count] = '\0';
279*5a6e8488SAndroid Build Coastguard Worker 
280*5a6e8488SAndroid Build Coastguard Worker 	ret = fprintf(out, bc_gen_label, label, buf);
281*5a6e8488SAndroid Build Coastguard Worker 
282*5a6e8488SAndroid Build Coastguard Worker 	free(buf);
283*5a6e8488SAndroid Build Coastguard Worker 
284*5a6e8488SAndroid Build Coastguard Worker 	return ret;
285*5a6e8488SAndroid Build Coastguard Worker 
286*5a6e8488SAndroid Build Coastguard Worker #endif // _WIN32
287*5a6e8488SAndroid Build Coastguard Worker }
288*5a6e8488SAndroid Build Coastguard Worker 
289*5a6e8488SAndroid Build Coastguard Worker /**
290*5a6e8488SAndroid Build Coastguard Worker  * This program generates C strings (well, actually, C char arrays) from text
291*5a6e8488SAndroid Build Coastguard Worker  * files. It generates 1 C source file. The resulting file has this structure:
292*5a6e8488SAndroid Build Coastguard Worker  *
293*5a6e8488SAndroid Build Coastguard Worker  * <Copyright Header>
294*5a6e8488SAndroid Build Coastguard Worker  *
295*5a6e8488SAndroid Build Coastguard Worker  * [<Label Extern>]
296*5a6e8488SAndroid Build Coastguard Worker  *
297*5a6e8488SAndroid Build Coastguard Worker  * <Char Array Extern>
298*5a6e8488SAndroid Build Coastguard Worker  *
299*5a6e8488SAndroid Build Coastguard Worker  * [<Preprocessor Guard Begin>]
300*5a6e8488SAndroid Build Coastguard Worker  * [<Label Definition>]
301*5a6e8488SAndroid Build Coastguard Worker  *
302*5a6e8488SAndroid Build Coastguard Worker  * <Char Array Definition>
303*5a6e8488SAndroid Build Coastguard Worker  * [<Preprocessor Guard End>]
304*5a6e8488SAndroid Build Coastguard Worker  *
305*5a6e8488SAndroid Build Coastguard Worker  * Anything surrounded by square brackets may not be in the final generated
306*5a6e8488SAndroid Build Coastguard Worker  * source file.
307*5a6e8488SAndroid Build Coastguard Worker  *
308*5a6e8488SAndroid Build Coastguard Worker  * The required command-line parameters are:
309*5a6e8488SAndroid Build Coastguard Worker  *
310*5a6e8488SAndroid Build Coastguard Worker  * input    Input filename.
311*5a6e8488SAndroid Build Coastguard Worker  * output   Output filename.
312*5a6e8488SAndroid Build Coastguard Worker  * exclude  Whether to exclude extra math-only stuff.
313*5a6e8488SAndroid Build Coastguard Worker  * name     The name of the char array.
314*5a6e8488SAndroid Build Coastguard Worker  *
315*5a6e8488SAndroid Build Coastguard Worker  * The optional parameters are:
316*5a6e8488SAndroid Build Coastguard Worker  *
317*5a6e8488SAndroid Build Coastguard Worker  * label        If given, a label for the char array. See the comment for the
318*5a6e8488SAndroid Build Coastguard Worker  *              output_label() function. It is meant as a "filename" for the
319*5a6e8488SAndroid Build Coastguard Worker  *              text when processed by bc and dc. If label is given, then the
320*5a6e8488SAndroid Build Coastguard Worker  *              <Label Extern> and <Label Definition> will exist in the
321*5a6e8488SAndroid Build Coastguard Worker  *              generated source file.
322*5a6e8488SAndroid Build Coastguard Worker  * define       If given, a preprocessor macro that should be used as a guard
323*5a6e8488SAndroid Build Coastguard Worker  *              for the char array and its label. If define is given, then
324*5a6e8488SAndroid Build Coastguard Worker  *              <Preprocessor Guard Begin> will exist in the form
325*5a6e8488SAndroid Build Coastguard Worker  *              "#if <define>" as part of the generated source file, and
326*5a6e8488SAndroid Build Coastguard Worker  *              <Preprocessor Guard End> will exist in the form
327*5a6e8488SAndroid Build Coastguard Worker  *              "endif // <define>".
328*5a6e8488SAndroid Build Coastguard Worker  * remove_tabs  If this parameter exists, it must be an integer. If it is
329*5a6e8488SAndroid Build Coastguard Worker  *              non-zero, then tabs are removed from the input file text before
330*5a6e8488SAndroid Build Coastguard Worker  *              outputting to the output char array.
331*5a6e8488SAndroid Build Coastguard Worker  *
332*5a6e8488SAndroid Build Coastguard Worker  * All text files that are transformed have license comments. This program finds
333*5a6e8488SAndroid Build Coastguard Worker  * the end of that comment and strips it out as well.
334*5a6e8488SAndroid Build Coastguard Worker  */
335*5a6e8488SAndroid Build Coastguard Worker int
main(int argc,char * argv[])336*5a6e8488SAndroid Build Coastguard Worker main(int argc, char* argv[])
337*5a6e8488SAndroid Build Coastguard Worker {
338*5a6e8488SAndroid Build Coastguard Worker 	char* in;
339*5a6e8488SAndroid Build Coastguard Worker 	FILE* out;
340*5a6e8488SAndroid Build Coastguard Worker 	const char* label;
341*5a6e8488SAndroid Build Coastguard Worker 	const char* define;
342*5a6e8488SAndroid Build Coastguard Worker 	char* name;
343*5a6e8488SAndroid Build Coastguard Worker 	unsigned int count, slashes, err = IO_ERR;
344*5a6e8488SAndroid Build Coastguard Worker 	bool has_label, has_define, remove_tabs, exclude_extra_math;
345*5a6e8488SAndroid Build Coastguard Worker 	size_t i;
346*5a6e8488SAndroid Build Coastguard Worker 
347*5a6e8488SAndroid Build Coastguard Worker 	if (argc < 5)
348*5a6e8488SAndroid Build Coastguard Worker 	{
349*5a6e8488SAndroid Build Coastguard Worker 		printf(bc_gen_usage, argv[0]);
350*5a6e8488SAndroid Build Coastguard Worker 		return INVALID_PARAMS;
351*5a6e8488SAndroid Build Coastguard Worker 	}
352*5a6e8488SAndroid Build Coastguard Worker 
353*5a6e8488SAndroid Build Coastguard Worker 	exclude_extra_math = (strtoul(argv[3], NULL, 10) != 0);
354*5a6e8488SAndroid Build Coastguard Worker 
355*5a6e8488SAndroid Build Coastguard Worker 	name = argv[4];
356*5a6e8488SAndroid Build Coastguard Worker 
357*5a6e8488SAndroid Build Coastguard Worker 	has_label = (argc > 5 && strcmp("", argv[5]) != 0);
358*5a6e8488SAndroid Build Coastguard Worker 	label = has_label ? argv[5] : "";
359*5a6e8488SAndroid Build Coastguard Worker 
360*5a6e8488SAndroid Build Coastguard Worker 	has_define = (argc > 6 && strcmp("", argv[6]) != 0);
361*5a6e8488SAndroid Build Coastguard Worker 	define = has_define ? argv[6] : "";
362*5a6e8488SAndroid Build Coastguard Worker 
363*5a6e8488SAndroid Build Coastguard Worker 	remove_tabs = (argc > 7 && atoi(argv[7]) != 0);
364*5a6e8488SAndroid Build Coastguard Worker 
365*5a6e8488SAndroid Build Coastguard Worker 	in = bc_read_file(argv[1]);
366*5a6e8488SAndroid Build Coastguard Worker 	if (in == NULL) return INVALID_INPUT_FILE;
367*5a6e8488SAndroid Build Coastguard Worker 
368*5a6e8488SAndroid Build Coastguard Worker 	open_file(&out, argv[2], "w");
369*5a6e8488SAndroid Build Coastguard Worker 	if (out == NULL) goto out_err;
370*5a6e8488SAndroid Build Coastguard Worker 
371*5a6e8488SAndroid Build Coastguard Worker 	if (fprintf(out, bc_gen_header, argv[1]) < 0) goto err;
372*5a6e8488SAndroid Build Coastguard Worker 	if (has_label && fprintf(out, bc_gen_label_extern, label) < 0) goto err;
373*5a6e8488SAndroid Build Coastguard Worker 	if (fprintf(out, bc_gen_name_extern, name) < 0) goto err;
374*5a6e8488SAndroid Build Coastguard Worker 	if (has_define && fprintf(out, bc_gen_ifdef, define) < 0) goto err;
375*5a6e8488SAndroid Build Coastguard Worker 	if (has_label && output_label(out, label, argv[1]) < 0) goto err;
376*5a6e8488SAndroid Build Coastguard Worker 	if (fprintf(out, bc_gen_name, name) < 0) goto err;
377*5a6e8488SAndroid Build Coastguard Worker 
378*5a6e8488SAndroid Build Coastguard Worker 	i = count = slashes = 0;
379*5a6e8488SAndroid Build Coastguard Worker 
380*5a6e8488SAndroid Build Coastguard Worker 	// This is where the end of the license comment is found.
381*5a6e8488SAndroid Build Coastguard Worker 	while (slashes < 2 && in[i] > 0)
382*5a6e8488SAndroid Build Coastguard Worker 	{
383*5a6e8488SAndroid Build Coastguard Worker 		if (slashes == 1 && in[i] == '*' && in[i + 1] == '/' &&
384*5a6e8488SAndroid Build Coastguard Worker 		    (in[i + 2] == '\n' || in[i + 2] == '\r'))
385*5a6e8488SAndroid Build Coastguard Worker 		{
386*5a6e8488SAndroid Build Coastguard Worker 			slashes += 1;
387*5a6e8488SAndroid Build Coastguard Worker 			i += 2;
388*5a6e8488SAndroid Build Coastguard Worker 		}
389*5a6e8488SAndroid Build Coastguard Worker 		else if (!slashes && in[i] == '/' && in[i + 1] == '*')
390*5a6e8488SAndroid Build Coastguard Worker 		{
391*5a6e8488SAndroid Build Coastguard Worker 			slashes += 1;
392*5a6e8488SAndroid Build Coastguard Worker 			i += 1;
393*5a6e8488SAndroid Build Coastguard Worker 		}
394*5a6e8488SAndroid Build Coastguard Worker 
395*5a6e8488SAndroid Build Coastguard Worker 		i += 1;
396*5a6e8488SAndroid Build Coastguard Worker 	}
397*5a6e8488SAndroid Build Coastguard Worker 
398*5a6e8488SAndroid Build Coastguard Worker 	// The file is invalid if the end of the license comment could not be found.
399*5a6e8488SAndroid Build Coastguard Worker 	if (in[i] == 0)
400*5a6e8488SAndroid Build Coastguard Worker 	{
401*5a6e8488SAndroid Build Coastguard Worker 		fprintf(stderr, "Could not find end of license comment\n");
402*5a6e8488SAndroid Build Coastguard Worker 		err = INVALID_INPUT_FILE;
403*5a6e8488SAndroid Build Coastguard Worker 		goto err;
404*5a6e8488SAndroid Build Coastguard Worker 	}
405*5a6e8488SAndroid Build Coastguard Worker 
406*5a6e8488SAndroid Build Coastguard Worker 	i += 1;
407*5a6e8488SAndroid Build Coastguard Worker 
408*5a6e8488SAndroid Build Coastguard Worker 	// Do not put extra newlines at the beginning of the char array.
409*5a6e8488SAndroid Build Coastguard Worker 	while (in[i] == '\n' || in[i] == '\r')
410*5a6e8488SAndroid Build Coastguard Worker 	{
411*5a6e8488SAndroid Build Coastguard Worker 		i += 1;
412*5a6e8488SAndroid Build Coastguard Worker 	}
413*5a6e8488SAndroid Build Coastguard Worker 
414*5a6e8488SAndroid Build Coastguard Worker 	// This loop is what generates the actual char array. It counts how many
415*5a6e8488SAndroid Build Coastguard Worker 	// chars it has printed per line in order to insert newlines at appropriate
416*5a6e8488SAndroid Build Coastguard Worker 	// places. It also skips tabs if they should be removed.
417*5a6e8488SAndroid Build Coastguard Worker 	while (in[i] != 0)
418*5a6e8488SAndroid Build Coastguard Worker 	{
419*5a6e8488SAndroid Build Coastguard Worker 		int val;
420*5a6e8488SAndroid Build Coastguard Worker 
421*5a6e8488SAndroid Build Coastguard Worker 		if (in[i] == '\r')
422*5a6e8488SAndroid Build Coastguard Worker 		{
423*5a6e8488SAndroid Build Coastguard Worker 			i += 1;
424*5a6e8488SAndroid Build Coastguard Worker 			continue;
425*5a6e8488SAndroid Build Coastguard Worker 		}
426*5a6e8488SAndroid Build Coastguard Worker 
427*5a6e8488SAndroid Build Coastguard Worker 		if (!remove_tabs || in[i] != '\t')
428*5a6e8488SAndroid Build Coastguard Worker 		{
429*5a6e8488SAndroid Build Coastguard Worker 			// Check for excluding something for extra math.
430*5a6e8488SAndroid Build Coastguard Worker 			if (in[i] == '{')
431*5a6e8488SAndroid Build Coastguard Worker 			{
432*5a6e8488SAndroid Build Coastguard Worker 				// If we found the start...
433*5a6e8488SAndroid Build Coastguard Worker 				if (!strncmp(in + i, bc_gen_ex_start, strlen(bc_gen_ex_start)))
434*5a6e8488SAndroid Build Coastguard Worker 				{
435*5a6e8488SAndroid Build Coastguard Worker 					if (exclude_extra_math)
436*5a6e8488SAndroid Build Coastguard Worker 					{
437*5a6e8488SAndroid Build Coastguard Worker 						// Get past the braces.
438*5a6e8488SAndroid Build Coastguard Worker 						i += 2;
439*5a6e8488SAndroid Build Coastguard Worker 
440*5a6e8488SAndroid Build Coastguard Worker 						// Find the end of the end.
441*5a6e8488SAndroid Build Coastguard Worker 						while (in[i] != '{' && strncmp(in + i, bc_gen_ex_end,
442*5a6e8488SAndroid Build Coastguard Worker 						                               strlen(bc_gen_ex_end)))
443*5a6e8488SAndroid Build Coastguard Worker 						{
444*5a6e8488SAndroid Build Coastguard Worker 							i += 1;
445*5a6e8488SAndroid Build Coastguard Worker 						}
446*5a6e8488SAndroid Build Coastguard Worker 
447*5a6e8488SAndroid Build Coastguard Worker 						i += strlen(bc_gen_ex_end);
448*5a6e8488SAndroid Build Coastguard Worker 
449*5a6e8488SAndroid Build Coastguard Worker 						// Skip the last newline.
450*5a6e8488SAndroid Build Coastguard Worker 						if (in[i] == '\r') i += 1;
451*5a6e8488SAndroid Build Coastguard Worker 						i += 1;
452*5a6e8488SAndroid Build Coastguard Worker 						continue;
453*5a6e8488SAndroid Build Coastguard Worker 					}
454*5a6e8488SAndroid Build Coastguard Worker 					else
455*5a6e8488SAndroid Build Coastguard Worker 					{
456*5a6e8488SAndroid Build Coastguard Worker 						i += strlen(bc_gen_ex_start);
457*5a6e8488SAndroid Build Coastguard Worker 
458*5a6e8488SAndroid Build Coastguard Worker 						// Skip the last newline.
459*5a6e8488SAndroid Build Coastguard Worker 						if (in[i] == '\r') i += 1;
460*5a6e8488SAndroid Build Coastguard Worker 						i += 1;
461*5a6e8488SAndroid Build Coastguard Worker 						continue;
462*5a6e8488SAndroid Build Coastguard Worker 					}
463*5a6e8488SAndroid Build Coastguard Worker 				}
464*5a6e8488SAndroid Build Coastguard Worker 				else if (!exclude_extra_math &&
465*5a6e8488SAndroid Build Coastguard Worker 				         !strncmp(in + i, bc_gen_ex_end, strlen(bc_gen_ex_end)))
466*5a6e8488SAndroid Build Coastguard Worker 				{
467*5a6e8488SAndroid Build Coastguard Worker 					i += strlen(bc_gen_ex_end);
468*5a6e8488SAndroid Build Coastguard Worker 
469*5a6e8488SAndroid Build Coastguard Worker 					// Skip the last newline.
470*5a6e8488SAndroid Build Coastguard Worker 					if (in[i] == '\r') i += 1;
471*5a6e8488SAndroid Build Coastguard Worker 					i += 1;
472*5a6e8488SAndroid Build Coastguard Worker 					continue;
473*5a6e8488SAndroid Build Coastguard Worker 				}
474*5a6e8488SAndroid Build Coastguard Worker 			}
475*5a6e8488SAndroid Build Coastguard Worker 
476*5a6e8488SAndroid Build Coastguard Worker 			// Print a tab if we are at the beginning of a line.
477*5a6e8488SAndroid Build Coastguard Worker 			if (!count && fputc('\t', out) == EOF) goto err;
478*5a6e8488SAndroid Build Coastguard Worker 
479*5a6e8488SAndroid Build Coastguard Worker 			// Print the character.
480*5a6e8488SAndroid Build Coastguard Worker 			val = fprintf(out, "%d,", in[i]);
481*5a6e8488SAndroid Build Coastguard Worker 			if (val < 0) goto err;
482*5a6e8488SAndroid Build Coastguard Worker 
483*5a6e8488SAndroid Build Coastguard Worker 			// Adjust the count.
484*5a6e8488SAndroid Build Coastguard Worker 			count += (unsigned int) val;
485*5a6e8488SAndroid Build Coastguard Worker 			if (count > MAX_WIDTH)
486*5a6e8488SAndroid Build Coastguard Worker 			{
487*5a6e8488SAndroid Build Coastguard Worker 				count = 0;
488*5a6e8488SAndroid Build Coastguard Worker 				if (fputc('\n', out) == EOF) goto err;
489*5a6e8488SAndroid Build Coastguard Worker 			}
490*5a6e8488SAndroid Build Coastguard Worker 		}
491*5a6e8488SAndroid Build Coastguard Worker 
492*5a6e8488SAndroid Build Coastguard Worker 		i += 1;
493*5a6e8488SAndroid Build Coastguard Worker 	}
494*5a6e8488SAndroid Build Coastguard Worker 
495*5a6e8488SAndroid Build Coastguard Worker 	// Make sure the end looks nice and insert the NUL byte at the end.
496*5a6e8488SAndroid Build Coastguard Worker 	if (!count && (fputc(' ', out) == EOF || fputc(' ', out) == EOF)) goto err;
497*5a6e8488SAndroid Build Coastguard Worker 	if (fprintf(out, "0\n};\n") < 0) goto err;
498*5a6e8488SAndroid Build Coastguard Worker 
499*5a6e8488SAndroid Build Coastguard Worker 	err = (has_define && fprintf(out, bc_gen_endif, define) < 0);
500*5a6e8488SAndroid Build Coastguard Worker 
501*5a6e8488SAndroid Build Coastguard Worker err:
502*5a6e8488SAndroid Build Coastguard Worker 	fclose(out);
503*5a6e8488SAndroid Build Coastguard Worker out_err:
504*5a6e8488SAndroid Build Coastguard Worker 	free(in);
505*5a6e8488SAndroid Build Coastguard Worker 	return (int) err;
506*5a6e8488SAndroid Build Coastguard Worker }
507