summaryrefslogtreecommitdiff
path: root/src/lib/libast/string/tokline.c
diff options
context:
space:
mode:
authorIgor Pashev <pashev.igor@gmail.com>2012-06-24 22:28:35 +0000
committerIgor Pashev <pashev.igor@gmail.com>2012-06-24 22:28:35 +0000
commit3950ffe2a485479f6561c27364d3d7df5a21d124 (patch)
tree468c6e14449d1b1e279222ec32f676b0311917d2 /src/lib/libast/string/tokline.c
downloadksh-upstream.tar.gz
Imported Upstream version 93u+upstream
Diffstat (limited to 'src/lib/libast/string/tokline.c')
-rw-r--r--src/lib/libast/string/tokline.c193
1 files changed, 193 insertions, 0 deletions
diff --git a/src/lib/libast/string/tokline.c b/src/lib/libast/string/tokline.c
new file mode 100644
index 0000000..b572651
--- /dev/null
+++ b/src/lib/libast/string/tokline.c
@@ -0,0 +1,193 @@
+/***********************************************************************
+* *
+* This software is part of the ast package *
+* Copyright (c) 1985-2011 AT&T Intellectual Property *
+* and is licensed under the *
+* Eclipse Public License, Version 1.0 *
+* by AT&T Intellectual Property *
+* *
+* A copy of the License is available at *
+* http://www.eclipse.org/org/documents/epl-v10.html *
+* (with md5 checksum b35adb5213ca9657e911e9befb180842) *
+* *
+* Information and Software Systems Research *
+* AT&T Research *
+* Florham Park NJ *
+* *
+* Glenn Fowler <gsf@research.att.com> *
+* David Korn <dgk@research.att.com> *
+* Phong Vo <kpv@research.att.com> *
+* *
+***********************************************************************/
+#pragma prototyped
+/*
+ * Glenn Fowler
+ * AT&T Research
+ *
+ * return an Sfio_t* to a file or string that
+ *
+ * splices \\n to single lines
+ * checks for "..." and '...' spanning newlines
+ * drops #...\n comments
+ *
+ * if <arg> is a file and first line matches
+ * #!!! <level> <message> !!!
+ * then error(<lev>,"%s: %s",<arg>,<msg>) called
+ *
+ * NOTE: seek disabled and string disciplines cannot be composed
+ * quoted \n translated to \r
+ */
+
+#include <ast.h>
+#include <error.h>
+#include <tok.h>
+
+typedef struct
+{
+ Sfdisc_t disc;
+ Sfio_t* sp;
+ int quote;
+ int* line;
+} Splice_t;
+
+/*
+ * the splicer
+ */
+
+static int
+spliceline(Sfio_t* s, int op, void* val, Sfdisc_t* ad)
+{
+ Splice_t* d = (Splice_t*)ad;
+ register char* b;
+ register int c;
+ register int n;
+ register int q;
+ register int j;
+ register char* e;
+ char* buf;
+
+ NoP(val);
+ switch (op)
+ {
+ case SF_CLOSING:
+ sfclose(d->sp);
+ return 0;
+ case SF_DPOP:
+ free(d);
+ return 0;
+ case SF_READ:
+ do
+ {
+ if (!(buf = sfgetr(d->sp, '\n', 0)) && !(buf = sfgetr(d->sp, '\n', -1)))
+ return 0;
+ n = sfvalue(d->sp);
+ q = d->quote;
+ j = 0;
+ (*d->line)++;
+ if (n > 1 && buf[n - 2] == '\\')
+ {
+ j = 1;
+ n -= 2;
+ if (q == '#')
+ {
+ n = 0;
+ continue;
+ }
+ }
+ else if (q == '#')
+ {
+ q = 0;
+ n = 0;
+ continue;
+ }
+ if (n > 0)
+ {
+ e = (b = buf) + n;
+ while (b < e)
+ {
+ if ((c = *b++) == '\\')
+ b++;
+ else if (c == q)
+ q = 0;
+ else if (!q)
+ {
+ if (c == '\'' || c == '"')
+ q = c;
+ else if (c == '#' && (b == (buf + 1) || (c = *(b - 2)) == ' ' || c == '\t'))
+ {
+ if (buf[n - 1] != '\n')
+ {
+ q = '#';
+ n = b - buf - 2;
+ }
+ else if (n = b - buf - 1)
+ buf[n - 1] = '\n';
+ break;
+ }
+ }
+ }
+ if (n > 0)
+ {
+ if (!j && buf[n - 1] != '\n' && (s->_flags & SF_STRING))
+ buf[n++] = '\n';
+ if (q && buf[n - 1] == '\n')
+ buf[n - 1] = '\r';
+ }
+ }
+ } while (n <= 0);
+ sfsetbuf(s, buf, n);
+ d->quote = q;
+ return 1;
+ default:
+ return 0;
+ }
+}
+
+/*
+ * open a stream to parse lines
+ *
+ * flags: 0 arg: open Sfio_t*
+ * flags: SF_READ arg: file name
+ * flags: SF_STRING arg: null terminated char*
+ *
+ * if line!=0 then it points to a line count that starts at 0
+ * and is incremented for each input line
+ */
+
+Sfio_t*
+tokline(const char* arg, int flags, int* line)
+{
+ Sfio_t* f;
+ Sfio_t* s;
+ Splice_t* d;
+ char* p;
+ char* e;
+
+ static int hidden;
+
+ if (!(d = newof(0, Splice_t, 1, 0)))
+ return 0;
+ if (!(s = sfopen(NiL, NiL, "s")))
+ {
+ free(d);
+ return 0;
+ }
+ if (!(flags & (SF_STRING|SF_READ)))
+ f = (Sfio_t*)arg;
+ else if (!(f = sfopen(NiL, arg, (flags & SF_STRING) ? "s" : "r")))
+ {
+ free(d);
+ sfclose(s);
+ return 0;
+ }
+ else if ((p = sfreserve(f, 0, 0)) && sfvalue(f) > 11 && strmatch(p, "#!!! +([-0-9]) *([!\n]) !!!\n*") && (e = strchr(p, '\n')))
+ {
+ flags = strtol(p + 5, &p, 10);
+ error(flags, "%s:%-.*s", arg, e - p - 4, p);
+ }
+ d->disc.exceptf = spliceline;
+ d->sp = f;
+ *(d->line = line ? line : &hidden) = 0;
+ sfdisc(s, (Sfdisc_t*)d);
+ return s;
+}