# Best abuse of CPP Don Yang The code for this entry can be found in prog.c ## Judges' comments: ### To use: make ./prog ### Try: echo "International Obfuscated C Code Contest 2020" | ./prog 23209 > code.c # compile without a PIN cc code.c -o decode # yes, you must compile with -DPIN= # compile with the wrong pin cc -DPIN=555 code.c -o wrongcoded # wait for it ... ./wrongcoded | hexdump -C # Use the wrong PIN: You're gonna have a bad time. # compile with a PIN cc -DPIN=23209 code.c -o decoded # wait for it ... ./decoded # weeeeeeeeeeeeee! You are having a good time! ### Selected Judges Remarks: You might think of this entry as code obfuscater with a code. If you use the wrong code, you're gonna have a bad time. What is compiling this entry about? [Having a good .. time](https://southpark.fandom.com/wiki/Asspen/Script)! This winning entry also comes with a [JavaScript timelapse spoiler](spoiler.html) that shows how this entry was written. ## Author's comments: ### Synopsis echo "Hello world" | ./mile 12345 > encoded.c gcc -DPIN=12345 encoded.c -o ./decoded && ./decoded ### Details Mile is a C code generator. Run without arguments for the help message below: usage: [PIN] < input.txt > output.c where [PIN] is an integer in the range of 1..65535 When run with an encoding PIN (first command line argument), Mile will read bytes from stdin and generate C code on stdout. Example: echo "Hello, world!" | ./mile 56789 > encoded.c This generated code will reproduce the same input bytes on stdout when executed, if it was compiled with the same encoding PIN. gcc -DPIN=56789 encode.c -o decoded The generated code reconstructs all data at compile time by seeding a random number generator with the given PIN to reproduce the input bytes, and simply does one fwrite() call at run time to print those bytes. Not specifying a PIN is a compile time error. Specify the wrong PIN and output code will still compile, but will likely produce garbage instead of the input bytes. If the original input was empty then any PIN works, but you might get a compile time warning complaining about empty arrays. Because all data is reconstructed at compile time, expect the compile time to be excruciatingly long if you try to encode too much data. Code generated by encoding a haiku will probably compile in seconds, a limerick may require tens of seconds, a sonnet will take minutes. Give Mile an epic poem and you can probably write a faster preprocessor before GCC is done (Clang probably already failed long before then). I see that there is a lack of preprocessor abuse in recent years, probably because programming in preprocessor is too tedious. Mile is meant to help with that by producing C code filled with at least 98% preprocessor lines for any input. Now you too can enjoy long compile times with minimal effort. ### Features + Output is randomized using system clock, so if you wait a second between runs, you should get a different generated code even for the same encoding PIN. + Customized bytecode encoding of preprocessor boilerplates, mostly for compression but should also make reverse engineering more fun. + CRC32 of the source code is embedded in the source code itself. + Code layout shaped to resemble "Mile" from "Didn't I Say to Make My Abilities Average in the Next Life?!". Mile is known to be able to talk to nanomachines directly, which is exactly what programming in preprocessor feels like. ### Compatibility Mile has been verified to work in these environments (also, compiles without warnings on all of these): + gcc 9.2.0 on Cygwin + gcc 8.3.0 on Linux + gcc 7.4.0 on Cygwin + gcc 6.1.0 on JSLinux + gcc 5.4.0 on Linux + clang 8.0.1 on Linux + clang 8.0.1 on Cygwin + tcc 0.9.27 on Linux + tcc 0.9.25 on JSLinux Output of Mile is a bit more taxing, and only works with a subset of the above: + gcc: works just fine when using 8.3.0 and below, but may trigger warning for large files. gcc 9.2.0 and above may crash with large files. + clang: works fine for small files, fails on large files due to [preprocessor bug](https://bugs.llvm.org/show_bug.cgi?id=44480). + tcc: works just fine on Linux. Fails silently on JSLinux. For people eager to encode large files, an extra utility has been packaged with this entry to preprocess the generated output to make them more acceptable to typical compilers. View the [cppp.cc](cppp.cc) source and then compile it with a c++ compiler of your choice. ----------------------------------------------------------------------------------------------------- (c) Copyright 1984-2020, [Leo Broukhis, Simon Cooper, Landon Curt Noll][judges] - All rights reserved This work is licensed under a [Creative Commons Attribution-ShareAlike 3.0 Unported License][cc]. [judges]: http://www.ioccc.org/judges.html [cc]: http://creativecommons.org/licenses/by-sa/3.0/ -----------------------------------------------------------------------------------------------------