Tock
Tock is an esolang/model of computation created by User:Silver. The name is because it puts a lot of focus on working in discrete steps, and Tick was taken.
Overview
There is a collection of functions taking an integer t
to a real. These functions can depend on the values that the other functions take for t-1
, but not for any other input, so f(t)
can be defined in terms of g(t-1)
but not g(t)
, g(t-2)
, g(t+1)
or g(3)
. Two functions, called halt
and out
, are special: the output of the program is the value of out(t)
for the smallest t
such that halt(t)
is zero.
Language
A program is a collection of functions, which must include functions called halt
and out
. Each function is defined as follows:
[function name] [initial value] { [expression] }
or:
[function name] [initial value] { [condition] : [expression]; [condition] : [expression]; otherwise : [expression] }
A function name can consist of alphabetical characters and underscores only. The initial value is a number.
A condition consists of two expressions related by one of =
, !=
, >
, <
, >=
or <=
. Clauses of a piecewise definition are checked sequentially, the otherwise clause is optional (but the program will throw an error if no clause matches).
An expression is an arithmetical expression using +
, -
, *
, /
, ^
, %
and ()
. It can contain both numbers and the names of other functions. Those names are interpreted as the values of those functions at t-1
.
Line comments are permitted using double slashes.
Examples
The following is a simple program that outputs 10:
halt 10 { halt - 1 } out 0 { out + 1 }
While the following outputs 20:
halt 1 { counter = 10: 0; otherwise: 1 } out 0 { counter * 2 } counter 0 { counter + 1 }
See fractran.tock in the implementation repository linked below for a more involved example.
Computational Class
Tock is Turing Complete, as shown by the existence of the following translation of Fractran:
halt 1 { pointer > [number of fractions]: 0; otherwise: 1; } out 0 { n } mode 0 { (mode + 1) % 4; // modes are: // 0 - load pointed at fraction // 1 - compare fraction to current n // 2 - update n and pointer accordingly // 3 - check if finished } pointer 0 { mode + (10 * current_match) = 12: 0; // return to start mode + (10 * current_match) = 2: pointer + 1; // next fraction otherwise: pointer; } current_fract_numerator 1 { pointer + ([number larger than number of fractions] * mode) = 0: [numerator of first fraction]; pointer + ([number larger than number of fractions] * mode) = 1: [numerator of second fraction]; ... otherwise: current_fract_numerator; } current_fract_denominator 1 { pointer + ([number larger than number of fractions] * mode) = 0: [denominator of first fraction]; pointer + ([number larger than number of fractions] * mode) = 1: [denominator of second fraction]; ... otherwise: current_fract_denominator; } current_match 0 { (n * current_fract_numerator) % current_fract_denominator = 0: 1; otherwise: 0; } n [input to the fractran program] { current_match + ([number larger than the number of fractions] * mode) = ([same large number] * 2) + 1: n * (current_fract_numerator / current_fract_denominator); otherwise: n; }
See fractran.tock in the implementation repository linked below for an example of using this to translate a fractran program.
Implementation
An implementation exists at [1]