Blame src/lib/libast/string/tokline.c

Packit Service a8c26c
/***********************************************************************
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*               This software is part of the ast package               *
Packit Service a8c26c
*          Copyright (c) 1985-2011 AT&T Intellectual Property          *
Packit Service a8c26c
*                      and is licensed under the                       *
Packit Service a8c26c
*                 Eclipse Public License, Version 1.0                  *
Packit Service a8c26c
*                    by AT&T Intellectual Property                     *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*                A copy of the License is available at                 *
Packit Service a8c26c
*          http://www.eclipse.org/org/documents/epl-v10.html           *
Packit Service a8c26c
*         (with md5 checksum b35adb5213ca9657e911e9befb180842)         *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*              Information and Software Systems Research               *
Packit Service a8c26c
*                            AT&T Research                             *
Packit Service a8c26c
*                           Florham Park NJ                            *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*                 Glenn Fowler <gsf@research.att.com>                  *
Packit Service a8c26c
*                  David Korn <dgk@research.att.com>                   *
Packit Service a8c26c
*                   Phong Vo <kpv@research.att.com>                    *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
***********************************************************************/
Packit Service a8c26c
#pragma prototyped
Packit Service a8c26c
/*
Packit Service a8c26c
 * Glenn Fowler
Packit Service a8c26c
 * AT&T Research
Packit Service a8c26c
 *
Packit Service a8c26c
 * return an Sfio_t* to a file or string that
Packit Service a8c26c
 *
Packit Service a8c26c
 *	splices \\n to single lines
Packit Service a8c26c
 *	checks for "..." and '...' spanning newlines
Packit Service a8c26c
 *	drops #...\n comments
Packit Service a8c26c
 *
Packit Service a8c26c
 * if <arg> is a file and first line matches
Packit Service a8c26c
 *	#!!! <level> <message> !!!
Packit Service a8c26c
 * then error(<lev>,"%s: %s",<arg>,<msg>) called
Packit Service a8c26c
 *
Packit Service a8c26c
 * NOTE: seek disabled and string disciplines cannot be composed
Packit Service a8c26c
 *	 quoted \n translated to \r
Packit Service a8c26c
 */
Packit Service a8c26c
Packit Service a8c26c
#include <ast.h>
Packit Service a8c26c
#include <error.h>
Packit Service a8c26c
#include <tok.h>
Packit Service a8c26c
Packit Service a8c26c
typedef struct
Packit Service a8c26c
{
Packit Service a8c26c
	Sfdisc_t	disc;
Packit Service a8c26c
	Sfio_t*		sp;
Packit Service a8c26c
	int		quote;
Packit Service a8c26c
	int*		line;
Packit Service a8c26c
} Splice_t;
Packit Service a8c26c
Packit Service a8c26c
/*
Packit Service a8c26c
 * the splicer
Packit Service a8c26c
 */
Packit Service a8c26c
Packit Service a8c26c
static int
Packit Service a8c26c
spliceline(Sfio_t* s, int op, void* val, Sfdisc_t* ad)
Packit Service a8c26c
{
Packit Service a8c26c
	Splice_t*	d = (Splice_t*)ad;
Packit Service a8c26c
	register char*	b;
Packit Service a8c26c
	register int	c;
Packit Service a8c26c
	register int	n;
Packit Service a8c26c
	register int	q;
Packit Service a8c26c
	register int	j;
Packit Service a8c26c
	register char*	e;
Packit Service a8c26c
	char*		buf;
Packit Service a8c26c
Packit Service a8c26c
	NoP(val);
Packit Service a8c26c
	switch (op)
Packit Service a8c26c
	{
Packit Service a8c26c
	case SF_CLOSING:
Packit Service a8c26c
		sfclose(d->sp);
Packit Service a8c26c
		return 0;
Packit Service a8c26c
	case SF_DPOP:
Packit Service a8c26c
		free(d);
Packit Service a8c26c
		return 0;
Packit Service a8c26c
	case SF_READ:
Packit Service a8c26c
		do
Packit Service a8c26c
		{
Packit Service a8c26c
			if (!(buf = sfgetr(d->sp, '\n', 0)) && !(buf = sfgetr(d->sp, '\n', -1)))
Packit Service a8c26c
				return 0;
Packit Service a8c26c
			n = sfvalue(d->sp);
Packit Service a8c26c
			q = d->quote;
Packit Service a8c26c
			j = 0;
Packit Service a8c26c
			(*d->line)++;
Packit Service a8c26c
			if (n > 1 && buf[n - 2] == '\\')
Packit Service a8c26c
			{
Packit Service a8c26c
				j = 1;
Packit Service a8c26c
				n -= 2;
Packit Service a8c26c
				if (q == '#')
Packit Service a8c26c
				{
Packit Service a8c26c
					n = 0;
Packit Service a8c26c
					continue;
Packit Service a8c26c
				}
Packit Service a8c26c
			}
Packit Service a8c26c
			else if (q == '#')
Packit Service a8c26c
			{
Packit Service a8c26c
				q = 0;
Packit Service a8c26c
				n = 0;
Packit Service a8c26c
				continue;
Packit Service a8c26c
			}
Packit Service a8c26c
			if (n > 0)
Packit Service a8c26c
			{
Packit Service a8c26c
				e = (b = buf) + n;
Packit Service a8c26c
				while (b < e)
Packit Service a8c26c
				{
Packit Service a8c26c
					if ((c = *b++) == '\\')
Packit Service a8c26c
						b++;
Packit Service a8c26c
					else if (c == q)
Packit Service a8c26c
						q = 0;
Packit Service a8c26c
					else if (!q)
Packit Service a8c26c
					{
Packit Service a8c26c
						if (c == '\'' || c == '"')
Packit Service a8c26c
							q = c;
Packit Service a8c26c
						else if (c == '#' && (b == (buf + 1) || (c = *(b - 2)) == ' ' || c == '\t'))
Packit Service a8c26c
						{
Packit Service a8c26c
							if (buf[n - 1] != '\n')
Packit Service a8c26c
							{
Packit Service a8c26c
								q = '#';
Packit Service a8c26c
								n = b - buf - 2;
Packit Service a8c26c
							}
Packit Service a8c26c
							else if (n = b - buf - 1)
Packit Service a8c26c
								buf[n - 1] = '\n';
Packit Service a8c26c
							break;
Packit Service a8c26c
						}
Packit Service a8c26c
					}
Packit Service a8c26c
				}
Packit Service a8c26c
				if (n > 0)
Packit Service a8c26c
				{
Packit Service a8c26c
					if (!j && buf[n - 1] != '\n' && (s->_flags & SF_STRING))
Packit Service a8c26c
						buf[n++] = '\n';
Packit Service a8c26c
					if (q && buf[n - 1] == '\n')
Packit Service a8c26c
						buf[n - 1] = '\r';
Packit Service a8c26c
				}
Packit Service a8c26c
			}
Packit Service a8c26c
		} while (n <= 0);
Packit Service a8c26c
		sfsetbuf(s, buf, n);
Packit Service a8c26c
		d->quote = q;
Packit Service a8c26c
		return 1;
Packit Service a8c26c
	default:
Packit Service a8c26c
		return 0;
Packit Service a8c26c
	}
Packit Service a8c26c
}
Packit Service a8c26c
Packit Service a8c26c
/*
Packit Service a8c26c
 * open a stream to parse lines
Packit Service a8c26c
 *
Packit Service a8c26c
 *	flags: 0		arg: open Sfio_t* 
Packit Service a8c26c
 *	flags: SF_READ		arg: file name
Packit Service a8c26c
 *	flags: SF_STRING	arg: null terminated char*
Packit Service a8c26c
 *
Packit Service a8c26c
 * if line!=0 then it points to a line count that starts at 0
Packit Service a8c26c
 * and is incremented for each input line
Packit Service a8c26c
 */
Packit Service a8c26c
Packit Service a8c26c
Sfio_t*
Packit Service a8c26c
tokline(const char* arg, int flags, int* line)
Packit Service a8c26c
{
Packit Service a8c26c
	Sfio_t*		f;
Packit Service a8c26c
	Sfio_t*		s;
Packit Service a8c26c
	Splice_t*	d;
Packit Service a8c26c
	char*		p;
Packit Service a8c26c
	char*		e;
Packit Service a8c26c
Packit Service a8c26c
	static int	hidden;
Packit Service a8c26c
Packit Service a8c26c
	if (!(d = newof(0, Splice_t, 1, 0)))
Packit Service a8c26c
		return 0;
Packit Service a8c26c
	if (!(s = sfopen(NiL, NiL, "s")))
Packit Service a8c26c
	{
Packit Service a8c26c
		free(d);
Packit Service a8c26c
		return 0;
Packit Service a8c26c
	}
Packit Service a8c26c
	if (!(flags & (SF_STRING|SF_READ)))
Packit Service a8c26c
		f = (Sfio_t*)arg;
Packit Service a8c26c
	else if (!(f = sfopen(NiL, arg, (flags & SF_STRING) ? "s" : "r")))
Packit Service a8c26c
	{
Packit Service a8c26c
		free(d);
Packit Service a8c26c
		sfclose(s);
Packit Service a8c26c
		return 0;
Packit Service a8c26c
	}
Packit Service a8c26c
	else if ((p = sfreserve(f, 0, 0)) && sfvalue(f) > 11 && strmatch(p, "#!!! +([-0-9]) *([!\n]) !!!\n*") && (e = strchr(p, '\n')))
Packit Service a8c26c
	{
Packit Service a8c26c
		flags = strtol(p + 5, &p, 10);
Packit Service a8c26c
		error(flags, "%s:%-.*s", arg, e - p - 4, p);
Packit Service a8c26c
	}
Packit Service a8c26c
	d->disc.exceptf = spliceline;
Packit Service a8c26c
	d->sp = f;
Packit Service a8c26c
	*(d->line = line ? line : &hidden) = 0;
Packit Service a8c26c
	sfdisc(s, (Sfdisc_t*)d);
Packit Service a8c26c
	return s;
Packit Service a8c26c
}