Author:
- Name: Jens Schweikhardt
Location: DE - Federal Republic of Germany (Germany)
To build:
make all
To use:
./yarng
or
./yarng integer_number
Try:
./try.sh
./yarng
# watch what it does
What is the difference between what is in the script and running yarng
without
an arg?
Judges’ remarks:
Run the program with a small argument, like 5 or 10, several times. Are the results different? Why?
Historical remarks:
At the time of judging, some non-gcc compilers that were not fully ANSI standard did not compile this entry correctly. Also at that time, some gcc and egcs implementations ran into a problem when building the entry. Doing:
make schweikh2
often produced an error of the form:
gcc -ansi schweikh2.alt.c -o schweikh2.alt
as: Error: cca00NzV.s, line 58: Missing " at end of string
"
as: Error: cca00NzV.s, line 58: malformed statement
*** Error code 1 (bu21)
because the line:
#line 10 ONE(O(1,1,2,6,0,6))
turned into the line:
# 9 "01\012"
(note, though, how that line is on line 11) which caused gcc to give to the assembler the following two lines:
.stabs "schweikh2.c
",132,0,0,Ltext1
and the lone "
after the #.file
line resulted in an assembly syntax error.
In some cases one had to compile using gcc -g
:
make schweikh2 CFLAGS=-g
to trigger this error.
With the proper text beyond the \012
, it was even possible to have all kinds
of fun adding inline assembly via #line
directives. :-)
For extra fun and credit:
If your compiler had this bug, you could transform the line in the C program to inject assembly instructions in such a way that the program would compile.
If you were able to do that, you could have tried adding assembly instructions so that it will compile, execute and do something interesting (or uninteresting, for that matter :-) ).
If you succeeded in doing that (especially on FreeBSD, Linux or SunOS/Solaris), you could have emailed the author for a free pat on the back (and maybe you still can :-) ).
Because of this bug, the code was changed to be instead:
#line 10 "01\015"
However in 2023 it was observed that it is gcc (at least some versions?) that has a problem with the compilation of the modified program, giving an internal compiler error:
:10:16: warning: type defaults to 'int' in declaration of 'zero' [-Wimplicit-int]
: In function 'main':
:12:1: warning: missing terminating " character
:12:19: internal compiler error: invalid built-in macro "__FILE__"
so the line:
#line 10 "01\015"
was changed to:
#line 10 ONE(O(1,1,2,6,0,6))
and now it works with both clang and gcc.
Author’s remarks:
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.
What this program does
My entry is a yarng
, which stands for, you guessed it, yet another
random number generator, and is pronounced ‘yawn’ due to its
incredible speed. Seriously, it exploits the randomness of quantum
mechanical events that build up the hardware as well as some other
non-deterministic factors. Here’s how it works:
- an alarm is scheduled to be delivered in a second
- an infinite loop is entered that just increments a counter
- on delivery of the alarm signal, the counter’s LSB is printed.
- while some other counter is not zero, go to step (1)
- exit
The yarng
therefore produces output at a rate of 1 baud. The number of
bits may be specified by an argument and defaults either to UINT_MAX
(least likely) or UINT_MAX+1
(most likely). Extra credit: why the
different values? How long would you need for an empirical test?
For example:
$ ./yarng 32
10111001010101110110010101001111
$ ./yarng 5
10011
Why I think this is obfuscated
Ever seen a
do for ... while
loop?I avoided
int
like the plague. Instead I used storage class specifiers (register
,auto
) to get implicitint
. The Standard allows implicitint
even in casts as you can see in(void(*)(register))main
. Where implicitint
is impossible, like in theint main
declaration I haveint
split across two lines. All indent programs I use insert space at the beginning of the continuation line and thereby introduce a syntax error.main()
(ab)used as a signal handler. Shocking. Yeah, this leads to undefined behaviour. I have yet to see a system where it does not work. If it doesn’t work for you, I’d appreciate if you mail me about it. Uh-oh, Schweikh did it, uttered “Works for me”. Don’t quote me on that!Even if you know how the program works, you just can’t predict the output.
Chained use of token pasting operators
##
with shuffled arguments.So you think you can’t assign to
__LINE__
and__FILE__
? Not so. The#line
preprocessor directive does the trick. But saying#line 10 "01\n"
is a little lame and I came up with??=line 10 ONE(O(1,1,2,6,0,6))
.__LINE__
is used as the signal number argument tosignal()
. If indenting or other editing changes the number of lines the program is likely to be killed after one second… Thanks to POSIX for specifyingSIGALRM
as14
. You’re SOL if your system does not conform. Get a real OS.All strings (or characters) needed are crammed into
__FILE__
using octal notation and then extracted using pointer + offset notation.Use of a
#
(null) preprocessor directive (hidden as a%:
digraph) Delete this line and prepare for a core dump.Identifiers reused with different capitalization:
one
,One
,ONE
,zero
,Zero
.ONE
is particularly nasty because it appears as a macro taking an arg and as a plain identifier inmain
’s parameters. The rule is that wheneverONE
is followed by(
it must be the macro.A quote from Kernighan and Plauger, The Elements of Programming Style, 2nd Ed., page 21:
"A useful way to decide if some piece of code is clear or not is the 'telephone test.' If someone could understand your code when read aloud over the telephone, it's clear enough. If not then it needs rewriting."
Now try the telephone test for ‘one = zero = atoi (ONE[zero-1])’.Don’t you love the aesthetics of symmetry:
write (1, 1+__FILE__+1, 1)
(telephone test this as well; is it 1, One, ONE, one? Maybe won?).
Where does the randomness come from?
Each bit obviously depends on the number of times the loop body is executed between two calls of the signal handler. But what determines this number?
- The accuracy of the clock driving the CPU.
- The accuracy of the clock driving the timer responsible for alarm signals.
- On multitasking OSes: The total time the process is scheduled to run between successive invocations of the signal handler.
How about the randomness of the yarng
? To measure these properties I
counted the number of bit sequences and compared them to what I would
expect from a generator with uniform distribution (all sequences
equally likely to occur).
It is interesting to note that yarng
’s maximum deviations are less
than or equal to rand(3)
’s. Randomness however tends to degrade under
high load averages on my system (Linux). This could be avoided by
allowing the process one second of execution time instead of wall
clock time (using the SIGVTALRM
sent by the virtual timer
instead of SIGALRM
).
I’m not surprised. Yarng
in the end depends on fluctuations of the
electrical field and the heat distribution in the oscillators, the
implementation and state of the scheduling algorithm, the system
load, number and temporal distribution of asynchronous events and
last but not least the phase of the moon. God does play dice.
Type yarng
and watch Him roll.
Inventory for 1998/schweikh2
Primary files
- schweikh2.c - entry source code
- Makefile - entry Makefile
- schweikh2.orig.c - original source code
- try.sh - script to try entry
Secondary files
- 1998_schweikh2.tar.bz2 - download entry tarball
- README.md - markdown source for this web page
- .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