1f3f50de6SPedro F. Giffuni /* $NetBSD: queue.c,v 1.5 2011/08/31 16:24:57 plunky Exp $ */
2b66a823bSGabor Kovesdan
34dc88ebeSGabor Kovesdan /*-
44d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
51de7b4b8SPedro F. Giffuni *
6*e738085bSDag-Erling Smørgrav * Copyright (c) 1999 James Howard and Dag-Erling Smørgrav
74dc88ebeSGabor Kovesdan * All rights reserved.
8df546c3bSKyle Evans * Copyright (c) 2020 Kyle Evans <kevans@FreeBSD.org>
94dc88ebeSGabor Kovesdan *
104dc88ebeSGabor Kovesdan * Redistribution and use in source and binary forms, with or without
114dc88ebeSGabor Kovesdan * modification, are permitted provided that the following conditions
124dc88ebeSGabor Kovesdan * are met:
134dc88ebeSGabor Kovesdan * 1. Redistributions of source code must retain the above copyright
144dc88ebeSGabor Kovesdan * notice, this list of conditions and the following disclaimer.
154dc88ebeSGabor Kovesdan * 2. Redistributions in binary form must reproduce the above copyright
164dc88ebeSGabor Kovesdan * notice, this list of conditions and the following disclaimer in the
174dc88ebeSGabor Kovesdan * documentation and/or other materials provided with the distribution.
184dc88ebeSGabor Kovesdan *
194dc88ebeSGabor Kovesdan * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
204dc88ebeSGabor Kovesdan * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
214dc88ebeSGabor Kovesdan * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
224dc88ebeSGabor Kovesdan * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
234dc88ebeSGabor Kovesdan * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
244dc88ebeSGabor Kovesdan * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
254dc88ebeSGabor Kovesdan * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
264dc88ebeSGabor Kovesdan * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
274dc88ebeSGabor Kovesdan * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
284dc88ebeSGabor Kovesdan * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
294dc88ebeSGabor Kovesdan * SUCH DAMAGE.
304dc88ebeSGabor Kovesdan */
314dc88ebeSGabor Kovesdan
324dc88ebeSGabor Kovesdan /*
334dc88ebeSGabor Kovesdan * A really poor man's queue. It does only what it has to and gets out of
344dc88ebeSGabor Kovesdan * Dodge. It is used in place of <sys/queue.h> to get a better performance.
354dc88ebeSGabor Kovesdan */
364dc88ebeSGabor Kovesdan
374dc88ebeSGabor Kovesdan #include <sys/param.h>
384dc88ebeSGabor Kovesdan #include <sys/queue.h>
394dc88ebeSGabor Kovesdan
404dc88ebeSGabor Kovesdan #include <stdlib.h>
414dc88ebeSGabor Kovesdan #include <string.h>
424dc88ebeSGabor Kovesdan
434dc88ebeSGabor Kovesdan #include "grep.h"
444dc88ebeSGabor Kovesdan
45df546c3bSKyle Evans typedef struct str qentry_t;
464dc88ebeSGabor Kovesdan
47df546c3bSKyle Evans static long long filled;
48df546c3bSKyle Evans static qentry_t *qend, *qpool;
494dc88ebeSGabor Kovesdan
50df546c3bSKyle Evans /*
51df546c3bSKyle Evans * qnext is the next entry to populate. qlist is where the list actually
52df546c3bSKyle Evans * starts, for the purposes of printing.
53df546c3bSKyle Evans */
54df546c3bSKyle Evans static qentry_t *qlist, *qnext;
55df546c3bSKyle Evans
56df546c3bSKyle Evans void
initqueue(void)57df546c3bSKyle Evans initqueue(void)
58df546c3bSKyle Evans {
59df546c3bSKyle Evans
60df546c3bSKyle Evans qlist = qnext = qpool = grep_calloc(Bflag, sizeof(*qpool));
61df546c3bSKyle Evans qend = qpool + (Bflag - 1);
62df546c3bSKyle Evans }
63df546c3bSKyle Evans
64df546c3bSKyle Evans static qentry_t *
advqueue(qentry_t * itemp)65df546c3bSKyle Evans advqueue(qentry_t *itemp)
66df546c3bSKyle Evans {
67df546c3bSKyle Evans
68df546c3bSKyle Evans if (itemp == qend)
69df546c3bSKyle Evans return (qpool);
70df546c3bSKyle Evans return (itemp + 1);
71df546c3bSKyle Evans }
724dc88ebeSGabor Kovesdan
73a4f3f02bSEd Maste /*
74a4f3f02bSEd Maste * Enqueue another line; return true if we've dequeued a line as a result
75a4f3f02bSEd Maste */
76a4f3f02bSEd Maste bool
enqueue(struct str * x)774dc88ebeSGabor Kovesdan enqueue(struct str *x)
784dc88ebeSGabor Kovesdan {
79df546c3bSKyle Evans qentry_t *item;
80df546c3bSKyle Evans bool rotated;
814dc88ebeSGabor Kovesdan
82df546c3bSKyle Evans item = qnext;
83df546c3bSKyle Evans qnext = advqueue(qnext);
84df546c3bSKyle Evans rotated = false;
854dc88ebeSGabor Kovesdan
86df546c3bSKyle Evans if (filled < Bflag) {
87df546c3bSKyle Evans filled++;
88df546c3bSKyle Evans } else if (filled == Bflag) {
89df546c3bSKyle Evans /* We had already filled up coming in; just rotate. */
90df546c3bSKyle Evans qlist = advqueue(qlist);
91df546c3bSKyle Evans rotated = true;
92df546c3bSKyle Evans free(item->dat);
93f3f50de6SPedro F. Giffuni }
9481c3f641SAlex Richardson /* len + 1 for NUL-terminator */
9581c3f641SAlex Richardson item->dat = grep_malloc(sizeof(char) * x->len + 1);
96df546c3bSKyle Evans item->len = x->len;
97df546c3bSKyle Evans item->line_no = x->line_no;
98df546c3bSKyle Evans item->boff = x->boff;
99df546c3bSKyle Evans item->off = x->off;
100df546c3bSKyle Evans memcpy(item->dat, x->dat, x->len);
10181c3f641SAlex Richardson item->dat[x->len] = '\0';
102df546c3bSKyle Evans item->file = x->file;
1034dc88ebeSGabor Kovesdan
104df546c3bSKyle Evans return (rotated);
1054dc88ebeSGabor Kovesdan }
1064dc88ebeSGabor Kovesdan
1074dc88ebeSGabor Kovesdan void
printqueue(void)1084dc88ebeSGabor Kovesdan printqueue(void)
1094dc88ebeSGabor Kovesdan {
110df546c3bSKyle Evans qentry_t *item;
1114dc88ebeSGabor Kovesdan
112df546c3bSKyle Evans item = qlist;
113df546c3bSKyle Evans do {
114df546c3bSKyle Evans /* Buffer must have ended early. */
115df546c3bSKyle Evans if (item->dat == NULL)
116df546c3bSKyle Evans break;
117df546c3bSKyle Evans
118df546c3bSKyle Evans grep_printline(item, '-');
119df546c3bSKyle Evans free(item->dat);
120df546c3bSKyle Evans item->dat = NULL;
121df546c3bSKyle Evans item = advqueue(item);
122df546c3bSKyle Evans } while (item != qlist);
123df546c3bSKyle Evans
124df546c3bSKyle Evans qlist = qnext = qpool;
125df546c3bSKyle Evans filled = 0;
1264dc88ebeSGabor Kovesdan }
1274dc88ebeSGabor Kovesdan
1284dc88ebeSGabor Kovesdan void
clearqueue(void)1294dc88ebeSGabor Kovesdan clearqueue(void)
1304dc88ebeSGabor Kovesdan {
131df546c3bSKyle Evans qentry_t *item;
1324dc88ebeSGabor Kovesdan
133df546c3bSKyle Evans item = qlist;
134df546c3bSKyle Evans do {
135df546c3bSKyle Evans free(item->dat);
136df546c3bSKyle Evans item->dat = NULL;
137df546c3bSKyle Evans item = advqueue(item);
138df546c3bSKyle Evans } while (item != qlist);
139df546c3bSKyle Evans
140df546c3bSKyle Evans qlist = qnext = qpool;
141df546c3bSKyle Evans filled = 0;
142f3f50de6SPedro F. Giffuni }
143