xref: /aosp_15_r20/external/libwebsockets/lib/misc/fts/trie-fd.c (revision 1c60b9aca93fdbc9b5f19b2d2194c91294b22281)
1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker  * libwebsockets - small server side websockets and web server implementation
3*1c60b9acSAndroid Build Coastguard Worker  *
4*1c60b9acSAndroid Build Coastguard Worker  * Copyright (C) 2010 - 2019 Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker  *
6*1c60b9acSAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a copy
7*1c60b9acSAndroid Build Coastguard Worker  * of this software and associated documentation files (the "Software"), to
8*1c60b9acSAndroid Build Coastguard Worker  * deal in the Software without restriction, including without limitation the
9*1c60b9acSAndroid Build Coastguard Worker  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10*1c60b9acSAndroid Build Coastguard Worker  * sell copies of the Software, and to permit persons to whom the Software is
11*1c60b9acSAndroid Build Coastguard Worker  * furnished to do so, subject to the following conditions:
12*1c60b9acSAndroid Build Coastguard Worker  *
13*1c60b9acSAndroid Build Coastguard Worker  * The above copyright notice and this permission notice shall be included in
14*1c60b9acSAndroid Build Coastguard Worker  * all copies or substantial portions of the Software.
15*1c60b9acSAndroid Build Coastguard Worker  *
16*1c60b9acSAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*1c60b9acSAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*1c60b9acSAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*1c60b9acSAndroid Build Coastguard Worker  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*1c60b9acSAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*1c60b9acSAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*1c60b9acSAndroid Build Coastguard Worker  * IN THE SOFTWARE.
23*1c60b9acSAndroid Build Coastguard Worker  */
24*1c60b9acSAndroid Build Coastguard Worker 
25*1c60b9acSAndroid Build Coastguard Worker #include "private-lib-core.h"
26*1c60b9acSAndroid Build Coastguard Worker #include "private-lib-misc-fts.h"
27*1c60b9acSAndroid Build Coastguard Worker 
28*1c60b9acSAndroid Build Coastguard Worker #include <stdio.h>
29*1c60b9acSAndroid Build Coastguard Worker #include <string.h>
30*1c60b9acSAndroid Build Coastguard Worker #include <assert.h>
31*1c60b9acSAndroid Build Coastguard Worker #include <fcntl.h>
32*1c60b9acSAndroid Build Coastguard Worker #include <sys/types.h>
33*1c60b9acSAndroid Build Coastguard Worker #include <sys/stat.h>
34*1c60b9acSAndroid Build Coastguard Worker 
35*1c60b9acSAndroid Build Coastguard Worker #define AC_COUNT_STASHED_CHILDREN 8
36*1c60b9acSAndroid Build Coastguard Worker 
37*1c60b9acSAndroid Build Coastguard Worker struct ch {
38*1c60b9acSAndroid Build Coastguard Worker 	jg2_file_offset ofs;
39*1c60b9acSAndroid Build Coastguard Worker 	char name[64];
40*1c60b9acSAndroid Build Coastguard Worker 	int inst;
41*1c60b9acSAndroid Build Coastguard Worker 	int child_agg;
42*1c60b9acSAndroid Build Coastguard Worker 	int name_length;
43*1c60b9acSAndroid Build Coastguard Worker 	int effpos;
44*1c60b9acSAndroid Build Coastguard Worker 	int descendents;
45*1c60b9acSAndroid Build Coastguard Worker };
46*1c60b9acSAndroid Build Coastguard Worker 
47*1c60b9acSAndroid Build Coastguard Worker struct wac {
48*1c60b9acSAndroid Build Coastguard Worker 	struct ch ch[AC_COUNT_STASHED_CHILDREN];
49*1c60b9acSAndroid Build Coastguard Worker 
50*1c60b9acSAndroid Build Coastguard Worker 	jg2_file_offset self;
51*1c60b9acSAndroid Build Coastguard Worker 	jg2_file_offset tifs;
52*1c60b9acSAndroid Build Coastguard Worker 	int child_count;
53*1c60b9acSAndroid Build Coastguard Worker 	int child;
54*1c60b9acSAndroid Build Coastguard Worker 
55*1c60b9acSAndroid Build Coastguard Worker 	int agg;
56*1c60b9acSAndroid Build Coastguard Worker 	int desc;
57*1c60b9acSAndroid Build Coastguard Worker 	char done_children;
58*1c60b9acSAndroid Build Coastguard Worker 	char once;
59*1c60b9acSAndroid Build Coastguard Worker };
60*1c60b9acSAndroid Build Coastguard Worker 
61*1c60b9acSAndroid Build Coastguard Worker struct linetable {
62*1c60b9acSAndroid Build Coastguard Worker 	struct linetable *next;
63*1c60b9acSAndroid Build Coastguard Worker 
64*1c60b9acSAndroid Build Coastguard Worker 	int chunk_line_number_start;
65*1c60b9acSAndroid Build Coastguard Worker 	int chunk_line_number_count;
66*1c60b9acSAndroid Build Coastguard Worker 
67*1c60b9acSAndroid Build Coastguard Worker 	off_t chunk_filepos_start;
68*1c60b9acSAndroid Build Coastguard Worker 
69*1c60b9acSAndroid Build Coastguard Worker 	off_t vli_ofs_in_index;
70*1c60b9acSAndroid Build Coastguard Worker };
71*1c60b9acSAndroid Build Coastguard Worker 
72*1c60b9acSAndroid Build Coastguard Worker static uint32_t
b32(unsigned char * b)73*1c60b9acSAndroid Build Coastguard Worker b32(unsigned char *b)
74*1c60b9acSAndroid Build Coastguard Worker {
75*1c60b9acSAndroid Build Coastguard Worker 	return (uint32_t)((b[0] << 24) | (b[1] << 16) | (b[2] << 8) | b[3]);
76*1c60b9acSAndroid Build Coastguard Worker }
77*1c60b9acSAndroid Build Coastguard Worker 
78*1c60b9acSAndroid Build Coastguard Worker static uint16_t
b16(unsigned char * b)79*1c60b9acSAndroid Build Coastguard Worker b16(unsigned char *b)
80*1c60b9acSAndroid Build Coastguard Worker {
81*1c60b9acSAndroid Build Coastguard Worker 	return (uint16_t)((b[0] << 8) | b[1]);
82*1c60b9acSAndroid Build Coastguard Worker }
83*1c60b9acSAndroid Build Coastguard Worker 
84*1c60b9acSAndroid Build Coastguard Worker static int
lws_fts_filepath(struct lws_fts_file * jtf,int filepath_index,char * result,size_t len,uint32_t * ofs_linetable,uint32_t * lines)85*1c60b9acSAndroid Build Coastguard Worker lws_fts_filepath(struct lws_fts_file *jtf, int filepath_index, char *result,
86*1c60b9acSAndroid Build Coastguard Worker 		 size_t len, uint32_t *ofs_linetable, uint32_t *lines)
87*1c60b9acSAndroid Build Coastguard Worker {
88*1c60b9acSAndroid Build Coastguard Worker 	unsigned char buf[256 + 15];
89*1c60b9acSAndroid Build Coastguard Worker 	uint32_t flen;
90*1c60b9acSAndroid Build Coastguard Worker 	int ra, bp = 0;
91*1c60b9acSAndroid Build Coastguard Worker 	size_t m;
92*1c60b9acSAndroid Build Coastguard Worker 	off_t o;
93*1c60b9acSAndroid Build Coastguard Worker 
94*1c60b9acSAndroid Build Coastguard Worker 	if (filepath_index > jtf->filepaths)
95*1c60b9acSAndroid Build Coastguard Worker 		return 1;
96*1c60b9acSAndroid Build Coastguard Worker 
97*1c60b9acSAndroid Build Coastguard Worker 	if (lseek(jtf->fd, (off_t)(jtf->filepath_table + (4 * (unsigned int)filepath_index)),
98*1c60b9acSAndroid Build Coastguard Worker 			SEEK_SET) < 0) {
99*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: unable to seek\n", __func__);
100*1c60b9acSAndroid Build Coastguard Worker 
101*1c60b9acSAndroid Build Coastguard Worker 		return 1;
102*1c60b9acSAndroid Build Coastguard Worker 	}
103*1c60b9acSAndroid Build Coastguard Worker 
104*1c60b9acSAndroid Build Coastguard Worker 	ra = (int)read(jtf->fd, buf, 4);
105*1c60b9acSAndroid Build Coastguard Worker 	if (ra < 0)
106*1c60b9acSAndroid Build Coastguard Worker 		return 1;
107*1c60b9acSAndroid Build Coastguard Worker 
108*1c60b9acSAndroid Build Coastguard Worker 	o = (off_t)b32(buf);
109*1c60b9acSAndroid Build Coastguard Worker 	if (lseek(jtf->fd, o, SEEK_SET) < 0) {
110*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: unable to seek\n", __func__);
111*1c60b9acSAndroid Build Coastguard Worker 
112*1c60b9acSAndroid Build Coastguard Worker 		return 1;
113*1c60b9acSAndroid Build Coastguard Worker 	}
114*1c60b9acSAndroid Build Coastguard Worker 
115*1c60b9acSAndroid Build Coastguard Worker 	ra = (int)read(jtf->fd, buf, sizeof(buf));
116*1c60b9acSAndroid Build Coastguard Worker 	if (ra < 0)
117*1c60b9acSAndroid Build Coastguard Worker 		return 1;
118*1c60b9acSAndroid Build Coastguard Worker 
119*1c60b9acSAndroid Build Coastguard Worker 	if (ofs_linetable)
120*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], ofs_linetable);
121*1c60b9acSAndroid Build Coastguard Worker 	else
122*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &flen);
123*1c60b9acSAndroid Build Coastguard Worker 	if (lines)
124*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], lines);
125*1c60b9acSAndroid Build Coastguard Worker 	else
126*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &flen);
127*1c60b9acSAndroid Build Coastguard Worker 	bp += rq32(&buf[bp], &flen);
128*1c60b9acSAndroid Build Coastguard Worker 
129*1c60b9acSAndroid Build Coastguard Worker 	m = flen;
130*1c60b9acSAndroid Build Coastguard Worker 	if (len - 1 < m)
131*1c60b9acSAndroid Build Coastguard Worker 		m = flen - 1;
132*1c60b9acSAndroid Build Coastguard Worker 
133*1c60b9acSAndroid Build Coastguard Worker 	strncpy(result, (char *)&buf[bp], m);
134*1c60b9acSAndroid Build Coastguard Worker 	result[m] = '\0';
135*1c60b9acSAndroid Build Coastguard Worker 	result[len - 1] = '\0';
136*1c60b9acSAndroid Build Coastguard Worker 
137*1c60b9acSAndroid Build Coastguard Worker 	return 0;
138*1c60b9acSAndroid Build Coastguard Worker }
139*1c60b9acSAndroid Build Coastguard Worker 
140*1c60b9acSAndroid Build Coastguard Worker /*
141*1c60b9acSAndroid Build Coastguard Worker  * returns -1 for fail or fd open on the trie file.
142*1c60b9acSAndroid Build Coastguard Worker  *
143*1c60b9acSAndroid Build Coastguard Worker  * *root is set to the position of the root trie entry.
144*1c60b9acSAndroid Build Coastguard Worker  * *flen is set to the length of the whole file
145*1c60b9acSAndroid Build Coastguard Worker  */
146*1c60b9acSAndroid Build Coastguard Worker 
147*1c60b9acSAndroid Build Coastguard Worker int
lws_fts_adopt(struct lws_fts_file * jtf)148*1c60b9acSAndroid Build Coastguard Worker lws_fts_adopt(struct lws_fts_file *jtf)
149*1c60b9acSAndroid Build Coastguard Worker {
150*1c60b9acSAndroid Build Coastguard Worker 	unsigned char buf[256];
151*1c60b9acSAndroid Build Coastguard Worker 	off_t ot;
152*1c60b9acSAndroid Build Coastguard Worker 
153*1c60b9acSAndroid Build Coastguard Worker 	if (read(jtf->fd, buf, TRIE_FILE_HDR_SIZE) != TRIE_FILE_HDR_SIZE) {
154*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: unable to read file header\n", __func__);
155*1c60b9acSAndroid Build Coastguard Worker 		goto bail;
156*1c60b9acSAndroid Build Coastguard Worker 	}
157*1c60b9acSAndroid Build Coastguard Worker 
158*1c60b9acSAndroid Build Coastguard Worker 	if (buf[0] != 0xca || buf[1] != 0x7a ||
159*1c60b9acSAndroid Build Coastguard Worker 	    buf[2] != 0x5f || buf[3] != 0x75) {
160*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: bad magic %02X %02X %02X %02X\n", __func__,
161*1c60b9acSAndroid Build Coastguard Worker 			 buf[0], buf[1], buf[2], buf[3]);
162*1c60b9acSAndroid Build Coastguard Worker 		goto bail;
163*1c60b9acSAndroid Build Coastguard Worker 	}
164*1c60b9acSAndroid Build Coastguard Worker 
165*1c60b9acSAndroid Build Coastguard Worker 	jtf->root = b32(&buf[4]);
166*1c60b9acSAndroid Build Coastguard Worker 
167*1c60b9acSAndroid Build Coastguard Worker 	ot = lseek(jtf->fd, 0, SEEK_END);
168*1c60b9acSAndroid Build Coastguard Worker 	if (ot < 0) {
169*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: unable to seek\n", __func__);
170*1c60b9acSAndroid Build Coastguard Worker 
171*1c60b9acSAndroid Build Coastguard Worker 		goto bail;
172*1c60b9acSAndroid Build Coastguard Worker 	}
173*1c60b9acSAndroid Build Coastguard Worker 	jtf->flen = (jg2_file_offset)ot;
174*1c60b9acSAndroid Build Coastguard Worker 
175*1c60b9acSAndroid Build Coastguard Worker 	if (jtf->flen != b32(&buf[8])) {
176*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: file size doesn't match expected\n", __func__);
177*1c60b9acSAndroid Build Coastguard Worker 
178*1c60b9acSAndroid Build Coastguard Worker 		goto bail;
179*1c60b9acSAndroid Build Coastguard Worker 	}
180*1c60b9acSAndroid Build Coastguard Worker 
181*1c60b9acSAndroid Build Coastguard Worker 	jtf->filepath_table = b32(&buf[12]);
182*1c60b9acSAndroid Build Coastguard Worker 	jtf->filepaths = (int)b32(&buf[16]);
183*1c60b9acSAndroid Build Coastguard Worker 
184*1c60b9acSAndroid Build Coastguard Worker 	return jtf->fd;
185*1c60b9acSAndroid Build Coastguard Worker 
186*1c60b9acSAndroid Build Coastguard Worker bail:
187*1c60b9acSAndroid Build Coastguard Worker 	return -1;
188*1c60b9acSAndroid Build Coastguard Worker }
189*1c60b9acSAndroid Build Coastguard Worker 
190*1c60b9acSAndroid Build Coastguard Worker struct lws_fts_file *
lws_fts_open(const char * filepath)191*1c60b9acSAndroid Build Coastguard Worker lws_fts_open(const char *filepath)
192*1c60b9acSAndroid Build Coastguard Worker {
193*1c60b9acSAndroid Build Coastguard Worker 	struct lws_fts_file *jtf;
194*1c60b9acSAndroid Build Coastguard Worker 
195*1c60b9acSAndroid Build Coastguard Worker 	jtf = lws_malloc(sizeof(*jtf), "fts open");
196*1c60b9acSAndroid Build Coastguard Worker 	if (!jtf)
197*1c60b9acSAndroid Build Coastguard Worker 		goto bail1;
198*1c60b9acSAndroid Build Coastguard Worker 
199*1c60b9acSAndroid Build Coastguard Worker 	jtf->fd = open(filepath, O_RDONLY);
200*1c60b9acSAndroid Build Coastguard Worker 	if (jtf->fd < 0) {
201*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: unable to open %s\n", __func__, filepath);
202*1c60b9acSAndroid Build Coastguard Worker 		goto bail2;
203*1c60b9acSAndroid Build Coastguard Worker 	}
204*1c60b9acSAndroid Build Coastguard Worker 
205*1c60b9acSAndroid Build Coastguard Worker 	if (lws_fts_adopt(jtf) < 0)
206*1c60b9acSAndroid Build Coastguard Worker 		goto bail3;
207*1c60b9acSAndroid Build Coastguard Worker 
208*1c60b9acSAndroid Build Coastguard Worker 	return jtf;
209*1c60b9acSAndroid Build Coastguard Worker 
210*1c60b9acSAndroid Build Coastguard Worker bail3:
211*1c60b9acSAndroid Build Coastguard Worker 	close(jtf->fd);
212*1c60b9acSAndroid Build Coastguard Worker bail2:
213*1c60b9acSAndroid Build Coastguard Worker 	lws_free(jtf);
214*1c60b9acSAndroid Build Coastguard Worker bail1:
215*1c60b9acSAndroid Build Coastguard Worker 	return NULL;
216*1c60b9acSAndroid Build Coastguard Worker }
217*1c60b9acSAndroid Build Coastguard Worker 
218*1c60b9acSAndroid Build Coastguard Worker void
lws_fts_close(struct lws_fts_file * jtf)219*1c60b9acSAndroid Build Coastguard Worker lws_fts_close(struct lws_fts_file *jtf)
220*1c60b9acSAndroid Build Coastguard Worker {
221*1c60b9acSAndroid Build Coastguard Worker 	close(jtf->fd);
222*1c60b9acSAndroid Build Coastguard Worker 	lws_free(jtf);
223*1c60b9acSAndroid Build Coastguard Worker }
224*1c60b9acSAndroid Build Coastguard Worker 
225*1c60b9acSAndroid Build Coastguard Worker #define grab(_pos, _size) { \
226*1c60b9acSAndroid Build Coastguard Worker 		bp = 0; \
227*1c60b9acSAndroid Build Coastguard Worker 		if (lseek(jtf->fd, (off_t)(_pos), SEEK_SET) < 0) { \
228*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("%s: unable to seek\n", __func__); \
229*1c60b9acSAndroid Build Coastguard Worker \
230*1c60b9acSAndroid Build Coastguard Worker 			goto bail; \
231*1c60b9acSAndroid Build Coastguard Worker 		} \
232*1c60b9acSAndroid Build Coastguard Worker \
233*1c60b9acSAndroid Build Coastguard Worker 		ra = (int)read(jtf->fd, buf, (size_t)(_size)); \
234*1c60b9acSAndroid Build Coastguard Worker 		if (ra < 0) \
235*1c60b9acSAndroid Build Coastguard Worker 			goto bail; \
236*1c60b9acSAndroid Build Coastguard Worker }
237*1c60b9acSAndroid Build Coastguard Worker 
238*1c60b9acSAndroid Build Coastguard Worker static struct linetable *
lws_fts_cache_chunktable(struct lws_fts_file * jtf,uint32_t ofs_linetable,struct lwsac ** linetable_head)239*1c60b9acSAndroid Build Coastguard Worker lws_fts_cache_chunktable(struct lws_fts_file *jtf, uint32_t ofs_linetable,
240*1c60b9acSAndroid Build Coastguard Worker 			 struct lwsac **linetable_head)
241*1c60b9acSAndroid Build Coastguard Worker {
242*1c60b9acSAndroid Build Coastguard Worker 	struct linetable *lt, *first = NULL, **prev = NULL;
243*1c60b9acSAndroid Build Coastguard Worker 	unsigned char buf[8];
244*1c60b9acSAndroid Build Coastguard Worker 	int line = 1, bp, ra;
245*1c60b9acSAndroid Build Coastguard Worker 	off_t cfs = 0;
246*1c60b9acSAndroid Build Coastguard Worker 
247*1c60b9acSAndroid Build Coastguard Worker 	*linetable_head = NULL;
248*1c60b9acSAndroid Build Coastguard Worker 
249*1c60b9acSAndroid Build Coastguard Worker 	do {
250*1c60b9acSAndroid Build Coastguard Worker 		grab(ofs_linetable, sizeof(buf));
251*1c60b9acSAndroid Build Coastguard Worker 
252*1c60b9acSAndroid Build Coastguard Worker 		lt = lwsac_use(linetable_head, sizeof(*lt), 0);
253*1c60b9acSAndroid Build Coastguard Worker 		if (!lt)
254*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
255*1c60b9acSAndroid Build Coastguard Worker 		if (!first)
256*1c60b9acSAndroid Build Coastguard Worker 			first = lt;
257*1c60b9acSAndroid Build Coastguard Worker 
258*1c60b9acSAndroid Build Coastguard Worker 		lt->next = NULL;
259*1c60b9acSAndroid Build Coastguard Worker 		if (prev)
260*1c60b9acSAndroid Build Coastguard Worker 			*prev = lt;
261*1c60b9acSAndroid Build Coastguard Worker 		prev = &lt->next;
262*1c60b9acSAndroid Build Coastguard Worker 
263*1c60b9acSAndroid Build Coastguard Worker 		lt->chunk_line_number_start = line;
264*1c60b9acSAndroid Build Coastguard Worker 		lt->chunk_line_number_count = b16(&buf[bp + 2]);
265*1c60b9acSAndroid Build Coastguard Worker 		lt->vli_ofs_in_index = (off_t)(ofs_linetable + 8);
266*1c60b9acSAndroid Build Coastguard Worker 		lt->chunk_filepos_start = cfs;
267*1c60b9acSAndroid Build Coastguard Worker 
268*1c60b9acSAndroid Build Coastguard Worker 		line += lt->chunk_line_number_count;
269*1c60b9acSAndroid Build Coastguard Worker 
270*1c60b9acSAndroid Build Coastguard Worker 		cfs += (int32_t)b32(&buf[bp + 4]);
271*1c60b9acSAndroid Build Coastguard Worker 		ofs_linetable += b16(&buf[bp]);
272*1c60b9acSAndroid Build Coastguard Worker 
273*1c60b9acSAndroid Build Coastguard Worker 	} while (b16(&buf[bp]));
274*1c60b9acSAndroid Build Coastguard Worker 
275*1c60b9acSAndroid Build Coastguard Worker 	return first;
276*1c60b9acSAndroid Build Coastguard Worker 
277*1c60b9acSAndroid Build Coastguard Worker bail:
278*1c60b9acSAndroid Build Coastguard Worker 	lwsac_free(linetable_head);
279*1c60b9acSAndroid Build Coastguard Worker 
280*1c60b9acSAndroid Build Coastguard Worker 	return NULL;
281*1c60b9acSAndroid Build Coastguard Worker }
282*1c60b9acSAndroid Build Coastguard Worker 
283*1c60b9acSAndroid Build Coastguard Worker static int
lws_fts_getfileoffset(struct lws_fts_file * jtf,struct linetable * ltstart,int line,off_t * _ofs)284*1c60b9acSAndroid Build Coastguard Worker lws_fts_getfileoffset(struct lws_fts_file *jtf, struct linetable *ltstart,
285*1c60b9acSAndroid Build Coastguard Worker 		      int line, off_t *_ofs)
286*1c60b9acSAndroid Build Coastguard Worker {
287*1c60b9acSAndroid Build Coastguard Worker 	struct linetable *lt = ltstart;
288*1c60b9acSAndroid Build Coastguard Worker 	unsigned char buf[LWS_FTS_LINES_PER_CHUNK * 5];
289*1c60b9acSAndroid Build Coastguard Worker 	uint32_t ll;
290*1c60b9acSAndroid Build Coastguard Worker 	off_t ofs;
291*1c60b9acSAndroid Build Coastguard Worker 	int bp, ra;
292*1c60b9acSAndroid Build Coastguard Worker 
293*1c60b9acSAndroid Build Coastguard Worker 	/* first figure out which chunk */
294*1c60b9acSAndroid Build Coastguard Worker 
295*1c60b9acSAndroid Build Coastguard Worker 	do {
296*1c60b9acSAndroid Build Coastguard Worker 		if (line >= lt->chunk_line_number_start &&
297*1c60b9acSAndroid Build Coastguard Worker 		    line < lt->chunk_line_number_start +
298*1c60b9acSAndroid Build Coastguard Worker 		    	    lt->chunk_line_number_count)
299*1c60b9acSAndroid Build Coastguard Worker 			break;
300*1c60b9acSAndroid Build Coastguard Worker 
301*1c60b9acSAndroid Build Coastguard Worker 		lt = lt->next;
302*1c60b9acSAndroid Build Coastguard Worker 	} while (lt);
303*1c60b9acSAndroid Build Coastguard Worker 
304*1c60b9acSAndroid Build Coastguard Worker 	if (!lt)
305*1c60b9acSAndroid Build Coastguard Worker 		goto bail;
306*1c60b9acSAndroid Build Coastguard Worker 
307*1c60b9acSAndroid Build Coastguard Worker 	/* we know it's in this chunk */
308*1c60b9acSAndroid Build Coastguard Worker 
309*1c60b9acSAndroid Build Coastguard Worker 	ofs = lt->chunk_filepos_start;
310*1c60b9acSAndroid Build Coastguard Worker 	line -= lt->chunk_line_number_start;
311*1c60b9acSAndroid Build Coastguard Worker 
312*1c60b9acSAndroid Build Coastguard Worker 	grab(lt->vli_ofs_in_index, sizeof(buf));
313*1c60b9acSAndroid Build Coastguard Worker 
314*1c60b9acSAndroid Build Coastguard Worker 	bp = 0;
315*1c60b9acSAndroid Build Coastguard Worker 	while (line) {
316*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &ll);
317*1c60b9acSAndroid Build Coastguard Worker 		ofs += (int32_t)ll;
318*1c60b9acSAndroid Build Coastguard Worker 		line--;
319*1c60b9acSAndroid Build Coastguard Worker 	}
320*1c60b9acSAndroid Build Coastguard Worker 
321*1c60b9acSAndroid Build Coastguard Worker 	/* we know the offset it is at in the original file */
322*1c60b9acSAndroid Build Coastguard Worker 
323*1c60b9acSAndroid Build Coastguard Worker 	*_ofs = ofs;
324*1c60b9acSAndroid Build Coastguard Worker 
325*1c60b9acSAndroid Build Coastguard Worker 	return 0;
326*1c60b9acSAndroid Build Coastguard Worker 
327*1c60b9acSAndroid Build Coastguard Worker bail:
328*1c60b9acSAndroid Build Coastguard Worker 	lwsl_info("%s: bail %d\n", __func__, line);
329*1c60b9acSAndroid Build Coastguard Worker 
330*1c60b9acSAndroid Build Coastguard Worker 	return 1;
331*1c60b9acSAndroid Build Coastguard Worker }
332*1c60b9acSAndroid Build Coastguard Worker 
333*1c60b9acSAndroid Build Coastguard Worker static int
ac_record(struct lws_fts_file * jtf,struct lwsac ** results_head,const char * needle,int pos,struct wac * s,int sp,uint32_t instances,uint32_t agg_instances,uint32_t children,struct lws_fts_result_autocomplete *** ppac)334*1c60b9acSAndroid Build Coastguard Worker ac_record(struct lws_fts_file *jtf, struct lwsac **results_head,
335*1c60b9acSAndroid Build Coastguard Worker 	  const char *needle, int pos, struct wac *s, int sp,
336*1c60b9acSAndroid Build Coastguard Worker 	  uint32_t instances, uint32_t agg_instances, uint32_t children,
337*1c60b9acSAndroid Build Coastguard Worker 	  struct lws_fts_result_autocomplete ***ppac)
338*1c60b9acSAndroid Build Coastguard Worker {
339*1c60b9acSAndroid Build Coastguard Worker 	struct lws_fts_result_autocomplete *ac;
340*1c60b9acSAndroid Build Coastguard Worker 	int n, m;
341*1c60b9acSAndroid Build Coastguard Worker 	char *p;
342*1c60b9acSAndroid Build Coastguard Worker 
343*1c60b9acSAndroid Build Coastguard Worker 	if (!instances && !agg_instances)
344*1c60b9acSAndroid Build Coastguard Worker 		return 1;
345*1c60b9acSAndroid Build Coastguard Worker 
346*1c60b9acSAndroid Build Coastguard Worker 	m = pos;
347*1c60b9acSAndroid Build Coastguard Worker 	for (n = 1; n <= sp; n++)
348*1c60b9acSAndroid Build Coastguard Worker 		m += s[n].ch[s[n].child - 1].name_length;
349*1c60b9acSAndroid Build Coastguard Worker 
350*1c60b9acSAndroid Build Coastguard Worker 	ac = lwsac_use(results_head, sizeof(*ac) + (unsigned int)m + 1, 0);
351*1c60b9acSAndroid Build Coastguard Worker 	if (!ac)
352*1c60b9acSAndroid Build Coastguard Worker 		return -1;
353*1c60b9acSAndroid Build Coastguard Worker 
354*1c60b9acSAndroid Build Coastguard Worker 	p = (char *)(ac + 1);
355*1c60b9acSAndroid Build Coastguard Worker 
356*1c60b9acSAndroid Build Coastguard Worker 	**ppac = ac;
357*1c60b9acSAndroid Build Coastguard Worker 	ac->next = NULL;
358*1c60b9acSAndroid Build Coastguard Worker 	*ppac = &ac->next;
359*1c60b9acSAndroid Build Coastguard Worker 	ac->instances = (int)instances;
360*1c60b9acSAndroid Build Coastguard Worker 	ac->agg_instances = (int)agg_instances;
361*1c60b9acSAndroid Build Coastguard Worker 	ac->ac_length = m;
362*1c60b9acSAndroid Build Coastguard Worker 	ac->has_children = !!children;
363*1c60b9acSAndroid Build Coastguard Worker 	ac->elided = 0;
364*1c60b9acSAndroid Build Coastguard Worker 
365*1c60b9acSAndroid Build Coastguard Worker 	memcpy(p, needle, (size_t)pos);
366*1c60b9acSAndroid Build Coastguard Worker 	p += pos;
367*1c60b9acSAndroid Build Coastguard Worker 
368*1c60b9acSAndroid Build Coastguard Worker 	for (n = 1; n <= sp; n++) {
369*1c60b9acSAndroid Build Coastguard Worker 		int w = s[n].child - 1;
370*1c60b9acSAndroid Build Coastguard Worker 
371*1c60b9acSAndroid Build Coastguard Worker 		memcpy(p, s[n].ch[w].name, (size_t)s[n].ch[w].name_length);
372*1c60b9acSAndroid Build Coastguard Worker 		p += s[n].ch[w].name_length;
373*1c60b9acSAndroid Build Coastguard Worker 	}
374*1c60b9acSAndroid Build Coastguard Worker 	p = (char *)(ac + 1);
375*1c60b9acSAndroid Build Coastguard Worker 	p[m] = '\0';
376*1c60b9acSAndroid Build Coastguard Worker 
377*1c60b9acSAndroid Build Coastguard Worker 	/*
378*1c60b9acSAndroid Build Coastguard Worker 	 * deduct this child's instance weight from his antecdents to track
379*1c60b9acSAndroid Build Coastguard Worker 	 * relative path attractiveness dynamically, after we already used its
380*1c60b9acSAndroid Build Coastguard Worker 	 * best results (children are sorted best-first)
381*1c60b9acSAndroid Build Coastguard Worker 	 */
382*1c60b9acSAndroid Build Coastguard Worker 	for (n = sp; n >= 0; n--) {
383*1c60b9acSAndroid Build Coastguard Worker 		s[n].ch[s[n].child - 1].child_agg -= (int)instances;
384*1c60b9acSAndroid Build Coastguard Worker 		s[n].agg -= (int)instances;
385*1c60b9acSAndroid Build Coastguard Worker 	}
386*1c60b9acSAndroid Build Coastguard Worker 
387*1c60b9acSAndroid Build Coastguard Worker 	return 0;
388*1c60b9acSAndroid Build Coastguard Worker }
389*1c60b9acSAndroid Build Coastguard Worker 
390*1c60b9acSAndroid Build Coastguard Worker struct lws_fts_result *
lws_fts_search(struct lws_fts_file * jtf,struct lws_fts_search_params * ftsp)391*1c60b9acSAndroid Build Coastguard Worker lws_fts_search(struct lws_fts_file *jtf, struct lws_fts_search_params *ftsp)
392*1c60b9acSAndroid Build Coastguard Worker {
393*1c60b9acSAndroid Build Coastguard Worker 	uint32_t children, instances, co, sl, agg, slt, chunk,
394*1c60b9acSAndroid Build Coastguard Worker 		 fileofs_tif_start, desc, agg_instances;
395*1c60b9acSAndroid Build Coastguard Worker 	int pos = 0, n, m, nl, bp, base = 0, ra, palm, budget, sp, ofd = -1;
396*1c60b9acSAndroid Build Coastguard Worker 	unsigned long long tf = (unsigned long long)lws_now_usecs();
397*1c60b9acSAndroid Build Coastguard Worker 	struct lws_fts_result_autocomplete **pac = NULL;
398*1c60b9acSAndroid Build Coastguard Worker 	char stasis, nac = 0, credible, needle[32];
399*1c60b9acSAndroid Build Coastguard Worker 	struct lws_fts_result_filepath *fp;
400*1c60b9acSAndroid Build Coastguard Worker 	struct lws_fts_result *result;
401*1c60b9acSAndroid Build Coastguard Worker 	unsigned char buf[4096];
402*1c60b9acSAndroid Build Coastguard Worker 	off_t o, child_ofs;
403*1c60b9acSAndroid Build Coastguard Worker 	struct wac s[128];
404*1c60b9acSAndroid Build Coastguard Worker 
405*1c60b9acSAndroid Build Coastguard Worker 	ftsp->results_head = NULL;
406*1c60b9acSAndroid Build Coastguard Worker 
407*1c60b9acSAndroid Build Coastguard Worker 	if (!ftsp->needle)
408*1c60b9acSAndroid Build Coastguard Worker 		return NULL;
409*1c60b9acSAndroid Build Coastguard Worker 
410*1c60b9acSAndroid Build Coastguard Worker 	nl = (int)strlen(ftsp->needle);
411*1c60b9acSAndroid Build Coastguard Worker 	if ((size_t)nl > sizeof(needle) - 2)
412*1c60b9acSAndroid Build Coastguard Worker 		return NULL;
413*1c60b9acSAndroid Build Coastguard Worker 
414*1c60b9acSAndroid Build Coastguard Worker 	result = lwsac_use(&ftsp->results_head, sizeof(*result), 0);
415*1c60b9acSAndroid Build Coastguard Worker 	if (!result)
416*1c60b9acSAndroid Build Coastguard Worker 		return NULL;
417*1c60b9acSAndroid Build Coastguard Worker 
418*1c60b9acSAndroid Build Coastguard Worker 	/* start with no results... */
419*1c60b9acSAndroid Build Coastguard Worker 
420*1c60b9acSAndroid Build Coastguard Worker 	result->autocomplete_head = NULL;
421*1c60b9acSAndroid Build Coastguard Worker 	pac = &result->autocomplete_head;
422*1c60b9acSAndroid Build Coastguard Worker 	result->filepath_head = NULL;
423*1c60b9acSAndroid Build Coastguard Worker 	result->duration_ms = 0;
424*1c60b9acSAndroid Build Coastguard Worker 	result->effective_flags = ftsp->flags;
425*1c60b9acSAndroid Build Coastguard Worker 
426*1c60b9acSAndroid Build Coastguard Worker 	palm = 0;
427*1c60b9acSAndroid Build Coastguard Worker 
428*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < nl; n++)
429*1c60b9acSAndroid Build Coastguard Worker 		needle[n] = (char)tolower(ftsp->needle[n]);
430*1c60b9acSAndroid Build Coastguard Worker 	needle[nl] = '\0';
431*1c60b9acSAndroid Build Coastguard Worker 
432*1c60b9acSAndroid Build Coastguard Worker 	o = (off_t)jtf->root;
433*1c60b9acSAndroid Build Coastguard Worker 	do {
434*1c60b9acSAndroid Build Coastguard Worker 		bp = 0;
435*1c60b9acSAndroid Build Coastguard Worker 		base = 0;
436*1c60b9acSAndroid Build Coastguard Worker 
437*1c60b9acSAndroid Build Coastguard Worker 		grab(o, sizeof(buf));
438*1c60b9acSAndroid Build Coastguard Worker 
439*1c60b9acSAndroid Build Coastguard Worker 		child_ofs = o + bp;
440*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &fileofs_tif_start);
441*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &children);
442*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &instances);
443*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &agg_instances);
444*1c60b9acSAndroid Build Coastguard Worker 		palm = pos;
445*1c60b9acSAndroid Build Coastguard Worker 
446*1c60b9acSAndroid Build Coastguard Worker 		/* the children follow here */
447*1c60b9acSAndroid Build Coastguard Worker 
448*1c60b9acSAndroid Build Coastguard Worker 		if (pos == nl) {
449*1c60b9acSAndroid Build Coastguard Worker 
450*1c60b9acSAndroid Build Coastguard Worker 			nac = 0;
451*1c60b9acSAndroid Build Coastguard Worker 			if (!fileofs_tif_start)
452*1c60b9acSAndroid Build Coastguard Worker 				/*
453*1c60b9acSAndroid Build Coastguard Worker 				 * we matched, but there are no instances of
454*1c60b9acSAndroid Build Coastguard Worker 				 * this, it's actually an intermediate
455*1c60b9acSAndroid Build Coastguard Worker 				 */
456*1c60b9acSAndroid Build Coastguard Worker 
457*1c60b9acSAndroid Build Coastguard Worker 				goto autocomp;
458*1c60b9acSAndroid Build Coastguard Worker 
459*1c60b9acSAndroid Build Coastguard Worker 			/* we leave with bp positioned at the instance list */
460*1c60b9acSAndroid Build Coastguard Worker 
461*1c60b9acSAndroid Build Coastguard Worker 			o = (off_t)fileofs_tif_start;
462*1c60b9acSAndroid Build Coastguard Worker 			grab(o, sizeof(buf));
463*1c60b9acSAndroid Build Coastguard Worker 			break;
464*1c60b9acSAndroid Build Coastguard Worker 		}
465*1c60b9acSAndroid Build Coastguard Worker 
466*1c60b9acSAndroid Build Coastguard Worker 		if (ra - bp < 1024) {
467*1c60b9acSAndroid Build Coastguard Worker 
468*1c60b9acSAndroid Build Coastguard Worker 			/*
469*1c60b9acSAndroid Build Coastguard Worker 			 * We don't have enough.  So reload the buffer starting
470*1c60b9acSAndroid Build Coastguard Worker 			 * at where we got to.
471*1c60b9acSAndroid Build Coastguard Worker 			 */
472*1c60b9acSAndroid Build Coastguard Worker 
473*1c60b9acSAndroid Build Coastguard Worker 			base += bp;
474*1c60b9acSAndroid Build Coastguard Worker 			grab(o + base, sizeof(buf));
475*1c60b9acSAndroid Build Coastguard Worker 		}
476*1c60b9acSAndroid Build Coastguard Worker 
477*1c60b9acSAndroid Build Coastguard Worker 		/* gets set if any child COULD match needle if it went on */
478*1c60b9acSAndroid Build Coastguard Worker 
479*1c60b9acSAndroid Build Coastguard Worker 		credible = 0;
480*1c60b9acSAndroid Build Coastguard Worker 		for (n = 0; (uint32_t)n < children; n++) {
481*1c60b9acSAndroid Build Coastguard Worker 			uint32_t inst;
482*1c60b9acSAndroid Build Coastguard Worker 
483*1c60b9acSAndroid Build Coastguard Worker 			bp += rq32(&buf[bp], &co);
484*1c60b9acSAndroid Build Coastguard Worker 			bp += rq32(&buf[bp], &inst);
485*1c60b9acSAndroid Build Coastguard Worker 			bp += rq32(&buf[bp], &agg);
486*1c60b9acSAndroid Build Coastguard Worker 			bp += rq32(&buf[bp], &desc);
487*1c60b9acSAndroid Build Coastguard Worker 			bp += rq32(&buf[bp], &sl);
488*1c60b9acSAndroid Build Coastguard Worker 
489*1c60b9acSAndroid Build Coastguard Worker 			if (sl > (uint32_t)(nl - pos)) {
490*1c60b9acSAndroid Build Coastguard Worker 
491*1c60b9acSAndroid Build Coastguard Worker 				/*
492*1c60b9acSAndroid Build Coastguard Worker 				 * it can't be a match because it's longer than
493*1c60b9acSAndroid Build Coastguard Worker 				 * our needle string (but that leaves it as a
494*1c60b9acSAndroid Build Coastguard Worker 				 * perfectly fine autocomplete candidate)
495*1c60b9acSAndroid Build Coastguard Worker 				 */
496*1c60b9acSAndroid Build Coastguard Worker 				size_t g = (size_t)(nl - pos);
497*1c60b9acSAndroid Build Coastguard Worker 
498*1c60b9acSAndroid Build Coastguard Worker 				/*
499*1c60b9acSAndroid Build Coastguard Worker 				 * "credible" means at least one child matches
500*1c60b9acSAndroid Build Coastguard Worker 				 * all the chars in needle up to as many as it
501*1c60b9acSAndroid Build Coastguard Worker 				 * has.  If not "credible" this path cannot
502*1c60b9acSAndroid Build Coastguard Worker 				 * match.
503*1c60b9acSAndroid Build Coastguard Worker 				 */
504*1c60b9acSAndroid Build Coastguard Worker 				if (!strncmp((char *)&buf[bp], &needle[pos], g))
505*1c60b9acSAndroid Build Coastguard Worker 					credible = 1;
506*1c60b9acSAndroid Build Coastguard Worker 				else
507*1c60b9acSAndroid Build Coastguard Worker 					/*
508*1c60b9acSAndroid Build Coastguard Worker 					 * deflate the parent agg using the
509*1c60b9acSAndroid Build Coastguard Worker 					 * knowledge this child is not on the
510*1c60b9acSAndroid Build Coastguard Worker 					 * path shown by the remainder of needle
511*1c60b9acSAndroid Build Coastguard Worker 					 */
512*1c60b9acSAndroid Build Coastguard Worker 					agg_instances -= agg;
513*1c60b9acSAndroid Build Coastguard Worker 
514*1c60b9acSAndroid Build Coastguard Worker 				nac = 0;
515*1c60b9acSAndroid Build Coastguard Worker 				bp += (int)sl;
516*1c60b9acSAndroid Build Coastguard Worker 				slt = 0;
517*1c60b9acSAndroid Build Coastguard Worker 				pos = palm;
518*1c60b9acSAndroid Build Coastguard Worker 				goto ensure;
519*1c60b9acSAndroid Build Coastguard Worker 			}
520*1c60b9acSAndroid Build Coastguard Worker 
521*1c60b9acSAndroid Build Coastguard Worker 			/* the comparison string potentially has huge length */
522*1c60b9acSAndroid Build Coastguard Worker 
523*1c60b9acSAndroid Build Coastguard Worker 			slt = sl;
524*1c60b9acSAndroid Build Coastguard Worker 			while (slt) {
525*1c60b9acSAndroid Build Coastguard Worker 
526*1c60b9acSAndroid Build Coastguard Worker 				/*
527*1c60b9acSAndroid Build Coastguard Worker 				 * the strategy is to compare whatever we have
528*1c60b9acSAndroid Build Coastguard Worker 				 * lying around, then bring in more if it didn't
529*1c60b9acSAndroid Build Coastguard Worker 				 * fail to match yet.  That way we don't bring
530*1c60b9acSAndroid Build Coastguard Worker 				 * in anything we could already have known was
531*1c60b9acSAndroid Build Coastguard Worker 				 * not needed due to a match fail.
532*1c60b9acSAndroid Build Coastguard Worker 				 */
533*1c60b9acSAndroid Build Coastguard Worker 
534*1c60b9acSAndroid Build Coastguard Worker 				chunk = (uint32_t)(ra - bp);
535*1c60b9acSAndroid Build Coastguard Worker 				if (chunk > slt)
536*1c60b9acSAndroid Build Coastguard Worker 					chunk = slt;
537*1c60b9acSAndroid Build Coastguard Worker 
538*1c60b9acSAndroid Build Coastguard Worker 				if ((chunk == 1 && needle[pos] != buf[bp]) ||
539*1c60b9acSAndroid Build Coastguard Worker 				    (chunk != 1 &&
540*1c60b9acSAndroid Build Coastguard Worker 				     memcmp(&needle[pos], &buf[bp], chunk))) {
541*1c60b9acSAndroid Build Coastguard Worker 
542*1c60b9acSAndroid Build Coastguard Worker 					/*
543*1c60b9acSAndroid Build Coastguard Worker 					 * it doesn't match... so nothing can
544*1c60b9acSAndroid Build Coastguard Worker 					 * autocomplete this...
545*1c60b9acSAndroid Build Coastguard Worker 					 */
546*1c60b9acSAndroid Build Coastguard Worker 					bp += (int)slt;
547*1c60b9acSAndroid Build Coastguard Worker 					slt = 0;
548*1c60b9acSAndroid Build Coastguard Worker 					nac = 1;
549*1c60b9acSAndroid Build Coastguard Worker 					goto ensure;
550*1c60b9acSAndroid Build Coastguard Worker 				}
551*1c60b9acSAndroid Build Coastguard Worker 
552*1c60b9acSAndroid Build Coastguard Worker 				slt -= chunk;
553*1c60b9acSAndroid Build Coastguard Worker 				pos += (int)chunk;
554*1c60b9acSAndroid Build Coastguard Worker 				bp += (int)chunk;
555*1c60b9acSAndroid Build Coastguard Worker 
556*1c60b9acSAndroid Build Coastguard Worker 				/* so far, it matches */
557*1c60b9acSAndroid Build Coastguard Worker 
558*1c60b9acSAndroid Build Coastguard Worker 				if (!slt) {
559*1c60b9acSAndroid Build Coastguard Worker 					/* we matched the whole thing */
560*1c60b9acSAndroid Build Coastguard Worker 					o = (int32_t)co;
561*1c60b9acSAndroid Build Coastguard Worker 					if (!co)
562*1c60b9acSAndroid Build Coastguard Worker 						goto bail;
563*1c60b9acSAndroid Build Coastguard Worker 					n = (int)children;
564*1c60b9acSAndroid Build Coastguard Worker 					credible = 1;
565*1c60b9acSAndroid Build Coastguard Worker 				}
566*1c60b9acSAndroid Build Coastguard Worker 
567*1c60b9acSAndroid Build Coastguard Worker ensure:
568*1c60b9acSAndroid Build Coastguard Worker 				/*
569*1c60b9acSAndroid Build Coastguard Worker 				 * do we have at least buf more to match, or the
570*1c60b9acSAndroid Build Coastguard Worker 				 * remainder of the string, whichever is less?
571*1c60b9acSAndroid Build Coastguard Worker 				 *
572*1c60b9acSAndroid Build Coastguard Worker 				 * bp may exceed sizeof(buf) on no match path
573*1c60b9acSAndroid Build Coastguard Worker 				 */
574*1c60b9acSAndroid Build Coastguard Worker 				chunk = sizeof(buf);
575*1c60b9acSAndroid Build Coastguard Worker 				if (slt < chunk)
576*1c60b9acSAndroid Build Coastguard Worker 					chunk = slt;
577*1c60b9acSAndroid Build Coastguard Worker 
578*1c60b9acSAndroid Build Coastguard Worker 				if (ra - bp >= (int)chunk)
579*1c60b9acSAndroid Build Coastguard Worker 					continue;
580*1c60b9acSAndroid Build Coastguard Worker 
581*1c60b9acSAndroid Build Coastguard Worker 				/*
582*1c60b9acSAndroid Build Coastguard Worker 				 * We don't have enough.  So reload buf starting
583*1c60b9acSAndroid Build Coastguard Worker 				 * at where we got to.
584*1c60b9acSAndroid Build Coastguard Worker 				 */
585*1c60b9acSAndroid Build Coastguard Worker 				base += bp;
586*1c60b9acSAndroid Build Coastguard Worker 				grab(o + base, sizeof(buf));
587*1c60b9acSAndroid Build Coastguard Worker 
588*1c60b9acSAndroid Build Coastguard Worker 			} /* while we are still comparing */
589*1c60b9acSAndroid Build Coastguard Worker 
590*1c60b9acSAndroid Build Coastguard Worker 		} /* for each child */
591*1c60b9acSAndroid Build Coastguard Worker 
592*1c60b9acSAndroid Build Coastguard Worker 		if ((uint32_t)n == children) {
593*1c60b9acSAndroid Build Coastguard Worker 			if (!credible)
594*1c60b9acSAndroid Build Coastguard Worker 				goto bail;
595*1c60b9acSAndroid Build Coastguard Worker 
596*1c60b9acSAndroid Build Coastguard Worker 			nac = 0;
597*1c60b9acSAndroid Build Coastguard Worker 			goto autocomp;
598*1c60b9acSAndroid Build Coastguard Worker 		}
599*1c60b9acSAndroid Build Coastguard Worker 	} while(1);
600*1c60b9acSAndroid Build Coastguard Worker 
601*1c60b9acSAndroid Build Coastguard Worker 	result->duration_ms = (int)(((uint64_t)lws_now_usecs() - tf) / 1000);
602*1c60b9acSAndroid Build Coastguard Worker 
603*1c60b9acSAndroid Build Coastguard Worker 	if (!instances && !children)
604*1c60b9acSAndroid Build Coastguard Worker 		return result;
605*1c60b9acSAndroid Build Coastguard Worker 
606*1c60b9acSAndroid Build Coastguard Worker 	/* the match list may easily exceed one read buffer load ... */
607*1c60b9acSAndroid Build Coastguard Worker 
608*1c60b9acSAndroid Build Coastguard Worker 	o += bp;
609*1c60b9acSAndroid Build Coastguard Worker 
610*1c60b9acSAndroid Build Coastguard Worker 	/*
611*1c60b9acSAndroid Build Coastguard Worker 	 * Only do the file match list if it was requested in the search flags
612*1c60b9acSAndroid Build Coastguard Worker 	 */
613*1c60b9acSAndroid Build Coastguard Worker 
614*1c60b9acSAndroid Build Coastguard Worker 	if (!(ftsp->flags & LWSFTS_F_QUERY_FILES))
615*1c60b9acSAndroid Build Coastguard Worker 		goto autocomp;
616*1c60b9acSAndroid Build Coastguard Worker 
617*1c60b9acSAndroid Build Coastguard Worker 	do {
618*1c60b9acSAndroid Build Coastguard Worker 		uint32_t fi, tot, line, ro, ofs_linetable, lines, fplen,
619*1c60b9acSAndroid Build Coastguard Worker 			*u, _o;
620*1c60b9acSAndroid Build Coastguard Worker 		struct lwsac *lt_head = NULL;
621*1c60b9acSAndroid Build Coastguard Worker 		struct linetable *ltst;
622*1c60b9acSAndroid Build Coastguard Worker 		char path[256], *pp;
623*1c60b9acSAndroid Build Coastguard Worker 		int footprint;
624*1c60b9acSAndroid Build Coastguard Worker 		off_t fo;
625*1c60b9acSAndroid Build Coastguard Worker 
626*1c60b9acSAndroid Build Coastguard Worker 		ofd = -1;
627*1c60b9acSAndroid Build Coastguard Worker 		grab(o, sizeof(buf));
628*1c60b9acSAndroid Build Coastguard Worker 
629*1c60b9acSAndroid Build Coastguard Worker 		ro = (uint32_t)o;
630*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &_o);
631*1c60b9acSAndroid Build Coastguard Worker 		o = (off_t)_o;
632*1c60b9acSAndroid Build Coastguard Worker 
633*1c60b9acSAndroid Build Coastguard Worker 		assert(!o || o > TRIE_FILE_HDR_SIZE);
634*1c60b9acSAndroid Build Coastguard Worker 
635*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &fi);
636*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &tot);
637*1c60b9acSAndroid Build Coastguard Worker 
638*1c60b9acSAndroid Build Coastguard Worker 		if (lws_fts_filepath(jtf, (int)fi, path, sizeof(path) - 1,
639*1c60b9acSAndroid Build Coastguard Worker 				     &ofs_linetable, &lines)) {
640*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("can't get filepath index %d\n", fi);
641*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
642*1c60b9acSAndroid Build Coastguard Worker 		}
643*1c60b9acSAndroid Build Coastguard Worker 
644*1c60b9acSAndroid Build Coastguard Worker 		if (ftsp->only_filepath && strcmp(path, ftsp->only_filepath))
645*1c60b9acSAndroid Build Coastguard Worker 			continue;
646*1c60b9acSAndroid Build Coastguard Worker 
647*1c60b9acSAndroid Build Coastguard Worker 		ltst = lws_fts_cache_chunktable(jtf, ofs_linetable, &lt_head);
648*1c60b9acSAndroid Build Coastguard Worker 		if (!ltst)
649*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
650*1c60b9acSAndroid Build Coastguard Worker 
651*1c60b9acSAndroid Build Coastguard Worker 		if (ftsp->flags & LWSFTS_F_QUERY_QUOTE_LINE) {
652*1c60b9acSAndroid Build Coastguard Worker 			ofd = open(path, O_RDONLY);
653*1c60b9acSAndroid Build Coastguard Worker 			if (ofd < 0) {
654*1c60b9acSAndroid Build Coastguard Worker 				lwsac_free(&lt_head);
655*1c60b9acSAndroid Build Coastguard Worker 				goto bail;
656*1c60b9acSAndroid Build Coastguard Worker 			}
657*1c60b9acSAndroid Build Coastguard Worker 		}
658*1c60b9acSAndroid Build Coastguard Worker 
659*1c60b9acSAndroid Build Coastguard Worker 		fplen = (uint32_t)strlen(path);
660*1c60b9acSAndroid Build Coastguard Worker 		footprint = (int)(sizeof(*fp) + fplen + 1);
661*1c60b9acSAndroid Build Coastguard Worker 		if (ftsp->flags & LWSFTS_F_QUERY_FILE_LINES) {
662*1c60b9acSAndroid Build Coastguard Worker 			/* line number and offset in file */
663*1c60b9acSAndroid Build Coastguard Worker 			footprint += (int)(2 * sizeof(uint32_t) * tot);
664*1c60b9acSAndroid Build Coastguard Worker 
665*1c60b9acSAndroid Build Coastguard Worker 			if (ftsp->flags & LWSFTS_F_QUERY_QUOTE_LINE)
666*1c60b9acSAndroid Build Coastguard Worker 				/* pointer to quote string */
667*1c60b9acSAndroid Build Coastguard Worker 				footprint += (int)(sizeof(void *) * tot);
668*1c60b9acSAndroid Build Coastguard Worker 		}
669*1c60b9acSAndroid Build Coastguard Worker 
670*1c60b9acSAndroid Build Coastguard Worker 		fp = lwsac_use(&ftsp->results_head, (unsigned int)footprint, 0);
671*1c60b9acSAndroid Build Coastguard Worker 		if (!fp) {
672*1c60b9acSAndroid Build Coastguard Worker 			lwsac_free(&lt_head);
673*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
674*1c60b9acSAndroid Build Coastguard Worker 		}
675*1c60b9acSAndroid Build Coastguard Worker 
676*1c60b9acSAndroid Build Coastguard Worker 		fp->filepath_length = (int)fplen;
677*1c60b9acSAndroid Build Coastguard Worker 		fp->lines_in_file = (int)lines;
678*1c60b9acSAndroid Build Coastguard Worker 		fp->matches = (int)tot;
679*1c60b9acSAndroid Build Coastguard Worker 		fp->matches_length = footprint - (int)sizeof(*fp) - (int)(fplen + 1);
680*1c60b9acSAndroid Build Coastguard Worker 		fp->next = result->filepath_head;
681*1c60b9acSAndroid Build Coastguard Worker 		result->filepath_head = fp;
682*1c60b9acSAndroid Build Coastguard Worker 
683*1c60b9acSAndroid Build Coastguard Worker 		/* line table first so it can be aligned */
684*1c60b9acSAndroid Build Coastguard Worker 
685*1c60b9acSAndroid Build Coastguard Worker 		u = (uint32_t*)(fp + 1);
686*1c60b9acSAndroid Build Coastguard Worker 
687*1c60b9acSAndroid Build Coastguard Worker 		if (ftsp->flags & LWSFTS_F_QUERY_FILE_LINES) {
688*1c60b9acSAndroid Build Coastguard Worker 
689*1c60b9acSAndroid Build Coastguard Worker 			/* for each line number */
690*1c60b9acSAndroid Build Coastguard Worker 
691*1c60b9acSAndroid Build Coastguard Worker 			for (n = 0; (uint32_t)n < tot; n++) {
692*1c60b9acSAndroid Build Coastguard Worker 
693*1c60b9acSAndroid Build Coastguard Worker 				unsigned char lbuf[256], *p;
694*1c60b9acSAndroid Build Coastguard Worker 				char ebuf[384];
695*1c60b9acSAndroid Build Coastguard Worker 				const char **v;
696*1c60b9acSAndroid Build Coastguard Worker 				int m;
697*1c60b9acSAndroid Build Coastguard Worker 
698*1c60b9acSAndroid Build Coastguard Worker 				if ((ra - bp) < 8) {
699*1c60b9acSAndroid Build Coastguard Worker 					base += bp;
700*1c60b9acSAndroid Build Coastguard Worker 					grab((int32_t)ro + base, sizeof(buf));
701*1c60b9acSAndroid Build Coastguard Worker 				}
702*1c60b9acSAndroid Build Coastguard Worker 
703*1c60b9acSAndroid Build Coastguard Worker 				bp += rq32(&buf[bp], &line);
704*1c60b9acSAndroid Build Coastguard Worker 				*u++ = line;
705*1c60b9acSAndroid Build Coastguard Worker 
706*1c60b9acSAndroid Build Coastguard Worker 				if (lws_fts_getfileoffset(jtf, ltst, (int)line, &fo))
707*1c60b9acSAndroid Build Coastguard Worker 					continue;
708*1c60b9acSAndroid Build Coastguard Worker 
709*1c60b9acSAndroid Build Coastguard Worker 				*u++ = (uint32_t)fo;
710*1c60b9acSAndroid Build Coastguard Worker 
711*1c60b9acSAndroid Build Coastguard Worker 				if (!(ftsp->flags & LWSFTS_F_QUERY_QUOTE_LINE))
712*1c60b9acSAndroid Build Coastguard Worker 					continue;
713*1c60b9acSAndroid Build Coastguard Worker 
714*1c60b9acSAndroid Build Coastguard Worker 				if (lseek(ofd, fo, SEEK_SET) < 0)
715*1c60b9acSAndroid Build Coastguard Worker 					continue;
716*1c60b9acSAndroid Build Coastguard Worker 
717*1c60b9acSAndroid Build Coastguard Worker 				m = (int)read(ofd, lbuf, sizeof(lbuf) - 1);
718*1c60b9acSAndroid Build Coastguard Worker 				if (m < 0)
719*1c60b9acSAndroid Build Coastguard Worker 					continue;
720*1c60b9acSAndroid Build Coastguard Worker 				lbuf[sizeof(lbuf) - 1] = '\0';
721*1c60b9acSAndroid Build Coastguard Worker 
722*1c60b9acSAndroid Build Coastguard Worker 				p = (unsigned char *)strchr((char *)lbuf, '\n');
723*1c60b9acSAndroid Build Coastguard Worker 				if (p)
724*1c60b9acSAndroid Build Coastguard Worker 					m = lws_ptr_diff(p, lbuf);
725*1c60b9acSAndroid Build Coastguard Worker 				lbuf[m] = '\0';
726*1c60b9acSAndroid Build Coastguard Worker 				p = (unsigned char *)strchr((char *)lbuf, '\r');
727*1c60b9acSAndroid Build Coastguard Worker 				if (p)
728*1c60b9acSAndroid Build Coastguard Worker 					m = lws_ptr_diff(p, lbuf);
729*1c60b9acSAndroid Build Coastguard Worker 				lbuf[m] = '\0';
730*1c60b9acSAndroid Build Coastguard Worker 
731*1c60b9acSAndroid Build Coastguard Worker 				lws_json_purify(ebuf, (const char *)lbuf,
732*1c60b9acSAndroid Build Coastguard Worker 						sizeof(ebuf) - 1, NULL);
733*1c60b9acSAndroid Build Coastguard Worker 				m = (int)strlen(ebuf);
734*1c60b9acSAndroid Build Coastguard Worker 
735*1c60b9acSAndroid Build Coastguard Worker 				p = lwsac_use(&ftsp->results_head, (unsigned int)m + 1, 0);
736*1c60b9acSAndroid Build Coastguard Worker 				if (!p) {
737*1c60b9acSAndroid Build Coastguard Worker 					lwsac_free(&lt_head);
738*1c60b9acSAndroid Build Coastguard Worker 					goto bail;
739*1c60b9acSAndroid Build Coastguard Worker 				}
740*1c60b9acSAndroid Build Coastguard Worker 
741*1c60b9acSAndroid Build Coastguard Worker 				memcpy(p, ebuf, (unsigned int)m);
742*1c60b9acSAndroid Build Coastguard Worker 				p[m] = '\0';
743*1c60b9acSAndroid Build Coastguard Worker 				v = (const char **)u;
744*1c60b9acSAndroid Build Coastguard Worker 				*v = (const char *)p;
745*1c60b9acSAndroid Build Coastguard Worker 				u += sizeof(const char *) / sizeof(uint32_t);
746*1c60b9acSAndroid Build Coastguard Worker 			}
747*1c60b9acSAndroid Build Coastguard Worker 		}
748*1c60b9acSAndroid Build Coastguard Worker 
749*1c60b9acSAndroid Build Coastguard Worker 		pp = ((char *)&fp[1]) + fp->matches_length;
750*1c60b9acSAndroid Build Coastguard Worker 		memcpy(pp, path, fplen);
751*1c60b9acSAndroid Build Coastguard Worker 		pp[fplen] = '\0';
752*1c60b9acSAndroid Build Coastguard Worker 
753*1c60b9acSAndroid Build Coastguard Worker 		if (ofd >= 0) {
754*1c60b9acSAndroid Build Coastguard Worker 			close(ofd);
755*1c60b9acSAndroid Build Coastguard Worker 			ofd = -1;
756*1c60b9acSAndroid Build Coastguard Worker 		}
757*1c60b9acSAndroid Build Coastguard Worker 
758*1c60b9acSAndroid Build Coastguard Worker 		lwsac_free(&lt_head);
759*1c60b9acSAndroid Build Coastguard Worker 
760*1c60b9acSAndroid Build Coastguard Worker 		if (ftsp->only_filepath)
761*1c60b9acSAndroid Build Coastguard Worker 			break;
762*1c60b9acSAndroid Build Coastguard Worker 
763*1c60b9acSAndroid Build Coastguard Worker 	} while (o);
764*1c60b9acSAndroid Build Coastguard Worker 
765*1c60b9acSAndroid Build Coastguard Worker 	/* sort the instance file list by results density */
766*1c60b9acSAndroid Build Coastguard Worker 
767*1c60b9acSAndroid Build Coastguard Worker 	do {
768*1c60b9acSAndroid Build Coastguard Worker 		struct lws_fts_result_filepath **prf, *rf1, *rf2;
769*1c60b9acSAndroid Build Coastguard Worker 
770*1c60b9acSAndroid Build Coastguard Worker 		stasis = 1;
771*1c60b9acSAndroid Build Coastguard Worker 
772*1c60b9acSAndroid Build Coastguard Worker 		/* bubble sort keeps going until nothing changed */
773*1c60b9acSAndroid Build Coastguard Worker 
774*1c60b9acSAndroid Build Coastguard Worker 		prf = &result->filepath_head;
775*1c60b9acSAndroid Build Coastguard Worker 		while (*prf) {
776*1c60b9acSAndroid Build Coastguard Worker 
777*1c60b9acSAndroid Build Coastguard Worker 			rf1 = *prf;
778*1c60b9acSAndroid Build Coastguard Worker 			rf2 = rf1->next;
779*1c60b9acSAndroid Build Coastguard Worker 
780*1c60b9acSAndroid Build Coastguard Worker 			if (rf2 && rf1->lines_in_file && rf2->lines_in_file &&
781*1c60b9acSAndroid Build Coastguard Worker 			    ((rf1->matches * 1000) / rf1->lines_in_file) <
782*1c60b9acSAndroid Build Coastguard Worker 			    ((rf2->matches * 1000) / rf2->lines_in_file)) {
783*1c60b9acSAndroid Build Coastguard Worker 				stasis = 0;
784*1c60b9acSAndroid Build Coastguard Worker 
785*1c60b9acSAndroid Build Coastguard Worker 				*prf = rf2;
786*1c60b9acSAndroid Build Coastguard Worker 				rf1->next = rf2->next;
787*1c60b9acSAndroid Build Coastguard Worker 				rf2->next = rf1;
788*1c60b9acSAndroid Build Coastguard Worker 			}
789*1c60b9acSAndroid Build Coastguard Worker 
790*1c60b9acSAndroid Build Coastguard Worker 			prf = &(*prf)->next;
791*1c60b9acSAndroid Build Coastguard Worker 		}
792*1c60b9acSAndroid Build Coastguard Worker 
793*1c60b9acSAndroid Build Coastguard Worker 	} while (!stasis);
794*1c60b9acSAndroid Build Coastguard Worker 
795*1c60b9acSAndroid Build Coastguard Worker autocomp:
796*1c60b9acSAndroid Build Coastguard Worker 
797*1c60b9acSAndroid Build Coastguard Worker 	if (!(ftsp->flags & LWSFTS_F_QUERY_AUTOCOMPLETE) || nac)
798*1c60b9acSAndroid Build Coastguard Worker 		return result;
799*1c60b9acSAndroid Build Coastguard Worker 
800*1c60b9acSAndroid Build Coastguard Worker 	/*
801*1c60b9acSAndroid Build Coastguard Worker 	 * autocomplete (ie, the descendent paths that yield the most hits)
802*1c60b9acSAndroid Build Coastguard Worker 	 *
803*1c60b9acSAndroid Build Coastguard Worker 	 * We actually need to spider the earliest terminal descendents from
804*1c60b9acSAndroid Build Coastguard Worker 	 * the child we definitely got past, and present the first n terminal
805*1c60b9acSAndroid Build Coastguard Worker 	 * strings.  The descendents are already sorted in order of highest
806*1c60b9acSAndroid Build Coastguard Worker 	 * aggregated hits in their descendents first, so simply collecting n
807*1c60b9acSAndroid Build Coastguard Worker 	 * earliest leaf children is enough.
808*1c60b9acSAndroid Build Coastguard Worker 	 *
809*1c60b9acSAndroid Build Coastguard Worker 	 * The leaf children may be quite deep down in a stack however.  So we
810*1c60b9acSAndroid Build Coastguard Worker 	 * have to go through all the walking motions collecting and retaining
811*1c60b9acSAndroid Build Coastguard Worker 	 * child into for when we come back up the walk.
812*1c60b9acSAndroid Build Coastguard Worker 	 *
813*1c60b9acSAndroid Build Coastguard Worker 	 * We can completely ignore file instances for this, we just need the
814*1c60b9acSAndroid Build Coastguard Worker 	 * earliest children.  And we can restrict how many children we stash
815*1c60b9acSAndroid Build Coastguard Worker 	 * in each stack level to eg, 5.
816*1c60b9acSAndroid Build Coastguard Worker 	 *
817*1c60b9acSAndroid Build Coastguard Worker 	 * child_ofs comes in pointing at the start of the trie entry that is
818*1c60b9acSAndroid Build Coastguard Worker 	 * to be the starting point for making suggestions.
819*1c60b9acSAndroid Build Coastguard Worker 	 */
820*1c60b9acSAndroid Build Coastguard Worker 
821*1c60b9acSAndroid Build Coastguard Worker 	budget = ftsp->max_autocomplete;
822*1c60b9acSAndroid Build Coastguard Worker 	base = 0;
823*1c60b9acSAndroid Build Coastguard Worker 	bp = 0;
824*1c60b9acSAndroid Build Coastguard Worker 	pac = &result->autocomplete_head;
825*1c60b9acSAndroid Build Coastguard Worker 	sp = 0;
826*1c60b9acSAndroid Build Coastguard Worker 	if (pos > (int)sizeof(s[sp].ch[0].name) - 1)
827*1c60b9acSAndroid Build Coastguard Worker 		pos = (int)sizeof(s[sp].ch[0].name) - 1;
828*1c60b9acSAndroid Build Coastguard Worker 
829*1c60b9acSAndroid Build Coastguard Worker 	memset(&s[sp], 0, sizeof(s[sp]));
830*1c60b9acSAndroid Build Coastguard Worker 
831*1c60b9acSAndroid Build Coastguard Worker 	s[sp].child = 1;
832*1c60b9acSAndroid Build Coastguard Worker 	s[sp].tifs = fileofs_tif_start;
833*1c60b9acSAndroid Build Coastguard Worker 	s[sp].self = (jg2_file_offset)child_ofs;
834*1c60b9acSAndroid Build Coastguard Worker 	s[sp].ch[0].effpos = pos;
835*1c60b9acSAndroid Build Coastguard Worker 
836*1c60b9acSAndroid Build Coastguard Worker 	if (pos == nl)
837*1c60b9acSAndroid Build Coastguard Worker 		n = ac_record(jtf, &ftsp->results_head, needle, pos, s, 0,
838*1c60b9acSAndroid Build Coastguard Worker 			      instances, agg_instances, children, &pac);
839*1c60b9acSAndroid Build Coastguard Worker 
840*1c60b9acSAndroid Build Coastguard Worker 	while (sp >= 0 && budget) {
841*1c60b9acSAndroid Build Coastguard Worker 		int nobump = 0;
842*1c60b9acSAndroid Build Coastguard Worker 		struct ch *tch = &s[sp].ch[s[sp].child - 1];
843*1c60b9acSAndroid Build Coastguard Worker 
844*1c60b9acSAndroid Build Coastguard Worker 		grab(child_ofs, sizeof(buf));
845*1c60b9acSAndroid Build Coastguard Worker 
846*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &fileofs_tif_start);
847*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &children);
848*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &instances);
849*1c60b9acSAndroid Build Coastguard Worker 		bp += rq32(&buf[bp], &agg_instances);
850*1c60b9acSAndroid Build Coastguard Worker 
851*1c60b9acSAndroid Build Coastguard Worker 		if (sp > 0 && s[sp - 1].done_children &&
852*1c60b9acSAndroid Build Coastguard Worker 		    tch->effpos + tch->name_length >= nl &&
853*1c60b9acSAndroid Build Coastguard Worker 		    tch->inst && fileofs_tif_start) {
854*1c60b9acSAndroid Build Coastguard Worker 			n = ac_record(jtf, &ftsp->results_head, needle, pos, s,
855*1c60b9acSAndroid Build Coastguard Worker 				      sp, (uint32_t)tch->inst, (uint32_t)tch->child_agg,
856*1c60b9acSAndroid Build Coastguard Worker 				      (uint32_t)tch->descendents, &pac);
857*1c60b9acSAndroid Build Coastguard Worker 			if (n < 0)
858*1c60b9acSAndroid Build Coastguard Worker 				goto bail;
859*1c60b9acSAndroid Build Coastguard Worker 			if (!n)
860*1c60b9acSAndroid Build Coastguard Worker 				if (--budget == 0)
861*1c60b9acSAndroid Build Coastguard Worker 					break;
862*1c60b9acSAndroid Build Coastguard Worker 		}
863*1c60b9acSAndroid Build Coastguard Worker 
864*1c60b9acSAndroid Build Coastguard Worker 		if (!s[sp].done_children && children) {
865*1c60b9acSAndroid Build Coastguard Worker 			s[sp].done_children = 1;
866*1c60b9acSAndroid Build Coastguard Worker 			sp++;
867*1c60b9acSAndroid Build Coastguard Worker 			memset(&s[sp], 0, sizeof(s[sp]));
868*1c60b9acSAndroid Build Coastguard Worker 			s[sp].tifs = fileofs_tif_start;
869*1c60b9acSAndroid Build Coastguard Worker 			s[sp].self = (jg2_file_offset)child_ofs;
870*1c60b9acSAndroid Build Coastguard Worker 
871*1c60b9acSAndroid Build Coastguard Worker 			for (n = 0; n < (int)children && s[sp].child_count <
872*1c60b9acSAndroid Build Coastguard Worker 					    (int)LWS_ARRAY_SIZE(s[0].ch); n++) {
873*1c60b9acSAndroid Build Coastguard Worker 				uint32_t slen, cho, agg, inst;
874*1c60b9acSAndroid Build Coastguard Worker 				int i = s[sp].child_count;
875*1c60b9acSAndroid Build Coastguard Worker 				struct ch *ch = &s[sp].ch[i];
876*1c60b9acSAndroid Build Coastguard Worker 				size_t max;
877*1c60b9acSAndroid Build Coastguard Worker 
878*1c60b9acSAndroid Build Coastguard Worker 				bp += rq32(&buf[bp], &cho);
879*1c60b9acSAndroid Build Coastguard Worker 				bp += rq32(&buf[bp], &inst);
880*1c60b9acSAndroid Build Coastguard Worker 				bp += rq32(&buf[bp], &agg);
881*1c60b9acSAndroid Build Coastguard Worker 				bp += rq32(&buf[bp], &desc);
882*1c60b9acSAndroid Build Coastguard Worker 				bp += rq32(&buf[bp], &slen);
883*1c60b9acSAndroid Build Coastguard Worker 
884*1c60b9acSAndroid Build Coastguard Worker 				max = slen;
885*1c60b9acSAndroid Build Coastguard Worker 				if (max > sizeof(ch->name) - 1)
886*1c60b9acSAndroid Build Coastguard Worker 					max = sizeof(ch->name) - 1;
887*1c60b9acSAndroid Build Coastguard Worker 
888*1c60b9acSAndroid Build Coastguard Worker 				strncpy(ch->name, (char *)&buf[bp], max);
889*1c60b9acSAndroid Build Coastguard Worker 				bp += (int)slen;
890*1c60b9acSAndroid Build Coastguard Worker 
891*1c60b9acSAndroid Build Coastguard Worker 				ch->name_length = (int)max;
892*1c60b9acSAndroid Build Coastguard Worker 				ch->name[sizeof(ch->name) - 1] = '\0';
893*1c60b9acSAndroid Build Coastguard Worker 				ch->inst = (int)inst;
894*1c60b9acSAndroid Build Coastguard Worker 				ch->effpos =
895*1c60b9acSAndroid Build Coastguard Worker 				       s[sp - 1].ch[s[sp - 1].child - 1].effpos;
896*1c60b9acSAndroid Build Coastguard Worker 
897*1c60b9acSAndroid Build Coastguard Worker 				ch->child_agg = (int)agg;
898*1c60b9acSAndroid Build Coastguard Worker 				ch->descendents = (int)desc;
899*1c60b9acSAndroid Build Coastguard Worker 
900*1c60b9acSAndroid Build Coastguard Worker 				/*
901*1c60b9acSAndroid Build Coastguard Worker 				 * if we have more needle chars than we matched
902*1c60b9acSAndroid Build Coastguard Worker 				 * to get this far, we can only allow potential
903*1c60b9acSAndroid Build Coastguard Worker 				 * matches that are consistent with the
904*1c60b9acSAndroid Build Coastguard Worker 				 * additional unmatched character(s)...
905*1c60b9acSAndroid Build Coastguard Worker 				 */
906*1c60b9acSAndroid Build Coastguard Worker 
907*1c60b9acSAndroid Build Coastguard Worker 				m = nl - ch->effpos;
908*1c60b9acSAndroid Build Coastguard Worker 				if (m > ch->name_length)
909*1c60b9acSAndroid Build Coastguard Worker 					m = ch->name_length;
910*1c60b9acSAndroid Build Coastguard Worker 
911*1c60b9acSAndroid Build Coastguard Worker 				if (m > 0 &&
912*1c60b9acSAndroid Build Coastguard Worker 				    strncmp(&needle[ch->effpos], ch->name, (unsigned int)m))
913*1c60b9acSAndroid Build Coastguard Worker 					continue;
914*1c60b9acSAndroid Build Coastguard Worker 
915*1c60b9acSAndroid Build Coastguard Worker 				ch->effpos += m;
916*1c60b9acSAndroid Build Coastguard Worker 				s[sp].ch[s[sp].child_count++].ofs = cho;
917*1c60b9acSAndroid Build Coastguard Worker 			}
918*1c60b9acSAndroid Build Coastguard Worker 
919*1c60b9acSAndroid Build Coastguard Worker 		}
920*1c60b9acSAndroid Build Coastguard Worker 
921*1c60b9acSAndroid Build Coastguard Worker 		while (sp >= 0 && s[sp].child >= s[sp].child_count) {
922*1c60b9acSAndroid Build Coastguard Worker 			s[sp].done_children = 0;
923*1c60b9acSAndroid Build Coastguard Worker 			sp--;
924*1c60b9acSAndroid Build Coastguard Worker 		}
925*1c60b9acSAndroid Build Coastguard Worker 
926*1c60b9acSAndroid Build Coastguard Worker 		/*
927*1c60b9acSAndroid Build Coastguard Worker 		 * Compare parent remaining agg vs parent's next siblings' still
928*1c60b9acSAndroid Build Coastguard Worker 		 * intact original agg... if the next sibling has more, abandon
929*1c60b9acSAndroid Build Coastguard Worker 		 * the parent path and go with the sibling... this keeps the
930*1c60b9acSAndroid Build Coastguard Worker 		 * autocomplete results related to popularity.
931*1c60b9acSAndroid Build Coastguard Worker 		 */
932*1c60b9acSAndroid Build Coastguard Worker 
933*1c60b9acSAndroid Build Coastguard Worker 		nobump = 0;
934*1c60b9acSAndroid Build Coastguard Worker 		n = sp - 1;
935*1c60b9acSAndroid Build Coastguard Worker 		while (n >= 0) {
936*1c60b9acSAndroid Build Coastguard Worker 			struct lws_fts_result_autocomplete *ac =
937*1c60b9acSAndroid Build Coastguard Worker 				(struct lws_fts_result_autocomplete *)pac;
938*1c60b9acSAndroid Build Coastguard Worker 
939*1c60b9acSAndroid Build Coastguard Worker 			if (s[n].child < s[n].child_count &&
940*1c60b9acSAndroid Build Coastguard Worker 			    s[n].ch[s[n].child - 1].child_agg <
941*1c60b9acSAndroid Build Coastguard Worker 			    	    s[n].ch[s[n].child].child_agg) {
942*1c60b9acSAndroid Build Coastguard Worker 
943*1c60b9acSAndroid Build Coastguard Worker 				if (pac)
944*1c60b9acSAndroid Build Coastguard Worker 					/*
945*1c60b9acSAndroid Build Coastguard Worker 					 * mark the autocomplete result that
946*1c60b9acSAndroid Build Coastguard Worker 					 * there were more children down his
947*1c60b9acSAndroid Build Coastguard Worker 					 * path that we skipped in these results
948*1c60b9acSAndroid Build Coastguard Worker 					 */
949*1c60b9acSAndroid Build Coastguard Worker 					ac->elided = 1;
950*1c60b9acSAndroid Build Coastguard Worker 
951*1c60b9acSAndroid Build Coastguard Worker 				for (m = n; m < sp + 1; m++)
952*1c60b9acSAndroid Build Coastguard Worker 					s[m].done_children = 0;
953*1c60b9acSAndroid Build Coastguard Worker 				sp = n;
954*1c60b9acSAndroid Build Coastguard Worker 				child_ofs = (off_t)s[sp].ch[s[sp].child++].ofs;
955*1c60b9acSAndroid Build Coastguard Worker 				nobump = 1;
956*1c60b9acSAndroid Build Coastguard Worker 			}
957*1c60b9acSAndroid Build Coastguard Worker 
958*1c60b9acSAndroid Build Coastguard Worker 			n--;
959*1c60b9acSAndroid Build Coastguard Worker 		}
960*1c60b9acSAndroid Build Coastguard Worker 
961*1c60b9acSAndroid Build Coastguard Worker 		if (nobump || sp < 0)
962*1c60b9acSAndroid Build Coastguard Worker 			continue;
963*1c60b9acSAndroid Build Coastguard Worker 
964*1c60b9acSAndroid Build Coastguard Worker 		child_ofs = (off_t)s[sp].ch[s[sp].child++].ofs;
965*1c60b9acSAndroid Build Coastguard Worker 	}
966*1c60b9acSAndroid Build Coastguard Worker 
967*1c60b9acSAndroid Build Coastguard Worker 	/* let's do a final sort into agg order */
968*1c60b9acSAndroid Build Coastguard Worker 
969*1c60b9acSAndroid Build Coastguard Worker 	do {
970*1c60b9acSAndroid Build Coastguard Worker 		struct lws_fts_result_autocomplete *ac1, *ac2;
971*1c60b9acSAndroid Build Coastguard Worker 
972*1c60b9acSAndroid Build Coastguard Worker 		stasis = 1;
973*1c60b9acSAndroid Build Coastguard Worker 
974*1c60b9acSAndroid Build Coastguard Worker 		/* bubble sort keeps going until nothing changed */
975*1c60b9acSAndroid Build Coastguard Worker 
976*1c60b9acSAndroid Build Coastguard Worker 		pac = &result->autocomplete_head;
977*1c60b9acSAndroid Build Coastguard Worker 		while (*pac) {
978*1c60b9acSAndroid Build Coastguard Worker 
979*1c60b9acSAndroid Build Coastguard Worker 			ac1 = *pac;
980*1c60b9acSAndroid Build Coastguard Worker 			ac2 = ac1->next;
981*1c60b9acSAndroid Build Coastguard Worker 
982*1c60b9acSAndroid Build Coastguard Worker 			if (ac2 && ac1->instances < ac2->instances) {
983*1c60b9acSAndroid Build Coastguard Worker 				stasis = 0;
984*1c60b9acSAndroid Build Coastguard Worker 
985*1c60b9acSAndroid Build Coastguard Worker 				*pac = ac2;
986*1c60b9acSAndroid Build Coastguard Worker 				ac1->next = ac2->next;
987*1c60b9acSAndroid Build Coastguard Worker 				ac2->next = ac1;
988*1c60b9acSAndroid Build Coastguard Worker 			}
989*1c60b9acSAndroid Build Coastguard Worker 
990*1c60b9acSAndroid Build Coastguard Worker 			pac = &(*pac)->next;
991*1c60b9acSAndroid Build Coastguard Worker 		}
992*1c60b9acSAndroid Build Coastguard Worker 
993*1c60b9acSAndroid Build Coastguard Worker 	} while (!stasis);
994*1c60b9acSAndroid Build Coastguard Worker 
995*1c60b9acSAndroid Build Coastguard Worker 	return result;
996*1c60b9acSAndroid Build Coastguard Worker 
997*1c60b9acSAndroid Build Coastguard Worker bail:
998*1c60b9acSAndroid Build Coastguard Worker 	if (ofd >= 0)
999*1c60b9acSAndroid Build Coastguard Worker 		close(ofd);
1000*1c60b9acSAndroid Build Coastguard Worker 
1001*1c60b9acSAndroid Build Coastguard Worker 	lwsl_info("%s: search ended up at bail\n", __func__);
1002*1c60b9acSAndroid Build Coastguard Worker 
1003*1c60b9acSAndroid Build Coastguard Worker 	return result;
1004*1c60b9acSAndroid Build Coastguard Worker }
1005