Thread extraction for polyadic instruction sequences by Bergstra, J. A. & Middelburg, C. A.
ar
X
iv
:0
80
2.
15
78
v3
  [
cs
.PL
]  
28
 Ju
l 2
00
9
Thread Extraction for
Polyadic Instruction Sequences⋆
J.A. Bergstra and C.A. Middelburg
Informatics Institute, Faculty of Science, University of Amsterdam,
Science Park 107, 1098 XG Amsterdam, the Netherlands
J.A.Bergstra@uva.nl,C.A.Middelburg@uva.nl
Abstract. In this paper, we study the phenomenon that instruction
sequences are split into fragments which somehow produce a joint be-
haviour. In order to bring this phenomenon better into the picture, we
formalize a simple mechanism by which several instruction sequence frag-
ments can produce a joint behaviour. We also show that, even in the case
of this simple mechanism, it is a non-trivial matter to explain by means
of a translation into a single instruction sequence what takes place on
execution of a collection of instruction sequence fragments.
Keywords: instruction sequence fragment; polyadic instruction sequence;
thread extraction; basic thread algebra; program algebra
MSC2000 codes: 68N15; 68Q05; 68Q55
1 Introduction
With the work presented in this paper, we carry on a line of research with
which a start was made in [4]. This line of research is concerned with sequential
programs that take the form of instruction sequences. Its working hypothesis is
that instruction sequence is a central notion of computer science, which merits
investigation for its own sake.
An instruction sequence is considered to produce on execution a behaviour
to be controlled by some execution environment. This behaviour proceeds by
performing steps in a sequential fashion. Each step performed actuates the pro-
cessing of an instruction by the execution environment in question. A reply
returned by this execution environment at completion of the processing of the
instruction determines how the behaviour proceeds further.
The following phenomenon presents itself: instruction sequences are split into
fragments which somehow produce a joint behaviour. The objective of this paper
is to bring this phenomenon better into the picture. To achieve this, we formalize
a simple mechanism by which several instruction sequence fragments can produce
a joint behaviour. We show that, even in the case of this simple mechanism, it is
⋆ This research has been partly carried out in the framework of the Jacquard-project
Symbiosis, which is funded by the Netherlands Organisation for Scientific Research
(NWO).
a non-trivial matter to explain by means of a translation into a single instruction
sequence what takes place on execution of a collection of instruction sequence
fragments.
The question is how a joint behaviour of the fragments in a collection of
fragments is achieved. The view of this matter is that there can only be a single
fragment being executed at any stage, but the fragment in question may make
any fragment in the collection the one being executed by means of a special
instruction for switching over execution to another fragment. This does not fit
in very well with the conception that the collection of fragments constitutes a
sequential program. To our knowledge, a theoretical understanding of this matter
has not yet been developed. This has motivated us to take up this topic.
The principal reason for splitting instruction sequences into fragments is
that the execution environment at hand sets bounds to the size of instruction
sequences. In the past, the phenomenon occurred explicitly in many software
systems. At present, it often occurs rather implicitly, e.g. on execution of pro-
grams written in contemporary object-oriented programming languages, such as
Java [1] and C# [11], classes are loaded as they are needed. The mechanisms in
question are improvements upon the simple mechanism considered in this paper,
but they are also much more complicated. We believe that it is useful to consider
the simple mechanism prior to the more complicated ones.
The instruction sequences taken for fragments are called polyadic instruc-
tion sequences in this paper. We introduce polyadic instruction sequences in the
setting of program algebra [4]. The starting-point of program algebra is the per-
ception of a program as a single-pass instruction sequence, i.e. a finite or infinite
sequence of instructions of which each instruction is executed at most once and
can be dropped after it has been executed or jumped over. This perception is
simple, appealing, and links up with practice.
The behaviours produced by instruction sequences on execution are modelled
by threads as considered in basic thread algebra [4].1 We take the view that
the possible joint behaviours produced by polyadic instruction sequences on
execution are threads as considered in basic thread algebra as well. In a system
that provides an execution environment for polyadic instruction sequences, a
polyadic instruction sequence must be loaded in order to become the one being
executed. Hence, making a polyadic instruction sequence the one being executed
can be looked upon as loading it for execution.
In [4], a hierarchy of program notations rooted in program algebra is pre-
sented. Included in this hierarchy are very simple program notations which are
close to existing assembly languages up to and including simple program nota-
tions that support structured programming by offering a rendering of conditional
and loop constructs. All of these program notations are referred to in this pa-
per, but only one of them is actually used. That program notation is introduced
under the name PGLD in [4].
1 In [4], basic thread algebra is introduced under the name basic polarized process
algebra.
2
This paper is organized as follows. First, we review basic thread algebra
and program algebra (Sections 2 and 3). After that, we give an overall picture
of the hierarchy of program notations rooted in program algebra and present
the program notation PGLD (Sections 4 and 5). Next, we introduce polyadic
instruction sequences in the setting of program algebra, explain the possible
joint behaviours of a collection of polyadic instruction sequences using basic
thread algebra, and give an example of the use of polyadic instruction sequences
(Sections 6 and 7). Following this, we extend basic thread algebra to allow for
threads to make use of services and give a description of instruction register file
services (Sections 8 and 9). After that, we show that, for each possible joint
behaviour of a collection of polyadic instruction sequences, a single instruction
sequence can be synthesized from the collection of polyadic instruction sequences
that produces on execution essentially the behaviour in question by making use of
an instruction register file service (Section 10). Finally, we make some concluding
remarks (Section 11).
In this paper, we only give brief summaries of program algebra and basic
thread algebra. Comprehensive introductions, including examples, can be found
in [4, 14].
2 Basic Thread Algebra
In this section, we review BTA (Basic Thread Algebra). BTA is a form of process
algebra which is concerned with the behaviour that sequential programs produce
on execution. Those behaviours are called threads.
In BTA, it is assumed that fixed but arbitrary finite sets A and I with
A ∩ I = ∅ and tau ∈ I have been given. The members of A are called basic
actions and the members of I are called internal actions. The members of A∪I
are referred to as actions. In previous work, we take in essence the singleton
set {tau} for I. The generalization made here permits internal actions with
differences relevant for analysis to be distinguished.
The operational intuition is that a thread has an execution environment
which processes each action performed by the thread. A thread performs actions
in a sequential fashion. Upon each action performed, a reply from the execution
environment of the thread determines how it proceeds. The possible replies are
T and F. Performing an internal action, always leads to the reply T.
Although BTA is one-sorted, we make this sort explicit. The reason for this
is that we will extend BTA with an additional sort in Section 8.
BTA has one sort: the sort T of threads. To build terms of sort T, BTA has
the following constants and operators:
– the inaction constant D :T;
– the termination constant S :T;
– for each a ∈ A∪I, the binary postconditional composition operator EaD :
T×T→ T.
3
Table 1. Axiom of BTA
x E ιD y = x E ιD x T1
Table 2. Axioms for guarded recursion
〈X|E〉 = 〈tX |E〉 if X= tX ∈ E RDP
E ⇒ X = 〈X|E〉 if X ∈ V(E) RSP
We assume that there are infinitely many variables of sort T, including x, y, z.
Terms of sort T are built as usual (see e.g. [15, 16]). We use infix notation for
the postconditional composition operator. We introduce basic action prefixing
as an abbreviation: a ◦ p abbreviates pEaD p.
The thread denoted by a closed term of the form pEaD q will first perform
a, and then proceed as the thread denoted by p if the reply from the execution
environment is T and proceed as the thread denoted by q if the reply from
the execution environment is F. The threads denoted by D and S will become
inactive and terminate successfully, respectively. A thread is inactive if it is
neither capable of performing any action nor capable of terminating successfully.
BTA has only one axiom. This axiom is given in Table 1. In this table, ι
stands for an arbitrary member of I.
Notice that each closed BTA term denotes a thread that will become inactive
or terminate after it has performed finitely many actions. Infinite threads can
be described by guarded recursion.
A guarded recursive specification over BTA is a set of recursion equations
E = {X = tX | X ∈ V }, where V is a set of variables of sort T and each tX is a
BTA term of the form D, S or tEaD t′ with t and t′ that contain only variables
from V . We write V(E) for the set of all variables that occur in E. We are
only interested in models of BTA in which guarded recursive specifications have
unique solutions, such as the projective limit model of BTA presented in [2]. A
thread that is the solution of a finite guarded recursive specification over BTA
is called a finite-state thread.
For each guarded recursive specification E and each X ∈ V(E), we introduce
a constant 〈X |E〉 of sort T standing for the unique solution of E for X . The
axioms for these constants are given in Table 2. In this table, we write 〈tX |E〉
for tX with, for all Y ∈ V(E), all occurrences of Y in tX replaced by 〈Y |E〉. X ,
tX and E stand for an arbitrary variable of sort T, an arbitrary BTA term of
sort T and an arbitrary guarded recursive specification over BTA, respectively.
Side conditions are added to restrict what X , tX and E stand for.
We will use the following abbreviation: aω, where a ∈ A ∪ I, abbreviates
〈X |{X = a ◦X}〉.
We will write BTA+REC for BTA extended with the constants for solutions
of guarded recursive specifications and axioms RDP and RSP.
Closed terms of sort T from the language of BTA+REC that denote the
same infinite thread cannot always be proved equal by means of the axioms of
4
Table 3. Approximation induction principle
V
n≥0 pin(x) = pin(y) ⇒ x = y AIP
pi0(x) = D P0
pin+1(S) = S P1
pin+1(D) = D P2
pin+1(xE aD y) = pin(x)E aD pin(y) P3
BTA+REC. We introduce AIP (Approximation Induction Principle) to remedy
this. AIP is based on the view that two threads are identical if their approxima-
tions up to any finite depth are identical. The approximation up to depth n of
a thread is obtained by cutting it off after performing a sequence of actions of
length n. In AIP, the approximation up to depth n is phrased in terms of the
unary projection operator pin : T → T. AIP and the axioms for the projection
operators are given in Table 3. In this table, a stands for an arbitrary member
of A∪ I.
We will write BTA+REC+AIP for BTA+REC extended with the projection
operators and the axioms from Table 3.
3 Program Algebra
In this section, we review PGA (ProGram Algebra). The perception of a program
as a single-pass instruction sequence is the starting-point of PGA.
In PGA, it is assumed that a fixed but arbitrary set A of basic instructions
has been given. PGA has the following primitive instructions :
– for each a ∈ A, a plain basic instruction a;
– for each a ∈ A, a positive test instruction +a;
– for each a ∈ A, a negative test instruction −a;
– for each l ∈ N, a forward jump instruction #l;
– a termination instruction !.
We write I for the set of all primitive instructions.
The intuition is that the execution of a basic instruction a produces either T
or F at its completion. In the case of a positive test instruction +a, a is executed
and execution proceeds with the next primitive instruction if T is produced.
Otherwise, the next primitive instruction is skipped and execution proceeds with
the primitive instruction following the skipped one. If there is no next instruction
to be executed, inaction occurs. In the case of a negative test instruction −a,
the role of the value produced is reversed. In the case of a plain basic instruction
a, execution always proceeds as if T is produced. The effect of a forward jump
instruction #l is that execution proceeds with the l-th next instruction. If l
equals 0 or the l-th next instruction does not exist, then #l results in inaction.
The effect of the termination instruction ! is that execution terminates.
PGA has the following constants and operators:
5
Table 4. Axioms of PGA
(x ; y) ; z = x ; (y ; z) PGA1
(xn)ω = xω PGA2
xω ; y = xω PGA3
(x ; y)ω = x ; (y ; x)ω PGA4
Table 5. Defining equations for thread extraction operation of PGA
|a| = a ◦ D
|a ; x| = a ◦ |x|
|+a| = a ◦ D
|+a ; x| = |x|E aD |#2 ; x|
|−a| = a ◦ D
|−a ; x| = |#2 ; x|E aD |x|
|#l| = D
|#0 ; x| = D
|#1 ; x| = |x|
|#l + 2 ; u| = D
|#l + 2 ; u ; x| = |#l + 1 ; x|
|!| = S
|! ; x| = S
– for each u ∈ I, an instruction constant u ;
– the binary concatenation operator ; ;
– the unary repetition operator ω .
We assume that there are infinitely many variables, including x, y, z. Terms are
built as usual. We use infix notation for the concatenation operator and postfix
notation for the repetition operator.
A closed PGA term is considered to denote a non-empty, finite or periodic
infinite sequence of primitive instructions.2 Closed PGA terms are considered
equal if they denote the same instruction sequence. The axioms for instruction
sequence equivalence are given in Table 4. In this table, n stands for an arbitrary
natural number greater than 0. For each PGA term P , the term Pn is defined by
induction on n as follows: P 1 = P and Pn+1 = P ;Pn. The equationXω = X ;Xω
is derivable. Each closed PGA term is derivably equal to one of the form P or
P ;Qω, where P and Q are closed PGA terms in which the repetition operator
does not occur.
Each closed PGA term P is considered to denote an instruction sequence
of which the behaviour is a finite-state thread, called the thread produced by
P . The set A of basic instructions is taken for the set A of basic actions. The
thread extraction operation | | determines, for each closed PGA term P , a finite
guarded recursive specification over BTA that defines the thread produced by
P . The thread extraction operation is defined by the equations given in Table 5
(for a ∈ A, l ∈ N and u ∈ I) and the rule that |#l ;x| = D if #l is the beginning
of an infinite chain of forward jumps. This rule is formalized in e.g. [8].
2 An infinite sequence is periodic if the set of all its subsequences is finite.
6
The behaviour of each closed PGA term, is a thread that is definable by a
finite guarded recursive specification over BTA. The other way round, each finite
guarded recursive specification over BTA in which no internal actions occur can
be translated into a closed PGA term of which the behaviour is the solution of
the finite guarded recursive specification concerned.
Closed PGA terms are considered to denote programs and therefore they
constitute an elementary program notation. Closed PGA terms are also called
PGA programs.
4 A Hierarchy of Program Notations Rooted in Program
Algebra
In [4], a hierarchy of program notations rooted in PGA is presented. The program
notations that appear in this hierarchy are PGA, PGLA, PGLB, PGLC, PGLD,
PGLDg, PGLE, and PGLS. The most interesting ones are PGLC, PGLD and
PGLS. PGLC and PGLD are close to existing assembly languages. The main
difference between them is that PGLC has relative jump instructions and PGLD
has absolute jump instructions. PGLS supports structured programming by of-
fering a rendering of conditional and loop constructs instead of (unstructured)
jump instructions.
For each of the program notations that appear in the hierarchy, except PGA,
a function is given in [4] by means of which each program from that program
notation is translated into a program from the first program notation lower in
the hierarchy that produces the same behaviour on execution. These functions
are called projections. Moreover, for each of the program notations that appear
in the hierarchy, except PGLE and PGLS, a function is given in [4] by means
of which each program from that program notation is translated into a program
from the first program notation higher in the hierarchy that produces the same
behaviour on execution. These functions are called embeddings. There does not
exist a function by which each PGLE program is translated into a PGLS program
that produces the same behaviour on execution because PGLS is strictly weaker
than PGLE. The names of the projections and embeddings referred to above are
given in Figure 1.
The program notations, projections, and embeddings referred to above are
defined in [4]. We refrain from giving all the definitions in question in this paper
as well, because most details of the program notations, projections, and embed-
dings do not matter here. The important point is that there exist a collection of
well-defined program notations rooted in an elementary program notation with
projections and embeddings between them. Only PGLD is actually used later
on in this paper. Therefore, PGLD is reviewed below in Section 5.
In Section 6, we will take special versions of the above-mentioned program no-
tations for the ones that may be used for fragments. Moreover, we will make use
of the projections pgla2pga and in addition the projections pglb2pga, pglc2pga,
. . . defined in the obvious way by composition of projections given in [4]. In Sec-
tion 10, we will make use of the embedding pglc2pgld and in addition the
7
PGLS
pgls2pgle ↓
PGLE
pgle2pgldg ↓ ↑ pgldg2pgle
PGLDg
pgldg2pgld ↓ ↑ pgld2pgldg
PGLD
pgld2pglc ↓ ↑ pglc2pgld
PGLC
pglc2pglb ↓ ↑ pglb2pglc
PGLB
pglb2pgla ↓ ↑ pgla2pglb
PGLA
pgla2pga ↓ ↑ pga2pgla
PGA
Fig. 1. Hierarchy of program notations rooted in PGA
embedding pga2pglc defined in the obvious way by composition of embeddings
given in [4].
5 The Program Notation PGLD
In this section, we review the program notation PGLD. This program notation
is reviewed because it will be used later on in Sections 7 and 10.
In PGLD, like in PGA, it is assumed that there is a fixed but arbitrary finite
set A of basic instructions. Again, the intuition is that the execution of a basic
instruction a produces either T or F at its completion.
PGLD has the following primitive instructions:
– for each a ∈ A, a plain basic instruction a;
– for each a ∈ A, a positive test instruction +a;
– for each a ∈ A, a negative test instruction −a;
– for each l ∈ N, an absolute jump instruction ##l.
PGLD programs have the form u1; . . . ;uk, where u1, . . . , uk are primitive in-
structions of PGLD.
The plain basic instructions, the positive test instructions, and the negative
test instructions are as in PGA. The effect of an absolute jump instruction ##l
is that execution proceeds with the l-th instruction of the program concerned.
If ##l is itself the l-th instruction, then ##l results in inaction. If l equals 0 or
l is greater than the length of the program, then termination occurs.
The function pgld2pga from the set of all PGLD programs to the set of all
PGA programs, which translates each PGLD program into a PGA program that
produces the same behaviour on execution, can be defined directly as follows:
pgld2pga(u1 ; . . . ; uk) = (ψ1(u1) ; . . . ; ψk(uk) ; ! ; !)
ω ,
8
where the auxiliary functions ψj from the set of all primitive instructions of
PGLD to the set of all primitive instructions of PGA are defined as follows
(1 ≤ j ≤ k):
ψj(##l) = #l − j if j ≤ l ≤ k ,
ψj(##l) = #k + 2− (j − l) if 0 < l < j ,
ψj(##l) = ! if l = 0 ∨ l > k ,
ψj(u) = u if u is not a jump instruction .
The idea is that each backward jump can be replaced by a forward jump if
the entire program is repeated. To enforce termination of the program after
execution of its last instruction if the last instruction is a plain basic instruction,
a positive test instruction or a negative test instruction, ! ; ! is appended to
ψ1(u1) ; . . . ; ψk(uk).
6 Polyadic Instruction Sequences
In this section, we formalize a simple mechanism by which several instruction
sequence fragments can produce a joint behaviour. The instruction sequence
fragments are viewed as instruction sequences that contain special instructions
for switching over execution from one fragment to another. The instruction se-
quences in question are called polyadic instruction sequences.
It is assumed that a special version of PGLA, PGLB, PGLC, PGLD, PGLDg,
PGLE or PGLS is used for each polyadic instruction sequence. Moreover, it is
assumed that a collection of polyadic instruction sequences between which exe-
cution can be switched takes the form of a sequence, called a polyadic instruction
sequence vector, in which each polyadic instruction sequence is coupled with the
program notation used for it.
Our general view on the way of achieving a joint behaviour of the polyadic
instruction sequences in a polyadic instruction sequence vector is as follows:
– there can only be a single polyadic instruction sequence being executed at
any stage;
– the polyadic instruction sequence in question may make any polyadic in-
struction sequence in the vector the one being executed;
– making another polyadic instruction sequence the one being executed is ef-
fected by executing a special instruction for switching over execution;
– any polyadic instruction sequence can be taken for the one being executed
initially.
In addition to special instructions for switching over execution, polyadic in-
struction sequences may contain two other kinds of special instructions:
– special instructions for putting instructions into instruction registers;
– special instructions of which the occurrences in a polyadic instruction se-
quence are replaced by instructions contained in instruction registers on
making the polyadic instruction sequence the one being executed.
9
The special instructions of the latter kind serve as instruction place-holders.
Their presence turns a polyadic instruction sequence into a parameterized in-
struction sequence of which the parameters are filled in each time it is made the
one being executed. This feature accounts for the use of the prefix polyadic. Its
merit is primarily that it allows for execution to proceed in effect from different
positions each time a polyadic instruction sequence is loaded for execution. An
example of this is given in Section 7.
We take the line that different program notations can be used for different
polyadic instruction sequences in a polyadic instruction sequence vector. On
making a polyadic instruction sequence in the vector the one being executed, it
is considered to be translated into a PGAp program.
PGAp is a variant of PGA in which the above-mentioned special instructions
are incorporated. In PGAp, it is assumed that there is a fixed but arbitrary finite
set Ac of core basic instructions. In PGAp, a basic instruction is either a core
basic instruction or a supplementary basic instruction.
PGAp has the following core primitive instructions :
– for each a ∈ Ac, a plain basic instruction a;
– for each a ∈ Ac, a positive test instruction +a;
– for each a ∈ Ac, a negative test instruction −a;
– for each l ∈ N, a forward jump instruction #l;
– a termination instruction !.
We write Ic for the set of all core primitive instructions. The core primitive
instructions of PGAp are the counterparts of the primitive instructions of PGA.
PGAp has the following supplementary basic instructions :
– for each i ∈ N, a switch-over instruction ###i;
– for each i ∈ N and u ∈ Ic, a put instruction $put:i:u;
– for each i ∈ N, a get instruction $get:i.
We write As for the set of all supplementary basic instructions. In the presence
of a polyadic instruction sequence vector, a switch-over instruction ###i is the
instruction for switching over execution to the i-th polyadic instruction sequence
in the vector. A put instruction $put:i:u is the instruction for putting instruction
u in the instruction register with number i. A get instruction $get:i is the in-
struction of which each occurrence in a polyadic instruction sequence is replaced
by the contents of the instruction register with number i on switching over exe-
cution to that polyadic instruction sequence. If a get instruction is encountered
in the polyadic instruction sequence being executed, inaction occurs.
The supplementary basic instructions of PGAp can be viewed as built-in ba-
sic instructions. However, as laid down below, supplementary basic instructions
do not occur in positive or negative test instructions. Thus, the core primi-
tive instructions and supplementary basic instructions make up the primitive
instructions of PGAp.
PGAp has the following constants and operators:
– for each u ∈ Ic ∪ As, an instruction constant u ;
10
– the binary concatenation operator ; ;
– the unary repetition operator ω .
The axioms of PGAp are the same as the axioms of PGA.
Suppose that in PGA the restriction is dropped that A must be a finite set.
Then PGAp can be viewed as the specialization of PGA obtained by taking the
set Ac∪As for A and excluding terms in which basic instructions from As occur in
positive or negative test instructions. Henceforth, we actually drop the restriction
that Amust be a finite set. This simplifies the definitions of the different program
notations that can be used for polyadic instruction sequences and also enables
the use of the functions pgla2pga, pglb2pga, etcetera for translating programs
in those program notations into PGAp programs.
The different program notations that can be used for polyadic instruction
sequences are PGLAp, PGLBp, PGLCp, PGLDp, PGLDgp, PGLEp, and PGLSp.
The set of all PGLAp programs is the subset of the set of all PGLA programs,
taking the set Ac∪As for A, in which the basic instructions from As do not occur
in positive or negative test instructions. The other program notations are defined
similarly. If the set Ac∪As is taken for A, the function pgla2pga translates each
PGLAp program into a PGAp program that produces the same behaviour on
execution. Similar remarks apply to the other program notations.
A polyadic instruction sequence is either a PGLAp program, a PGLBp pro-
gram, a PGLCp program, a PGLDp program, a PGLDgp program, a PGLEp
program or a PGLSp program.
A polyadic instruction sequence vector is a sequence of pairs consisting of a
polyadic instruction sequence and a member of the set {A,B,C,D,Dg,E, S} of
program notation indices.
Let α be a polyadic instruction sequence vector, let P1, . . . , Pn and c1, . . . , cn
be polyadic instruction sequences and program notation indices, respectively,
such that α = 〈(P1, c1)〉 y . . . y 〈(Pn, cn)〉,3 and let i ∈ [1, n]. Then we write
pg(α, i) and pgn(α, i) for Pi and ci, respectively.
Let α be a polyadic instruction sequence vector of length n, and let i ∈ [1, n].
Then program notation index pgn(α, i) indicates which program notation is used
for polyadic instruction sequence pg(α, i). A stands for PGLAp, B stands for
PGLBp, etcetera. The program notation used is made explicit because it can-
not always be determined uniquely from the polyadic instruction sequence con-
cerned, whereas the behaviour that this polyadic instruction sequence produces
on execution may be different for each of the program notations in question.
Below, we use special notation relating to the program notation indices. Let
c be a program notation index. Then we write prj c for the projection pgla2pga
if c = A, the projection pglb2pga if c = B, etcetera.
The set of instruction registers that contain an instruction and the contents
of each of those registers matter when a polyadic instruction sequence is made
3 We write D∗ for the set of all finite sequences with elements from set D. We use
the following notation for finite sequences: 〈 〉 for the empty sequence, 〈d〉 for the
sequence having d as sole element, σ y σ′ for the concatenation of finite sequences
σ and σ′, and len(σ) for the length of finite sequence σ.
11
the one being executed. That makes us introduce the notion of an instruction
register file state and special notation relating to this notion.
An instruction register file state is a function σ : I → Ic, where I is a finite
subset of N.
Let p be a PGAp program and σ be an instruction register file state. Then
we write p[σ] for p with, for all i ∈ dom(σ), all occurrences of $get:i in p replaced
by σ(i).
Let i, n ∈ N be such that 1 ≤ i ≤ n, let α be a polyadic instruction sequence
vector of length n, and let σ be an instruction register file state. Then we write
valid(α, i, σ) to indicate that instructions of the form $get:i do not occur in
prj
pgn(α,i)(pg(α, i))[σ].
An obvious choice of the thread extraction operation of PGAp is the thread
extraction operation of PGA, taking the set Ac ∪ As for A, restricted to the
set of closed terms of PGAp. This thread extraction operation is considered not
to be the proper one, because it treats the supplementary basic instructions
as arbitrary basic instructions and thus disregards the fixed effects that they
produce on execution. Moreover, this thread extraction operation requires that
in BTA the restriction is dropped that A must be a finite set.
As regards the proper thread extraction for PGAp, the idea is that it yields,
for each PGAp program P , a function that determines, for each polyadic instruc-
tion sequence vector α, a finite guarded recursive specification over BTA that
defines the thread that is the joint behaviour of P and the polyadic instruction
sequences in α in the case where P is the polyadic instruction sequence being
executed initially. Because this behaviour depends upon the set of instruction
registers that contain an instruction and the contents of each of those registers,
we need a thread extraction operation for each instruction register file state.
In the thread extraction for PGAp, it is assumed that gl ∈ I. The internal
action gl (generate and load) represents the internal activity involved in switching
over execution. The internal actions tau and gl are distinguished because tau is
considered to represent a negligible internal activity, whereas gl is considered to
represent a substantial internal activity.
For each instruction register file state σ, we introduce the thread extraction
operation | |σ. These thread extraction operations are defined by the equations
given in Table 6 (for a ∈ A, l, i ∈ N, u ∈ Ic ∪ As and v ∈ Ic) and the rule that
|#l ;X |σ(α) = D if #l is the beginning of an infinite chain of forward jumps.
We can couple nominal indices as labels with some of the polyadic instruction
sequences in a polyadic instruction sequence vector. This would permit the use
of alternative switch-over instructions with nominal indices instead of ordinal
indices, like with the goto instructions from PGLDg. In the notational style
of [3], the form of those alternative switch-over instructions would be ###[i].
7 Example
To illustrate the mechanism formalized in Section 6, we consider in this section
the splitting of a PGLD program P of 10000 instructions into two fragments.
12
Table 6. Defining equations for thread extraction operations of PGAp
|a|σ(α) = a ◦ D
|a ; x|σ(α) = a ◦ |x|σ(α)
|+a|σ(α) = a ◦ D
|+a ; x|σ(α) = |x|σ(α)E aD |#2 ; x|σ(α)
|−a|σ(α) = a ◦ D
|−a ; x|σ(α) = |#2 ; x|σ(α)E aD |x|σ(α)
|#l|σ(α) = D
|#0 ; x|σ(α) = D
|#1 ; x|σ(α) = |x|σ(α)
|#l + 2 ; u|σ(α) = D
|#l + 2 ; u ; x|σ(α) = |#l + 1 ; x|σ(α)
|!|σ(α) = S
|! ; x|σ(α) = S
|###i|σ(α) = gl ◦ |prj pgn(α,i)(pg(α, i))[σ]|σ(α) if 1 ≤ i ≤ n ∧ valid(α, i, σ)
|###i|σ(α) = D if 1 ≤ i ≤ n ∧ ¬valid(α, i, σ)
|###i|σ(α) = S if i = 0 ∨ i > n
|###i ; x|σ(α) = gl ◦ |prj pgn(α,i)(pg(α, i))[σ]|σ(α) if 1 ≤ i ≤ n ∧ valid(α, i, σ)
|###i ; x|σ(α) = D if 1 ≤ i ≤ n ∧ ¬valid(α, i, σ)
|###i ; x|σ(α) = S if i = 0 ∨ i > n
|$put:i:v|σ(α) = tau ◦ D
|$put:i:v ; x|σ(α) = tau ◦ |x|σ⊕[i7→v](α)
|$get:i|σ(α) = D
|$get:i ; x|σ(α) = D
We write ν1(l) for the number of absolute jump instructions ##l
′ with l′ >
5000 from position 1 up to position l and ν2(l) for the number of absolute jump
instructions ##l′ with l′ ≤ 5000 from position 5001 up to position l.
The polyadic instruction sequence P ′ corresponding to the first half of P is
obtained from the first half of P as follows:
– the instruction $get:1 is prefixed to it;
– each absolute jump instruction ##l with l ≤ 5000 is replaced by the absolute
jump instructions ##l′, where l′ = l + ν1(l) + 1;
– each absolute jump instruction ##l with l > 5000 is replaced by the in-
struction sequence $put:2:#l′ ; ###2, where l′ = (l − 5000) + ν2(l − 5000);
and the polyadic instruction sequence P ′′ corresponding to the second half of P
is obtained from the second half of P as follows:
13
– the instruction $get:2 is prefixed to it;
– each absolute jump instruction ##l with l > 5000 is replaced by the absolute
jump instructions ##l′, where l′ = (l − 5000) + ν2(l − 5000) + 1;
– each absolute jump instruction ##l with l ≤ 5000 is replaced by the in-
struction sequence $put:1:#l′ ; ###1, where l′ = l+ ν1(l).
Notice that the positions occurring in jump instructions are adapted to the
prefixing of a get instruction to each half of P and the replacement of each jump
instructions that gives rise to a jump into the other half of P by two instructions.
For any instruction register file state σ, we have that |P | coincides with
|$put:1:#1 ; ###1|σ(〈(P ′, D)〉 y 〈(P ′′, D)〉) after the presence of the internal
actions tau and gl in the latter behaviour has been concealed. In Section 8, we
will introduce operators to conceal the presence of internal actions.
In this section, we have illustrated by means of an example that splitting
a program into fragments is relatively simple. In Section 10, we will show that
synthesizing a program from a collection of fragments is fairly complicated.
8 Threads-Services Interaction and Abstraction
A thread may make use of services. That is, it may perform certain actions for
the purpose of having itself affected by a service that takes them as commands
to be processed. The processing of an action may involve a change of state of the
service and at completion of the processing of the action the service returns a
reply value to the thread. The reply value determines how the thread proceeds.
In this section, we review the use operators, which are concerned with threads
making such use of services.4 We also introduce abstraction operators. They
serve for concealment of the presence of internal actions, which arise among other
things from the use operators. Both use operators and abstraction operators will
be used later on in Section 10.
It is assumed that a fixed but arbitrary finite set F of foci and a fixed but
arbitrary finite setM of methods have been given. Each focus plays the role of a
name of a service provided by the execution environment that can be requested
to process a command. Each method plays the role of a command proper. For
the set A of basic actions, we take the set {f.m | f ∈ F ,m ∈M}. Performing a
basic action f.m is taken as making a request to the service named f to process
command m.
A service H consists of
– a set S of states ;
– an effect function eff :M× S → S;
– a yield function yld :M× S → {T,F,B};
– an initial state s0 ∈ S;
4 This version of the use mechanism was first introduced in [7]. In later papers, it is
also called thread-service composition.
14
Table 7. Axioms for use operators
S /f H = S TSU1
D /f H = D TSU2
(ι ◦ x) /f H = ι ◦ (x /f H) TSU3
(xE g.mD y) /f H = (x /f H)E g.mD (y /f H) if f 6= g TSU4
(xE f.mD y) /f H = tau ◦ (x /f
∂
∂m
H) if H(m) = T TSU5
(xE f.mD y) /f H = tau ◦ (y /f
∂
∂m
H) if H(m) = F TSU6
(xE f.mD y) /f H = D if H(m) = B TSU7
satisfying the following condition:
∀m ∈M, s ∈ S • (yld(m, s) = B ⇒ ∀m′ ∈M • yld(m′, eff (m, s)) = B) .
The set S contains the states in which the service may be, and the functions eff
and yld give, for each method m and state s, the state and reply, respectively,
that result from processing m in state s. By the condition imposed on services,
once the service has returned B as reply, it keeps returning B as reply.
Let H = (S, eff , yld , s0) be a service and letm ∈M. Then the derived service
of H after processingm, written ∂
∂m
H , is the service (S, eff , yld , eff (m, s0)); and
the reply of H after processing m, written H(m), is yld(m, s0).
When a thread makes a request to the service to process m:
– if H(m) 6= B, then the request is accepted, the reply is H(m), and the service
proceeds as ∂
∂m
H ;
– if H(m) = B, then the request is rejected and the service proceeds as a
service that rejects any request.
We introduce the sort S of services and, for each f ∈ F , the binary use
operator /f :T×S→ T. The axioms for these operators are given in Table 7.
In this table, ι stands for an arbitrary member of I, f and g stand for arbitrary
foci from F , and m stands for an arbitrary method from M.
Intuitively, p /f H is the thread that results from processing all basic actions
performed by thread p that are of the form f.m by service H . When a basic
action of the form f.m performed by thread p is processed by service H , it is
turned into the internal action tau and postconditional composition is removed in
favour of action prefixing on the basis of the reply value produced. This intuition
is covered by axioms TSU5 and TSU6. Axioms TSU3 and TSU4 express that
internal actions and basic actions of the form g.m with f 6= g are not processed.
Axiom TSU7 expresses that inaction occurs when a basic action to be processed
is not accepted.
Let T stand for either BTA, BTA+REC or BTA+REC+AIP. Then we will
write T+TSU for T , taking the set {f.m | f ∈ F ,m ∈M} for A, extended with
the use operators and the axioms from Table 7.
15
Table 8. Axioms for abstraction operators
τι(S) = S TT1
τι(D) = D TT2
τι(ι ◦ x) = τι(x) TT3
τι(xE aD y) = τι(x)E aD τι(y) if a 6= ι TT4
V
n≥0 τι(pin(x)) = τι(pin(y)) ⇒ τι(x) = τι(y) TT5
For each ι ∈ I, we introduce the unary abstraction operator τι : T → T to
conceal the presence of internal action ι in the case where its presence does not
matter. The axioms for the abstraction operators are given in Table 8. In this
table, a stands for an arbitrary action from A∪ I.
A main difference between the version of the use mechanism introduced here
and the version of the use mechanism introduced in [10] is that the former version
does not incorporate abstraction and the latter version incorporates abstraction.
Let T stand for either BTA, BTA+REC, BTA+REC+AIP, BTA+TSU,
BTA+REC+TSU or BTA+REC+AIP+TSU. Then we will write T+ABSTR
for T extended with the abstraction operators and the axioms from Table 8.
For each ι ∈ I, the equation τι(ι
ω) = D is derivable from the axioms of
BTA+REC+AIP+ABSTR.
9 Instruction Register File Services
In this section, we describe services that make up register files with a finite
set of registers that can contain instructions from a finite set of core primitive
instructions. These services will be used in Section 10.
It is assumed that a fixed but arbitrary set I ⊆ N such that I = [1, n] for
some n ∈ N and a fixed but arbitrary finite set U ⊆ Ic have been given. The set
I is considered to contain the positions of the registers in the instruction register
file and the set U is considered to contain the instructions that can be put in
those registers.
We write IRFS for the set
⋃
I′⊆I I
′ → U . The members of IRFS are consid-
ered to be the possible instruction register file states. It is assumed that a fixed
but arbitrary bijection θ : IRFS → [1, card(IRFS )] has been given.
The instruction register file services accept the following methods:
– for each i ∈ I and u ∈ U , a register put method put:i:u;
– for each j ∈ rng(θ), a register file test method eq:j.
We write Mirf for the set {put:i:u | i ∈ I ∧ u ∈ U} ∪ {eq:j | j ∈ rng(θ)}.
It is assumed that Mirf ⊆M.
The methods accepted by instruction register file services can be explained
as follows:
– put:i:u : the contents of register i becomes instruction u and the reply is T;
16
– eq:j : if the state of the instruction register file equals θ−1(j), then nothing
changes and the reply is T; otherwise nothing changes and the reply is F.
Let s ∈ IRFS∪{↑}, where ↑ /∈ IRFS . Then the instruction register file service
with initial state s, written IRFs, is the service (IRFS , eff , yld , s), where the
functions eff and yld are defined as follows (σ ∈ IRFS ∪ {↑}):5
eff (put:i:u, σ) = σ ⊕ [i 7→ u] ,
eff (eq:j, σ) = σ ,
eff (m,σ) = ↑ if m 6∈ Mirf ,
eff (m, ↑) = ↑ ,
yld(put:i:n, σ) = T ,
yld(eq:j, σ) = T if θ(σ) = j ,
yld(eq:j, σ) = F if θ(σ) 6= j ,
yld(m,σ) = B if m 6∈ Mirf ,
yld(m, ↑) = B .
10 Program Synthesis
In order to establish a connection between collections of instruction sequence
fragments and instruction sequences, we show in this section that, for each pos-
sible joint behaviour of a collection of instruction sequence fragments, a single
instruction sequence can be synthesized from the collection that produces on
execution essentially the behaviour in question by making use of an instruction
register file service. More precisely, we show that, for each PGAp program P
and polyadic instruction sequence vector α, a PGA program P ′ can be synthe-
sized from P and α such that, for all relevant instruction register file states σ,
|P ′| /irf IRFσ coincides with |P |σ(α) after the presence of the internal actions
tau and gl has been concealed.
Let P be a PGAp program and α be a polyadic instruction sequence vector.
The general idea is that:
– each polyadic instruction sequence in α is translated into a PGAp program
and an appropriate finite collection of instances of this PGAp program in
which occurrences of get instructions are replaced by core primitive instruc-
tions is generated;
– P and all the generated programs are translated into PGLCp programs and
these PGLCp programs are concatenated;
– the resulting PGLCp program is translated into a PGLDp program and this
program is translated into a PGLD program by replacing all occurrences of
the supplementary instructions by core primitive instructions as follows:
• a switch-over instruction ###i is replaced by an absolute jump instruc-
tion whose effect is a jump to the beginning of an appended instruction
sequence whose execution leads, after the state of the instruction register
file has been found by a linear search, to a jump to the beginning of the
5 We use the following notation for functions: [ ] for the empty function; [d 7→ r] for
the function f with dom(f) = {d} such that f(d) = r; and f ⊕ g for the function
h with dom(h) = dom(f) ∪ dom(g) such that, for all d ∈ dom(h), h(d) = f(d) if
d 6∈ dom(g) and h(d) = g(d) otherwise.
17
right instance of the PGAp program that corresponds to the ith polyadic
instruction sequence in α;
• a put instruction $put:i:u is simply replaced by the plain basic instruction
irf.put:i:u;
• a get instruction $get:i is simply replaced by the absolute jump instruc-
tion whose effect is a jump to the position of the instruction itself.
A collection of instances of the PGAp program corresponding to a polyadic
instruction sequence in α is considered appropriate if it includes all instances
that may become the one being executed. P and all the generated programs are
translated into PGLCp programs because PGLCp programs are relocatable: they
can be concatenated without disturbing the meaning of jump instructions. The
PGLCp program resulting from the concatenation is translated into a PGLDp
program before the supplementary instructions are replaced because the replace-
ment of a switch-over instruction by an absolute jump instruction is simpler than
its replacement by a relative jump instruction.
Following the general idea outlined above, we define a function pgap2pgld
that yields, for each PGAp program P , a function that gives, for each polyadic
instruction sequence vector α, a PGLD program P ′ such that, for each relevant
instruction register file service state σ, |pgld2pga(P ′)| /irf IRFσ coincides with
|P |σ(α) after the presence of the internal actions tau and gl has been concealed.
The function pgap2pgld from the set of all PGAp programs to the set all
functions from the set of all polyadic instruction sequence vectors to the set of
all PGLD programs is defined as follows:
pgap2pgld(x)(α) =
translate(pglc2pgld(expand(x)(α))) ;
+irf.eq:1 ; ##l1,1 ; . . . ; +irf.eq:n
′ ; ##l1,n′ ;
...
+irf.eq:1 ; ##ln,1 ; . . . ; +irf.eq:n
′ ; ##ln,n′ ,
where n = len(α), n′ = max(rng(θ)), the function expand from the set of all
PGAp programs to the set all functions from the set of all polyadic instruction
sequence vectors to the set of all PGLCp programs is defined as follows:
expand(x)(α) =
pga2pglc(x) ;
pga2pglc(gen(α, 1, θ−1(1))) ; . . . ; pga2pglc(gen(α, 1, θ−1(n′))) ;
...
pga2pglc(gen(α, n, θ−1(1))) ; . . . ; pga2pglc(gen(α, n, θ−1(n′))) ,
where n = len(α), n′ = max(rng(θ)), and the function gen from the set
of all polyadic instruction sequence vectors, the set of all natural numbers
and the set of all instruction register file states to the set of all PGAp
programs is defined as follows:
18
gen(α, i, σ) = prj
pgn(α,i)(pg(α, i))[σ] if 1 ≤ i ≤ len(α) ∧ valid(α, i, σ) ,
gen(α, i, σ) = #0 if 1 ≤ i ≤ len(α) ∧ ¬valid(α, i, σ) ,
gen(α, i, σ) = ! if i = 0 ∨ i > len(α) ,
the function translate from the set of all PGLDp programs to the set of all PGLD
programs is defined as follows:
translate(u1 ; . . . ; uk) = ψ1(u1) ; . . . ; ψ1(uk) ,
where the functions ψj from the set of all primitive instructions of PGLDp
to the set of all primitive instructions of PGLD are defined as follows
(1 ≤ j ≤ k):
ψj(###i) = ##li if 1 ≤ i ≤ len(α) ,
ψj(###i) = ! if i = 0 ∨ i > len(α) ,
ψj($put:i:u) = irf.put:i:u ,
ψj($get:i) = ##j ,
ψj(u) = u if u is not a supplementary basic instruction ,
where for each i ∈ [1, len(α)]:
li = len(pga2pglc(x))
+
∑
h∈[1,len(α)],h′∈rng(θ)
len(pga2pglc(prj
pgn(α,h)(pg(α, h))[θ
−1(h′)]))
+ 2 ·max(rng(θ)) · (i− 1) ,
and for each i ∈ [1, len(α)] and j ∈ rng(θ):
li,j = len(pga2pglc(x))
+
∑
h∈[1,i−1],h′∈rng(θ)
len(pga2pglc(prj
pgn(α,h)(pg(α, h))[θ
−1(h′)]))
+
∑
h′∈[1,j−1]
len(pga2pglc(prj
pgn(α,i)(pg(α, i))[θ
−1(h′)])) .
The following theorem states rigorously that, for any PGAp program P and
polyadic instruction sequence vector α, for all relevant instruction register file
states σ, |pgld2pga(pgap2pgld(P )(α))| /irf IRFσ coincides with |P |σ(α) after
the presence of the internal actions tau and gl has been concealed.
Theorem 1. Let P be a PGAp program and α be a polyadic instruction sequence
vector, and let n be the highest number occurring in instructions of the form
$put:i:u or $get:i in P or α. Take the interval [1, n] for I and the set of all core
primitive instructions occurring in instructions of the form $put:i:u in P or α
for U , and let σ ∈ IRFS. Then τtau(|pgld2pga(pgap2pgld(P )(α))| /irf IRFσ) =
τtau(τgl(|P |σ(α))).
19
Proof. The proof of Theorem 1 follows the same line as the proof of Theorem 1
from [5] given in that paper. Here, we give only a brief outline of the proof of
the current theorem.
The proof of this theorem proceeds as follows: (i) we give a set T of closed
terms of sortT with τtau(τgl(|P |σ(α))) ∈ T , a set T ′ of closed terms of sortT with
τtau(|pgld2pga(pgap2pgld(P )(α))| /irf IRFσ) ∈ T ′, and a bijection β : T → T ′;
(ii) we show that there exists a set E consisting of one derivable equation p = p′
for each p ∈ T such that, for all equations p = p′ in E:
– β(p) = p′′ is a derivable equation if p′′ is p′ with, for all q ∈ T , all occurrences
of q in p′ replaced by β(q);
– p′ ∈ T only if p′ can be rewritten to a q′ 6∈ T using the equations in E from
left to right.
This means that τtau(τgl(|P |σ(α))) and τtau(|pgld2pga(pgap2pgld(P )(α))| /irf
IRFσ) denote solutions of the same guarded recursive specification. Because
guarded recursive specifications have unique solutions, it follows immediately
that τtau(|pgld2pga(pgap2pgld(P )(α))| /irf IRFσ) = τtau(τgl(|P |σ(α))). ⊓⊔
In the proof outlined above, an apposite indexing of the closed terms in the
sets T and T ′ facilitates the definition of the bijection β. Yet, this definition is
much more complicated than the definition of the bijection needed in the proof
from [5] referred to.
The definition of pgap2pgld shows that the synthesis of single instruction se-
quences from collections of instruction sequence fragments is fairly complicated.
11 Conclusions
We have given a formalization of a simple mechanism by which several instruc-
tion sequence fragments can produce a joint behaviour. Thread extraction pro-
vides the possible joint behaviours of a collection of instruction sequence frag-
ments. We have shown that, for each possible joint behaviour of a collection
of instruction sequence fragments, a single instruction sequence can be synthe-
sized from the collection that produces on execution essentially the behaviour in
question by making use of a service. This program synthesis is reminiscent of the
service-based variant of projection semantics for program notations used in [6].
The program synthesis shows that it is a non-trivial matter to explain by means
of a translation into a single instruction sequence what takes place on execution
of a collection of instruction sequence fragments.
In this paper, an instruction sequence fragment in a collection of instruction
sequence fragments has two attributes: an ordinal index (position) and a program
notation index. We have also mentioned that a nominal index (label) could be
an optional attribute. Many other attributes that are relevant in practice can be
imagined, e.g. modification date, author, tester, owner, user, and security level.
In this paper, we have restricted ourselves to attributes that are indispensable
for a theoretical understanding.
20
The question arises whether all aspects of the mechanism formalized in this
paper can be dealt with at the level of threads. This issue has been investigated
in [9]. It happens that not all aspects can be dealt with at the level of threads. In
particular, the ability to replace special instructions in an instruction sequence
fragment by different ordinary instructions every time execution is switched over
to that fragment cannot be dealt with at the level of threads. Threads turn out
to be too abstract to deal with the mechanism in full.
It is sometimes suggested that program algebra is closely related to Kleene
algebra [13] and omega algebra [12] – an extension of Kleene algebra with an
infinite iteration operator. In program algebra, programs are considered at a
more concrete level than in Kleene algebra and omega algebra. For instance,
Kleene algebra and omega algebra are not concerned with instruction sequences,
and program algebra is not concerned with non-determinism.
References
1. Arnold, K., Gosling, J.: The Java Programming Language. Addison-Wesley, Read-
ing, MA (1996)
2. Bergstra, J.A., Bethke, I.: Polarized process algebra and program equivalence. In:
J.C.M. Baeten, J.K. Lenstra, J. Parrow, G.J. Woeginger (eds.) Proceedings 30th
ICALP, Lecture Notes in Computer Science, vol. 2719, pp. 1–21. Springer-Verlag
(2003)
3. Bergstra, J.A., Bethke, I.: Predictable and reliable program code: Virtual machine
based projection semantics. In: J.A. Bergstra, M. Burgess (eds.) Handbook of
Network and Systems Administration, pp. 653–685. Elsevier, Amsterdam (2007)
4. Bergstra, J.A., Loots, M.E.: Program algebra for sequential code. Journal of Logic
and Algebraic Programming 51(2), 125–156 (2002)
5. Bergstra, J.A., Middelburg, C.A.: Instruction sequences with dynamically instan-
tiated instructions. Electronic Report PRG0710, Programming Research Group,
University of Amsterdam (2007). Available from http://www.science.uva.nl/
research/prog/publications.html. Also available from http://arxiv.org/:
arXiv:0711.4217v3 [cs.PL]
6. Bergstra, J.A., Middelburg, C.A.: Instruction sequences with indirect jumps. Sci-
entific Annals of Computer Science 17, 19–46 (2007)
7. Bergstra, J.A., Middelburg, C.A.: Thread algebra for strategic interleaving. Formal
Aspects of Computing 19(4), 445–474 (2007)
8. Bergstra, J.A., Middelburg, C.A.: Program algebra with a jump-shift instruction.
Journal of Applied Logic 6(4), 553–563 (2008)
9. Bergstra, J.A., Middelburg, C.A.: Thread algebra for poly-threading. Elec-
tronic Report PRG0810, Programming Research Group, University of Ams-
terdam (2008). Available from http://www.science.uva.nl/research/prog/
publications.html. Also available from http://arxiv.org/: arXiv:0803.0378v2
[cs.LO]
10. Bergstra, J.A., Ponse, A.: Combining programs and state machines. Journal of
Logic and Algebraic Programming 51(2), 175–192 (2002)
11. Bishop, J., Horspool, N.: C# Concisely. Addison-Wesley, Reading, MA (2004)
12. Cohen, E.: Seperation and reduction. In: R. Backhouse, J.N. Oliveira (eds.) MCP
2000, Lecture Notes in Computer Science, vol. 1837, pp. 45–59. Springer-Verlag
(2000)
21
13. Kozen, D.: A completeness theorem for Kleene algebras and the algebra of regular
events. Information and Computation 110(2), 366–390 (1994)
14. Ponse, A., van der Zwaag, M.B.: An introduction to program and thread algebra.
In: A. Beckmann, et al. (eds.) CiE 2006, Lecture Notes in Computer Science, vol.
3988, pp. 445–458. Springer-Verlag (2006)
15. Sannella, D., Tarlecki, A.: Algebraic preliminaries. In: E. Astesiano, H.J. Kreowski,
B. Krieg-Bru¨ckner (eds.) Algebraic Foundations of Systems Specification, pp. 13–
30. Springer-Verlag, Berlin (1999)
16. Wirsing, M.: Algebraic specification. In: J. van Leeuwen (ed.) Handbook of Theo-
retical Computer Science, vol. B, pp. 675–788. Elsevier, Amsterdam (1990)
22
