xref: /titanic_50/usr/src/lib/libcmd/common/revlib.c (revision 3e14f97f673e8a630f076077de35afdd43dc1587)
1da2e3ebdSchin /***********************************************************************
2da2e3ebdSchin *                                                                      *
3da2e3ebdSchin *               This software is part of the ast package               *
4*3e14f97fSRoger A. Faulkner *          Copyright (c) 1992-2010 AT&T Intellectual Property          *
5da2e3ebdSchin *                      and is licensed under the                       *
6da2e3ebdSchin *                  Common Public License, Version 1.0                  *
77c2fbfb3SApril Chin *                    by AT&T Intellectual Property                     *
8da2e3ebdSchin *                                                                      *
9da2e3ebdSchin *                A copy of the License is available at                 *
10da2e3ebdSchin *            http://www.opensource.org/licenses/cpl1.0.txt             *
11da2e3ebdSchin *         (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9)         *
12da2e3ebdSchin *                                                                      *
13da2e3ebdSchin *              Information and Software Systems Research               *
14da2e3ebdSchin *                            AT&T Research                             *
15da2e3ebdSchin *                           Florham Park NJ                            *
16da2e3ebdSchin *                                                                      *
17da2e3ebdSchin *                 Glenn Fowler <gsf@research.att.com>                  *
18da2e3ebdSchin *                  David Korn <dgk@research.att.com>                   *
19da2e3ebdSchin *                                                                      *
20da2e3ebdSchin ***********************************************************************/
21da2e3ebdSchin #pragma prototyped
22da2e3ebdSchin /*
23da2e3ebdSchin  * common support for tail and rev
24da2e3ebdSchin  */
25da2e3ebdSchin 
26da2e3ebdSchin #include	<cmd.h>
27da2e3ebdSchin #include	<rev.h>
28da2e3ebdSchin 
29da2e3ebdSchin #define BUFSIZE			SF_BUFSIZE
30da2e3ebdSchin #define rounddown(n,size)	(((n)-1)&~((size)-1))
31da2e3ebdSchin 
32da2e3ebdSchin /*
33da2e3ebdSchin  * copy the lines starting at offset <start> from in <in> to <out>
34da2e3ebdSchin  * in reverse order
35da2e3ebdSchin  */
rev_line(Sfio_t * in,Sfio_t * out,off_t start)36da2e3ebdSchin int rev_line(Sfio_t *in, Sfio_t *out, off_t start)
37da2e3ebdSchin {
38da2e3ebdSchin 	register char *cp, *cpold;
39da2e3ebdSchin 	register int n, nleft=0;
40da2e3ebdSchin 	char buff[BUFSIZE];
41da2e3ebdSchin 	off_t offset;
42da2e3ebdSchin 	if(sfseek(in,(off_t)0,SEEK_CUR) < 0)
43da2e3ebdSchin 	{
44da2e3ebdSchin 		Sfio_t *tmp = sftmp(4*SF_BUFSIZE);
45da2e3ebdSchin 		if(!tmp)
46da2e3ebdSchin 			return(-1);
47da2e3ebdSchin 		if(start>0 && sfmove(in, (Sfio_t*)0, start, -1) != start)
48da2e3ebdSchin 			return(-1);
49da2e3ebdSchin 		if(sfmove(in, tmp, SF_UNBOUND, -1) < 0 || !sfeof(in) || sferror(tmp))
50da2e3ebdSchin 			return(-1);
51da2e3ebdSchin 		in = tmp;
52da2e3ebdSchin 		start=0;
53da2e3ebdSchin 	}
54da2e3ebdSchin 	if((offset = sfseek(in,(off_t)0,SEEK_END)) <= start)
55da2e3ebdSchin 		return(0);
56da2e3ebdSchin 	offset = rounddown(offset,BUFSIZE);
57da2e3ebdSchin 	while(1)
58da2e3ebdSchin 	{
59da2e3ebdSchin 		n = BUFSIZE;
60da2e3ebdSchin 		if(offset < start)
61da2e3ebdSchin 		{
62da2e3ebdSchin 			n -= (start-offset);
63da2e3ebdSchin 			offset = start;
64da2e3ebdSchin 		}
65da2e3ebdSchin 		sfseek(in, offset, SEEK_SET);
66da2e3ebdSchin 		if((n=sfread(in, buff, n)) <=0)
67da2e3ebdSchin 			break;
68da2e3ebdSchin 		cp = buff+n;
69da2e3ebdSchin 		n = *buff;
70da2e3ebdSchin 		*buff = '\n';
71da2e3ebdSchin 		while(1)
72da2e3ebdSchin 		{
73da2e3ebdSchin 			cpold = cp;
74da2e3ebdSchin 			if(nleft==0)
75da2e3ebdSchin 				cp--;
76da2e3ebdSchin 			if(cp==buff)
77da2e3ebdSchin 			{
78da2e3ebdSchin 				nleft= 1;
79da2e3ebdSchin 				break;
80da2e3ebdSchin 			}
81da2e3ebdSchin 			while(*--cp != '\n');
82da2e3ebdSchin 			if(cp==buff && n!='\n')
83da2e3ebdSchin 			{
84da2e3ebdSchin 				*cp = n;
85da2e3ebdSchin 				nleft += cpold-cp;
86da2e3ebdSchin 				break;
87da2e3ebdSchin 			}
88da2e3ebdSchin 			else
89da2e3ebdSchin 				cp++;
90da2e3ebdSchin 			if(sfwrite(out,cp,cpold-cp) < 0)
91da2e3ebdSchin 				return(-1);
92da2e3ebdSchin 			if(nleft)
93da2e3ebdSchin 			{
94da2e3ebdSchin 				if(nleft==1)
95da2e3ebdSchin 					sfputc(out,'\n');
96da2e3ebdSchin 				else if(sfmove(in,out,nleft,-1) != nleft)
97da2e3ebdSchin 					return(-1);
98da2e3ebdSchin 				nleft = 0;
99da2e3ebdSchin 			}
100da2e3ebdSchin 		}
101da2e3ebdSchin 		if(offset <= start)
102da2e3ebdSchin 			break;
103da2e3ebdSchin 		offset -= BUFSIZE;
104da2e3ebdSchin 	}
105da2e3ebdSchin 	if(nleft)
106da2e3ebdSchin 	{
107da2e3ebdSchin 		sfseek(in, start, SEEK_SET);
108da2e3ebdSchin 		if(sfmove(in,out,nleft,-1) != nleft)
109da2e3ebdSchin 			return(-1);
110da2e3ebdSchin 	}
111da2e3ebdSchin 	return(0);
112da2e3ebdSchin }
113