FRACTRAN


FRACTRAN is a Turing-complete esoteric programming language invented by the mathematician John Conway. A FRACTRAN program is an ordered list of positive fractions together with an initial positive integer input n. The program is run by updating the integer n as follows:
  1. for the first fraction f in the list for which nf is an integer, replace n by nf
  2. repeat this rule until no fraction in the list produces an integer when multiplied by n, then halt.
gives the following formula for primes in FRACTRAN:
Starting with n=2, this FRACTRAN program generates the following sequence of integers:
After 2, this sequence contains the following powers of 2:
which are the prime powers of 2.

Understanding a FRACTRAN program

A FRACTRAN program can be seen as a type of register machine where the registers are stored in prime exponents in the argument n.
Using Gödel numbering, a positive integer n can encode an arbitrary number of arbitrarily large positive integer variables. The value of each variable is encoded as the exponent of a prime number in the prime factorization of the integer. For example, the integer
represents a register state in which one variable holds the value 2 and two other variables hold the value 1. All other variables hold the value 0.
A FRACTRAN program is an ordered list of positive fractions. Each fraction represents an instruction that tests one or more variables, represented by the prime factors of its denominator. For example:
tests v2 and v5. If and, then it subtracts 2 from v2 and 1 from v5 and adds 1 to v3 and 1 to v7. For example:
Since the FRACTRAN program is just a list of fractions, these test-decrement-increment instructions are the only allowed instructions in the FRACTRAN language. In addition the following restrictions apply:

Addition

The simplest FRACTRAN program is a single instruction such as
This program can be represented as a algorithm as follows:
FRACTRAN
Instruction
ConditionAction
v2 > 0Subtract 1 from v2
Add 1 to v3
v2 = 0Stop

Given an initial input of the form, this program will compute the sequence,, etc., until eventually, after steps, no factors of 2 remain and the product with no longer yields an integer; the machine then stops with a final output of. It therefore adds two integers together.

Multiplication

We can create a "multiplier" by "looping" through the "adder". In order to do this we need to introduce states into our algorithm. This algorithm will take a number and produce :
Current StateConditionActionNext State
Av7 > 0Subtract 1 from v7
Add 1 to v3
A
Av7 = 0 and
v2 > 0
Subtract 1 from v2B
Av7 = 0 and
v2 = 0 and
v3 > 0
Subtract 1 from v3A
Av7 = 0 and
v2 = 0 and
v3 = 0
Stop
Bv3 > 0Subtract 1 from v3
Add 1 to v5
Add 1 to v7
B
Bv3 = 0-A

State B is a loop that adds v3 to v5 and also moves v3 to v7, and state A is an outer control loop that repeats the loop in state B v2 times. State A also restores the value of v3 from v7 after the loop in state B has completed.
We can implement states using new variables as state indicators. The state indicators for state B will be v11 and v13. Note that we require two state control indicators for one loop; a primary flag and a secondary flag. Because each indicator is consumed whenever it is tested, we need a secondary indicator to say "continue in the current state"; this secondary indicator is swapped back to the primary indicator in the next instruction, and the loop continues.
Adding FRACTRAN state indicators and instructions to the multiplication algorithm table, we have:
FRACTRAN
Instruction
Current StateState
Indicators
ConditionActionNext State
A-v7 > 0Subtract 1 from v7
Add 1 to v3
A
A-v7 = 0 and
v2 > 0
Subtract 1 from v2B
A-v7 = 0 and
v2 = 0 and
v3 > 0
Subtract 1 from v3A
A-v7 = 0 and
v2 = 0 and
v3 = 0
Stop
Bv11, v13v3 > 0Subtract 1 from v3
Add 1 to v5
Add 1 to v7
B
Bv11, v13v3 = 0-A

When we write out the FRACTRAN instructions, we must put the state A instructions last, because state A has no state indicators - it is the default state if no state indicators are set. So as a FRACTRAN program, the multiplier becomes:
With input 2a3b this program produces output 5ab.

Subtraction and division

In a similar way, we can create a FRACTRAN "subtracter", and repeated subtractions allow us to create a "quotient and remainder" algorithm as follows:
FRACTRAN
Instruction
Current StateState
Indicators
ConditionActionNext State
Av11, v13v2 > 0 and
v3 > 0
Subtract 1 from v2
Subtract 1 from v3
Add 1 to v7
A
Av11, v13v2 = 0 and
v3 > 0
Subtract 1 from v3X
Av11, v13v3 = 0Add 1 to v5B
Bv17, v19v7 > 0Subtract 1 from v7
Add 1 to v3
B
Bv17, v19v7 = 0-A
Xv3 > 0Subtract 1 from v3X
Xv3 = 0Stop

Writing out the FRACTRAN program, we have:
and input 2n3d11 produces output 5q7r where n = qd + r and 0 ≤ r < d.

Conway's prime algorithm

Conway's prime generating algorithm above is essentially a quotient and remainder algorithm within two loops. Given input of the form where 0 ≤ m < n, the algorithm tries to divide n+1 by each number from n down to 1, until it finds the largest number k that is a divisor of n+1. It then returns 2n+1 7k-1 and repeats. The only times that the sequence of state numbers generated by the algorithm produces a power of 2 is when k is 1, which only occurs if the exponent of 2 is a prime. A step-by-step explanation of Conway's algorithm can be found in Havil.

Other examples

The following FRACTRAN program:
calculates the Hamming weight H of the binary expansion of a i.e. the number of 1s in the binary expansion of a. Given input 2a, its output is 13H. The program can be analysed as follows:
FRACTRAN
Instruction
Current StateState
Indicators
ConditionActionNext State
Av5, v11v2 > 1Subtract 2 from v2
Add 1 to v3
A
Av5, v11v2 = 1Subtract 1 from v2
Add 1 to v13
B
Av5, v11v2 = 0-B
B-v3 > 0Subtract 1 from v3
Add 1 to v2
B
B-v3 = 0 and
v7 > 0
Subtract 1 from v7
Add 1 to v2
A
B-v3 = 0 and
v7 = 0 and
v2 > 0
Subtract 1 from v2
add 1 to v7
B
B-v2 = 0 and
v3 = 0 and
v7 = 0
Stop