Twigs - Linux 512b (inérciaDemoparty 2020)
data:image/s3,"s3://crabby-images/dfc47/dfc47e89370ba0388b1ce72594317f77b0c34e3f" alt=""
My first tiny program which is basically a POC and
demonstrated that some of my engraving-like IFS fractals
can be reduced down to a small program.
The process was simple on this one because it is a
straightforward port from the Processing sketch, there was four
issues if i recall :
- generating a 512 bytes binary with C (lots of build tweaks)
- finding an okay tiny PRNG,
settled for
this one but i don't extract the high bit and loop, instead i
just use it as is
- reducing the original expressions so it compress better, this
step also came with small compromises but i think the end result is
still pretty close to the original image
- tweaking the code until GCC + LZMA generated small code (mostly
random guess fun on constants and compiler hints !)
There was no saturation on the party release so pixels value
loop around if run for too long, it can be seen on the image above
on parts of the leaves, a decision i had to make because i couldn't
fit saturation
arithmetic at the time, later found a "cheap" solution by using
SIMD
SSE2 instructions.
Also found a quite obvious optimization (code redundancy)
after the party release which resulted in 8 bytes gain on the final
binary.
The PRNG part can also use the RDRAND instruction with
a ~6 bytes gain for a small compatibility price (won't works on old
CPU).
The code is full C
with some minimal assembly language for the fbdev API calls, first
version also use floats.
One advantage of doing it full C (there is no ELF header
tricks on the party version) is that it both works on 32 and 64
bits systems without issues, porting it is thus fairly easy.
The final binary is compressed with
LZMA and a
small shell stub is added at the front of the binary, the stub
execute some command-line instructions at the start of the
executable and unpack the binary somewhere then run it.
Not too fond of the stub method due to the shell /
requirements but it works okay for this kind of size target and it
is a cheap / fast method to shrink the executable.
Perhaps this program could have been reduced to 256 bytes with
assembly and fixed-point
arithmetic.
The binary size vary with the framebuffer resolution (it is
fixed), party version is reduced to 497 bytes at 1024x768.
I later improved the code so that everything fit into 426
bytes at 1920x1080 without the compression / stub phase, it also
use fixed-point arithmetic and a custom ELF header with fbdev setup
code in assembly which helped to reduce the binary size a lot
.
data:image/s3,"s3://crabby-images/2b83b/2b83b2530552001a75f61b495a621ada1e8b56e4" alt=""
what if you zoom out ? (also did a quick
post-processed
version later)
data:image/s3,"s3://crabby-images/9a3fb/9a3fb403bc32d8d283e8db9db055ebb0fe0d163a" alt=""
the original
back to top
data:image/s3,"s3://crabby-images/d6d9f/d6d9f7ddad85fc6077d0b29299878a506dbba65e" alt=""
data:image/s3,"s3://crabby-images/20419/204190d137879aac8af036fb1e67375e757d2323" alt=""
data:image/s3,"s3://crabby-images/5d633/5d63342e168ef56f677a7fcb12350f03584df5cd" alt="Licence Creative Commons"