[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
This manual documents how to run, install and use the Algol 60 Interpreter.
2. The Goal of the Interpreter | The goal of the Interpreter. | |
3. Installation of the Algol 60 Interpreter | About the installation of ‘a60’. | |
4. Algol 60 Command Line Options | Command line options supported by ‘a60’. | |
5. The Implementation | About the implementation. | |
9. Some Examples | Examples. | |
10. Parser and Runtime Messages | ||
11. About Bugs and Bug Reports | Known Bugs and more. | |
12. Bibliography and References |
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
This Algol 60 interpreter is based upon the “Revised Report on the Algorithmic Language Algol 60” [RRA60].
At school, a long time ago, I learned Algol 60 in a completely theoretical manner. Later I learned Algol 68 and C (and more ...).
The concept of call-by-name never left my mind, and so I started to write this Algol 60 interpreter: Made for fun and a call-by-name.
Here is an example:
'begin' 'integer' 'procedure' one; 'begin' write (`one called \n'); one := 1 'end'; 'procedure' foo (n); 'integer' n; 'if' n > 0 'then' foo ( n - one ); foo (5) 'end' |
The parameter ‘n’ in ‘foo (n)’ is called by name. Every time ‘n - one’ is evaluated, ‘n’ is evaluated by name. Guess how many times ‘one’ is called: 5, 10, 15 ?
Guess or prove ? – I want to run the example and see the result. And now you can do like me.
This was the main goal: call-by-name.
Many things were later added, and now the defining description of the “Revised Report on the Algorithmic Language Algol 60” is nearly (hopefully) fulfilled.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
A60 now runs on Un*x machines and PC’s.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
Since version v0.18 a configure script is provided.
Simply run ./configure
followed by make
.
For your convenience the old Makefile is still avail as Makefile.unx.
If configure does not work for you, follow this old instructions:
Glance through the Makefile and change the FLAGS as appropriate:
_POSIX_SOURCE
define this when compiling for a Posix compliant System. This should work and suffice for many Systems including SunOS and Linux.
USG
define this when compiling for a System V Un*x. For a BSD system define nothing; this is the default.
VPRINTF_MISSING
define this if your system does not provide the vprintf () function. This is used in err.c.
REALLOC_MISSING
define this if your system does not provide the realloc () function. This is used in util.c.
ALLOCA_MISSING
define this if your system does not provide the alloca () function. This is only used by bison. If you are not using the bison generated parser, this define is not used.
NO_LIMITS_H
define this if your system has no header limits.h, defining
LONG_MIN
and LONG_MAX
. (don’t care: set it if
you’re in doubt)
NO_ENUMS
define this if your compiler bombs on enums and you have changed the enum declarations in the header files. Look to ENUM.README for more about this (normally you will not).
DEBUG
define this if you would like to include general debug code (normally you will not).
PARSEDEBUG
define this if you would like to include the debug code for the parser (normally you will not).
MEMORY_STATISTICS
define this if you would like to include some code for computing statistics about the amount of heap and stack used (normally you will not).
For installation adjust BINDIR to point to the destination for the “a60” binary, and LIBDIR to point to the destination of the “a60-mkc.inc” file. If you don’t want this, set them to ‘/tmp’; they are only used, if C output is being compiled. MANDIR and MANSUFF are used to install the “a60.man” manual page.
Ah, we are back to normality:
Say make
to compile.
If you would like to make the simple edit-and-go xa60 application, say
make xa60
.
If you would like to run the test suite, say make test
, and
hopefully no differences between the expected output and the actual
output will be found.
Say make install
to install the binary, the manpage and the
include-file.
Say make xa60-install
to install the xa60 binary and the xa60
manpage.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
I’ve compiled the sources with QuickC v2.0 using qc-makeit.bat. The project file is qc-a60.mak. The compiler itself runs short of memory when running the optimiser, so the a60-ptab.c module had better be compiled without it.
C code generation is possible, but I’ve tried it only with few examples, because the large generated macros cannot be compiled properly.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
When you invoke Algol 60 ...
Without arguments, the program text is read from standard input, and executed upon reaching EOF.
The available options:
Print the usage message and exit.
Print the Version string and exit.
Be verbose processing the input. The version string is displayed too.
Don’t run the input; only parse and check.
Do not check or execute the input; parse only. (This was useful for debugging the interpreter.)
Trace line numbers when running the input.
Follow strict a60 convention. Skip whitespace in entire input, except in strings. Keywords must be enclosed in single quotes.
Create C output from the input. This is an experimental option which changes a60 into something like a60-to-c.
Create C output from the input, like the option -c, but then invokes the C compiler and creates an executable (hopefully).
Place the output in file file. This is used, if C code is created (via the -c option) or if the input is compiled (via the -C option).
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
6. Representation of Algol 60 Code | About Keywords and recognized input | |
7. Builtin Functions | Toplevel functions | |
8. C-code creation |
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
There is a strict form of the input which conforms to RRA60 and also a simple form.
The strict form:
Keywords are expected to be enclosed in single quotes: ’. For example: ’begin’, ’for’, ’if’, ’end’.
The case of letters is insignificant in keywords. For example: ’begin’ is the same as ’Begin’, ’integer’ loopvar is the same as ’INTEGER’ loopvar.
Whitespace characters are skipped in the input, except in strings. For example: ’integer’ greatnumber is the same as ’integer’ great number, and the same as ’ i n t e g e r’ great n u m b e r.
Strings are expected to be enclosed in double quotes, or in a backquote and a quote. For example: "This is a string", ‘This is a string’. The ’\’ is recognized as a escape character (like C syntax). "\n" is a linefeed, "\"" is a double-quote and "\\" is a backslash.
The simple form:
Keywords are written like identifiers. For example: begin, for, if, end. White spaces are recognized to separate tokens. Therefore, it is illegal to use: integer great number;
The simple form is used if no quoted keyword is scanned. RRA60 conformance can be forced with the ‘-strict’ option.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
entier, abs, sign, sqrt, sin, cos, arctan, exp: implemented as described in RRA60.
rand, pi: random number generation and the constant “pi”:
'real' 'procedure' rand; 'code' |
returns a random number between 0.0 (inclusive) and 1.0 (exclusive). The randomness of “rand” is not very robust.
'real' 'procedure' pi; 'code' |
returns the constant “pi”.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
The input and output functions use channel numbers to read from or to write to. The range of the channel numbers is from 0 to 15 included. The channel numbers 0, 1 and 2 are taken from the standard channel numbering known als 0 = stdin (standard input), 1 = stdout (standard output) and 2 = stderr (standard error).
The channels 2 to 15 were mapped to files. The first use of a channel determines the direction: If it is an output function, the file is opened in write mode, if it is a input function, the file is opened in read mode.
The filename is read from the environment variable “FILE_n” where n is the channel number. If the environment variable is not set the name “FILE_n” is used with n set to the channel number. So if the environment variable FILE_3 is set to data.txt writing to channel 3 will write to the file data.txt. If this environment variable is not set writing to channel 3 will write to the file FILE_3.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
length, outstring, insymbol, outsymbol
'integer' 'procedure' length (string); 'string' string; 'code'; |
returns the length of the string string.
'procedure' outstring (channel, value); 'value' channel; 'integer' channel; 'string' value; 'code'; |
send the string value to the channel channel. Channel 1 is stdout (standard output) and channel 2 is stderr (standard error).
'procedure' write (string); 'string' string; 'code'; |
Prints the string string to standard output. This is the same behavior as outstring (1, string).
'procedure' insymbol (channel, string, value); 'value' channel; 'integer' channel, value; 'string' string; 'code'; |
A character is read from channel channel. If the character is found in string, the index is assigned to value with a starting index of 0. If the character is not found, the negative character code is assigned to value. Channel 0 is stdin (standard input).
'procedure' outsymbol (channel, string, source); 'value' channel, source; 'integer' channel, source; 'string' string; 'code'; |
Prints the character at the source position of string string to channel channel. The posistion is counted from 0. If source is a negative value, -source is sent to the channel and the string is ignored. Channel 1 is stdout (standard output) and channel 2 is stderr (standard error).
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
'procedure' print (value, f1, f2); 'value' value, f1, f2; 'real' value; 'integer' f1, f2; 'code'; |
The value value is printed with f1 and f2 used as format. [ still missing: *** describe f1 and f2 *** ] The output is printed to standard output.
'procedure' inreal (channel, value); 'value' channel; 'integer' channel; 'real' value; 'code'; |
Reads a real number from channel channel and assigns it to value. Channel 0 is stdin (standard input).
'procedure' ininteger (channel, value); 'value' channel; 'integer' channel; 'integer value; 'code'; |
Reads a integer type number from channel channel and assigns it to value. Channel 0 is stdin (standard input).
'procedure' outreal (channel, value); 'value' channel, value; 'integer' channel; 'real' value; 'code'; |
Prints the value value to channel channel. Channel 1 is stdout (standard output) and channel 2 is stderr (standard error).
'procedure' outinteger (channel, value); 'value' channel, value; 'integer' channel, value; 'code'; |
Prints the value value to channel channel. Channel 1 is stdout (standard output) and channel 2 is stderr (standard error).
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
'procedure' vprint (...); 'code'; |
Vprint prints the variable arguments to the standard output.
The output is terminated with a newline-character. Numbers are
printed width a fixed with (about 14 characters).
For example: vprint ("Foo: ", 12, 99.9).
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
[** Still not finished **]
C-code creation for less complex programs is now possible. The resulting code is somewhat faster (example whetstones: about a factor of 50).
Call-by-name procedures must be expandable into C macros. The other procedures are translated into C functions.
Problems / Restrictions:
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
Example 1:
'begin' write ("Hi!\n") 'end' |
Assume these three lines are in a file named ‘hi.a60’. Run it with the call ‘a60 hi’: It produces the output:
Hi! |
Example 2:
'begin' 'integer' 'procedure' fakul (n); 'value' n; 'integer' n; 'begin' 'if' n < 1 'then' fakul := 1 'else' fakul := n * fakul (n - 1) 'end'; 'integer' result; outstring (1, "See fakul (5): "); result := fakul (5); outinteger (1, result); outstring (1, "\n"); 'end' |
This will produce the output:
See fakul (5): 120 |
Example 3:
The classic call-by-name example: The “Jensen Device”:
[ Note: Here the keywords are not quoted; this is not RRA60 compliant, but usable as an extension of NASE A60. ]
begin procedure jdev ( i, n, s, x ); begin s := 0; for i := 1 step 1 until n do s := s + x; end; integer NN; NN := 100; begin integer i; real sum; integer array arr [1 : NN]; for i := 1 step 1 until NN do arr[i] := i; jdev (i, NN, sum, arr [i]); outstring (1, `See the sum: '); outreal (1, sum); outstring (1, `\n') end end |
This will produce the output:
See the sum: 5050 |
The clever part is the loop-variable used in jdev which is passed by name and used as index in the array “arr [i]”.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
[ *** not yet - sorry *** ]
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
Surely there are many bugs. Of interest are any core dumps: regardless of correct input or not and compile-time and run-time misbehavior, this should never happen. Secondary are the elegance and efficiency of the implementation.
Please report bugs to Erik Schoenfelder (schoenfr@web.de). Hopefully I will have enough time to reply.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
Revised Report on the Algorithmic Language Algol 60.
Communications of the ACM
[Top] | [Contents] | [Index] | [ ? ] |
[Top] | [Contents] | [Index] | [ ? ] |
This document was generated by bond on July 28, 2014 using texi2html 1.82.
The buttons in the navigation panels have the following meaning:
Button | Name | Go to | From 1.2.3 go to |
---|---|---|---|
[ < ] | Back | Previous section in reading order | 1.2.2 |
[ > ] | Forward | Next section in reading order | 1.2.4 |
[ << ] | FastBack | Beginning of this chapter or previous chapter | 1 |
[ Up ] | Up | Up section | 1.2 |
[ >> ] | FastForward | Next chapter | 2 |
[Top] | Top | Cover (top) of document | |
[Contents] | Contents | Table of contents | |
[Index] | Index | Index | |
[ ? ] | About | About (help) |
where the Example assumes that the current position is at Subsubsection One-Two-Three of a document of the following structure:
This document was generated by bond on July 28, 2014 using texi2html 1.82.