xref: /aosp_15_r20/external/bc/src/parse.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  * Code common to the parsers.
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 <stddef.h>
38*5a6e8488SAndroid Build Coastguard Worker #include <stdlib.h>
39*5a6e8488SAndroid Build Coastguard Worker #include <string.h>
40*5a6e8488SAndroid Build Coastguard Worker 
41*5a6e8488SAndroid Build Coastguard Worker #include <limits.h>
42*5a6e8488SAndroid Build Coastguard Worker 
43*5a6e8488SAndroid Build Coastguard Worker #include <parse.h>
44*5a6e8488SAndroid Build Coastguard Worker #include <program.h>
45*5a6e8488SAndroid Build Coastguard Worker #include <vm.h>
46*5a6e8488SAndroid Build Coastguard Worker 
47*5a6e8488SAndroid Build Coastguard Worker void
bc_parse_updateFunc(BcParse * p,size_t fidx)48*5a6e8488SAndroid Build Coastguard Worker bc_parse_updateFunc(BcParse* p, size_t fidx)
49*5a6e8488SAndroid Build Coastguard Worker {
50*5a6e8488SAndroid Build Coastguard Worker 	p->fidx = fidx;
51*5a6e8488SAndroid Build Coastguard Worker 	p->func = bc_vec_item(&p->prog->fns, fidx);
52*5a6e8488SAndroid Build Coastguard Worker }
53*5a6e8488SAndroid Build Coastguard Worker 
54*5a6e8488SAndroid Build Coastguard Worker inline void
bc_parse_pushName(const BcParse * p,char * name,bool var)55*5a6e8488SAndroid Build Coastguard Worker bc_parse_pushName(const BcParse* p, char* name, bool var)
56*5a6e8488SAndroid Build Coastguard Worker {
57*5a6e8488SAndroid Build Coastguard Worker 	bc_parse_pushIndex(p, bc_program_search(p->prog, name, var));
58*5a6e8488SAndroid Build Coastguard Worker }
59*5a6e8488SAndroid Build Coastguard Worker 
60*5a6e8488SAndroid Build Coastguard Worker /**
61*5a6e8488SAndroid Build Coastguard Worker  * Updates the function, then pushes the instruction and the index. This is a
62*5a6e8488SAndroid Build Coastguard Worker  * convenience function.
63*5a6e8488SAndroid Build Coastguard Worker  * @param p     The parser.
64*5a6e8488SAndroid Build Coastguard Worker  * @param inst  The instruction to push.
65*5a6e8488SAndroid Build Coastguard Worker  * @param idx   The index to push.
66*5a6e8488SAndroid Build Coastguard Worker  */
67*5a6e8488SAndroid Build Coastguard Worker static inline void
bc_parse_pushInstIdx(BcParse * p,uchar inst,size_t idx)68*5a6e8488SAndroid Build Coastguard Worker bc_parse_pushInstIdx(BcParse* p, uchar inst, size_t idx)
69*5a6e8488SAndroid Build Coastguard Worker {
70*5a6e8488SAndroid Build Coastguard Worker 	bc_parse_push(p, inst);
71*5a6e8488SAndroid Build Coastguard Worker 	bc_parse_pushIndex(p, idx);
72*5a6e8488SAndroid Build Coastguard Worker }
73*5a6e8488SAndroid Build Coastguard Worker 
74*5a6e8488SAndroid Build Coastguard Worker void
bc_parse_addString(BcParse * p)75*5a6e8488SAndroid Build Coastguard Worker bc_parse_addString(BcParse* p)
76*5a6e8488SAndroid Build Coastguard Worker {
77*5a6e8488SAndroid Build Coastguard Worker 	size_t idx;
78*5a6e8488SAndroid Build Coastguard Worker 
79*5a6e8488SAndroid Build Coastguard Worker 	idx = bc_program_addString(p->prog, p->l.str.v);
80*5a6e8488SAndroid Build Coastguard Worker 
81*5a6e8488SAndroid Build Coastguard Worker 	// Push the string info.
82*5a6e8488SAndroid Build Coastguard Worker 	bc_parse_pushInstIdx(p, BC_INST_STR, idx);
83*5a6e8488SAndroid Build Coastguard Worker }
84*5a6e8488SAndroid Build Coastguard Worker 
85*5a6e8488SAndroid Build Coastguard Worker static void
bc_parse_addNum(BcParse * p,const char * string)86*5a6e8488SAndroid Build Coastguard Worker bc_parse_addNum(BcParse* p, const char* string)
87*5a6e8488SAndroid Build Coastguard Worker {
88*5a6e8488SAndroid Build Coastguard Worker 	BcProgram* prog = p->prog;
89*5a6e8488SAndroid Build Coastguard Worker 	size_t idx;
90*5a6e8488SAndroid Build Coastguard Worker 
91*5a6e8488SAndroid Build Coastguard Worker 	// XXX: This function has an implicit assumption: that string is a valid C
92*5a6e8488SAndroid Build Coastguard Worker 	// string with a nul terminator. This is because of the unchecked array
93*5a6e8488SAndroid Build Coastguard Worker 	// accesses below. I can't check this with an assert() because that could
94*5a6e8488SAndroid Build Coastguard Worker 	// lead to out-of-bounds access.
95*5a6e8488SAndroid Build Coastguard Worker 	//
96*5a6e8488SAndroid Build Coastguard Worker 	// XXX: In fact, just for safety's sake, assume that this function needs a
97*5a6e8488SAndroid Build Coastguard Worker 	// non-empty string with a nul terminator, just in case bc_parse_zero or
98*5a6e8488SAndroid Build Coastguard Worker 	// bc_parse_one change in the future, which I doubt.
99*5a6e8488SAndroid Build Coastguard Worker 
100*5a6e8488SAndroid Build Coastguard Worker 	BC_SIG_ASSERT_LOCKED;
101*5a6e8488SAndroid Build Coastguard Worker 
102*5a6e8488SAndroid Build Coastguard Worker 	// Special case 0.
103*5a6e8488SAndroid Build Coastguard Worker 	if (bc_parse_zero[0] == string[0] && bc_parse_zero[1] == string[1])
104*5a6e8488SAndroid Build Coastguard Worker 	{
105*5a6e8488SAndroid Build Coastguard Worker 		bc_parse_push(p, BC_INST_ZERO);
106*5a6e8488SAndroid Build Coastguard Worker 		return;
107*5a6e8488SAndroid Build Coastguard Worker 	}
108*5a6e8488SAndroid Build Coastguard Worker 
109*5a6e8488SAndroid Build Coastguard Worker 	// Special case 1.
110*5a6e8488SAndroid Build Coastguard Worker 	if (bc_parse_one[0] == string[0] && bc_parse_one[1] == string[1])
111*5a6e8488SAndroid Build Coastguard Worker 	{
112*5a6e8488SAndroid Build Coastguard Worker 		bc_parse_push(p, BC_INST_ONE);
113*5a6e8488SAndroid Build Coastguard Worker 		return;
114*5a6e8488SAndroid Build Coastguard Worker 	}
115*5a6e8488SAndroid Build Coastguard Worker 
116*5a6e8488SAndroid Build Coastguard Worker 	if (bc_map_insert(&prog->const_map, string, prog->consts.len, &idx))
117*5a6e8488SAndroid Build Coastguard Worker 	{
118*5a6e8488SAndroid Build Coastguard Worker 		BcConst* c;
119*5a6e8488SAndroid Build Coastguard Worker 		BcId* id = bc_vec_item(&prog->const_map, idx);
120*5a6e8488SAndroid Build Coastguard Worker 
121*5a6e8488SAndroid Build Coastguard Worker 		// Get the index.
122*5a6e8488SAndroid Build Coastguard Worker 		idx = id->idx;
123*5a6e8488SAndroid Build Coastguard Worker 
124*5a6e8488SAndroid Build Coastguard Worker 		// Push an empty constant.
125*5a6e8488SAndroid Build Coastguard Worker 		c = bc_vec_pushEmpty(&prog->consts);
126*5a6e8488SAndroid Build Coastguard Worker 
127*5a6e8488SAndroid Build Coastguard Worker 		// Set the fields. We reuse the string in the ID (allocated by
128*5a6e8488SAndroid Build Coastguard Worker 		// bc_map_insert()), because why not?
129*5a6e8488SAndroid Build Coastguard Worker 		c->val = id->name;
130*5a6e8488SAndroid Build Coastguard Worker 		c->base = BC_NUM_BIGDIG_MAX;
131*5a6e8488SAndroid Build Coastguard Worker 
132*5a6e8488SAndroid Build Coastguard Worker 		// We need this to be able to tell that the number has not been
133*5a6e8488SAndroid Build Coastguard Worker 		// allocated.
134*5a6e8488SAndroid Build Coastguard Worker 		bc_num_clear(&c->num);
135*5a6e8488SAndroid Build Coastguard Worker 	}
136*5a6e8488SAndroid Build Coastguard Worker 	else
137*5a6e8488SAndroid Build Coastguard Worker 	{
138*5a6e8488SAndroid Build Coastguard Worker 		BcId* id = bc_vec_item(&prog->const_map, idx);
139*5a6e8488SAndroid Build Coastguard Worker 		idx = id->idx;
140*5a6e8488SAndroid Build Coastguard Worker 	}
141*5a6e8488SAndroid Build Coastguard Worker 
142*5a6e8488SAndroid Build Coastguard Worker 	bc_parse_pushInstIdx(p, BC_INST_NUM, idx);
143*5a6e8488SAndroid Build Coastguard Worker }
144*5a6e8488SAndroid Build Coastguard Worker 
145*5a6e8488SAndroid Build Coastguard Worker void
bc_parse_number(BcParse * p)146*5a6e8488SAndroid Build Coastguard Worker bc_parse_number(BcParse* p)
147*5a6e8488SAndroid Build Coastguard Worker {
148*5a6e8488SAndroid Build Coastguard Worker #if BC_ENABLE_EXTRA_MATH
149*5a6e8488SAndroid Build Coastguard Worker 	char* exp = strchr(p->l.str.v, 'e');
150*5a6e8488SAndroid Build Coastguard Worker 	size_t idx = SIZE_MAX;
151*5a6e8488SAndroid Build Coastguard Worker 
152*5a6e8488SAndroid Build Coastguard Worker 	// Do we have a number in scientific notation? If so, add a nul byte where
153*5a6e8488SAndroid Build Coastguard Worker 	// the e is.
154*5a6e8488SAndroid Build Coastguard Worker 	if (exp != NULL)
155*5a6e8488SAndroid Build Coastguard Worker 	{
156*5a6e8488SAndroid Build Coastguard Worker 		idx = ((size_t) (exp - p->l.str.v));
157*5a6e8488SAndroid Build Coastguard Worker 		*exp = 0;
158*5a6e8488SAndroid Build Coastguard Worker 	}
159*5a6e8488SAndroid Build Coastguard Worker #endif // BC_ENABLE_EXTRA_MATH
160*5a6e8488SAndroid Build Coastguard Worker 
161*5a6e8488SAndroid Build Coastguard Worker 	bc_parse_addNum(p, p->l.str.v);
162*5a6e8488SAndroid Build Coastguard Worker 
163*5a6e8488SAndroid Build Coastguard Worker #if BC_ENABLE_EXTRA_MATH
164*5a6e8488SAndroid Build Coastguard Worker 	// If we have a number in scientific notation...
165*5a6e8488SAndroid Build Coastguard Worker 	if (exp != NULL)
166*5a6e8488SAndroid Build Coastguard Worker 	{
167*5a6e8488SAndroid Build Coastguard Worker 		bool neg;
168*5a6e8488SAndroid Build Coastguard Worker 
169*5a6e8488SAndroid Build Coastguard Worker 		// Figure out if the exponent is negative.
170*5a6e8488SAndroid Build Coastguard Worker 		neg = (*((char*) bc_vec_item(&p->l.str, idx + 1)) == BC_LEX_NEG_CHAR);
171*5a6e8488SAndroid Build Coastguard Worker 
172*5a6e8488SAndroid Build Coastguard Worker 		// Add the number and instruction.
173*5a6e8488SAndroid Build Coastguard Worker 		bc_parse_addNum(p, bc_vec_item(&p->l.str, idx + 1 + neg));
174*5a6e8488SAndroid Build Coastguard Worker 		bc_parse_push(p, BC_INST_LSHIFT + neg);
175*5a6e8488SAndroid Build Coastguard Worker 	}
176*5a6e8488SAndroid Build Coastguard Worker #endif // BC_ENABLE_EXTRA_MATH
177*5a6e8488SAndroid Build Coastguard Worker }
178*5a6e8488SAndroid Build Coastguard Worker 
179*5a6e8488SAndroid Build Coastguard Worker void
bc_parse_text(BcParse * p,const char * text,BcMode mode)180*5a6e8488SAndroid Build Coastguard Worker bc_parse_text(BcParse* p, const char* text, BcMode mode)
181*5a6e8488SAndroid Build Coastguard Worker {
182*5a6e8488SAndroid Build Coastguard Worker 	BC_SIG_LOCK;
183*5a6e8488SAndroid Build Coastguard Worker 
184*5a6e8488SAndroid Build Coastguard Worker 	// Make sure the pointer isn't invalidated.
185*5a6e8488SAndroid Build Coastguard Worker 	p->func = bc_vec_item(&p->prog->fns, p->fidx);
186*5a6e8488SAndroid Build Coastguard Worker 	bc_lex_text(&p->l, text, mode);
187*5a6e8488SAndroid Build Coastguard Worker 
188*5a6e8488SAndroid Build Coastguard Worker 	BC_SIG_UNLOCK;
189*5a6e8488SAndroid Build Coastguard Worker }
190*5a6e8488SAndroid Build Coastguard Worker 
191*5a6e8488SAndroid Build Coastguard Worker void
bc_parse_reset(BcParse * p)192*5a6e8488SAndroid Build Coastguard Worker bc_parse_reset(BcParse* p)
193*5a6e8488SAndroid Build Coastguard Worker {
194*5a6e8488SAndroid Build Coastguard Worker 	BC_SIG_ASSERT_LOCKED;
195*5a6e8488SAndroid Build Coastguard Worker 
196*5a6e8488SAndroid Build Coastguard Worker 	// Reset the function if it isn't main and switch to main.
197*5a6e8488SAndroid Build Coastguard Worker 	if (p->fidx != BC_PROG_MAIN)
198*5a6e8488SAndroid Build Coastguard Worker 	{
199*5a6e8488SAndroid Build Coastguard Worker 		bc_func_reset(p->func);
200*5a6e8488SAndroid Build Coastguard Worker 		bc_parse_updateFunc(p, BC_PROG_MAIN);
201*5a6e8488SAndroid Build Coastguard Worker 	}
202*5a6e8488SAndroid Build Coastguard Worker 
203*5a6e8488SAndroid Build Coastguard Worker 	// Reset the lexer.
204*5a6e8488SAndroid Build Coastguard Worker 	p->l.i = p->l.len;
205*5a6e8488SAndroid Build Coastguard Worker 	p->l.t = BC_LEX_EOF;
206*5a6e8488SAndroid Build Coastguard Worker 
207*5a6e8488SAndroid Build Coastguard Worker #if BC_ENABLED
208*5a6e8488SAndroid Build Coastguard Worker 	if (BC_IS_BC)
209*5a6e8488SAndroid Build Coastguard Worker 	{
210*5a6e8488SAndroid Build Coastguard Worker 		// Get rid of the bc parser state.
211*5a6e8488SAndroid Build Coastguard Worker 		p->auto_part = false;
212*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_npop(&p->flags, p->flags.len - 1);
213*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_popAll(&p->exits);
214*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_popAll(&p->conds);
215*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_popAll(&p->ops);
216*5a6e8488SAndroid Build Coastguard Worker 	}
217*5a6e8488SAndroid Build Coastguard Worker #endif // BC_ENABLED
218*5a6e8488SAndroid Build Coastguard Worker 
219*5a6e8488SAndroid Build Coastguard Worker 	// Reset the program. This might clear the error.
220*5a6e8488SAndroid Build Coastguard Worker 	bc_program_reset(p->prog);
221*5a6e8488SAndroid Build Coastguard Worker 
222*5a6e8488SAndroid Build Coastguard Worker 	// Jump if there is an error.
223*5a6e8488SAndroid Build Coastguard Worker 	if (BC_ERR(vm->status)) BC_JMP;
224*5a6e8488SAndroid Build Coastguard Worker }
225*5a6e8488SAndroid Build Coastguard Worker 
226*5a6e8488SAndroid Build Coastguard Worker #if BC_DEBUG
227*5a6e8488SAndroid Build Coastguard Worker void
bc_parse_free(BcParse * p)228*5a6e8488SAndroid Build Coastguard Worker bc_parse_free(BcParse* p)
229*5a6e8488SAndroid Build Coastguard Worker {
230*5a6e8488SAndroid Build Coastguard Worker 	BC_SIG_ASSERT_LOCKED;
231*5a6e8488SAndroid Build Coastguard Worker 
232*5a6e8488SAndroid Build Coastguard Worker 	assert(p != NULL);
233*5a6e8488SAndroid Build Coastguard Worker 
234*5a6e8488SAndroid Build Coastguard Worker #if BC_ENABLED
235*5a6e8488SAndroid Build Coastguard Worker 	if (BC_IS_BC)
236*5a6e8488SAndroid Build Coastguard Worker 	{
237*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_free(&p->flags);
238*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_free(&p->exits);
239*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_free(&p->conds);
240*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_free(&p->ops);
241*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_free(&p->buf);
242*5a6e8488SAndroid Build Coastguard Worker 	}
243*5a6e8488SAndroid Build Coastguard Worker #endif // BC_ENABLED
244*5a6e8488SAndroid Build Coastguard Worker 
245*5a6e8488SAndroid Build Coastguard Worker 	bc_lex_free(&p->l);
246*5a6e8488SAndroid Build Coastguard Worker }
247*5a6e8488SAndroid Build Coastguard Worker #endif // BC_DEBUG
248*5a6e8488SAndroid Build Coastguard Worker 
249*5a6e8488SAndroid Build Coastguard Worker void
bc_parse_init(BcParse * p,BcProgram * prog,size_t func)250*5a6e8488SAndroid Build Coastguard Worker bc_parse_init(BcParse* p, BcProgram* prog, size_t func)
251*5a6e8488SAndroid Build Coastguard Worker {
252*5a6e8488SAndroid Build Coastguard Worker #if BC_ENABLED
253*5a6e8488SAndroid Build Coastguard Worker 	uint16_t flag = 0;
254*5a6e8488SAndroid Build Coastguard Worker #endif // BC_ENABLED
255*5a6e8488SAndroid Build Coastguard Worker 
256*5a6e8488SAndroid Build Coastguard Worker 	BC_SIG_ASSERT_LOCKED;
257*5a6e8488SAndroid Build Coastguard Worker 
258*5a6e8488SAndroid Build Coastguard Worker 	assert(p != NULL && prog != NULL);
259*5a6e8488SAndroid Build Coastguard Worker 
260*5a6e8488SAndroid Build Coastguard Worker #if BC_ENABLED
261*5a6e8488SAndroid Build Coastguard Worker 	if (BC_IS_BC)
262*5a6e8488SAndroid Build Coastguard Worker 	{
263*5a6e8488SAndroid Build Coastguard Worker 		// We always want at least one flag set on the flags stack.
264*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_init(&p->flags, sizeof(uint16_t), BC_DTOR_NONE);
265*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_push(&p->flags, &flag);
266*5a6e8488SAndroid Build Coastguard Worker 
267*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_init(&p->exits, sizeof(BcInstPtr), BC_DTOR_NONE);
268*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_init(&p->conds, sizeof(size_t), BC_DTOR_NONE);
269*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_init(&p->ops, sizeof(BcLexType), BC_DTOR_NONE);
270*5a6e8488SAndroid Build Coastguard Worker 		bc_vec_init(&p->buf, sizeof(char), BC_DTOR_NONE);
271*5a6e8488SAndroid Build Coastguard Worker 
272*5a6e8488SAndroid Build Coastguard Worker 		p->auto_part = false;
273*5a6e8488SAndroid Build Coastguard Worker 	}
274*5a6e8488SAndroid Build Coastguard Worker #endif // BC_ENABLED
275*5a6e8488SAndroid Build Coastguard Worker 
276*5a6e8488SAndroid Build Coastguard Worker 	bc_lex_init(&p->l);
277*5a6e8488SAndroid Build Coastguard Worker 
278*5a6e8488SAndroid Build Coastguard Worker 	// Set up the function.
279*5a6e8488SAndroid Build Coastguard Worker 	p->prog = prog;
280*5a6e8488SAndroid Build Coastguard Worker 	bc_parse_updateFunc(p, func);
281*5a6e8488SAndroid Build Coastguard Worker }
282