Author:
- Name: Dave Burton
Location: US - United States of America (United States)
To build:
make
To use:
./prog [-tcksri] < file.c
Try:
./try.sh
If you get really stuck, try:
man ./tac.1
Judges’ remarks:
They say size isn’t everything, and in the case of IOCCC iocccsize.c that is saying something! What is this program weighing and how much does it weigh?
Author’s remarks:
tac - tokenize and count C and derivative languages:
tac
computes C program size
(obfuscated / otherwise)
by splitting code as tokens small,
ignoring space, then counting all.
It tries quite hard to act the same
as counting tool of I-OCCC fame.
Some bugs were fixed, it’s faster too,
and does more things, just add some glue!
It does far more than print the size:
the token part’s a better prize!
With each upon on a single line,
how many things can you design?
Included is a keyword sorter,
de-obfuscator, freq reporter.
With code produced as little parts,
just add your own creative arts.
But first you have to figure out
the braces part, without a doubt!
And then the code - it isn’t easy:
for as else? That’s rather cheesy!
The rest of it is standard fare:
expressions strange, the globals bare,
confusing symbols one and oh,
and precedence you have to know.
The keyword list was made external,
because of standards change eternal.
Thus other languages can now,
be counted just like C, somehow!
I hope you find this code obscure
enough to win, and thus procure
a public place for all to see
how badly I can butcher C!
Said another way….
tac
is the inverse of cat
(1): it un-concatenates its input into C tokens,
writing the token stream to stdout
. There are options to suppress the token
ids, summarize the token counts, and several options to allow it to be
compatible with iocccsize
.
The token-based output is quite useful in analyzing source code (see below),
and the counter is more accurate than iocccsize
(see below); but this can be
fixed by using the -c
compatibility flag, restoring the following problems as
in the official tool:
- Whitespace within a string is not counted; this seems wrong, as these chars are significant.
- “
{;}
” is not counted within a double-quoted string. - Whitespace preceding a comment is
char
-counted, unless the comment starts a line. -k
processes the reserved words within//
comments(!), but not/*
, changing the counts.
The following bugs are always corrected:
- Treats
/*/
as a complete comment (/*/int if for/*/
has three keywords) -k
enables//*
or within a line comment/*
to initiate a comment block until following*/
-k
processes double quote characters in//
comments - thus a solitary quote consumes everything until the next quote- Single quotes are not handled correctly (
'"'
starts a double quote iniocccsize
) #ifndef
is omitted as a keyword (but#ifdef
is allowed?)I
is not a reserved word.tac
correctly penetrates 23rd century cloaking technology. ;-)
There is one (known) remaining problem with detection:
- Comments between preprocessor hash and keyword are not replaced with whitespace;
the following counts #include as two tokens,
#
andinclude
(not as a reserved word):
#/*this will not be counted*/include/*correctly*/<stdio.h>
Fixing this properly hampers backwards compatibility in counting. This requires
eliding comments during the translation phases (ISO 5.1.1.2, phase 3), and yet still
counting words correctly absent -k
.
Backwards Compatibility:
tac
was run over all 366 previous IOCCC winning entries:
find ~/src/obc -type f -name \*.c | wc
The discrepancies found are documented and explained in the file discrepancies.html.
find ~/src/obc -type f -a -name "*.c" | xargs ./spotcheck.sh ./prog | ./spotdiff.sh |
grep -v keep | diff -bw - discrep* | grep "[<>] cl "
In summary, there are only 6 unique entries out of 366 that have any variation
in Rule 2 counts; in all cases, the differences are due to bugs within
iocccsize
, described in detail in the file
discrepancies.html. There are only 40
files (40/366, ~11%, including originals and variations) that differ at all, 31
only in word count; most word count differences appear related to comment block
detection, weighted more towards recent entries (2011 and onward = 20/40 ~50%).
This (minor) variation in word count values has not been addressed since
it is not a qualifying metric, nor is it very useful in a programming context:
whitespace surrounds English words, not necessarily C tokens, as in a+=1;
.
Accuracy (versus compatibility) is demonstrated through a series of small test cases designed to allow exhaustive, hand-counted values for comparison with the tool values. These test cases were vital in debugging and regression testing, and provided a way to determine which tool was correct when there were differences.
A version of this tool in more clearly written C
is presented for the Judge’s consideration as a more accurate replacement foriocccsize
.
This obfuscated entry is derived from (and compatible with) that code,
but due to obfuscation, has had some significant, deep changes for the contest.
Nevertheless, the more clearly written code remains a spoiler for this entry.
NB: iocccsize
gets a different answer from tac
on its own
iocccsize.c source code;
tac
gets the correct answer. This is due to the aforementioned bugs within iocccsize
,
proved by fixing
iocccsize.c with the included patch, so iocccsize
reports
the correct answer for itself.
But wait… There’s More!
There is no limit on line length, file length, comment length, or identifier length.
tac
features an accurate tokenizer; which presents each C token on a line by itself.
This feature enables interesting analyses of C (and C++) code that’s hard to get any other way.
For instance, here is a simple token counter, useful in finding repeated long tokens, or operator frequency, or counting the references to identifiers, constants, or breadth of use of the language:
#!/usr/bin/env bash
cat $* | ./prog -t | sort | uniq -c | sort -k1nr
And here is a C keyword frequency counter:
#!/usr/bin/env bash
function iskeyword {
awk 'BEGIN{f="c11";while(getline<f)k[$1]=0} {if($1 in k)k[$1]++}END{for(i in k)print k[i],i}'
}
cat $* | ./prog -t | iskeyword | sort -k1nr
This script was used to “optimize” the reserved word order so the most frequent IOCCC winning entry keywords are checked first.
An interesting aside: as might be expected, obfuscation has changed the frequency of keyword distribution over time. The top five for the decades:
1980s 1990s 2000s 2010s
----- ----- ----- -----
94 if 532 char 375 if 254 int
69 for 524 if 372 char 217 for
59 while 417 int 332 int 198 if
48 char 223 for 184 for 121 return
43 int 223 void 165 return 104 char
The keyword list is external to the program, and is easily changed, sorted, checked, verified. This allows such additional programs in concert with the tool’s primary operation.
A poor-man’s de-obfuscator can be based upon the output of tac -t
and a
handful of simple rules (and another use of the external keyword file). A more
refined version of this is included in the file unob.sh, but the
simple code below is a serviceable obfuscated C de-obfuscator in a scripting
language. It really is this easy with tac
:
#!/usr/bin/env bash
script='
BEGIN {
last=nl="\n";
f="c11"; while(getline <f > 0) kw[$0]++; close(f);
}
function iskw(a) { return a in kw }
function indent(a) { return sprintf("%*s", n*3, " ") }
function newline() { if (!infor && last != nl) printf last=nl; }
function show(a) {
if (last==nl) printf "%s", indent()
printf "%s%s", space(), a
last=a
}
function space() {
return iskw(last) ||
(last ~ /[A-Za-z0-9_+-\/%^[&\]\)=:<>;]$/ && $0 !~ /[:;()\[\],]/) ? " " : ""
}
/^\(/ { ++paren }
/^\)/ { --paren }
/^for/ { newline(); infor=1 }
infor && /^;/ { ++infor; show($0 " "); next }
infor==1 && /^:/ { ++infor }
infor>1 && paren==0 { infor=0 }
/^\?/ { tern++; n++; newline(); show($0 " "); next }
tern && /^:/ { newline(); show($0); --tern; --n; next }
/^#/ { newline(); show($0); newline(); next }
/^;/ { show($0); newline(); next }
/^{/ { show($0); ++n; newline(); next }
/^}/ { n--; newline(); show($0); newline(); next }
{ show($0) }
'
cat $* | sed 's/#include/##include/' | cpp -E -trigraphs |
sed 's/^# .*$//' | sed 's/^#//' |
./prog -t | awk "$script"
On the question of obfuscation:
I assume you noticed the braces?
Every brace is sacred,
Every brace is great.
If a brace is wasted,
Clang gets quite irate.Keywords follow the 2010s top five, without the
if
, for interesting flow control.Sometimes a
for
can be anelse
- inconceivable!I see the
getopt(3)
. But “where’s the beef?”No character constants, several
int
constants, and one string (hint: which does not encode letters) – how does this code parse code?Interesting aside: how many is “several
int
constants”?./prog < prog.c 2>/dev/null | awk ‘$1~/260/{a[$2]++}END{for(i in a)print i,a[i]}’ … | wc
Which numbers are most often used?
… | sort -k2nr | sed 5q
The code describes its function by careful arrangement of variables up front…
…coupled with a description of the typical IOCCC contestant, or at least the author
Why shouldn’t trigraph parsing be written in trigraph?
Where iocccsize.c mocks, this code flaunts: “no matter how well you may think you understand this code, you don’t, so don’t mess with it. :-)”
O,0,l,1
are used to confusing effect, local names obscure global names.Globals are used to pass information between routines: don’t reorder “unrelated” statements….
How does the compatibility mode account for the fundamentally different manner of computation?
The algorithm is quite efficient; reserved word detection and trigraph parsing particularly so.
This may be the first obfuscated submission that provides the means for its own de-obfuscation.
Rule 2:
The keyword list is externalized as an include file. This presents a useful
feature for the source code: not only can the reserved word script above be
produced, but the tool can be rebuilt with any set of reserved words, just by
using a different list. For instance: removing all the secondary keywords;
removing all the C11 keywords; trying just K&R C. Since C++ and Java share the
same operators as C, just change the keyword list and tac
will correctly
tokenize both of these; the tokens >>>
and ::
can be handled with a short
post-filter (tokenfix.sh, included).
I believe this entry may also satisfy the request for gratuitous use of all the
C11 keywords? Whether this context satisfies “intended C language context” is a
matter of interpretation. ;-) And tac
is ready for the IOCCC++ contest. And
the IOJCC trash can.
The following reserved word files are included:
- kandr from my venerable 1978 18th printing “The C Programming Language”
- v7unix 7th edition Unix source code, extracted from c00.c
- kandr2 from my 1988 1st printing “The C Programming Language”, 2e
- ansi ANSI X3.159-1989
- c99 ISO/IEC 9899:1999(E)
- c11 ISO/IEC 9899:201x(E) N1570
- c++98 http://en.cppreference.com/w/cpp/keyword
- c++11 ISO/IEC 14882:2011(E) N3337 2012-01-16
- c++14 ISO/IEC 14882:2014(E) N4296 2014-11-09
- java8 http://docs.oracle.com/javase/specs/index.html
- ioccc.kw.freq c11 + additional words, sorted on frequency of occurrence in IOCCC entries
NB: The keyword file used in this code is derived from the list in iocccsize.c,
which is neither complete (#define
, #ifndef
, #undef
are missing
yes, Virginia knows about `#define` omitted on purpose
nor correct (many more are added: I
, true
, bool
, compl
, …):
(sed -n '1p' c11; sed -n '2,$p' ioccc.kw.freq | sort) | comm -3 - c11
That is an obfuscated way to say “diff”. But it also more clearly shows the, um, diffs.
The usage string is similarly externalized, providing a better usability string than
the limits would otherwise allow. If this is considered bending the rules too far,
just -DU=O
to get a passible, if cryptic, usage message using no additional octets
than without the define.
This also allows the builder to configure the amount of help a utility should provide
(you want a manpage? Because that’s how you get a manpage.) – a rather
significant point of contention made moot.
There is definitely abuse of the Rule2 counter, exploiting a “feature” of
iocccsize
: iocccsize
does not Rule2 count {
, }
and ;
when followed by
whitespace – even within a string! Thus: Freecode, a simple, uncounted, highly
obfuscated encoding of a central part of this program. (The reserved word list
is not Freecoded, both because it is useful apart from the program, and because
Freecode is not entirely free.)
- Some restrictions apply. Not approved for all uses. Some assembly required.
Compilation:
The code is ANSI compliant, but it does have a string longer than 509 characters. It has been tested on OSX and Linux, Clang and GCC.
The code uses trigraphs, ironically: it parses trigraphs with trigraph code.
Because that’s funny. And it remains obscure after, so trigraphs are not used
for obfuscation. Thus -trigraphs -Wno-trigraphs
(which is also humorous).
It’s obfuscated C. Thus -Wno-parentheses -Wno-empty-body -Wno-char-subscripts
.
The code uses unsigned char
s, but C strings are signed
. Thus -Wno-pointer-sign
.
NB: This is not a portability concern, the one string holds only printable ASCII characters.
Thus:
cc -ansi -Wall -trigraphs -Wno-trigraphs -Wno-parentheses -Wno-empty-body -Wno-char-subscripts -Wno-pointer-sign -DU=O -DW=\"keywords\" -o prog prog.c
Final remarks:
Cody Boone Ferguson was relentless in his pursuit of bugs.
Thanks to his reports, the version of unob.sh is stronger,
tac
groks digraphs, tokenfix.sh corrects for missing digraphs
in prog.c, and the program
manpage.sh was added to the corpus.
manpage.sh is a useful program that turns ASCII versions of man
pages (e.g., tac(1) man page) into real man pages. It works for C and C-like
languages, command lines, and was designed to handle man page sections 1-8. It
works well on most inputs, but tends to need some raw nroff for more fancy
constructions, such as combined arguments (e.g. the -r
and -s
arguments to tac
are not recognised when written as -rs
in prose).
Try:
./manpage.sh tac
./manpage.sh -h
./manpage.sh -h | ./manpage.sh
./manpage.sh -h | ./manpage.sh -R | less
./manpage.sh -h | ./manpage.sh -P | lpr
NOTE: in 2023 the tac.man
was updated to proper man format and renamed
tac.1.
Inventory for 2018/burton2
Primary files
- prog.c - entry source code
- Makefile - entry Makefile
- prog.orig.c - original source code
- discrepancies.html - field-by-field difference between iocccsize and entry’s ioc
- freqcount.sh - frequency count utility
- tac.1 - tac.c man page
- tokcount.sh - count the tokens produced by ./prog -t
- try.sh - script to try entry
- fixed_iocccsize.c - iocccsize.c with author patch applied
- iocccbug01.c - test cases demonstrating problems with iocccsize
- iocccbug02.c - test cases demonstrating problems with iocccsize
- iocccbug03.c - test cases demonstrating problems with iocccsize
- iocccbug04.c - test cases demonstrating problems with iocccsize
- iocccbug05.c - test cases demonstrating problems with iocccsize
- iocccbug06.c - test cases demonstrating problems with iocccsize
- ioccc.kw.freq - C keywords sorted by 201x frequency
- ioc_without_quote - test suite data
- manpage.sh - turns ASCII versions of man into real man pages
- mkkeywords.sh - reads a file from variations - produces string for prog.c
- results.k - results with -k option
- tac.c - tokenize and count C-like languages
- test01.c - test case source code
- test02.c - test case source code
- test03.c - test case source code
- test04.c - test case source code
- test05.c - test case source code
- test06.c - test case source code
- test07.c - test case source code
- test08.c - test case source code
- test09.c - test case source code
- test10.c - test case source code
- test11.c - test case source code
- test12.c - test case source code
- test13.c - test case source code
- test14.c - test case source code
- tokenfix.sh - short post-filter
- unob.sh - refined version of de-obfuscator
- ansi - reserved words for ANSI C standard
- author-README.txt - author supplied README
- c++11 - reserved words for C++11 standard
- c11 - reserved words for C11 standard
- c++14 - reserved words for C++14 standard
- c++98 - reserved words for C++98 standard
- c99 - reserved words for C99 standard
- java8 - reserved words for Java v8
- kandr2 - reserved words for K&R2 C
- kandr - reserved words for K&R C
- patch - author supplied patch to fix iocccsize.c source
- results - results without -k
- runtest.sh - test suite
- spotcheck.sh - script to test a lot of files vs iocccsize
- v7unix - reserved words for a v7 UNIX C standard
- spotdiff.sh - summarize the results of spotcheck
- src.doc.txt - deobfuscation hints about the source code
Secondary files
- 2018_burton2.tar.bz2 - download entry tarball
- README.md - markdown source for this web page
- discrepancies.md - markdown source for discrepancies.html
- .entry.json - entry summary and manifest in JSON
- .gitignore - list of files that should not be committed under git
- .path - directory path from top level directory
- index.html - this web page