Merge pull request #8 from arnoldrobbins/master
Fix multiple long-standing bugs, improve test suite.
This commit is contained in:
commit
b619480f94
225
ChangeLog
Normal file
225
ChangeLog
Normal file
@ -0,0 +1,225 @@
|
|||||||
|
2018-08-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* awktest.tar (testdir/T.expr): Fix test for unary plus.
|
||||||
|
|
||||||
|
2018-08-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* REGRESS: Extract tests if necessary, set PATH to include '.'.
|
||||||
|
* regdir/beebe.tar (Makefile): Fix longwrds test to prefix
|
||||||
|
sort with LC_ALL=C.
|
||||||
|
* awktest.tar: Updated from fixed test suite, directory
|
||||||
|
it extracts is now called 'testdir' to match what's in top-level
|
||||||
|
REGRESS script.
|
||||||
|
* regdir: Removed, as Brian wants to keep the test suite in
|
||||||
|
the tar file.
|
||||||
|
|
||||||
|
2018-08-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* FIXES, lib.c, run.c, makefile, main.c: Merge from Brian's tree.
|
||||||
|
* REGRESS: New file, from Brian.
|
||||||
|
* awktest.tar: Restored from Brian's tree.
|
||||||
|
|
||||||
|
2018-08-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* awkgram.y (UPLUS): New token. In the grammar, call op1()
|
||||||
|
with it.
|
||||||
|
* maketab.c (proc): Add entry for UPLUS.
|
||||||
|
* run.c (arith): Handle UPLUS.
|
||||||
|
* main.c (version): Updated.
|
||||||
|
* bugs-fixed/unary-plus.awk, bugs-fixed/unary-plus.bad,
|
||||||
|
bugs-fixed/unary-plus.ok: New files.
|
||||||
|
|
||||||
|
2018-08-10 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* TODO: Updated.
|
||||||
|
* awk.1: Improve use of macros, add some additional explanation
|
||||||
|
in a few places, alphabetize list of variables.
|
||||||
|
|
||||||
|
2018-08-08 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* awk.h (Cell): Add new field `fmt' to track xFMT value used
|
||||||
|
for a string conversion.
|
||||||
|
[CONVC, CONVO]: New flag macros.
|
||||||
|
* bugs-fixed/README: Updated.
|
||||||
|
* bugs-fixed/string-conv.awk, bugs-fixed/string-conv.bad,
|
||||||
|
bugs-fixed/string-conv.ok: New files.
|
||||||
|
* main.c (version): Updated.
|
||||||
|
* proto.h (flags2str): Add declaration.
|
||||||
|
* tran.c (setfval): Clear CONVC and CONVO flags and set vp->fmt
|
||||||
|
to NULL.
|
||||||
|
(setsval): Ditto. Add large comment and new code to manage
|
||||||
|
correct conversion of number to string based on various flags
|
||||||
|
and the value of vp->fmt. The idea is to not convert again
|
||||||
|
if xFMT is the same as before and we're doing the same conversion.
|
||||||
|
Otherwise, clear the old flags, set the new, and reconvert.
|
||||||
|
(flags2str): New function. For debug prints and for use from a debugger.
|
||||||
|
|
||||||
|
2018-08-05 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
Fix filename conflicts in regdir where the only difference was
|
||||||
|
in letter case. This caused problems on Windows systems.
|
||||||
|
|
||||||
|
* regdir/Compare.T1: Renamed from regdir/Compare.T.
|
||||||
|
* regdir/t.delete0: Renamed from regdir/t.delete.
|
||||||
|
* regdir/t.getline1: Renamed from regdir/t.getline.
|
||||||
|
* regdir/t.redir1: Renamed from regdir/t.redir.
|
||||||
|
* regdir/t.split1: Renamed from regdir/t.split.
|
||||||
|
* regdir/t.sub0: Renamed from regdir/t.sub.
|
||||||
|
* regdir/REGRESS: Adjusted.
|
||||||
|
|
||||||
|
2018-08-04 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
With scalpel, tweasers, magnifying glass and bated breath,
|
||||||
|
borrow code from the NetBSD version of nawk to fix the years-old
|
||||||
|
bug whereby decrementing the value of NF did not change the
|
||||||
|
record.
|
||||||
|
|
||||||
|
* lib.c (fldbld): Set donerec to 1 when done.
|
||||||
|
(setlastfld): New function.
|
||||||
|
* proto.h (setlastfld): Add declaration.
|
||||||
|
* run.c (copycell): Make code smarter about flags (from NetBSD code).
|
||||||
|
* tran.c (setfree): New function.
|
||||||
|
* tran.c (setfval): Normalize negative zero to positive zero.
|
||||||
|
If setting NF, clear donerec and call setlastfld().
|
||||||
|
(setsval): Remove call to save_old_OFS(). If setting OFS, call
|
||||||
|
recbld(). If setting NF, clear donerec and call setlastfld().
|
||||||
|
|
||||||
|
As part of the process, revert OFS-related changes of 2018-05-22:
|
||||||
|
|
||||||
|
* awk.h (saveOFS, saveOFSlen, save_old_OFS): Remove declarations.
|
||||||
|
* lib.c (recbld): Use *OFS instead of saveOFS.
|
||||||
|
* run.c (saveOFS, saveOFSlen, save_old_OFS): Remove.
|
||||||
|
* tran.c (syminit): Remove initialization of saveOFS and saveOFSlen.
|
||||||
|
|
||||||
|
General stuff that goes along with all this:
|
||||||
|
|
||||||
|
* bugs-fixed/README: Updated.
|
||||||
|
* bugs-fixed/decr-NF.awk, bugs-fixed/decr-NF.bad,
|
||||||
|
bugs-fixed/decr-NF.ok: New files.
|
||||||
|
* main.c (version): Updated.
|
||||||
|
* regdir/README.TESTS: Fix awk book title.
|
||||||
|
* regdir/T.misc: Revise test to match fixed code.
|
||||||
|
* run.c (format): Increase size of buffer used for %a test. (Unrelated
|
||||||
|
to NF or OFS, but fixes a compiler complaint.)
|
||||||
|
|
||||||
|
2018-06-07 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* regdir/beebe.tar: Fix longwrds.ok so that the test will pass.
|
||||||
|
The file was incorrectly sorted.
|
||||||
|
|
||||||
|
2018-06-06 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* regdir/T.lilly: Fix the bug again in the second instance
|
||||||
|
of the code. Thanks to BWK for pointing this out.
|
||||||
|
|
||||||
|
2018-05-31 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* regdir/T.lilly: Fix a syntax error and ordering bug
|
||||||
|
in creating the 'foo' file.
|
||||||
|
|
||||||
|
2018-05-23 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* awk.1: Remove standalone 'awk' at the top of file, it messed up
|
||||||
|
the formatting. Arrange built-in variable list in alphabetical
|
||||||
|
order.
|
||||||
|
|
||||||
|
2018-05-23 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* main.c (version): Add my email address and a date so that
|
||||||
|
users can tell this isn't straight BWK awk.
|
||||||
|
* README.md: Minor updates.
|
||||||
|
* TODO: Updated.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
Add POSIX-required formats %a and %A.
|
||||||
|
|
||||||
|
* run.c (format): Check for %a support in C library. If there,
|
||||||
|
allow %a and %A as valid formats.
|
||||||
|
* TODO: Updated.
|
||||||
|
* bugs-fixed/README: Updated.
|
||||||
|
* bugs-fixed/a-format.awk, bugs-fixed/a-format.bad,
|
||||||
|
bugs-fixed/a-format.ok: New files.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* FIXES: Restored a line from a much earlier version that
|
||||||
|
apparently got lost when the dates were reordered.
|
||||||
|
* TODO: Updated.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* README.md: New file.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* regdir/echo.c, regdir/time.c: Minor fixes to compile without
|
||||||
|
warning on current GCC / Linux.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* TODO: New file.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* makefile (gitadd, gitpush): Remove these targets. They
|
||||||
|
should not be automated and were incorrect for things that
|
||||||
|
would be done regularly.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
Fix nawk so that [[:blank:]] only matches space and tab instead
|
||||||
|
of any whitespace character, originally made May 10, 2018.
|
||||||
|
See bugs-fixed/space.awk.
|
||||||
|
|
||||||
|
This appears to have been a thinko on Brian's part.
|
||||||
|
|
||||||
|
* b.c (charclasses): Use xisblank() function for [[:blank:]].
|
||||||
|
* bugs-fixed/README: Updated.
|
||||||
|
* bugs-fixed/space.awk, bugs-fixed/space.bad,
|
||||||
|
bugs-fixed/space.ok: New files.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* .gitignore: New file.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
Fix nawk to provide reasonable exit status for system(),
|
||||||
|
a la gawk, originally made March 12, 2016. See
|
||||||
|
bugs-fixed/system-status.awk.
|
||||||
|
|
||||||
|
* run.c (bltin): For FSYSTEM, use the macros defined for wait(2)
|
||||||
|
to produce a reasonable exit value, instead of doing a floating-point
|
||||||
|
division by 256.
|
||||||
|
* awk.1: Document the return status values.
|
||||||
|
* bugs-fixed/README: Updated.
|
||||||
|
* bugs-fixed/system-status.awk, bugs-fixed/system-status.bad,
|
||||||
|
bugs-fixed/system-status.ok: New files.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
Bug fix with respect to rebuilding a record, originally
|
||||||
|
made August 19, 2014. See bugs-fixed/ofs-rebuild.awk.
|
||||||
|
|
||||||
|
* awk.h (saveOFS, saveOFSlen): Declare new variables.
|
||||||
|
* lib.c (recbld): Use them when rebuilding the record.
|
||||||
|
* run.c (saveOFS, saveOFSlen): Define new variables.
|
||||||
|
(save_old_OFS): New function to save OFS aside.
|
||||||
|
* tran.c (syminit): Initialize saveOFS and saveOFSlen.
|
||||||
|
(setsval): If setting a field, call save_old_OFS().
|
||||||
|
* bugs-fixed/README, bugs-fixed/ofs-rebuild.awk,
|
||||||
|
bugs-fixed/ofs-rebuild.bad, bugs-fixed/ofs-rebuild.ok: New files.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* makefile (YACC): Use bison.
|
||||||
|
|
||||||
|
2018-05-22 Arnold D. Robbins <arnold@skeeve.com>
|
||||||
|
|
||||||
|
* ChangeLog: Created.
|
||||||
|
* regdir: Created. Based on contents of awktest.a.
|
||||||
|
* .gitattributes: Created, to preserve CR LF in regdir/t.crlf.
|
||||||
|
* awktest.a: Removed.
|
||||||
|
* regdir/T.gawk, regdir/T.latin1: Updated from awktest.tar.
|
||||||
|
* awktest.tar: Removed.
|
2
FIXES
2
FIXES
@ -514,6 +514,8 @@ May 12, 1998:
|
|||||||
Mar 12, 1998:
|
Mar 12, 1998:
|
||||||
added -V to print version number and die.
|
added -V to print version number and die.
|
||||||
|
|
||||||
|
[notify dave kerns, dkerns@dacsoup.ih.lucent.com]
|
||||||
|
|
||||||
Feb 11, 1998:
|
Feb 11, 1998:
|
||||||
subtle silent bug in lex.c: if the program ended with a number
|
subtle silent bug in lex.c: if the program ended with a number
|
||||||
longer than 1 digit, part of the input would be pushed back and
|
longer than 1 digit, part of the input would be pushed back and
|
||||||
|
17
REGRESS
17
REGRESS
@ -1,3 +1,20 @@
|
|||||||
|
#! /bin/sh
|
||||||
|
|
||||||
|
if [ -d testdir ]
|
||||||
|
then
|
||||||
|
true # do nothing
|
||||||
|
elif [ -f awktest.tar ]
|
||||||
|
then
|
||||||
|
echo extracting testdir
|
||||||
|
tar -xpf awktest.tar
|
||||||
|
else
|
||||||
|
echo $0: No testdir directory and no awktest.tar to extract it from! >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
cd testdir
|
cd testdir
|
||||||
pwd
|
pwd
|
||||||
|
PATH=.:$PATH
|
||||||
|
export PATH
|
||||||
|
|
||||||
REGRESS
|
REGRESS
|
||||||
|
20
TODO
Normal file
20
TODO
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
Fri Aug 10 11:11:11 IDT 2018
|
||||||
|
============================
|
||||||
|
|
||||||
|
A semi-random list of things to look into.
|
||||||
|
|
||||||
|
1. DONE. Make sure the FIXES file has everything after the date order
|
||||||
|
was reversed.
|
||||||
|
|
||||||
|
2. DONE. Look into the problem that NF-- doesn't throw away the last field.
|
||||||
|
This a long-standing issue.
|
||||||
|
|
||||||
|
3. DONE. Add suport for %a and %A printf formats.
|
||||||
|
|
||||||
|
4. DONE. Look into the problem that string conversions are sticky; once
|
||||||
|
done they never change, even if OFMT or CONVFMT do. Similarly, a
|
||||||
|
conversion from OFMT becomes the permanent string value, which is
|
||||||
|
incorrect.
|
||||||
|
|
||||||
|
5. MAYBE: Improve the test suite enough so that it can be cleaned up after
|
||||||
|
it was run, and old results can be saved for comparison.
|
164
awk.1
164
awk.1
@ -7,7 +7,6 @@
|
|||||||
.fi
|
.fi
|
||||||
.ft 1
|
.ft 1
|
||||||
..
|
..
|
||||||
awk
|
|
||||||
.TH AWK 1
|
.TH AWK 1
|
||||||
.CT 1 files prog_other
|
.CT 1 files prog_other
|
||||||
.SH NAME
|
.SH NAME
|
||||||
@ -36,7 +35,7 @@ awk \- pattern-directed scanning and processing language
|
|||||||
scans each input
|
scans each input
|
||||||
.I file
|
.I file
|
||||||
for lines that match any of a set of patterns specified literally in
|
for lines that match any of a set of patterns specified literally in
|
||||||
.IR prog
|
.I prog
|
||||||
or in one or more files
|
or in one or more files
|
||||||
specified as
|
specified as
|
||||||
.B \-f
|
.B \-f
|
||||||
@ -53,7 +52,7 @@ The file name
|
|||||||
.B \-
|
.B \-
|
||||||
means the standard input.
|
means the standard input.
|
||||||
Any
|
Any
|
||||||
.IR file
|
.I file
|
||||||
of the form
|
of the form
|
||||||
.I var=value
|
.I var=value
|
||||||
is treated as an assignment, not a filename,
|
is treated as an assignment, not a filename,
|
||||||
@ -70,12 +69,12 @@ any number of
|
|||||||
options may be present.
|
options may be present.
|
||||||
The
|
The
|
||||||
.B \-F
|
.B \-F
|
||||||
.IR fs
|
.I fs
|
||||||
option defines the input field separator to be the regular expression
|
option defines the input field separator to be the regular expression
|
||||||
.IR fs .
|
.IR fs .
|
||||||
.PP
|
.PP
|
||||||
An input line is normally made up of fields separated by white space,
|
An input line is normally made up of fields separated by white space,
|
||||||
or by regular expression
|
or by the regular expression
|
||||||
.BR FS .
|
.BR FS .
|
||||||
The fields are denoted
|
The fields are denoted
|
||||||
.BR $1 ,
|
.BR $1 ,
|
||||||
@ -87,7 +86,7 @@ If
|
|||||||
.BR FS
|
.BR FS
|
||||||
is null, the input line is split into one field per character.
|
is null, the input line is split into one field per character.
|
||||||
.PP
|
.PP
|
||||||
A pattern-action statement has the form
|
A pattern-action statement has the form:
|
||||||
.IP
|
.IP
|
||||||
.IB pattern " { " action " }
|
.IB pattern " { " action " }
|
||||||
.PP
|
.PP
|
||||||
@ -101,7 +100,7 @@ An action is a sequence of statements.
|
|||||||
A statement can be one of the following:
|
A statement can be one of the following:
|
||||||
.PP
|
.PP
|
||||||
.EX
|
.EX
|
||||||
.ta \w'\f(CWdelete array[expression]'u
|
.ta \w'\f(CWdelete array[expression]\fR'u
|
||||||
.RS
|
.RS
|
||||||
.nf
|
.nf
|
||||||
.ft CW
|
.ft CW
|
||||||
@ -145,7 +144,7 @@ The operators
|
|||||||
are also available in expressions.
|
are also available in expressions.
|
||||||
Variables may be scalars, array elements
|
Variables may be scalars, array elements
|
||||||
(denoted
|
(denoted
|
||||||
.IB x [ i ] )
|
.IB x [ i ] \fR)
|
||||||
or fields.
|
or fields.
|
||||||
Variables are initialized to the null string.
|
Variables are initialized to the null string.
|
||||||
Array subscripts may be any string,
|
Array subscripts may be any string,
|
||||||
@ -161,11 +160,11 @@ The
|
|||||||
.B print
|
.B print
|
||||||
statement prints its arguments on the standard output
|
statement prints its arguments on the standard output
|
||||||
(or on a file if
|
(or on a file if
|
||||||
.BI > file
|
.BI > " file
|
||||||
or
|
or
|
||||||
.BI >> file
|
.BI >> " file
|
||||||
is present or on a pipe if
|
is present or on a pipe if
|
||||||
.BI | cmd
|
.BI | " cmd
|
||||||
is present), separated by the current output field separator,
|
is present), separated by the current output field separator,
|
||||||
and terminated by the output record separator.
|
and terminated by the output record separator.
|
||||||
.I file
|
.I file
|
||||||
@ -176,7 +175,8 @@ identical string values in different statements denote
|
|||||||
the same open file.
|
the same open file.
|
||||||
The
|
The
|
||||||
.B printf
|
.B printf
|
||||||
statement formats its expression list according to the format
|
statement formats its expression list according to the
|
||||||
|
.I format
|
||||||
(see
|
(see
|
||||||
.IR printf (3)).
|
.IR printf (3)).
|
||||||
The built-in function
|
The built-in function
|
||||||
@ -189,13 +189,13 @@ flushes any buffered output for the file or pipe
|
|||||||
.IR expr .
|
.IR expr .
|
||||||
.PP
|
.PP
|
||||||
The mathematical functions
|
The mathematical functions
|
||||||
|
.BR atan2 ,
|
||||||
|
.BR cos ,
|
||||||
.BR exp ,
|
.BR exp ,
|
||||||
.BR log ,
|
.BR log ,
|
||||||
.BR sqrt ,
|
|
||||||
.BR sin ,
|
.BR sin ,
|
||||||
.BR cos ,
|
|
||||||
and
|
and
|
||||||
.BR atan2
|
.B sqrt
|
||||||
are built in.
|
are built in.
|
||||||
Other built-in functions:
|
Other built-in functions:
|
||||||
.TF length
|
.TF length
|
||||||
@ -203,7 +203,8 @@ Other built-in functions:
|
|||||||
.B length
|
.B length
|
||||||
the length of its argument
|
the length of its argument
|
||||||
taken as a string,
|
taken as a string,
|
||||||
or of
|
number of elements in an array for an array argument,
|
||||||
|
or length of
|
||||||
.B $0
|
.B $0
|
||||||
if no argument.
|
if no argument.
|
||||||
.TP
|
.TP
|
||||||
@ -218,14 +219,18 @@ and returns the previous seed.
|
|||||||
.B int
|
.B int
|
||||||
truncates to an integer value
|
truncates to an integer value
|
||||||
.TP
|
.TP
|
||||||
.BI substr( s , " m" , " n\fB)
|
\fBsubstr(\fIs\fB, \fIm\fR [\fB, \fIn\^\fR]\fB)\fR
|
||||||
the
|
the
|
||||||
.IR n -character
|
.IR n -character
|
||||||
substring of
|
substring of
|
||||||
.I s
|
.I s
|
||||||
that begins at position
|
that begins at position
|
||||||
.IR m
|
.I m
|
||||||
counted from 1.
|
counted from 1.
|
||||||
|
If no
|
||||||
|
.IR m ,
|
||||||
|
use the rest of the string
|
||||||
|
.I
|
||||||
.TP
|
.TP
|
||||||
.BI index( s , " t" )
|
.BI index( s , " t" )
|
||||||
the position in
|
the position in
|
||||||
@ -246,14 +251,14 @@ and
|
|||||||
.B RLENGTH
|
.B RLENGTH
|
||||||
are set to the position and length of the matched string.
|
are set to the position and length of the matched string.
|
||||||
.TP
|
.TP
|
||||||
.BI split( s , " a" , " fs\fB)
|
\fBsplit(\fIs\fB, \fIa \fR[\fB, \fIfs\^\fR]\fB)\fR
|
||||||
splits the string
|
splits the string
|
||||||
.I s
|
.I s
|
||||||
into array elements
|
into array elements
|
||||||
.IB a [1] ,
|
.IB a [1] \fR,
|
||||||
.IB a [2] ,
|
.IB a [2] \fR,
|
||||||
\&...,
|
\&...,
|
||||||
.IB a [ n ] ,
|
.IB a [ n ] \fR,
|
||||||
and returns
|
and returns
|
||||||
.IR n .
|
.IR n .
|
||||||
The separation is done with the regular expression
|
The separation is done with the regular expression
|
||||||
@ -266,7 +271,7 @@ is not given.
|
|||||||
An empty string as field separator splits the string
|
An empty string as field separator splits the string
|
||||||
into one array element per character.
|
into one array element per character.
|
||||||
.TP
|
.TP
|
||||||
.BI sub( r , " t" , " s\fB)
|
\fBsub(\fIr\fB, \fIt \fR[, \fIs\^\fR]\fB)
|
||||||
substitutes
|
substitutes
|
||||||
.I t
|
.I t
|
||||||
for the first occurrence of the regular expression
|
for the first occurrence of the regular expression
|
||||||
@ -279,7 +284,7 @@ is not given,
|
|||||||
.B $0
|
.B $0
|
||||||
is used.
|
is used.
|
||||||
.TP
|
.TP
|
||||||
.B gsub
|
\fBgsub(\fIr\fB, \fIt \fR[, \fIs\^\fR]\fB)
|
||||||
same as
|
same as
|
||||||
.B sub
|
.B sub
|
||||||
except that all occurrences of the regular expression
|
except that all occurrences of the regular expression
|
||||||
@ -295,12 +300,22 @@ the string resulting from formatting
|
|||||||
according to the
|
according to the
|
||||||
.IR printf (3)
|
.IR printf (3)
|
||||||
format
|
format
|
||||||
.I fmt
|
.IR fmt .
|
||||||
.TP
|
.TP
|
||||||
.BI system( cmd )
|
.BI system( cmd )
|
||||||
executes
|
executes
|
||||||
.I cmd
|
.I cmd
|
||||||
and returns its exit status
|
and returns its exit status. This will be \-1 upon error,
|
||||||
|
.IR cmd 's
|
||||||
|
exit status upon a normal exit,
|
||||||
|
256 +
|
||||||
|
.I sig
|
||||||
|
upon death-by-signal, where
|
||||||
|
.I sig
|
||||||
|
is the number of the murdering signal,
|
||||||
|
or 512 +
|
||||||
|
.I sig
|
||||||
|
if there was a core dump.
|
||||||
.TP
|
.TP
|
||||||
.BI tolower( str )
|
.BI tolower( str )
|
||||||
returns a copy of
|
returns a copy of
|
||||||
@ -321,7 +336,7 @@ sets
|
|||||||
.B $0
|
.B $0
|
||||||
to the next input record from the current input file;
|
to the next input record from the current input file;
|
||||||
.B getline
|
.B getline
|
||||||
.BI < file
|
.BI < " file
|
||||||
sets
|
sets
|
||||||
.B $0
|
.B $0
|
||||||
to the next record from
|
to the next record from
|
||||||
@ -359,7 +374,7 @@ Isolated regular expressions
|
|||||||
in a pattern apply to the entire line.
|
in a pattern apply to the entire line.
|
||||||
Regular expressions may also occur in
|
Regular expressions may also occur in
|
||||||
relational expressions, using the operators
|
relational expressions, using the operators
|
||||||
.BR ~
|
.B ~
|
||||||
and
|
and
|
||||||
.BR !~ .
|
.BR !~ .
|
||||||
.BI / re /
|
.BI / re /
|
||||||
@ -383,8 +398,12 @@ A relational expression is one of the following:
|
|||||||
.br
|
.br
|
||||||
.BI ( expr , expr,... ") in " array-name
|
.BI ( expr , expr,... ") in " array-name
|
||||||
.PP
|
.PP
|
||||||
where a relop is any of the six relational operators in C,
|
where a
|
||||||
and a matchop is either
|
.I relop
|
||||||
|
is any of the six relational operators in C,
|
||||||
|
and a
|
||||||
|
.I matchop
|
||||||
|
is either
|
||||||
.B ~
|
.B ~
|
||||||
(matches)
|
(matches)
|
||||||
or
|
or
|
||||||
@ -405,57 +424,68 @@ and after the last.
|
|||||||
and
|
and
|
||||||
.B END
|
.B END
|
||||||
do not combine with other patterns.
|
do not combine with other patterns.
|
||||||
|
They may appear multiple times in a program and execute
|
||||||
|
in the order they are read by
|
||||||
|
.IR awk .
|
||||||
.PP
|
.PP
|
||||||
Variable names with special meanings:
|
Variable names with special meanings:
|
||||||
.TF FILENAME
|
.TF FILENAME
|
||||||
.TP
|
.TP
|
||||||
|
.B ARGC
|
||||||
|
argument count, assignable.
|
||||||
|
.TP
|
||||||
|
.B ARGV
|
||||||
|
argument array, assignable;
|
||||||
|
non-null members are taken as filenames.
|
||||||
|
.TP
|
||||||
.B CONVFMT
|
.B CONVFMT
|
||||||
conversion format used when converting numbers
|
conversion format used when converting numbers
|
||||||
(default
|
(default
|
||||||
.BR "%.6g" )
|
.BR "%.6g" ).
|
||||||
|
.TP
|
||||||
|
.B ENVIRON
|
||||||
|
array of environment variables; subscripts are names.
|
||||||
|
.TP
|
||||||
|
.B FILENAME
|
||||||
|
the name of the current input file.
|
||||||
|
.TP
|
||||||
|
.B FNR
|
||||||
|
ordinal number of the current record in the current file.
|
||||||
.TP
|
.TP
|
||||||
.B FS
|
.B FS
|
||||||
regular expression used to separate fields; also settable
|
regular expression used to separate fields; also settable
|
||||||
by option
|
by option
|
||||||
.BI \-F fs.
|
.BI \-F fs\fR.
|
||||||
.TP
|
.TP
|
||||||
.BR NF
|
.BR NF
|
||||||
number of fields in the current record
|
number of fields in the current record.
|
||||||
.TP
|
.TP
|
||||||
.B NR
|
.B NR
|
||||||
ordinal number of the current record
|
ordinal number of the current record.
|
||||||
.TP
|
|
||||||
.B FNR
|
|
||||||
ordinal number of the current record in the current file
|
|
||||||
.TP
|
|
||||||
.B FILENAME
|
|
||||||
the name of the current input file
|
|
||||||
.TP
|
|
||||||
.B RS
|
|
||||||
input record separator (default newline)
|
|
||||||
.TP
|
|
||||||
.B OFS
|
|
||||||
output field separator (default blank)
|
|
||||||
.TP
|
|
||||||
.B ORS
|
|
||||||
output record separator (default newline)
|
|
||||||
.TP
|
.TP
|
||||||
.B OFMT
|
.B OFMT
|
||||||
output format for numbers (default
|
output format for numbers (default
|
||||||
.BR "%.6g" )
|
.BR "%.6g" ).
|
||||||
|
.TP
|
||||||
|
.B OFS
|
||||||
|
output field separator (default space).
|
||||||
|
.TP
|
||||||
|
.B ORS
|
||||||
|
output record separator (default newline).
|
||||||
|
.TP
|
||||||
|
.B RLENGTH
|
||||||
|
the length of a string matched by
|
||||||
|
.BR match .
|
||||||
|
.TP
|
||||||
|
.B RS
|
||||||
|
input record separator (default newline).
|
||||||
|
.TP
|
||||||
|
.B RSTART
|
||||||
|
the start position of a string matched by
|
||||||
|
.BR match .
|
||||||
.TP
|
.TP
|
||||||
.B SUBSEP
|
.B SUBSEP
|
||||||
separates multiple subscripts (default 034)
|
separates multiple subscripts (default 034).
|
||||||
.TP
|
|
||||||
.B ARGC
|
|
||||||
argument count, assignable
|
|
||||||
.TP
|
|
||||||
.B ARGV
|
|
||||||
argument array, assignable;
|
|
||||||
non-null members are taken as filenames
|
|
||||||
.TP
|
|
||||||
.B ENVIRON
|
|
||||||
array of environment variables; subscripts are names.
|
|
||||||
.PD
|
.PD
|
||||||
.PP
|
.PP
|
||||||
Functions may be defined (at the position of a pattern-action statement) thus:
|
Functions may be defined (at the position of a pattern-action statement) thus:
|
||||||
@ -486,7 +516,7 @@ BEGIN { FS = ",[ \et]*|[ \et]+" }
|
|||||||
.EE
|
.EE
|
||||||
.ns
|
.ns
|
||||||
.IP
|
.IP
|
||||||
Same, with input fields separated by comma and/or blanks and tabs.
|
Same, with input fields separated by comma and/or spaces and tabs.
|
||||||
.PP
|
.PP
|
||||||
.EX
|
.EX
|
||||||
.nf
|
.nf
|
||||||
@ -512,13 +542,13 @@ BEGIN { # Simulate echo(1)
|
|||||||
.fi
|
.fi
|
||||||
.EE
|
.EE
|
||||||
.SH SEE ALSO
|
.SH SEE ALSO
|
||||||
|
.IR grep (1),
|
||||||
.IR lex (1),
|
.IR lex (1),
|
||||||
.IR sed (1)
|
.IR sed (1)
|
||||||
.br
|
.br
|
||||||
A. V. Aho, B. W. Kernighan, P. J. Weinberger,
|
A. V. Aho, B. W. Kernighan, P. J. Weinberger,
|
||||||
.I
|
.IR "The AWK Programming Language" ,
|
||||||
The AWK Programming Language,
|
Addison-Wesley, 1988. ISBN 0-201-07981-X.
|
||||||
Addison-Wesley, 1988. ISBN 0-201-07981-X
|
|
||||||
.SH BUGS
|
.SH BUGS
|
||||||
There are no explicit conversions between numbers and strings.
|
There are no explicit conversions between numbers and strings.
|
||||||
To force an expression to be treated as a number add 0 to it;
|
To force an expression to be treated as a number add 0 to it;
|
||||||
@ -527,3 +557,7 @@ to force it to be treated as a string concatenate
|
|||||||
.br
|
.br
|
||||||
The scope rules for variables in functions are a botch;
|
The scope rules for variables in functions are a botch;
|
||||||
the syntax is worse.
|
the syntax is worse.
|
||||||
|
.br
|
||||||
|
POSIX-standard interval expressions in regular expressions are not supported.
|
||||||
|
.br
|
||||||
|
Only eight-bit characters sets are handled correctly.
|
||||||
|
5
awk.h
5
awk.h
@ -81,7 +81,8 @@ typedef struct Cell {
|
|||||||
char *nval; /* name, for variables only */
|
char *nval; /* name, for variables only */
|
||||||
char *sval; /* string value */
|
char *sval; /* string value */
|
||||||
Awkfloat fval; /* value as number */
|
Awkfloat fval; /* value as number */
|
||||||
int tval; /* type info: STR|NUM|ARR|FCN|FLD|CON|DONTFREE */
|
int tval; /* type info: STR|NUM|ARR|FCN|FLD|CON|DONTFREE|CONVC|CONVO */
|
||||||
|
char *fmt; /* CONVFMT/OFMT value used to convert from number */
|
||||||
struct Cell *cnext; /* ptr to next if chained */
|
struct Cell *cnext; /* ptr to next if chained */
|
||||||
} Cell;
|
} Cell;
|
||||||
|
|
||||||
@ -109,6 +110,8 @@ extern Cell *rlengthloc; /* RLENGTH */
|
|||||||
#define FCN 040 /* this is a function name */
|
#define FCN 040 /* this is a function name */
|
||||||
#define FLD 0100 /* this is a field $1, $2, ... */
|
#define FLD 0100 /* this is a field $1, $2, ... */
|
||||||
#define REC 0200 /* this is $0 */
|
#define REC 0200 /* this is $0 */
|
||||||
|
#define CONVC 0400 /* string was converted from number via CONVFMT */
|
||||||
|
#define CONVO 01000 /* string was converted from number via OFMT */
|
||||||
|
|
||||||
|
|
||||||
/* function types */
|
/* function types */
|
||||||
|
@ -86,7 +86,7 @@ Node *arglist = 0; /* list of args for current function */
|
|||||||
%left CAT
|
%left CAT
|
||||||
%left '+' '-'
|
%left '+' '-'
|
||||||
%left '*' '/' '%'
|
%left '*' '/' '%'
|
||||||
%left NOT UMINUS
|
%left NOT UMINUS UPLUS
|
||||||
%right POWER
|
%right POWER
|
||||||
%right DECR INCR
|
%right DECR INCR
|
||||||
%left INDIRECT
|
%left INDIRECT
|
||||||
@ -357,7 +357,7 @@ term:
|
|||||||
| term '%' term { $$ = op2(MOD, $1, $3); }
|
| term '%' term { $$ = op2(MOD, $1, $3); }
|
||||||
| term POWER term { $$ = op2(POWER, $1, $3); }
|
| term POWER term { $$ = op2(POWER, $1, $3); }
|
||||||
| '-' term %prec UMINUS { $$ = op1(UMINUS, $2); }
|
| '-' term %prec UMINUS { $$ = op1(UMINUS, $2); }
|
||||||
| '+' term %prec UMINUS { $$ = $2; }
|
| '+' term %prec UMINUS { $$ = op1(UPLUS, $2); }
|
||||||
| NOT term %prec UMINUS { $$ = op1(NOT, notnull($2)); }
|
| NOT term %prec UMINUS { $$ = op1(NOT, notnull($2)); }
|
||||||
| BLTIN '(' ')' { $$ = op2(BLTIN, itonp($1), rectonode()); }
|
| BLTIN '(' ')' { $$ = op2(BLTIN, itonp($1), rectonode()); }
|
||||||
| BLTIN '(' patlist ')' { $$ = op2(BLTIN, itonp($1), $3); }
|
| BLTIN '(' patlist ')' { $$ = op2(BLTIN, itonp($1), $3); }
|
||||||
|
BIN
awktest.tar
BIN
awktest.tar
Binary file not shown.
2
b.c
2
b.c
@ -749,7 +749,7 @@ struct charclass {
|
|||||||
{ "alnum", 5, isalnum },
|
{ "alnum", 5, isalnum },
|
||||||
{ "alpha", 5, isalpha },
|
{ "alpha", 5, isalpha },
|
||||||
#ifndef HAS_ISBLANK
|
#ifndef HAS_ISBLANK
|
||||||
{ "blank", 5, isspace }, /* was isblank */
|
{ "blank", 5, xisblank },
|
||||||
#else
|
#else
|
||||||
{ "blank", 5, isblank },
|
{ "blank", 5, isblank },
|
||||||
#endif
|
#endif
|
||||||
|
25
bugs-fixed/README
Normal file
25
bugs-fixed/README
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
List of bugs fixed.
|
||||||
|
|
||||||
|
1. ofs-rebuild: OFS value used to rebuild the record was incorrect.
|
||||||
|
Fixed August 19, 2014. Revised fix August 2018.
|
||||||
|
|
||||||
|
2. system-status: Instead of a floating-point division by 256, use
|
||||||
|
the wait(2) macros to create a reasonable exit status. Fixed March 12, 2016.
|
||||||
|
|
||||||
|
3. space: Use provided xisblank() function instead of ispace() for
|
||||||
|
matching [[:blank:]].
|
||||||
|
|
||||||
|
4. a-format: Add POSIX standard %a and %A to supported formats. Check
|
||||||
|
at runtime that this format is available.
|
||||||
|
|
||||||
|
5. decr-NF: Decrementing NF did not change $0. This is a decades-old
|
||||||
|
bug. There are interactions with the old and new value of OFS as well.
|
||||||
|
Most of the fix came from the NetBSD awk.
|
||||||
|
|
||||||
|
6. string-conv: String conversions of scalars were sticky. Once a
|
||||||
|
conversion to string happened, even with OFMT, that value was used until
|
||||||
|
a new numeric value was assigned, even if OFMT differed from CONVFMT,
|
||||||
|
and also if CONVFMT changed.
|
||||||
|
|
||||||
|
7. unary-plus: Unary plus on a string constant returned the string.
|
||||||
|
Instead, it should convert the value to numeric and give that value.
|
3
bugs-fixed/a-format.awk
Normal file
3
bugs-fixed/a-format.awk
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
BEGIN {
|
||||||
|
printf("%a\n", 42)
|
||||||
|
}
|
3
bugs-fixed/a-format.bad
Normal file
3
bugs-fixed/a-format.bad
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
nawk: weird printf conversion %a
|
||||||
|
source line number 2
|
||||||
|
%a42
|
1
bugs-fixed/a-format.ok
Normal file
1
bugs-fixed/a-format.ok
Normal file
@ -0,0 +1 @@
|
|||||||
|
0x1.5p+5
|
11
bugs-fixed/decr-NF.awk
Normal file
11
bugs-fixed/decr-NF.awk
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
BEGIN {
|
||||||
|
$0 = "a b c d e f"
|
||||||
|
print NF
|
||||||
|
OFS = ":"
|
||||||
|
NF--
|
||||||
|
print $0
|
||||||
|
print NF
|
||||||
|
NF++
|
||||||
|
print $0
|
||||||
|
print NF
|
||||||
|
}
|
5
bugs-fixed/decr-NF.bad
Normal file
5
bugs-fixed/decr-NF.bad
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
6
|
||||||
|
a b c d e f
|
||||||
|
5
|
||||||
|
a b c d e f
|
||||||
|
6
|
5
bugs-fixed/decr-NF.ok
Normal file
5
bugs-fixed/decr-NF.ok
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
6
|
||||||
|
a:b:c:d:e
|
||||||
|
5
|
||||||
|
a:b:c:d:e:
|
||||||
|
6
|
17
bugs-fixed/ofs-rebuild.awk
Normal file
17
bugs-fixed/ofs-rebuild.awk
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
# The bug here is that nawk should use the value of OFS that
|
||||||
|
# was current when $0 became invalid to rebuild the record.
|
||||||
|
|
||||||
|
BEGIN {
|
||||||
|
OFS = ":"
|
||||||
|
$0 = "a b c d e f g"
|
||||||
|
$3 = "3333"
|
||||||
|
# Conceptually, $0 should now be "a:b:3333:d:e:f:g"
|
||||||
|
|
||||||
|
# Change OFS after (conceptually) rebuilding the record
|
||||||
|
OFS = "<>"
|
||||||
|
|
||||||
|
# Unmodifed nawk prints "a<>b<>3333<>d<>e<>f<>g" because
|
||||||
|
# it delays rebuilding $0 until it's needed, and then it uses
|
||||||
|
# the current value of OFS. Oops.
|
||||||
|
print
|
||||||
|
}
|
1
bugs-fixed/ofs-rebuild.bad
Normal file
1
bugs-fixed/ofs-rebuild.bad
Normal file
@ -0,0 +1 @@
|
|||||||
|
a<>b<>3333<>d<>e<>f<>g
|
1
bugs-fixed/ofs-rebuild.ok
Normal file
1
bugs-fixed/ofs-rebuild.ok
Normal file
@ -0,0 +1 @@
|
|||||||
|
a:b:3333:d:e:f:g
|
22
bugs-fixed/space.awk
Normal file
22
bugs-fixed/space.awk
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
BEGIN {
|
||||||
|
c[" "] = "\" \""
|
||||||
|
c["\a"] = "\\a"
|
||||||
|
c["\b"] = "\\b"
|
||||||
|
c["\f"] = "\\f"
|
||||||
|
c["\n"] = "\\n"
|
||||||
|
c["\r"] = "\\r"
|
||||||
|
c["\t"] = "\\t"
|
||||||
|
c["\v"] = "\\v"
|
||||||
|
|
||||||
|
sort = "LC_ALL=C sort"
|
||||||
|
|
||||||
|
for (i in c)
|
||||||
|
printf("%s %s [[:space:]]\n", c[i],
|
||||||
|
i ~ /[[:space:]]/ ? "~" : "!~") | sort
|
||||||
|
|
||||||
|
for (i in c)
|
||||||
|
printf("%s %s [[:blank:]]\n", c[i],
|
||||||
|
i ~ /[[:blank:]]/ ? "~" : "!~") | sort
|
||||||
|
|
||||||
|
close(sort)
|
||||||
|
}
|
16
bugs-fixed/space.bad
Normal file
16
bugs-fixed/space.bad
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
" " ~ [[:blank:]]
|
||||||
|
" " ~ [[:space:]]
|
||||||
|
\a !~ [[:blank:]]
|
||||||
|
\a !~ [[:space:]]
|
||||||
|
\b !~ [[:blank:]]
|
||||||
|
\b !~ [[:space:]]
|
||||||
|
\f ~ [[:blank:]]
|
||||||
|
\f ~ [[:space:]]
|
||||||
|
\n ~ [[:blank:]]
|
||||||
|
\n ~ [[:space:]]
|
||||||
|
\r ~ [[:blank:]]
|
||||||
|
\r ~ [[:space:]]
|
||||||
|
\t ~ [[:blank:]]
|
||||||
|
\t ~ [[:space:]]
|
||||||
|
\v ~ [[:blank:]]
|
||||||
|
\v ~ [[:space:]]
|
16
bugs-fixed/space.ok
Normal file
16
bugs-fixed/space.ok
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
" " ~ [[:blank:]]
|
||||||
|
" " ~ [[:space:]]
|
||||||
|
\a !~ [[:blank:]]
|
||||||
|
\a !~ [[:space:]]
|
||||||
|
\b !~ [[:blank:]]
|
||||||
|
\b !~ [[:space:]]
|
||||||
|
\f !~ [[:blank:]]
|
||||||
|
\f ~ [[:space:]]
|
||||||
|
\n !~ [[:blank:]]
|
||||||
|
\n ~ [[:space:]]
|
||||||
|
\r !~ [[:blank:]]
|
||||||
|
\r ~ [[:space:]]
|
||||||
|
\t ~ [[:blank:]]
|
||||||
|
\t ~ [[:space:]]
|
||||||
|
\v !~ [[:blank:]]
|
||||||
|
\v ~ [[:space:]]
|
13
bugs-fixed/string-conv.awk
Normal file
13
bugs-fixed/string-conv.awk
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
BEGIN {
|
||||||
|
OFMT = ">>%.6g<<"
|
||||||
|
a = 12.1234
|
||||||
|
print "a =", a
|
||||||
|
b = a ""
|
||||||
|
print "1 ->", b
|
||||||
|
CONVFMT = "%2.2f"
|
||||||
|
b = a ""
|
||||||
|
print "2 ->", b
|
||||||
|
CONVFMT = "%.12g"
|
||||||
|
b = a ""
|
||||||
|
print "3 ->", b
|
||||||
|
}
|
4
bugs-fixed/string-conv.bad
Normal file
4
bugs-fixed/string-conv.bad
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
a = >>12.1234<<
|
||||||
|
1 -> >>12.1234<<
|
||||||
|
2 -> >>12.1234<<
|
||||||
|
3 -> >>12.1234<<
|
4
bugs-fixed/string-conv.ok
Normal file
4
bugs-fixed/string-conv.ok
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
a = >>12.1234<<
|
||||||
|
1 -> 12.1234
|
||||||
|
2 -> 12.12
|
||||||
|
3 -> 12.1234
|
19
bugs-fixed/system-status.awk
Normal file
19
bugs-fixed/system-status.awk
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
# Unmodified nawk prints the 16 bit exit status divided by 256, but
|
||||||
|
# does so using floating point arithmetic, yielding strange results.
|
||||||
|
#
|
||||||
|
# The fix is to use the various macros defined for wait(2) and to
|
||||||
|
# use the signal number + 256 for death by signal, or signal number + 512
|
||||||
|
# for death by signal with core dump.
|
||||||
|
|
||||||
|
BEGIN {
|
||||||
|
status = system("exit 42")
|
||||||
|
print "normal status", status
|
||||||
|
|
||||||
|
status = system("kill -HUP $$")
|
||||||
|
print "death by signal status", status
|
||||||
|
|
||||||
|
status = system("kill -ABRT $$")
|
||||||
|
print "death by signal with core dump status", status
|
||||||
|
|
||||||
|
system("rm -f core*")
|
||||||
|
}
|
3
bugs-fixed/system-status.bad
Normal file
3
bugs-fixed/system-status.bad
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
normal status 42
|
||||||
|
death by signal status 0.00390625
|
||||||
|
death by signal with core dump status 0.523438
|
3
bugs-fixed/system-status.ok
Normal file
3
bugs-fixed/system-status.ok
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
normal status 42
|
||||||
|
death by signal status 257
|
||||||
|
death by signal with core dump status 518
|
4
bugs-fixed/unary-plus.awk
Normal file
4
bugs-fixed/unary-plus.awk
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
BEGIN {
|
||||||
|
print +"q"
|
||||||
|
print +"43.12345678912345678"
|
||||||
|
}
|
2
bugs-fixed/unary-plus.bad
Normal file
2
bugs-fixed/unary-plus.bad
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
q
|
||||||
|
43.12345678912345678
|
2
bugs-fixed/unary-plus.ok
Normal file
2
bugs-fixed/unary-plus.ok
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
0
|
||||||
|
43.1235
|
14
lib.c
14
lib.c
@ -356,6 +356,7 @@ void fldbld(void) /* create fields from current record */
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
setfval(nfloc, (Awkfloat) lastfld);
|
setfval(nfloc, (Awkfloat) lastfld);
|
||||||
|
donerec = 1; /* restore */
|
||||||
if (dbg) {
|
if (dbg) {
|
||||||
for (j = 0; j <= lastfld; j++) {
|
for (j = 0; j <= lastfld; j++) {
|
||||||
p = fldtab[j];
|
p = fldtab[j];
|
||||||
@ -387,6 +388,19 @@ void newfld(int n) /* add field n after end of existing lastfld */
|
|||||||
setfval(nfloc, (Awkfloat) n);
|
setfval(nfloc, (Awkfloat) n);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setlastfld(int n) /* set lastfld cleaning fldtab cells if necessary */
|
||||||
|
{
|
||||||
|
if (n > nfields)
|
||||||
|
growfldtab(n);
|
||||||
|
|
||||||
|
if (lastfld < n)
|
||||||
|
cleanfld(lastfld+1, n);
|
||||||
|
else
|
||||||
|
cleanfld(n+1, lastfld);
|
||||||
|
|
||||||
|
lastfld = n;
|
||||||
|
}
|
||||||
|
|
||||||
Cell *fieldadr(int n) /* get nth field */
|
Cell *fieldadr(int n) /* get nth field */
|
||||||
{
|
{
|
||||||
if (n < 0)
|
if (n < 0)
|
||||||
|
2
main.c
2
main.c
@ -22,7 +22,7 @@ ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
|
|||||||
THIS SOFTWARE.
|
THIS SOFTWARE.
|
||||||
****************************************************************/
|
****************************************************************/
|
||||||
|
|
||||||
const char *version = "version 20130105";
|
const char *version = "version 20130105 + fixes by arnold@skeeve.com 20180822";
|
||||||
|
|
||||||
#define DEBUG
|
#define DEBUG
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
8
makefile
8
makefile
@ -26,13 +26,13 @@ CFLAGS = -g
|
|||||||
CFLAGS = -O2
|
CFLAGS = -O2
|
||||||
CFLAGS =
|
CFLAGS =
|
||||||
|
|
||||||
CC = gcc -Wall -g -Wwrite-strings
|
#CC = gcc -Wall -g -Wwrite-strings
|
||||||
CC = gcc -g -Wall -pedantic
|
CC = gcc -g -Wall -pedantic
|
||||||
CC = gcc -O4 -Wall -pedantic -fno-strict-aliasing
|
#CC = gcc -O4 -Wall -pedantic -fno-strict-aliasing
|
||||||
CC = gcc -fprofile-arcs -ftest-coverage # then gcov f1.c; cat f1.c.gcov
|
#CC = gcc -fprofile-arcs -ftest-coverage # then gcov f1.c; cat f1.c.gcov
|
||||||
|
|
||||||
YACC = bison -d -y
|
YACC = bison -d -y
|
||||||
YACC = yacc -d
|
# YACC = yacc -d
|
||||||
#YFLAGS = -d -S
|
#YFLAGS = -d -S
|
||||||
# -S uses sprintf in yacc parser instead of sprint
|
# -S uses sprintf in yacc parser instead of sprint
|
||||||
|
|
||||||
|
@ -62,6 +62,7 @@ struct xx
|
|||||||
{ DIVIDE, "arith", " / " },
|
{ DIVIDE, "arith", " / " },
|
||||||
{ MOD, "arith", " % " },
|
{ MOD, "arith", " % " },
|
||||||
{ UMINUS, "arith", " -" },
|
{ UMINUS, "arith", " -" },
|
||||||
|
{ UPLUS, "arith", " +" },
|
||||||
{ POWER, "arith", " **" },
|
{ POWER, "arith", " **" },
|
||||||
{ PREINCR, "incrdecr", "++" },
|
{ PREINCR, "incrdecr", "++" },
|
||||||
{ POSTINCR, "incrdecr", "++" },
|
{ POSTINCR, "incrdecr", "++" },
|
||||||
|
3
proto.h
3
proto.h
@ -124,6 +124,7 @@ extern void setclvar(char *);
|
|||||||
extern void fldbld(void);
|
extern void fldbld(void);
|
||||||
extern void cleanfld(int, int);
|
extern void cleanfld(int, int);
|
||||||
extern void newfld(int);
|
extern void newfld(int);
|
||||||
|
extern void setlastfld(int);
|
||||||
extern int refldbld(const char *, const char *);
|
extern int refldbld(const char *, const char *);
|
||||||
extern void recbld(void);
|
extern void recbld(void);
|
||||||
extern Cell *fieldadr(int);
|
extern Cell *fieldadr(int);
|
||||||
@ -193,3 +194,5 @@ extern Cell *gsub(Node **, int);
|
|||||||
|
|
||||||
extern FILE *popen(const char *, const char *);
|
extern FILE *popen(const char *, const char *);
|
||||||
extern int pclose(FILE *);
|
extern int pclose(FILE *);
|
||||||
|
|
||||||
|
extern const char *flags2str(int flags);
|
||||||
|
51
run.c
51
run.c
@ -31,6 +31,8 @@ THIS SOFTWARE.
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/wait.h>
|
||||||
#include "awk.h"
|
#include "awk.h"
|
||||||
#include "ytab.h"
|
#include "ytab.h"
|
||||||
|
|
||||||
@ -323,14 +325,18 @@ Cell *copycell(Cell *x) /* make a copy of a cell in a temp */
|
|||||||
{
|
{
|
||||||
Cell *y;
|
Cell *y;
|
||||||
|
|
||||||
|
/* copy is not constant or field */
|
||||||
|
|
||||||
y = gettemp();
|
y = gettemp();
|
||||||
|
y->tval = x->tval & ~(CON|FLD|REC);
|
||||||
y->csub = CCOPY; /* prevents freeing until call is over */
|
y->csub = CCOPY; /* prevents freeing until call is over */
|
||||||
y->nval = x->nval; /* BUG? */
|
y->nval = x->nval; /* BUG? */
|
||||||
if (isstr(x))
|
if (isstr(x) /* || x->ctype == OCELL */) {
|
||||||
y->sval = tostring(x->sval);
|
y->sval = tostring(x->sval);
|
||||||
|
y->tval &= ~DONTFREE;
|
||||||
|
} else
|
||||||
|
y->tval |= DONTFREE;
|
||||||
y->fval = x->fval;
|
y->fval = x->fval;
|
||||||
y->tval = x->tval & ~(CON|FLD|REC|DONTFREE); /* copy is not constant or field */
|
|
||||||
/* is DONTFREE right? */
|
|
||||||
return y;
|
return y;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -817,6 +823,17 @@ int format(char **pbuf, int *pbufsize, const char *s, Node *a) /* printf-like co
|
|||||||
char *buf = *pbuf;
|
char *buf = *pbuf;
|
||||||
int bufsize = *pbufsize;
|
int bufsize = *pbufsize;
|
||||||
|
|
||||||
|
static int first = 1;
|
||||||
|
static int have_a_format = 0;
|
||||||
|
|
||||||
|
if (first) {
|
||||||
|
char buf[100];
|
||||||
|
|
||||||
|
sprintf(buf, "%a", 42.0);
|
||||||
|
have_a_format = (strcmp(buf, "0x1.5p+5") == 0);
|
||||||
|
first = 0;
|
||||||
|
}
|
||||||
|
|
||||||
os = s;
|
os = s;
|
||||||
p = buf;
|
p = buf;
|
||||||
if ((fmt = (char *) malloc(fmtsz)) == NULL)
|
if ((fmt = (char *) malloc(fmtsz)) == NULL)
|
||||||
@ -859,6 +876,12 @@ int format(char **pbuf, int *pbufsize, const char *s, Node *a) /* printf-like co
|
|||||||
adjbuf(&buf, &bufsize, fmtwd+1+p-buf, recsize, &p, "format4");
|
adjbuf(&buf, &bufsize, fmtwd+1+p-buf, recsize, &p, "format4");
|
||||||
|
|
||||||
switch (*s) {
|
switch (*s) {
|
||||||
|
case 'a': case 'A':
|
||||||
|
if (have_a_format)
|
||||||
|
flag = *s;
|
||||||
|
else
|
||||||
|
flag = 'f';
|
||||||
|
break;
|
||||||
case 'f': case 'e': case 'g': case 'E': case 'G':
|
case 'f': case 'e': case 'g': case 'E': case 'G':
|
||||||
flag = 'f';
|
flag = 'f';
|
||||||
break;
|
break;
|
||||||
@ -901,6 +924,8 @@ int format(char **pbuf, int *pbufsize, const char *s, Node *a) /* printf-like co
|
|||||||
p += strlen(p);
|
p += strlen(p);
|
||||||
sprintf(p, "%s", t);
|
sprintf(p, "%s", t);
|
||||||
break;
|
break;
|
||||||
|
case 'a':
|
||||||
|
case 'A':
|
||||||
case 'f': sprintf(p, fmt, getfval(x)); break;
|
case 'f': sprintf(p, fmt, getfval(x)); break;
|
||||||
case 'd': sprintf(p, fmt, (long) getfval(x)); break;
|
case 'd': sprintf(p, fmt, (long) getfval(x)); break;
|
||||||
case 'u': sprintf(p, fmt, (int) getfval(x)); break;
|
case 'u': sprintf(p, fmt, (int) getfval(x)); break;
|
||||||
@ -1003,7 +1028,7 @@ Cell *arith(Node **a, int n) /* a[0] + a[1], etc. also -a[0] */
|
|||||||
x = execute(a[0]);
|
x = execute(a[0]);
|
||||||
i = getfval(x);
|
i = getfval(x);
|
||||||
tempfree(x);
|
tempfree(x);
|
||||||
if (n != UMINUS) {
|
if (n != UMINUS && n != UPLUS) {
|
||||||
y = execute(a[1]);
|
y = execute(a[1]);
|
||||||
j = getfval(y);
|
j = getfval(y);
|
||||||
tempfree(y);
|
tempfree(y);
|
||||||
@ -1033,6 +1058,8 @@ Cell *arith(Node **a, int n) /* a[0] + a[1], etc. also -a[0] */
|
|||||||
case UMINUS:
|
case UMINUS:
|
||||||
i = -i;
|
i = -i;
|
||||||
break;
|
break;
|
||||||
|
case UPLUS: /* handled by getfval(), above */
|
||||||
|
break;
|
||||||
case POWER:
|
case POWER:
|
||||||
if (j >= 0 && modf(j, &v) == 0.0) /* pos integer exponent */
|
if (j >= 0 && modf(j, &v) == 0.0) /* pos integer exponent */
|
||||||
i = ipow(i, (int) j);
|
i = ipow(i, (int) j);
|
||||||
@ -1479,6 +1506,7 @@ Cell *bltin(Node **a, int n) /* builtin functions. a[0] is type, a[1] is arg lis
|
|||||||
Node *nextarg;
|
Node *nextarg;
|
||||||
FILE *fp;
|
FILE *fp;
|
||||||
void flush_all(void);
|
void flush_all(void);
|
||||||
|
int status = 0;
|
||||||
|
|
||||||
t = ptoi(a[0]);
|
t = ptoi(a[0]);
|
||||||
x = execute(a[1]);
|
x = execute(a[1]);
|
||||||
@ -1515,7 +1543,20 @@ Cell *bltin(Node **a, int n) /* builtin functions. a[0] is type, a[1] is arg lis
|
|||||||
break;
|
break;
|
||||||
case FSYSTEM:
|
case FSYSTEM:
|
||||||
fflush(stdout); /* in case something is buffered already */
|
fflush(stdout); /* in case something is buffered already */
|
||||||
u = (Awkfloat) system(getsval(x)) / 256; /* 256 is unix-dep */
|
status = system(getsval(x));
|
||||||
|
u = status;
|
||||||
|
if (status != -1) {
|
||||||
|
if (WIFEXITED(status)) {
|
||||||
|
u = WEXITSTATUS(status);
|
||||||
|
} else if (WIFSIGNALED(status)) {
|
||||||
|
u = WTERMSIG(status) + 256;
|
||||||
|
#ifdef WCOREDUMP
|
||||||
|
if (WCOREDUMP(status))
|
||||||
|
u += 256;
|
||||||
|
#endif
|
||||||
|
} else /* something else?!? */
|
||||||
|
u = 0;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case FRAND:
|
case FRAND:
|
||||||
/* in principle, rand() returns something in 0..RAND_MAX */
|
/* in principle, rand() returns something in 0..RAND_MAX */
|
||||||
|
156
tran.c
156
tran.c
@ -67,6 +67,18 @@ Cell *literal0;
|
|||||||
|
|
||||||
extern Cell **fldtab;
|
extern Cell **fldtab;
|
||||||
|
|
||||||
|
static void
|
||||||
|
setfree(Cell *vp)
|
||||||
|
{
|
||||||
|
if (&vp->sval == FS || &vp->sval == RS ||
|
||||||
|
&vp->sval == OFS || &vp->sval == ORS ||
|
||||||
|
&vp->sval == OFMT || &vp->sval == CONVFMT ||
|
||||||
|
&vp->sval == FILENAME || &vp->sval == SUBSEP)
|
||||||
|
vp->tval |= DONTFREE;
|
||||||
|
else
|
||||||
|
vp->tval &= ~DONTFREE;
|
||||||
|
}
|
||||||
|
|
||||||
void syminit(void) /* initialize symbol table with builtin vars */
|
void syminit(void) /* initialize symbol table with builtin vars */
|
||||||
{
|
{
|
||||||
literal0 = setsymtab("0", "0", 0.0, NUM|STR|CON|DONTFREE, symtab);
|
literal0 = setsymtab("0", "0", 0.0, NUM|STR|CON|DONTFREE, symtab);
|
||||||
@ -282,6 +294,7 @@ Awkfloat setfval(Cell *vp, Awkfloat f) /* set float val of a Cell */
|
|||||||
{
|
{
|
||||||
int fldno;
|
int fldno;
|
||||||
|
|
||||||
|
f += 0.0; /* normalise negative zero to positive zero */
|
||||||
if ((vp->tval & (NUM | STR)) == 0)
|
if ((vp->tval & (NUM | STR)) == 0)
|
||||||
funnyvar(vp, "assign to");
|
funnyvar(vp, "assign to");
|
||||||
if (isfld(vp)) {
|
if (isfld(vp)) {
|
||||||
@ -290,13 +303,18 @@ Awkfloat setfval(Cell *vp, Awkfloat f) /* set float val of a Cell */
|
|||||||
if (fldno > *NF)
|
if (fldno > *NF)
|
||||||
newfld(fldno);
|
newfld(fldno);
|
||||||
dprintf( ("setting field %d to %g\n", fldno, f) );
|
dprintf( ("setting field %d to %g\n", fldno, f) );
|
||||||
|
} else if (&vp->fval == NF) {
|
||||||
|
donerec = 0; /* mark $0 invalid */
|
||||||
|
setlastfld(f);
|
||||||
|
dprintf( ("setting NF to %g\n", f) );
|
||||||
} else if (isrec(vp)) {
|
} else if (isrec(vp)) {
|
||||||
donefld = 0; /* mark $1... invalid */
|
donefld = 0; /* mark $1... invalid */
|
||||||
donerec = 1;
|
donerec = 1;
|
||||||
}
|
}
|
||||||
if (freeable(vp))
|
if (freeable(vp))
|
||||||
xfree(vp->sval); /* free any previous string */
|
xfree(vp->sval); /* free any previous string */
|
||||||
vp->tval &= ~STR; /* mark string invalid */
|
vp->tval &= ~(STR|CONVC|CONVO); /* mark string invalid */
|
||||||
|
vp->fmt = NULL;
|
||||||
vp->tval |= NUM; /* mark number ok */
|
vp->tval |= NUM; /* mark number ok */
|
||||||
if (f == -0) /* who would have thought this possible? */
|
if (f == -0) /* who would have thought this possible? */
|
||||||
f = 0;
|
f = 0;
|
||||||
@ -318,6 +336,7 @@ char *setsval(Cell *vp, const char *s) /* set string val of a Cell */
|
|||||||
{
|
{
|
||||||
char *t;
|
char *t;
|
||||||
int fldno;
|
int fldno;
|
||||||
|
Awkfloat f;
|
||||||
|
|
||||||
dprintf( ("starting setsval %p: %s = \"%s\", t=%o, r,f=%d,%d\n",
|
dprintf( ("starting setsval %p: %s = \"%s\", t=%o, r,f=%d,%d\n",
|
||||||
(void*)vp, NN(vp->nval), s, vp->tval, donerec, donefld) );
|
(void*)vp, NN(vp->nval), s, vp->tval, donerec, donefld) );
|
||||||
@ -332,16 +351,28 @@ char *setsval(Cell *vp, const char *s) /* set string val of a Cell */
|
|||||||
} else if (isrec(vp)) {
|
} else if (isrec(vp)) {
|
||||||
donefld = 0; /* mark $1... invalid */
|
donefld = 0; /* mark $1... invalid */
|
||||||
donerec = 1;
|
donerec = 1;
|
||||||
|
} else if (&vp->sval == OFS) {
|
||||||
|
if (donerec == 0)
|
||||||
|
recbld();
|
||||||
}
|
}
|
||||||
t = tostring(s); /* in case it's self-assign */
|
t = s ? tostring(s) : tostring(""); /* in case it's self-assign */
|
||||||
if (freeable(vp))
|
if (freeable(vp))
|
||||||
xfree(vp->sval);
|
xfree(vp->sval);
|
||||||
vp->tval &= ~NUM;
|
vp->tval &= ~(NUM|CONVC|CONVO);
|
||||||
vp->tval |= STR;
|
vp->tval |= STR;
|
||||||
vp->tval &= ~DONTFREE;
|
vp->fmt = NULL;
|
||||||
|
setfree(vp);
|
||||||
dprintf( ("setsval %p: %s = \"%s (%p) \", t=%o r,f=%d,%d\n",
|
dprintf( ("setsval %p: %s = \"%s (%p) \", t=%o r,f=%d,%d\n",
|
||||||
(void*)vp, NN(vp->nval), t,t, vp->tval, donerec, donefld) );
|
(void*)vp, NN(vp->nval), t,t, vp->tval, donerec, donefld) );
|
||||||
return(vp->sval = t);
|
vp->sval = t;
|
||||||
|
if (&vp->fval == NF) {
|
||||||
|
donerec = 0; /* mark $0 invalid */
|
||||||
|
f = getfval(vp);
|
||||||
|
setlastfld(f);
|
||||||
|
dprintf( ("setting NF to %g\n", f) );
|
||||||
|
}
|
||||||
|
|
||||||
|
return(vp->sval);
|
||||||
}
|
}
|
||||||
|
|
||||||
Awkfloat getfval(Cell *vp) /* get float val of a Cell */
|
Awkfloat getfval(Cell *vp) /* get float val of a Cell */
|
||||||
@ -373,17 +404,78 @@ static char *get_str_val(Cell *vp, char **fmt) /* get string val of a Cel
|
|||||||
fldbld();
|
fldbld();
|
||||||
else if (isrec(vp) && donerec == 0)
|
else if (isrec(vp) && donerec == 0)
|
||||||
recbld();
|
recbld();
|
||||||
if (isstr(vp) == 0) {
|
|
||||||
if (freeable(vp))
|
/*
|
||||||
xfree(vp->sval);
|
* ADR: This is complicated and more fragile than is desirable.
|
||||||
if (modf(vp->fval, &dtemp) == 0) /* it's integral */
|
* Retrieving a string value for a number associates the string
|
||||||
sprintf(s, "%.30g", vp->fval);
|
* value with the scalar. Previously, the string value was
|
||||||
else
|
* sticky, meaning if converted via OFMT that became the value
|
||||||
sprintf(s, *fmt, vp->fval);
|
* (even though POSIX wants it to be via CONVFMT). Or if CONVFMT
|
||||||
vp->sval = tostring(s);
|
* changed after a string value was retrieved, the original value
|
||||||
vp->tval &= ~DONTFREE;
|
* was maintained and used. Also not per POSIX.
|
||||||
vp->tval |= STR;
|
*
|
||||||
|
* We work around this design by adding two additional flags,
|
||||||
|
* CONVC and CONVO, indicating how the string value was
|
||||||
|
* obtained (via CONVFMT or OFMT) and _also_ maintaining a copy
|
||||||
|
* of the pointer to the xFMT format string used for the
|
||||||
|
* conversion. This pointer is only read, **never** dereferenced.
|
||||||
|
* The next time we do a conversion, if it's coming from the same
|
||||||
|
* xFMT as last time, and the pointer value is different, we
|
||||||
|
* know that the xFMT format string changed, and we need to
|
||||||
|
* redo the conversion. If it's the same, we don't have to.
|
||||||
|
*
|
||||||
|
* There are also several cases where we don't do a conversion,
|
||||||
|
* such as for a field (see the checks below).
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Don't duplicate the code for actually updating the value */
|
||||||
|
#define update_str_val(vp) \
|
||||||
|
{ \
|
||||||
|
if (freeable(vp)) \
|
||||||
|
xfree(vp->sval); \
|
||||||
|
if (modf(vp->fval, &dtemp) == 0) /* it's integral */ \
|
||||||
|
sprintf(s, "%.30g", vp->fval); \
|
||||||
|
else \
|
||||||
|
sprintf(s, *fmt, vp->fval); \
|
||||||
|
vp->sval = tostring(s); \
|
||||||
|
vp->tval &= ~DONTFREE; \
|
||||||
|
vp->tval |= STR; \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isstr(vp) == 0) {
|
||||||
|
update_str_val(vp);
|
||||||
|
if (fmt == OFMT) {
|
||||||
|
vp->tval &= ~CONVC;
|
||||||
|
vp->tval |= CONVO;
|
||||||
|
} else {
|
||||||
|
/* CONVFMT */
|
||||||
|
vp->tval &= ~CONVO;
|
||||||
|
vp->tval |= CONVC;
|
||||||
|
}
|
||||||
|
vp->fmt = *fmt;
|
||||||
|
} else if ((vp->tval & DONTFREE) != 0 || ! isnum(vp) || isfld(vp)) {
|
||||||
|
goto done;
|
||||||
|
} else if (isstr(vp)) {
|
||||||
|
if (fmt == OFMT) {
|
||||||
|
if ((vp->tval & CONVC) != 0
|
||||||
|
|| ((vp->tval & CONVO) != 0 && vp->fmt != *fmt)) {
|
||||||
|
update_str_val(vp);
|
||||||
|
vp->tval &= ~CONVC;
|
||||||
|
vp->tval |= CONVO;
|
||||||
|
vp->fmt = *fmt;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* CONVFMT */
|
||||||
|
if ((vp->tval & CONVO) != 0
|
||||||
|
|| ((vp->tval & CONVC) != 0 && vp->fmt != *fmt)) {
|
||||||
|
update_str_val(vp);
|
||||||
|
vp->tval &= ~CONVO;
|
||||||
|
vp->tval |= CONVC;
|
||||||
|
vp->fmt = *fmt;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
done:
|
||||||
dprintf( ("getsval %p: %s = \"%s (%p)\", t=%o\n",
|
dprintf( ("getsval %p: %s = \"%s (%p)\", t=%o\n",
|
||||||
(void*)vp, NN(vp->nval), vp->sval, vp->sval, vp->tval) );
|
(void*)vp, NN(vp->nval), vp->sval, vp->sval, vp->tval) );
|
||||||
return(vp->sval);
|
return(vp->sval);
|
||||||
@ -457,3 +549,37 @@ char *qstring(const char *is, int delim) /* collect string up to next delim */
|
|||||||
*bp++ = 0;
|
*bp++ = 0;
|
||||||
return (char *) buf;
|
return (char *) buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const char *flags2str(int flags)
|
||||||
|
{
|
||||||
|
static const struct ftab {
|
||||||
|
const char *name;
|
||||||
|
int value;
|
||||||
|
} flagtab[] = {
|
||||||
|
{ "NUM", NUM },
|
||||||
|
{ "STR", STR },
|
||||||
|
{ "DONTFREE", DONTFREE },
|
||||||
|
{ "CON", CON },
|
||||||
|
{ "ARR", ARR },
|
||||||
|
{ "FCN", FCN },
|
||||||
|
{ "FLD", FLD },
|
||||||
|
{ "REC", REC },
|
||||||
|
{ "CONVC", CONVC },
|
||||||
|
{ "CONVO", CONVO },
|
||||||
|
{ NULL, 0 }
|
||||||
|
};
|
||||||
|
static char buf[100];
|
||||||
|
int i;
|
||||||
|
char *cp = buf;
|
||||||
|
|
||||||
|
for (i = 0; flagtab[i].name != NULL; i++) {
|
||||||
|
if ((flags & flagtab[i].value) != 0) {
|
||||||
|
if (cp > buf)
|
||||||
|
*cp++ = '|';
|
||||||
|
strcpy(cp, flagtab[i].name);
|
||||||
|
cp += strlen(cp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user