Selfie is a project of the Computational Systems Group at the Department of Computer Sciences of the University of Salzburg in Austria.
For further information and support please refer to http://selfie.cs.uni-salzburg.at
The first step is to produce a binary that runs on your computer. To do that on a Linux or Mac OS X system use gcc
in a terminal to compile selfie.c
:
gcc -w -m32 -D'main(a, b)=main(int argc, char **argv)' selfie.c -o selfie
This produces from selfie.c
an executable called selfie
as directed by the -o
option. The executable contains the C* compiler, the mipster emulator, and the hypster hypervisor. The -w
option suppresses warnings that can be ignored for now. The -m32
option makes the compiler generate a 32-bit executable (which may require installing gcc-multiarch or gcc-multilib, depending on your system). Selfie only supports 32-bit architectures. The -D
option is needed to bootstrap the main function declaration. The char
data type is not available in C* but required by gcc
.
We have not tested selfie on Windows systems ourselves but users have reported success. Since the only unusual requirement is to have a C compiler installed that supports generating 32-bit binaries, compiling selfie on Windows should be straightforward to do.
Once you have successfully compiled selfie you may invoke it in a terminal as follows:
./selfie { -c source | -o binary | -s assembly | -l binary } [ -m size ... | -d size ... | -y size ... ]
The order in which the options are provided matters for taking full advantage of self-referentiality.
The -c
option invokes the C* compiler on the given source
file producing MIPSter code that is stored internally.
The -o
option writes MIPSter code produced by the most recent compiler invocation to the given binary
file.
The -s
option writes MIPSter assembly of the MIPSter code produced by the most recent compiler invocation including approximate source line numbers to the given assembly
file.
The -l
option loads MIPSter code from the given binary
file. The -o
and -s
options can also be used after the -l
option. However, in this case the -s
option does not generate approximate source line numbers.
The -m
option invokes the mipster emulator to execute MIPSter code most recently loaded or produced by a compiler invocation. The emulator creates a machine instance with size
MB of memory. The source
or binary
name of the MIPSter code and any remaining ...
arguments are passed to the main function of the code. The -d
option is similar to the -m
option except that mipster outputs each executed instruction, its approximate source line number, if available, and the relevant machine state.
The -y
option invokes the hypster hypervisor to execute MIPSter code similar to the mipster emulator. The difference to mipster is that hypster creates MIPSter virtual machines rather than a MIPSter emulator to execute the code.
To compile selfie.c
for mipster and hypster use the following command:
$ ./selfie -c selfie.c -o selfie.m
This produces a MIPSter binary file called selfie.m
that implements selfie.
To execute selfie.m
by mipster use the following command:
$ ./selfie -l selfie.m -m 1
This is semantically equivalent to executing selfie
without any arguments:
$ ./selfie
To execute selfie.m
by hypster use the following command:
$ ./selfie -l selfie.m -y 1
This is semantically equivalent to executing selfie.m
by mipster and thus selfie
without any arguments. There is a difference in output though since mipster reports code execution profiles whereas hypster does not.
Here is an example of how to perform self-compilation of selfie.c
:
$ ./selfie -c selfie.c -o selfie1.m -m 2 -c selfie.c -o selfie2.m
$ diff -s selfie1.m selfie2.m
Files selfie1.m and selfie2.m are identical
Note that at least 2MB of memory is required.
The following example shows how to perform self-execution of selfie.c
. In this case we invoke the emulator to invoke itself which then invokes the compiler to compile itself:
$ ./selfie -c selfie.c -o selfie1.m -m 4 -l selfie1.m -m 2 -c selfie.c -o selfie2.m
$ diff -s selfie1.m selfie2.m
Files selfie1.m and selfie2.m are identical
Note that the example may take several hours to complete. Also, an emulator instance A running an emulator instance B needs more memory than B, say, 4MB rather than 2MB in the example here.
The previous example can also be done by running hypster on mipster. This is significantly faster since hypster does not create a second emulator instance on top of the first emulator instance. Instead, hypster creates a virtual machine to execute selfie that runs concurrently to hypster on the first emulator instance:
$ ./selfie -c selfie.c -o selfie1.m -m 4 -l selfie1.m -y 2 -c selfie.c -o selfie2.m
$ diff -s selfie1.m selfie2.m
Files selfie1.m and selfie2.m are identical
We may even run hypster on hypster on mipster which is still reasonably fast since there is still only one emulator instance involved and hypster itself does not add much overhead:
$ ./selfie -c selfie.c -o selfie1.m -m 8 -l selfie1.m -y 4 -l selfie1.m -y 2 -c selfie.c -o selfie2.m
$ diff -s selfie1.m selfie2.m
Files selfie1.m and selfie2.m are identical
To compile any C* source and execute it right away in a single invocation of selfie
without generating a MIPSter binary use:
$ ./selfie -c any-cstar-file.c -m 1 "arguments for any-cstar-file.c"
Equivalently, you may also use a selfie-compiled version of selfie
and have the emulator execute selfie to compile any C* source and then execute it right away with hypster on the same emulator instance:
$ ./selfie -c selfie.c -m 1 -c any-cstar-file.c -y 1 "arguments for any-cstar-file.c"
You may also generate MIPSter binaries both ways which will then be identical:
$ ./selfie -c any-cstar-file.c -o any-cstar-file1.m
$ ./selfie -c selfie.c -m 1 -c any-cstar-file.c -o any-cstar-file2.m
$ diff -s any-cstar-file1.m any-cstar-file2.m
Files any-cstar-file1.m and any-cstar-file2.m are identical
This can also be done in a single invocation of selfie
:
$ ./selfie -c any-cstar-file.c -o any-cstar-file1.m -c selfie.c -m 1 -c any-cstar-file.c -o any-cstar-file2.m
$ diff -s any-cstar-file1.m any-cstar-file2.m
Files any-cstar-file1.m and any-cstar-file2.m are identical
The generated MIPSter binaries can then be loaded and executed as follows:
$ ./selfie -l any-cstar-file1.m -m 1 "arguments for any-cstar-file1.m"
Console messages always begin with the name of the source or binary file currently running. The emulator also shows the amount of memory allocated for its machine instance and how execution terminated (exit code).
MIPSter assembly for selfie
and any other C* file is generated as follows:
$ ./selfie -c selfie.c -s selfie.s
If the assembly code is generated from a binary generated by the compiler (and not loaded from a file) approximate source line numbers are included in the assembly file.
Verbose debugging information is printed with the -d
option, for example:
$ ./selfie -c selfie.c -d 1
Similarly, if the executed binary is generated by the compiler (and not loaded from a file) approximate source line numbers are included in the debug information.