IOCCC image by Matt Zucker

The International Obfuscated C Code Contest

2005/mikeash - Best use of parenthesis

Author:

To build:

    make

Bugs and (Mis)features:

The current status of this entry is:

STATUS: INABIAF - please DO NOT fix

For more detailed information see 2005/mikeash in bugs.html.

To use:

    ./mikeash

NOTE: the author stated that they tested this under i386 machines but this works under 64-bit (x86_64, arm64) machines too.

Try:

    ./try.sh

Judges’ remarks:

Having problems speaking code? Do you LISP? Parenthetically speaking, this entry takes advantage of C99 features, while speaking parenthetically. Best of all, it self-reproduces!

John McCarthy might not have imagined it quite like this! :-)

Author’s remarks:

This program is a self-reproducing program, which is on the IOCCC’s list of discouraged program types. Worse, it’s a self-reproducing program written in Common Lisp. What’s this doing as an entry to the IOCCC?

Start over: this program, when compiled with a C99 compiler, is an interpreter for an extremely limited and highly broken subset of Common Lisp.

It requires C99 because of C99’s ability to declare variables in the initializer of a for loop. Unfortunately, C99 mode typically warns about implicitly declared functions, and preprocessor directives prevent the program from being valid Common Lisp. As a result, several warnings will be printed about implicit declarations of functions. There are also warnings about unknown escape sequences in the constant string, due to different string escape semantics in C and Lisp, and some implicit integer-to-pointer conversions due to the implicitly-declared functions.

Try:

    echo '(format t "~s" (+ 2 2 ))' | ./mikeash

You should see 4" echoed on your command prompt. The " after the 4 and the lack of a newline are due to the somewhat broken implementation of the format function. More sophisticated arithmetic expressions may be used:

    echo '(format t "~s" (* (+ 2 2 ) (- 5 (/ 9 3 ))))' | ./mikeash

This evaluates the expression (2+2) * (5 - 9/3) giving the correct result, 8" (with the same issue of no newline after the "). Note that due to the highly ad-hoc nature of the tokenizer, the spaces before the closing parentheses are mandatory.

The program can also be run in an interactive mode, by simply executing ./mikeash and then typing lisp expressions, but this is difficult due to the lack of proper editing facilities and the total lack of error recovery in the interpreter. Typos will most likely crash the program.

Aside from being required to put spaces between tokens and closing parentheses, there are other limitations on the dialect of Common Lisp that this program understands:

Basically, the LISP interpreter is good for some basic math operations, and for running itself. To do the latter, try this:

    ./mikeash < mikeash.c > mikeash2.c
    diff mikeash.c mikeash2.c

The program can also be run from an implementation of common lisp. To do this, run the common lisp compiler and then type the following command:

    (load "mikeash.c")

The source code of mikeash.c will be printed.

Obfuscation

Lispob is obfuscated in several ways. First, it’s actually two programs in one, which actually share a small amount of code. Running a C code beautifier will destroy the LISP program contained within. Second and more importantly, it is very badly written, which will presumably confuse people who are used to looking at well-implemented interpreters.

The program comes in at just under 2k of significant characters. However, this size is exaggerated, because it contains a complete copy of itself for the self-reproducing LISP program, making it twice the size. This doubling effect made it something of a challenge to fit inside the IOCCC’s size limit.

This code should be highly portable. It was tested under Mac OS X and Linux on i386, but it should run elsewhere. It requires ASCII due to hardcoding various character constants. Other than this requirement, it should run anywhere there’s a conforming C99 compiler and a standard C library.

Inventory for 2005/mikeash

Primary files

Secondary files


Jump to: top