203 lines
5.6 KiB
Awk
203 lines
5.6 KiB
Awk
#!/bin/awk -f
|
|
|
|
# scripts/dfn.awk - process a .dfn file
|
|
#
|
|
# Copyright (c) 2013-2014 Glenn Randers-Pehrson
|
|
#
|
|
# This code is released under the libpng license.
|
|
# For conditions of distribution and use, see the disclaimer
|
|
# and license in png.h
|
|
|
|
# The output of this script is written to the file given by
|
|
# the variable 'out', which should be set on the command line.
|
|
# Error messages are printed to stdout and if any are printed
|
|
# the script will exit with error code 1.
|
|
|
|
BEGIN{
|
|
out="/dev/null" # as a flag
|
|
out_count=0 # count of output lines
|
|
err=0 # set if an error occurred
|
|
sort=0 # sort the output
|
|
array[""]=""
|
|
}
|
|
|
|
# The output file must be specified before any input:
|
|
NR==1 && out == "/dev/null" {
|
|
print "out=output.file must be given on the command line"
|
|
# but continue without setting the error code; this allows the
|
|
# script to be checked easily
|
|
}
|
|
|
|
# Output can be sorted; two lines are recognized
|
|
$1 == "PNG_DFN_START_SORT"{
|
|
sort=0+$2
|
|
next
|
|
}
|
|
|
|
$1 ~ /^PNG_DFN_END_SORT/{
|
|
# Do a very simple, slow, sort; notice that blank lines won't be
|
|
# output by this
|
|
for (entry in array) {
|
|
while (array[entry] != "") {
|
|
key = entry
|
|
value = array[key]
|
|
array[key] = ""
|
|
|
|
for (alt in array) {
|
|
if (array[alt] != "" && alt < key) {
|
|
array[key] = value
|
|
value = array[alt]
|
|
key = alt
|
|
array[alt] = ""
|
|
}
|
|
}
|
|
|
|
print value >out
|
|
}
|
|
}
|
|
sort=0
|
|
next
|
|
}
|
|
|
|
/^[^"]*PNG_DFN *".*"[^"]*$/{
|
|
# A definition line, apparently correctly formatted; extract the
|
|
# definition then replace any doubled "" that remain with a single
|
|
# double quote. Notice that the original doubled double quotes
|
|
# may have been split by tokenization
|
|
#
|
|
# Sometimes GCC splits the PNG_DFN lines; we know this has happened
|
|
# if the quotes aren't closed and must read another line. In this
|
|
# case it is essential to reject lines that start with '#' because those
|
|
# are introduced #line directives.
|
|
orig=$0
|
|
line=$0
|
|
lineno=FNR
|
|
if (lineno == "") lineno=NR
|
|
|
|
if (sub(/^[^"]*PNG_DFN *"/,"",line) != 1) {
|
|
print "line", lineno ": processing failed:"
|
|
print orig
|
|
err=1
|
|
next
|
|
} else {
|
|
++out_count
|
|
}
|
|
|
|
# Now examine quotes within the value:
|
|
#
|
|
# @" - delete this and any following spaces
|
|
# "@ - delete this and any preceding spaces
|
|
# @' - replace this by a double quote
|
|
#
|
|
# This allows macro substitution by the C compiler thus:
|
|
#
|
|
# #define first_name John
|
|
# #define last_name Smith
|
|
#
|
|
# PNG_DFN"#define name @'@" first_name "@ @" last_name "@@'"
|
|
#
|
|
# Might get C preprocessed to:
|
|
#
|
|
# PNG_DFN "#define foo @'@" John "@ @" Smith "@@'"
|
|
#
|
|
# Which this script reduces to:
|
|
#
|
|
# #define name "John Smith"
|
|
#
|
|
while (1) {
|
|
# While there is an @" remove it and the next "@
|
|
if (line ~ /@"/) {
|
|
if (line ~ /@".*"@/) {
|
|
# Do this special case first to avoid swallowing extra spaces
|
|
# before or after the @ stuff:
|
|
if (!sub(/@" *"@/, "", line)) {
|
|
# Ok, do it in pieces - there has to be a non-space between the
|
|
# two. NOTE: really weird things happen if a leading @" is
|
|
# lost - the code will error out below (I believe).
|
|
if (!sub(/@" */, "", line) || !sub(/ *"@/, "", line)) {
|
|
print "line", lineno, ": internal error:", orig
|
|
exit 1
|
|
}
|
|
}
|
|
}
|
|
|
|
# There is no matching "@. Assume a split line
|
|
else while (1) {
|
|
if (getline nextline) {
|
|
# If the line starts with '#' it is a preprocessor line directive
|
|
# from cc -E; skip it:
|
|
if (nextline !~ /^#/) {
|
|
line = line " " nextline
|
|
break
|
|
}
|
|
} else {
|
|
# This is end-of-input - probably a missing "@ on the first line:
|
|
print "line", lineno ": unbalanced @\" ... \"@ pair"
|
|
err=1
|
|
next
|
|
}
|
|
}
|
|
|
|
# Keep going until all the @" have gone
|
|
continue
|
|
}
|
|
|
|
# Attempt to remove a trailing " (not preceded by '@') - if this can
|
|
# be done, stop now; if not assume a split line again
|
|
if (sub(/"[^"]*$/, "", line))
|
|
break
|
|
|
|
# Read another line
|
|
while (1) {
|
|
if (getline nextline) {
|
|
if (nextline !~ /^#/) {
|
|
line = line " " nextline
|
|
# Go back to stripping @" "@ pairs
|
|
break
|
|
}
|
|
} else {
|
|
print "line", lineno ": unterminated PNG_DFN string"
|
|
err=1
|
|
next
|
|
}
|
|
}
|
|
}
|
|
|
|
# Put any needed double quotes in (at the end, because these would otherwise
|
|
# interfere with the processing above.)
|
|
gsub(/@'/,"\"", line)
|
|
|
|
# Remove any trailing spaces (not really required, but for
|
|
# editorial consistency
|
|
sub(/ *$/, "", line)
|
|
|
|
# Remove trailing CR
|
|
sub(/
|
|
$/, "", line)
|
|
|
|
if (sort) {
|
|
if (split(line, parts) < sort) {
|
|
print "line", lineno ": missing sort field:", line
|
|
err=1
|
|
} else
|
|
array[parts[sort]] = line
|
|
}
|
|
|
|
else
|
|
print line >out
|
|
next
|
|
}
|
|
|
|
/PNG_DFN/{
|
|
print "line", NR, "incorrectly formatted PNG_DFN line:"
|
|
print $0
|
|
err = 1
|
|
}
|
|
|
|
END{
|
|
if (out_count > 0 || err > 0)
|
|
exit err
|
|
|
|
print "no definition lines found"
|
|
exit 1
|
|
}
|