13
1
Golf a Purple Interpreter
Purple is an esolang which is designed with two main purposes:
- To be a minimization of Aubergine, since there just aren't enough self-modifying one-instruction languages around.
- To admit the possibility of terrifyingly small golfed interpreters. My first pass at a reasonably full-featured Python 2 interpreter is only 702 bytes, and I'm sure a more experienced golfer could shave quite a bit from that.
Your goal is to write an interpreter for this language.
Information on Purple:
A Purple program is a sequence of characters placed into an infinite, addressable memory array such that the first character of the program is placed at address zero. The rest of the array (both before and after where the Purple program is stored) is initialized to zero.
There are three registers in Purple, called a and b and i, each of which can hold a signed integer and is initialized to zero. i is also the instruction pointer, and always points to the currently executing Purple instruction.
Each cycle, the interpreter will read a sequence of three contiguous characters beginning from the memory location indicated by the instruction pointer and attempt to execute this sequence as the Purple instruction. Afterwards, the instruction pointer is always incremented by 3.
Syntactically, the Purple instruction consists of three characters (or encodings thereof) in a row, like "xyz".
The first character x can be any of the following:
abABio
These symbols have the following meaning:
a - Place the result in register a.
b - Place the result in register b.
A - Place the result in the location in memory referred to by register a.
B - Place the result in the location in memory referred to by register b.
i - Set the instruction pointer to the result.
o - Output the result to stdout.
The other two bytes y and z can be any of the following:
abABio1
Each of these symbols has the following meaning:
a - Return the contents of register a.
b - Return the contents of register b.
A - Return the contents of the memory array at the address stored in register a.
B - Return the contents of the memory array at the address stored in register b.
i - Return the contents of register i (the instruction pointer).
o - Return the value of a single character read from stdin.
1 - Return the literal numeric value 1.
After fetching the instruction, the Purple interpreter will evaluate y and then z, subtract the result of z from the result of y, and then perform the action indicated by x on the difference.
If the sequence of three characters (or encodings thereof) is not a valid Purple instruction, the interpreter halts immediately without giving any errors.
Your interpreter must:
- Be a complete program, not a function.
- Never output to stderr, unless EOF is read.
- Behave identically to the reference implementation on all well-formed inputs that do not involve very large numbers, including the test programs given below. (Well, identically up to timing--it can run slower, but not by too much!)
You may provide the program to the interpreter in any form you wish: read it from a file, embed it in the program as a string, or read it from stdin.
Test cases:
The program
ooo
when run with input
z!
should yield
Y
The program
bbboobiii
when run with input
It's a cat program.
(or any other input) should yield
It's a cat program.
(or whatever input it received) and then start over and do the same thing again.
The program
Aoab11bi1bABoAaiba
when run with input
0
should yield
0
and then halt, but when run with input
1
should continue outputting
1
forever.
The program
b1bbb1oAbabaa1ab1Ab1Bi1b
should yield
b1bbb1oAbabaa1ab1Ab1Bi1b
The program
aA1aa1bb1oAbbi1bb1bbAb1Bi1b Purple is the awesomest! Why haven't you tried it yet?
!dlroW ,olleG
should yield
Hello, World!
Scoring:
This is code-golf, so shortest source in bytes, as potentially modified by the following bonus, wins.
Bonus:
- -10% if your interpreter reads a filename from either stdin or from a command line argument and loads the program from the file.
1What is the size of the memory cells? bytes, characters (unicode ones?), (arbitrary) large integers? It looks like you are using "character" and "byte" with the same meaning. – Paŭlo Ebermann – 2015-12-01T20:07:59.057
@PaŭloEbermann my guess is that's implementation-specific; for example I need to use
uint32
for characters and MAXINT for ints – cat – 2015-12-01T20:10:04.313What should be the value of
o
when the input is closed (i.e. EOF)? And what would be read after that? – Paŭlo Ebermann – 2015-12-01T20:42:07.523@PaŭloEbermann do whatever the reference implementation does in this situation. (I'm fairly certain it gives 0 on EOF.) – quintopia – 2015-12-01T22:13:07.003
@quintopia actually, your cat program demonstrates that the reference implementation throws an exception in this case (
TypeError: ord() expected a character, but string of length 0 found
), at least here (Python 2.7.6 on Ubuntu 14.04) – read(1) from a file at EOF gives an empty string, and ord on that is not defined. I guess finishing the program without an exception would work too, as we are not allowed to print to stderr? – Paŭlo Ebermann – 2015-12-01T23:20:36.550@PaŭloEbermann Handle it any way you wish then. I'll allow writing to STDERR when reading EOF. – quintopia – 2015-12-02T00:20:16.597
I put quite a few hours and quite a bit of hope into doing this in Go, only to realise the tape needs to be actually negatively addressable. Welp, there goes that project. – cat – 2015-12-02T02:01:19.017
2@sysreq Is that really a blocker? Your implementation could simply have two tapes, one for negative and one for positive indices. (Yes, this will take a bit more code, but not that much, I think.) – Paŭlo Ebermann – 2015-12-02T16:58:41.927
Having to figure out which of two tapes to assign to and such seems rather long, but I'll try it.
Suppose I write an interpreter/runtime that can read from a file: if I make the runtime basically allow Purple to bootstrap and interpret itself (unless someone would like to provide a kernel that ships with Purple instructions builtin), does that actually count as Purple interpreting Purple? – cat – 2015-12-03T12:40:57.367
because there needs to be some sort of chicken-and-egg-bootstrap-resolver, but if that's not written in Purple then does it qualify? – cat – 2015-12-03T12:41:58.680
1@sysreq basically, a Purple self-interpreter would be a program that reads a Purple program from stdin and then does whatever that program would do. The first Purple program (the interpreter) can run in whatever interpreter you like. A program that completely overwrites the lowest memory addresses with the input, then deletes itself before somehow jumping to the read code would qualify (though I don't think this is actually possible). – quintopia – 2015-12-03T19:10:25.107
2I came so close to having a runtime capable of self-interpretation, but I was too late. – cat – 2015-12-10T12:44:51.693
Post it anyway if you get it. I'll throw a love your way. – quintopia – 2015-12-10T14:41:38.163