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 = <->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, <_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(<_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(<_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(<_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(<_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