(very early draft)
Comment in Stargaze starts with ;
.
Only the strings #t
and #f
are boolean values; they evaluate to true and false respectively.
There’s only one kind of syntax for string literals: "{string-piece}*"
, where string-piece
is one of:
- Normal characters.
- Escape sequences, which starts with
\
and is of the following format:\r
,\n
,\b
,\t
,\\
,\"
,\v
,\f
, for carriage return, linefeed, backspace, horizontal tab, slash, double quote, vertical tab and form feed respectively.\xHEX
, whereHEX
is the codepoint of character in hexadecimal, e.g.\x61
is the same asa
.
Two kinds of syntax for character literals:
#\NAME
whereNAME
is the name for the character.NAME
is often the character itself (e.g.#\a
refers to the charactera
), but exceptions exist; see below.#ch{HEX}
whereHEX
is the codepoint in hexadecimal, e.g.#ch{61}
is the same as#\a
.
As for the first kind, Scheme R4RS only explicitly declares a #\space
and a #\newline
. Stargaze (partly) follows the convention in MIT Scheme, except that #\newline
always corresponds to the ASCII LF regardless of the operating system. For this reason the following “character name”s are also defined in Stargaze:
- All of the standard ASCII names for non-printing characters are supported. They can be used by prefixing with
#\
, e.g.#\NUL
:NUL SOH STX ETX EOT ENQ ACK BEL BS HT LF VT FF CR SO SI DLE DC1 DC2 DC3 DC4 NAK SYN ETB CAN EM SUB ESC FS GS RS US DEL
- A few other names are also defined:
#\esc
, for#\ESC
. MIT Scheme used the string#\altmode
, which we#\backspace
for#\BS
#\linefeed
for#\LF
(also, as stated above,#\newline
)#\tab
for#\HT
#\return
for#\CR
Vector literals can be constructed by surrounding values with #{
and }
.
Four kinds of quote exists in Stargaze:
quote
/'
, which turns whatever it prefixed into “the thing itself”; e.g.(if #t 3 4)
evaluates to3
but'(if #t 3 4)
evaluates to(if #t 3 4)
(that is, a list containingif
,#t
,3
and4
.)qquote
/`
, which works like normal quote, but any of the sub-expression within can be “unquoted” and thus get evaluated like normal.unquote
/,
, which evaluates whatever it prefixed within aquasiquote
-ed context. e.g.`(if (leq 3 4) 3 4)
evaluates to(if (leq 3 4) 3 4)
but`(if ,(leq 3 4) 3 4)
evaluates to(if #t 3 4)
.
These quotes are equivalent to their s-expr counterparts at all time, e.g. ''a
would be equivalent to (quote (quote a))
which would evaluates to (quote a)
(which also means that (list? ''a)
would evaluate to #t
.)
(def NAME BODY)
: Bind the valueBODY
to the nameNAME
in the current environment.(fn ARGLIST BODY)
: Anonymous function.(if COND THEN_CLAUSE ELSE_CLAUSE)
: Simple branching.(cond (COND1 CLAUSE1) ...)
: Multiway branching.(let ((NAME1 EXP1) ...) BODY ...)
:let
-binding.(letrec ((NAME1 EXP1) ...) BODY ...)
:letrec
-binding.(quote ...)
: Quote its argument as symbolic values(set! NAME VALUE)
: AssignVALUE
to the nameNAME
.(begin EXP1 ...)
: ExecutesEXP1
and the rest in the order they appear in the argument list. Returns the evaluated value of the last argument..(equal EXP1 EXP2)
: Check ifEXP1
andEXP2
has the same value.
(include STR)
: Include a file.(import MOD RENAME?)
:MOD
can be one of two options:- A string, which refers to the module.
(STR PREFIX)
, which prefix all exported name inSTR
withPREFIX
.- e.g. assumes that
mymodule
exportsfunc1
,func2
andfunc3
;(import ("mymodule" xyz))
would importfunc1
asxyzfunc1
,func2
asxyzfunc2
,func3
asxyzfunc3
.
- e.g. assumes that
RENAME
is one of two followings:- Nothing, in which case the form would be
(import MOD)
; this would import all exported name inMOD
. - A list of
(NAME NEWNAME)
, which would only import the exported names described byNAME
, which would bind toNEWNAME
when importing. This renaming takes precedence over prefixes, so(import ("abc" abc) (myfunc1 defmyfunc1))
would importmyfunc1
from module"abc"
intodefmyfunc1
instead ofabcmyfunc1
orabcdefmyfunc1
.
- Nothing, in which case the form would be
- Note that
(import MOD)
means import all names fromMOD
, while(import MOD ())
means import all the exported names in()
, which means none of the exported names.
in the imported module; NEW_NAME
would be the effective name in the importing module for NAME
in the imported module.
(export NAME ...)
: Export
(atom? EXP)
: Check ifEXP
is an atom, i.e. a symbol, an integer, a boolean, a string, a character, or the empty list.(closure? EXP)
: Check ifEXP
is a closure.(primitive? EXP)
: Check ifEXP
is a primitive.(procedure? EXP)
: Check ifEXP
is a closure or primitive.- Note that in Stargaze lingo a “primitive” refers to primitives that can be applied like a function (unlike things like
fn
andquote
, which are called “special forms” and are called “syntax forms” in other LISP-like languages) and a “closure” refers to any anonymous function created during the course of execution.
- Note that in Stargaze lingo a “primitive” refers to primitives that can be applied like a function (unlike things like
(int? EXP)
: Check ifEXP
is an integer.(add EXP ...)
: Return the sum of the arguments.(sub EXP1 ...)
: ReturnEXP1 - EXP2 - ...
.(mul EXP ...)
: Return the product of the arguments.(div EXP1 ...)
: ReturnEXP1 div EXP2 div ...
(mod EXP1 EXP2)
: ReturnEXP1 % EXP2
.(leq EXP1 EXP2)
,(lt EXP1 EXP2)
,(geq EXP1 EXP2)
,(gt EXP1 EXP2)
: Check ifEXP1
is smaller or equal than, strictly smaller than, greater or equal than, and strictly greater thanEXP2
respectively.
(float? EXP)
: Check ifEXP
is an integer.(addf EXP ...)
: Return the sum of the arguments.(subf EXP1 ...)
: ReturnEXP1 - EXP2 - ...
.(mulf EXP ...)
: Return the product of the arguments.(divf EXP1 ...)
: ReturnEXP1 div EXP2 div ...
(float INT)
: ConvertINT
to a floating point number.- Returns the argument itself if it is already a floating-point number.
(floor FLOAT)
,(ceil FLOAT)
,(round FLOAT)
,(trunc FLOAT)
:floor
returns the first integer that’s smaller than or equal to its arugment.ceil
returns the first integer that’s bigger than or equal to its argument.round
rounds its argument to the nearest integer.trunc
directly removes the decimal part.- Returns the argument itself if it is already an integer.
(leqf EXP1 EXP2)
,(ltf EXP1 EXP2)
,(geqf EXP1 EXP2)
,(gtf EXP1 EXP2)
: Check ifEXP1
is smaller or equal than, strictly smaller than, greater or equal than, and strictly greater thanEXP2
respectively.(eqnum EXP1 EXP2)
: Return ifEXP1
andEXP2
has the same numerical value.
(cons EXP1 EXP2)
: Return the pair ofEXP1
andEXP2
(car EXP1)
: Return the first component of the pairEXP1
.(cdr EXP1)
: Return the second component of the pairEXP2
.(set-car! EXP NEWCAR)
: Set the first component ofEXP
toNEWCAR
.(set-cdr! EXP NEWCDR)
: Set the second component ofEXP
toNEWCAR
.(w/car EXP NEWCAR)
: Equivalent to(cons NEWCAR (cdr EXP))
.(w/cdr EXP NEWCDR)
: Equivalent to(cons (car EXP) NEWCDR)
.
(chr INT)
: ConvertINT
into the corresponding character.(ord CHAR)
: ConvertCHAR
into the corresponding integer.(char? EXP)
: Check ifEXP
is a character.
(strref STR I)
: Retrieve theI
-th (starting from 0) character of the stringSTR
.(substr STR START END?)
: Retrieve the substring of the stringSTR
starting from indexSTART
to indexEND
.END
is optional; if it’s not provided, this function takes the substring starting from indexSTART
to the end of the string.(strappend STR1 ...)
: Return the string appended together from its arguments.(strsym STR)
: Convert a string to a symbol.(str? EXP)
: Check ifEXP
is a string.
(and EXP1 ...)
: Returns#f
if one of theEXP
evalueates to#f
, or else returns the last argument.(or EXP1 ...)
: Returns the first value that is not#f
(if any); or else, returns#f
.(not EXP)
: IfEXP
evaluates to#f
then return#t
; returns#f
otherwise.
(symstr SYM)
: Convert a symbol to a string.(sym? EXP)
: Check ifEXP
is a symbol.
(list EXP1 ...)
: Combine its arguments into a list.(nil? EXP)
: Check ifEXP
is an empty list.
(vec? EXP)
: Check ifEXP
is a vector.(vector EXP1 ...)
: Combine its arguments into a vector.(listvec LIST)
: ConvertLIST
into a vector.(veclist VEC)
: ConverVEC
into a list(vecref VEC INT)
: Return theINT
-th element fromVEC
.(mkvec INT)
: Create a vector of sizeINT
.(vecset! VEC INT VALUE)
: Set theINT
-th element ofVEC
to valueVALUE
.(vec++ EXP1 ...)
:(veclen VECTOR)
:
stdin
,stdout
,stderr
: Standard input, standard output and standard error.(eof? EXP)
: Check ifEXP
is the EOF object.(openinput STR)
: Open fileSTR
as input.(openoutput STR MODE?)
: Open fileSTR
as output.- When
MODE
is not used (i.e. the form is(openoutput STR)
), the file is opened for writing. - When
MODE
is"a"
, the file is opened for appending.
- When
(close FILE)
: Close a file.(readch FILE)
: Read a character from an input file.(writech FILE)
: Write a character to an output file.
Iteration is important (at least for now) since we don’t have tail-call optimization.
(while COND BODY)
: Repeatedly executeBODY
untilCOND
evaluates to false. Returns nil.
A few functions that should be able to be defined using the language itself is defined as primitives for performance, even if there aren’t much performance to begin with…
(length LIST)
: Return the length ofLIST
.(append EXP ...)
/(list++ EXP ...)
: CombinesEXP
and the rest into one list.(reverse LIST)
: Return a reversed version ofLIST
.(map F LIST1 ...)
: Returns a new list whose members are the results of applyingF
onLIST1
and the rest, e.g.(map add (list 3 4) (list 5 6) (list 7 8))
is equivalent to(list (add 3 5 7) (add 4 6 8))
.(filter F LIST)
: Returns a new list consisting of all the members ofLIST
that satisfiesF
.F
should be a function that takes 1 argument and returns a boolean.(member EXP LIST)
: Check ifEXP
is in the listLIST
. If it is, returns the part ofLIST
starting fromEXP
; if not, return the false value.(assoc EXP LIST)
: Check ifEXP
is in the assoc listLIST
. An assoc list in LISP-like languages is a kind of list that consists of key-value pairs. If there is a key-value pair that usesEXP
as the key, this primitive will return that key-value pair; or else, it will return the false value.
(bit~ INT)
: Bitwise not.(bit& INT ...)
: Bitwise and.(bit^ INT ...)
: Bitwise xor.(bit| INT ...)
: Bitwise or.(bit<< INT INT)
: Left shift.(bit>> INT INT)
: Right shift (left-side padded with 0)
(mkstr N CHAR)
: Makes a string of lengthN
consisting of only characterCHAR
.(strlen STR)
: Returns the length of the stringSTR
.(strlist STR)
: Convert a string to a list of characters.(strvec STR)
: Convert a string to a vector of characters.(str++ STR1 ...)
: ConcatenatesSTR1
and the rest into one single string.