IOCCC image by Matt Zucker

The International Obfuscated C Code Contest

2006/monge - Best compiled graphics

Graphical fractal generator

Author:

To build:

Make sure you have the SDL1 (not SDL2!) development environment installed.

    make

There is alternate version that lets you redefine the width and height and number of iterations to use. See Alternate code below.

Bugs and (Mis)features:

The current status of this entry is:

STATUS: doesn’t work with some platforms - please help us fix it
STATUS: INABIAF - please DO NOT fix

For more detailed information see 2006/monge in bugs.html.

To use:

    ./monge expression ...

Incorrect formulas will ungracefully crash the program.

Try:

    ./try.sh

Alternate code:

This alternate code lets you specify the width and height of the image as well as the number of iterations to perform.

Alternate build:

To use the default values, the same as monge.c, just do:

    make alt

But if you wish to actually make use of it you will want to redefine one or more of the variables W, H and I. The defaults are (respectively) 400, 300 and \128. If you wish to change the image size to 500x500 but keep the iterations the same then do:

    make clobber WIDTH=500 HEIGHT=500 alt

If you wish to keep the dimensions the same but change the iterations to 512:

    make clobber ITERATIONS=512 alt

If you specify a value less than 1 for any of these it sets it back to the default. The value 1 was selected arbitrarily and such a small number as 1 will not do anything useful but as long as it’s at least 1 it’ll not be redefined. It does not try taking care of overflows but such large values would be impractical anyway.

Alternate use:

Use monge.alt exactly as you would monge above.

Alternate try:

    ./try.alt.sh

Judges’ remarks:

For those who are familiar with the previous IOCCC entries, this program is best described as “1994/tvr meets 2001/bellard”. Here you have mouse-manipulated fractals AND generated binary code, all in one package.

A more appropriate award could have been “Best abuse of the rules”, because on a non-x86-based architecture this program is, quite predictably, a dud.

At the time of judging we were too mesmerized by the graphics to realize it; and, after all, this entry does take a special effort to work on both i386 and x86_64. Portable it is! :)

Author’s remarks:

For the impatient:

  1. Compile using the Makefile, or just run:
    gcc monge.c -o monge -O3 `sdl-config --libs --cflags`
  1. Run:
    ./monge "z = 0" "z = z*z + c; Abs2(z) < 4"
  1. Keep clicking with left or right button to zoom in or out.

  2. Use the number pad + and - to increase/decrease the iteration count when needed.

  3. Enjoy, unless you’re more interested in trying to understand how it works. :)

Introduction

This is a fractal generator that supports custom formulas and real time zoom. The program needs to be run with two arguments, the first being a semicolon separated list of assignments to do once per pixel, and the second a semicolon separated list of assignments and conditions that will be executed (and conditions checked) for each iteration (up to the maximum). Using the left and right mouse buttons you will be able to zoom in and out (a la XaoS project), the zooming algorithm is a bit slower than XaoS’s but it should give slightly better results.

Formulas can use all a-z (lowercase) variables, and the special variables c and i are respectively set by default to the complex coordinate of the pixel and to the complex imaginary i.

All the constant numbers will be interpreted as real double values. Each expression must be an assignment ([variable] = [expr]) or a comparison with < or > ([expr1] < [expr2] or [expr1] > [expr2]), and you can add spaces as you like (as understood by the C function isspace()).

Supported operations and functions are:

Here are a few examples of fractals you can draw:

    ./monge "z=1" "z=z*z+c; Abs2(z)<4"
    ./monge "z=c" "z=z*z+c; Abs2(z-c)>0.0001"
    ./monge "z=c; c=0.31+i*0.5" "z=z*z+c; Abs2(z)<4"
    ./monge "z=c; c=0.31+i*0.5" "z=z*z+c; Abs2(z-c)>0.0001"
    ./monge "z=c" "p=z; z=0.6666*z+0.3333/(z*z); Abs2(p-z) > 0.001"
    ./monge "z=0" "p=z; z=z-(z*z*z + (c-1)*z - c)/(3*z*z+c-1); Abs2(p-z) > 0.001"
    ./monge "z=0; q=0" "t=z; z=z*z+Re(c)+Im(c)*q; q=t; Abs2(z)<4"
    ./monge "z=c; c=0.56667-i*0.5; q=0" "t=z; z=z*z+Re(c)+Im(c)*q; q=t; Abs2(z)<4"

Compilation and portability

This program will only work on x86 (with an x87 FPU) or x86_64 machines, but it requires the SDL library.

Another system requirement is the mmap(2) function (as #defined at the beginning of the program). If it is not available the macro M(a) will have to be replaced with a system dependent function that allocates readable, writable AND executable memory (it will not be possible to make this program run on paranoid systems (like OpenBSD IIRC) that do not allow rwx memory).

However I was able to compile and make work the program under Linux (Debian/x86 and openSUSE/x86_64) and under Cygwin/x86. A friend of mine was able to make it work under macOS too.

If you want you can change the window width and height, or the default number of iterations (that one can be tweaked at runtime, anyway) by changing the definitions of W, H and I at the beginning of the code.

Caveats (I just a selected few of them!)

NOTICE to those who wish for a greater challenge:

If you want a greater challenge, don’t read any further: just try to understand the program via the source.

If you get stuck, come back and read below for additional hints and information.

Obfuscation

Sure, I don’t want to deprive you of the pleasure of digging into the infernal mess created by my corrupted mind (writing this remark I noticed that I was ending sentences with ‘;’ instead of with ‘.’, and I worried for my sanity), but just in case…

I used many obfuscation techniques, including a few ones that are different from the tricks used in most common IOCCC program (more ‘high level’, in the sense that they require an understanding of how all the program works to be worked out).

This program is a formula parser that outputs machine code that calculates the formula and check the given conditions. The machine code targets the x87 stack based FPU, and it is almost identical for x86 and x86_64 (except for the comparison instructions, that have to be generated differently). The program keeps track of the FPU stack (the FPU has a cyclic stack of 8 registers) and will automatically swap to and from the main memory when required (i.e., adds instructions to the generated machine code to take care of this issue).

The formulas are parsed recursively and machine code for the inner iteration loop is also generated.

After generating the machine code, a big event loop creates (or zooms into) the fractal executing the generated code when needed, and then waits for input events.

The zooming algorithm works by remembering for each pixel the floating point offset to where the value was actually calculated. When zooming, the best match is searched, and if the offset is too big (>1) the pixel is recalculated.

Inventory for 2006/monge

Primary files

Secondary files


Jump to: top