xref: /freebsd/usr.bin/grep/queue.c (revision 5e3934b15a2741b2de6b217e77dc9d798d740804)
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