xref: /freebsd/usr.sbin/fifolog/lib/fifolog_reader.c (revision b3e7694832e81d7a904a10f525f8797b753bf0d3)
1662cb04cSPoul-Henning Kamp /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
31de7b4b8SPedro F. Giffuni  *
4662cb04cSPoul-Henning Kamp  * Copyright (c) 2005-2008 Poul-Henning Kamp
5662cb04cSPoul-Henning Kamp  * All rights reserved.
6662cb04cSPoul-Henning Kamp  *
7662cb04cSPoul-Henning Kamp  * Redistribution and use in source and binary forms, with or without
8662cb04cSPoul-Henning Kamp  * modification, are permitted provided that the following conditions
9662cb04cSPoul-Henning Kamp  * are met:
10662cb04cSPoul-Henning Kamp  * 1. Redistributions of source code must retain the above copyright
11662cb04cSPoul-Henning Kamp  *    notice, this list of conditions and the following disclaimer.
12662cb04cSPoul-Henning Kamp  * 2. Redistributions in binary form must reproduce the above copyright
13662cb04cSPoul-Henning Kamp  *    notice, this list of conditions and the following disclaimer in the
14662cb04cSPoul-Henning Kamp  *    documentation and/or other materials provided with the distribution.
15662cb04cSPoul-Henning Kamp  *
16662cb04cSPoul-Henning Kamp  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17662cb04cSPoul-Henning Kamp  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18662cb04cSPoul-Henning Kamp  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19662cb04cSPoul-Henning Kamp  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20662cb04cSPoul-Henning Kamp  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21662cb04cSPoul-Henning Kamp  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22662cb04cSPoul-Henning Kamp  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23662cb04cSPoul-Henning Kamp  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24662cb04cSPoul-Henning Kamp  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25662cb04cSPoul-Henning Kamp  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26662cb04cSPoul-Henning Kamp  * SUCH DAMAGE.
27662cb04cSPoul-Henning Kamp  */
28662cb04cSPoul-Henning Kamp 
29662cb04cSPoul-Henning Kamp #include <stdio.h>
30662cb04cSPoul-Henning Kamp #include <unistd.h>
31662cb04cSPoul-Henning Kamp #include <assert.h>
32662cb04cSPoul-Henning Kamp #include <err.h>
33662cb04cSPoul-Henning Kamp #include <time.h>
34662cb04cSPoul-Henning Kamp #include <string.h>
35662cb04cSPoul-Henning Kamp #include <stdlib.h>
36662cb04cSPoul-Henning Kamp #include <zlib.h>
37662cb04cSPoul-Henning Kamp #include <sys/endian.h>
38662cb04cSPoul-Henning Kamp 
39662cb04cSPoul-Henning Kamp #include "fifolog.h"
40662cb04cSPoul-Henning Kamp #include "libfifolog.h"
41662cb04cSPoul-Henning Kamp #include "libfifolog_int.h"
42662cb04cSPoul-Henning Kamp #include "miniobj.h"
43662cb04cSPoul-Henning Kamp 
44662cb04cSPoul-Henning Kamp /*--------------------------------------------------------------------*/
45662cb04cSPoul-Henning Kamp 
46662cb04cSPoul-Henning Kamp struct fifolog_reader {
47662cb04cSPoul-Henning Kamp 	unsigned		magic;
48662cb04cSPoul-Henning Kamp #define FIFOLOG_READER_MAGIC	0x1036d139
49662cb04cSPoul-Henning Kamp 	struct fifolog_file	*ff;
50662cb04cSPoul-Henning Kamp 	unsigned		olen;
51662cb04cSPoul-Henning Kamp 	unsigned char		*obuf;
52662cb04cSPoul-Henning Kamp 	time_t			now;
53662cb04cSPoul-Henning Kamp };
54662cb04cSPoul-Henning Kamp 
55662cb04cSPoul-Henning Kamp struct fifolog_reader *
fifolog_reader_open(const char * fname)56662cb04cSPoul-Henning Kamp fifolog_reader_open(const char *fname)
57662cb04cSPoul-Henning Kamp {
58662cb04cSPoul-Henning Kamp 	const char *retval;
59662cb04cSPoul-Henning Kamp 	struct fifolog_reader *fr;
60662cb04cSPoul-Henning Kamp 	int i;
61662cb04cSPoul-Henning Kamp 
625ddd0f16SMarcelo Araujo 	fr = calloc(1, sizeof(*fr));
63662cb04cSPoul-Henning Kamp 	if (fr == NULL)
64662cb04cSPoul-Henning Kamp 		err(1, "Cannot malloc");
65662cb04cSPoul-Henning Kamp 
66662cb04cSPoul-Henning Kamp 	retval = fifolog_int_open(&fr->ff, fname, 0);
67662cb04cSPoul-Henning Kamp 	if (retval != NULL)
68662cb04cSPoul-Henning Kamp 		err(1, "%s", retval);
69662cb04cSPoul-Henning Kamp 
7060e600c8SPedro F. Giffuni 	fr->obuf = calloc(16, fr->ff->recsize);
71662cb04cSPoul-Henning Kamp 	if (fr->obuf == NULL)
72662cb04cSPoul-Henning Kamp 		err(1, "Cannot malloc");
7360e600c8SPedro F. Giffuni 	fr->olen = fr->ff->recsize * 16;
74662cb04cSPoul-Henning Kamp 
75662cb04cSPoul-Henning Kamp 	i = inflateInit(fr->ff->zs);
76662cb04cSPoul-Henning Kamp 	assert(i == Z_OK);
77662cb04cSPoul-Henning Kamp 
78662cb04cSPoul-Henning Kamp 	fr->magic = FIFOLOG_READER_MAGIC;
79662cb04cSPoul-Henning Kamp 	return (fr);
80662cb04cSPoul-Henning Kamp }
81662cb04cSPoul-Henning Kamp 
82662cb04cSPoul-Henning Kamp /*
83662cb04cSPoul-Henning Kamp  * Find the next SYNC block
84662cb04cSPoul-Henning Kamp  *
85662cb04cSPoul-Henning Kamp  * Return:
86662cb04cSPoul-Henning Kamp  *	0 - empty fifolog
87662cb04cSPoul-Henning Kamp  *	1 - found sync block
88662cb04cSPoul-Henning Kamp  *	2 - would have wrapped around
89662cb04cSPoul-Henning Kamp  *	3 - End of written log.
90662cb04cSPoul-Henning Kamp  */
91662cb04cSPoul-Henning Kamp 
92662cb04cSPoul-Henning Kamp static int
fifolog_reader_findsync(const struct fifolog_file * ff,off_t * o)93662cb04cSPoul-Henning Kamp fifolog_reader_findsync(const struct fifolog_file *ff, off_t *o)
94662cb04cSPoul-Henning Kamp {
95662cb04cSPoul-Henning Kamp 	int e;
96662cb04cSPoul-Henning Kamp 	unsigned seq, seqs;
97662cb04cSPoul-Henning Kamp 
98662cb04cSPoul-Henning Kamp 	assert(*o < ff->logsize);
99662cb04cSPoul-Henning Kamp 	e = fifolog_int_read(ff, *o);
100662cb04cSPoul-Henning Kamp 	if (e)
101ba2b5429SPoul-Henning Kamp 		err(1, "Read error (%d) while looking for SYNC", e);
102662cb04cSPoul-Henning Kamp 	seq = be32dec(ff->recbuf);
103662cb04cSPoul-Henning Kamp 	if (*o == 0 && seq == 0)
104662cb04cSPoul-Henning Kamp 		return (0);
105662cb04cSPoul-Henning Kamp 
106662cb04cSPoul-Henning Kamp 	if (ff->recbuf[4] & FIFOLOG_FLG_SYNC)
107662cb04cSPoul-Henning Kamp 		return (1);		/* That was easy... */
108662cb04cSPoul-Henning Kamp 	while(1) {
109662cb04cSPoul-Henning Kamp 		assert(*o < ff->logsize);
110662cb04cSPoul-Henning Kamp 		(*o)++;
111662cb04cSPoul-Henning Kamp 		seq++;
112662cb04cSPoul-Henning Kamp 		if (*o == ff->logsize)
113662cb04cSPoul-Henning Kamp 			return (2);	/* wraparound */
114662cb04cSPoul-Henning Kamp 		e = fifolog_int_read(ff, *o);
115662cb04cSPoul-Henning Kamp 		if (e)
116ba2b5429SPoul-Henning Kamp 			err(1, "Read error (%d) while looking for SYNC", e);
117662cb04cSPoul-Henning Kamp 		seqs = be32dec(ff->recbuf);
118662cb04cSPoul-Henning Kamp 		if (seqs != seq)
119662cb04cSPoul-Henning Kamp 			return (3);		/* End of log */
120662cb04cSPoul-Henning Kamp 		if (ff->recbuf[4] & FIFOLOG_FLG_SYNC)
121662cb04cSPoul-Henning Kamp 			return (1);		/* Bingo! */
122662cb04cSPoul-Henning Kamp 	}
123662cb04cSPoul-Henning Kamp }
124662cb04cSPoul-Henning Kamp 
125662cb04cSPoul-Henning Kamp /*
126662cb04cSPoul-Henning Kamp  * Seek out a given timestamp
127662cb04cSPoul-Henning Kamp  */
128662cb04cSPoul-Henning Kamp 
129662cb04cSPoul-Henning Kamp off_t
fifolog_reader_seek(const struct fifolog_reader * fr,time_t t0)130662cb04cSPoul-Henning Kamp fifolog_reader_seek(const struct fifolog_reader *fr, time_t t0)
131662cb04cSPoul-Henning Kamp {
132662cb04cSPoul-Henning Kamp 	off_t o, s, st;
133662cb04cSPoul-Henning Kamp 	time_t t, tt;
134662cb04cSPoul-Henning Kamp 	unsigned seq, seqs;
135662cb04cSPoul-Henning Kamp 	const char *retval;
136662cb04cSPoul-Henning Kamp 	int e;
137662cb04cSPoul-Henning Kamp 
138662cb04cSPoul-Henning Kamp 	CHECK_OBJ_NOTNULL(fr, FIFOLOG_READER_MAGIC);
139662cb04cSPoul-Henning Kamp 
140662cb04cSPoul-Henning Kamp 	/*
141662cb04cSPoul-Henning Kamp 	 * First, find the first SYNC block
142662cb04cSPoul-Henning Kamp 	 */
143662cb04cSPoul-Henning Kamp 	o = 0;
144662cb04cSPoul-Henning Kamp 	e = fifolog_reader_findsync(fr->ff, &o);
145662cb04cSPoul-Henning Kamp 	if (e == 0)
146662cb04cSPoul-Henning Kamp 		return (0);			/* empty fifolog */
147662cb04cSPoul-Henning Kamp 	assert(e == 1);
148662cb04cSPoul-Henning Kamp 
149662cb04cSPoul-Henning Kamp 	assert(fr->ff->recbuf[4] & FIFOLOG_FLG_SYNC);
150662cb04cSPoul-Henning Kamp 	seq = be32dec(fr->ff->recbuf);
151662cb04cSPoul-Henning Kamp 	t = be32dec(fr->ff->recbuf + 5);
152662cb04cSPoul-Henning Kamp 
153662cb04cSPoul-Henning Kamp 	if (t > t0) {
154662cb04cSPoul-Henning Kamp 		/* Check if there is a second older part we can use */
155662cb04cSPoul-Henning Kamp 		retval = fifolog_int_findend(fr->ff, &s);
156662cb04cSPoul-Henning Kamp 		if (retval != NULL)
157662cb04cSPoul-Henning Kamp 			err(1, "%s", retval);
1589e7948d8SPoul-Henning Kamp 		s++;
159662cb04cSPoul-Henning Kamp 		e = fifolog_reader_findsync(fr->ff, &s);
160662cb04cSPoul-Henning Kamp 		if (e == 0)
161662cb04cSPoul-Henning Kamp 			return (0);		/* empty fifolog */
162662cb04cSPoul-Henning Kamp 		if (e == 1) {
163662cb04cSPoul-Henning Kamp 			o = s;
164662cb04cSPoul-Henning Kamp 			seq = be32dec(fr->ff->recbuf);
165662cb04cSPoul-Henning Kamp 			t = be32dec(fr->ff->recbuf + 5);
166662cb04cSPoul-Henning Kamp 		}
167662cb04cSPoul-Henning Kamp 	}
168662cb04cSPoul-Henning Kamp 
169662cb04cSPoul-Henning Kamp 	/* Now do a binary search to find the sync block right before t0 */
170662cb04cSPoul-Henning Kamp 	s = st = (fr->ff->logsize - o) / 2;
171662cb04cSPoul-Henning Kamp 	while (s > 1) {
172662cb04cSPoul-Henning Kamp 		/* We know we shouldn't wrap */
173662cb04cSPoul-Henning Kamp 		if (o + st > fr->ff->logsize + 1) {
174662cb04cSPoul-Henning Kamp 			s = st = s / 2;
175662cb04cSPoul-Henning Kamp 			continue;
176662cb04cSPoul-Henning Kamp 		}
177662cb04cSPoul-Henning Kamp 		e = fifolog_int_read(fr->ff, o + st);
178ba2b5429SPoul-Henning Kamp 		if (e) {
179ba2b5429SPoul-Henning Kamp 			s = st = s / 2;
180ba2b5429SPoul-Henning Kamp 			continue;
181ba2b5429SPoul-Henning Kamp 		}
182662cb04cSPoul-Henning Kamp 		/* If not in same part, sequence won't match */
183662cb04cSPoul-Henning Kamp 		seqs = be32dec(fr->ff->recbuf);
184662cb04cSPoul-Henning Kamp 		if (seqs != seq + st) {
185662cb04cSPoul-Henning Kamp 			s = st = s / 2;
186662cb04cSPoul-Henning Kamp 			continue;
187662cb04cSPoul-Henning Kamp 		}
188662cb04cSPoul-Henning Kamp 		/* If not sync block, try next */
189662cb04cSPoul-Henning Kamp 		if (!(fr->ff->recbuf[4] & FIFOLOG_FLG_SYNC)) {
190662cb04cSPoul-Henning Kamp 			st++;
191662cb04cSPoul-Henning Kamp 			continue;
192662cb04cSPoul-Henning Kamp 		}
193662cb04cSPoul-Henning Kamp 		/* Check timestamp */
194662cb04cSPoul-Henning Kamp 		tt = be32dec(fr->ff->recbuf + 5);
195662cb04cSPoul-Henning Kamp 		if (tt >= t0) {
196662cb04cSPoul-Henning Kamp 			s = st = s / 2;
197662cb04cSPoul-Henning Kamp 			continue;
198662cb04cSPoul-Henning Kamp 		}
199662cb04cSPoul-Henning Kamp 		o += st;
200662cb04cSPoul-Henning Kamp 		seq = seqs;
201662cb04cSPoul-Henning Kamp 	}
202662cb04cSPoul-Henning Kamp 	fprintf(stderr, "Read from %jx\n", o * fr->ff->recsize);
203662cb04cSPoul-Henning Kamp 	return (o);
204662cb04cSPoul-Henning Kamp }
205662cb04cSPoul-Henning Kamp 
206662cb04cSPoul-Henning Kamp static unsigned char *
fifolog_reader_chop(struct fifolog_reader * fr,fifolog_reader_render_t * func,void * priv)207662cb04cSPoul-Henning Kamp fifolog_reader_chop(struct fifolog_reader *fr, fifolog_reader_render_t *func, void *priv)
208662cb04cSPoul-Henning Kamp {
209662cb04cSPoul-Henning Kamp 	u_char *p, *q;
210662cb04cSPoul-Henning Kamp 	uint32_t v, w, u;
211662cb04cSPoul-Henning Kamp 
212662cb04cSPoul-Henning Kamp 	p = fr->obuf;
213662cb04cSPoul-Henning Kamp 	q = fr->obuf + (fr->olen - fr->ff->zs->avail_out);
214662cb04cSPoul-Henning Kamp 
215662cb04cSPoul-Henning Kamp 	while (1) {
216662cb04cSPoul-Henning Kamp 		/* Make sure we have a complete header */
217662cb04cSPoul-Henning Kamp 		if (p + 5 >= q)
218662cb04cSPoul-Henning Kamp 			return (p);
219662cb04cSPoul-Henning Kamp 		w = 4;
220662cb04cSPoul-Henning Kamp 		u = be32dec(p);
221662cb04cSPoul-Henning Kamp 		if (u & FIFOLOG_TIMESTAMP) {
222662cb04cSPoul-Henning Kamp 			fr->now = be32dec(p + 4);
223662cb04cSPoul-Henning Kamp 			w += 4;
224662cb04cSPoul-Henning Kamp 		}
225662cb04cSPoul-Henning Kamp 		if (u & FIFOLOG_LENGTH) {
226662cb04cSPoul-Henning Kamp 			v = p[w];
227662cb04cSPoul-Henning Kamp 			w++;
228c6bfb71aSPoul-Henning Kamp 			if (p + w + v >= q)
229c6bfb71aSPoul-Henning Kamp 				return (p);
230662cb04cSPoul-Henning Kamp 		} else {
231662cb04cSPoul-Henning Kamp 			for (v = 0; p + v + w < q && p[v + w] != '\0'; v++)
232662cb04cSPoul-Henning Kamp 				continue;
233662cb04cSPoul-Henning Kamp 			if (p + v + w >= q)
234662cb04cSPoul-Henning Kamp 				return (p);
235662cb04cSPoul-Henning Kamp 			v++;
236662cb04cSPoul-Henning Kamp 		}
237662cb04cSPoul-Henning Kamp 		func(priv, fr->now, u, p + w, v);
238662cb04cSPoul-Henning Kamp 		p += w + v;
239662cb04cSPoul-Henning Kamp 	}
240662cb04cSPoul-Henning Kamp }
241662cb04cSPoul-Henning Kamp 
242662cb04cSPoul-Henning Kamp /*
243662cb04cSPoul-Henning Kamp  * Process fifolog until end of written log or provided timestamp
244662cb04cSPoul-Henning Kamp  */
245662cb04cSPoul-Henning Kamp 
246662cb04cSPoul-Henning Kamp void
fifolog_reader_process(struct fifolog_reader * fr,off_t from,fifolog_reader_render_t * func,void * priv,time_t end)247662cb04cSPoul-Henning Kamp fifolog_reader_process(struct fifolog_reader *fr, off_t from, fifolog_reader_render_t *func, void *priv, time_t end)
248662cb04cSPoul-Henning Kamp {
249662cb04cSPoul-Henning Kamp 	uint32_t seq, lseq;
250662cb04cSPoul-Henning Kamp 	off_t o = from;
251662cb04cSPoul-Henning Kamp 	int i, e;
252662cb04cSPoul-Henning Kamp 	time_t t;
253662cb04cSPoul-Henning Kamp 	u_char *p, *q;
254662cb04cSPoul-Henning Kamp 	z_stream *zs;
255662cb04cSPoul-Henning Kamp 
256662cb04cSPoul-Henning Kamp 	CHECK_OBJ_NOTNULL(fr, FIFOLOG_READER_MAGIC);
257662cb04cSPoul-Henning Kamp 	zs = fr->ff->zs;
258662cb04cSPoul-Henning Kamp 	lseq = 0;
259662cb04cSPoul-Henning Kamp 	while (1) {
260662cb04cSPoul-Henning Kamp 		e = fifolog_int_read(fr->ff, o);
261662cb04cSPoul-Henning Kamp 		if (e)
262ba2b5429SPoul-Henning Kamp 			err(1, "Read error (%d)", e);
263662cb04cSPoul-Henning Kamp 		if (++o >= fr->ff->logsize)
264662cb04cSPoul-Henning Kamp 			o = 0;
265662cb04cSPoul-Henning Kamp 		seq = be32dec(fr->ff->recbuf);
266662cb04cSPoul-Henning Kamp 		if (lseq != 0 && seq != lseq + 1)
267662cb04cSPoul-Henning Kamp 			break;
268662cb04cSPoul-Henning Kamp 		lseq = seq;
269662cb04cSPoul-Henning Kamp 		zs->avail_in = fr->ff->recsize - 5;
270662cb04cSPoul-Henning Kamp 		zs->next_in = fr->ff->recbuf + 5;
271662cb04cSPoul-Henning Kamp 		if (fr->ff->recbuf[4] & FIFOLOG_FLG_1BYTE)
272662cb04cSPoul-Henning Kamp 			zs->avail_in -= fr->ff->recbuf[fr->ff->recsize - 1];
273662cb04cSPoul-Henning Kamp 		if (fr->ff->recbuf[4] & FIFOLOG_FLG_4BYTE)
274662cb04cSPoul-Henning Kamp 			zs->avail_in -=
275662cb04cSPoul-Henning Kamp 			    be32dec(fr->ff->recbuf + fr->ff->recsize - 4);
276662cb04cSPoul-Henning Kamp 		if (fr->ff->recbuf[4] & FIFOLOG_FLG_SYNC) {
277662cb04cSPoul-Henning Kamp 			i = inflateReset(zs);
278662cb04cSPoul-Henning Kamp 			assert(i == Z_OK);
279662cb04cSPoul-Henning Kamp 			zs->next_out = fr->obuf;
280662cb04cSPoul-Henning Kamp 			zs->avail_out = fr->olen;
281662cb04cSPoul-Henning Kamp 			t = be32dec(fr->ff->recbuf + 5);
282662cb04cSPoul-Henning Kamp 			if (t > end)
283662cb04cSPoul-Henning Kamp 				break;
284662cb04cSPoul-Henning Kamp 			zs->next_in += 4;
285662cb04cSPoul-Henning Kamp 			zs->avail_in -= 4;
286662cb04cSPoul-Henning Kamp 		}
287662cb04cSPoul-Henning Kamp 
288662cb04cSPoul-Henning Kamp 		while(zs->avail_in > 0) {
289662cb04cSPoul-Henning Kamp 			i = inflate(zs, 0);
290662cb04cSPoul-Henning Kamp 			if (i == Z_BUF_ERROR) {
291662cb04cSPoul-Henning Kamp #if 1
292662cb04cSPoul-Henning Kamp 				fprintf(stderr,
293662cb04cSPoul-Henning Kamp 				    "Z_BUF_ERROR [%d,%d] [%d,%d,%d]\n",
294662cb04cSPoul-Henning Kamp 				    (int)(zs->next_in - fr->ff->recbuf),
295662cb04cSPoul-Henning Kamp 				    zs->avail_in,
296662cb04cSPoul-Henning Kamp 				    (int)(zs->next_out - fr->obuf),
297662cb04cSPoul-Henning Kamp 				    zs->avail_out, fr->olen);
298662cb04cSPoul-Henning Kamp 				exit (250);
299662cb04cSPoul-Henning Kamp #else
300662cb04cSPoul-Henning Kamp 
301662cb04cSPoul-Henning Kamp 				i = Z_OK;
302662cb04cSPoul-Henning Kamp #endif
303662cb04cSPoul-Henning Kamp 			}
304662cb04cSPoul-Henning Kamp 			if (i == Z_STREAM_END) {
305662cb04cSPoul-Henning Kamp 				i = inflateReset(zs);
306662cb04cSPoul-Henning Kamp 			}
30736dc3731SGiorgos Keramidas 			if (i != Z_OK) {
308662cb04cSPoul-Henning Kamp 				fprintf(stderr, "inflate = %d\n", i);
30936dc3731SGiorgos Keramidas 				exit (250);
31036dc3731SGiorgos Keramidas 			}
311662cb04cSPoul-Henning Kamp 			assert(i == Z_OK);
312662cb04cSPoul-Henning Kamp 			if (zs->avail_out != fr->olen) {
313662cb04cSPoul-Henning Kamp 				q = fr->obuf + (fr->olen - zs->avail_out);
314662cb04cSPoul-Henning Kamp 				p = fifolog_reader_chop(fr, func, priv);
315662cb04cSPoul-Henning Kamp 				if (p < q)
316662cb04cSPoul-Henning Kamp 					(void)memmove(fr->obuf, p, q - p);
317662cb04cSPoul-Henning Kamp 				zs->avail_out = fr->olen - (q - p);
318662cb04cSPoul-Henning Kamp 				zs->next_out = fr->obuf + (q - p);
319662cb04cSPoul-Henning Kamp 			}
320662cb04cSPoul-Henning Kamp 		}
321662cb04cSPoul-Henning Kamp 	}
322662cb04cSPoul-Henning Kamp }
323