Copyright © 2000-2004, Felix L. Winkelmann All rights reserved.

Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:

CHICKEN - A practical and portable Scheme system. Version 1, Build 63

CHICKEN - A practical and portable Scheme system. Version 1, Build 63

Table of Contents


Next: , Up: (dir)

CHICKEN

CHICKEN - A practical and portable Scheme system User's manual (Version 1, Build 63)

(c) 2000-2004, Felix L. Winkelmann All rights reserved. Translated to LaTeX by Peter Keller. Translated to texinfo by Linh Dang.

--- The Detailed Node Listing ---

Using the compiler

Using the interpreter

Supported language

Non-standard macros and special forms

Unit library

Unit eval

Unit extras

Unit posix

Unit utils

Unit lolevel

Unit tinyclos

Interface to external functions and variables

The Easy Foreign Function Interface

chicken-setup

Additional files

test-infrastructure.scm


Next: , Previous: Top, Up: Top

1 Introduction

CHICKEN is a compiler that translates Scheme source files into C, which in turn can be fed to a C-compiler to generate a standalone executable. This principle, which is used by several existing compilers, achieves high portability because C is implemented on nearly all available platforms.

This package is distributed under the BSD license and as such is free to use and modify. An interpreter is also available and can be used as a scripting environment or for testing programs before compilation.

The method of compilation and the design of the runtime-system follow closely Henry Baker's CONS Should Not CONS Its Arguments, Part II: Cheney on the M.T.A. paper and expose a number of interesting properties: consing (creation of data on the heap) is relatively inexpensive, because a generational garbage collection scheme is used, in which short-lived data structures are reclaimed extremely quickly. Moreover, call-with-current-continuation is practically for free and CHICKEN does not suffer under any performance penalties if first-class continuations are used in complex ways. The generated C code is fully tail-recursive.

Some of the features supported by CHICKEN:


Next: , Previous: Introduction, Up: Top

2 Basic mode of operation

The compiler translates Scheme source code into fairly portable C that can be compiled and linked with most available C compilers. CHICKEN supports the generation of executables and libraries, linked either statically or dynamically. Compiled Scheme code can be loaded dynamically, or can be embedded in applications written in other languages. Separate compilation of modules is fully supported.

The most portable way of creating separately linkable entities is supported by so-called units. A unit is a single compiled object module that contains a number of toplevel expressions that are executed either when the unit is the main unit or if the unit is used. To use a unit, the unit has to be declareed as used, like this:

(declare (uses UNITNAME))

The toplevel expressions of used units are executed in the order in which the units appear in the uses declaration. Units may be used multiple times and uses declarations may be circular (the unit is initialized at most once). To compile a file as a unit, add a unit declaration:

(declare (unit UNITNAME))

When compiling different object modules, make sure to have one main unit. This unit is called initially and initializes all used units before executing its toplevel expressions. The main-unit has no unit declaration.

Another method of using definitions in separate source files is to include them. This simply inserts the code in a given file into the current file:

(include "FILENAME")

One important thing: macro definitions are only available when processed by include or visit. Macro definitions in separate units are not available, since they are defined at compile time, i.e the time when that other unit was compiled (macros defined using the low-level macro system can optionally be available at runtime, see define-macro in Substitution forms and macros).

On platforms that support dynamic loading of compiled code (like Windows and most ELF based systems like Linux or BSD), it is not necessary to use unit declarations. Here code can be compiled into a shared object (.so) and loaded dynamically into a running application.


Next: , Previous: Basic mode of operation, Up: Top

3 Using the compiler

The interface to chicken is intentionally simple. System dependent makefiles, shell-scripts or batch-files should perform any necessary steps before and after invocation of chicken. On UNIX-compatible systems, a shell script named chicken-config is supplied that emits the correct options for the host system's C compiler. Enter

chicken-config -help

on the command line for a list of available options. On most UNIX systems, a Scheme script named csc provides a much simpler interface to the Scheme- and C-compilers and linker. Enter

csc -help

on the command line for more information. A batch-file (csc.bat) with the same functionality is also available for Windows platforms. The Windows version of csc (csc.bat) does not accept more than 8 arguments.


Next: , Up: Using the compiler

3.1 Command line format

chicken FILENAME {OPTION}

FILENAME is the complete pathname of the source file that is to be translated into C. A filename argument of “-” specifies that the source text should be read from standard input. Note that the filename has to be the first argument to chicken. Possible options are:

-analyze-only
Stop compilation after first analysis pass.
-benchmark-mode
Equivalent to -debug-level 0 -optimize-level 3 -fixnum-arithmetic -disable-interrupts -block -lambda-lift.
-block
Enable block-compilation. When this option is specified, the compiler assumes that global variables are not modified outside this compilation-unit. Specifically, toplevel bindings are not seen by eval and unused toplevel bindings are removed.
-case-insensitive
Enables the reader to read symbols case insensitive. The default is to read case sensitive (in violation of R5RS). This option registers the case-insensitive feature identifier.
-check-syntax
Aborts compilation process after macro-expansion and syntax checks.
-compress-literals THRESHOLD
Compiles quoted literals that exceed the size THRESHOLD as strings and parse the strings at run-time. This reduces the size of the code and speeds up compile-times of the host C compiler, but has a small run-time performance penalty. The size of a literal is computed by counting recursively the objects in the literal, so a vector counts as 1 plus the count of the elements, a pair counts as the counts of the car and the cdr, respectively. All other objects count 1.
-debug MODES
Enables one or more compiler debugging modes. MODES is a string of characters that select debugging information about the compiler that will be printed to standard output.
t
show time needed for compilation
b
show breakdown of time needed for each compiler pass
o
show performed optimizations
r
show invocation parameters
s
show program-size information and other statistics
a
show node-matching during simplification
p
show execution of compiler sub-passes
l
show lambda-lifting information
m
show GC statistics during compilation
n
print the line-number database
c
print every expression before macro-expansion
e
lists all exported toplevel bindings
x
display information about experimental features
D
when printing nodes, use node-tree output
N
show the real-name mapping table
U
show expressions after the secondary user pass
0
show database before lambda-lifting pass
L
show expressions after lambda-lifting
F
show output of “easy” FFI parser
P
show execution of outer partitioning
Q
show execution of middle partitioning
R
show execution of inner partitioning
1
show source expressions
2
show canonicalized expressions
3
show expressions converted into CPS
4
show database after each analysis pass
5
show expressions after each optimization pass
6
show expressions after each inlining pass
7
show expressions after complete optimization
8
show database after final analysis
9
show expressions after closure conversion

-debug-level LEVEL
Selects amount of debug-information. LEVEL should be an integer.
-disable-c-syntax-checks
Disable basic syntax checking of embedded C code fragments.
-disable-interrupts
Equivalent to the (disable-interrupts) declaration. No interrupt-checks are generated for compiled programs.
-disable-stack-overflow-checks
Disables detection of stack overflows. This is equivalent to running the compiled executable with the -:o runtime option.
-dynamic
This option should be used when compiling files intended to be loaded dynamically into a running Scheme program.
-epilogue FILENAME
Includes the file named FILENAME at the end of the compiled source file. The include-path is not searched. This option may be given multiple times.
-explicit-use
Disables automatic use of the units library, eval and extras. Use this option if compiling a library unit instead of an application unit.
-extend FILENAME
Loads a Scheme source file or compiled Scheme program (on systems that support it) before compilation commences. This feature can be used to extend the compiler. This option may be given multiple times. The file is also searched in the current include path and in the extension-repository.
-feature SYMBOL
Registers SYMBOL to be a valid feature identifier for cond-expand.
-ffi
Compile C/C++ code and generate Scheme bindings. This is effectively equivalent to wrapping the code in #>! ... <#.
-ffi-define SYMBOL
Defines a macro that will be accessible in foreign-parse declarations.
-ffi-include-path PATH
Set include path for “easy” FFI parser.
-fixnum-arithmetic
Equivalent to (fixnum-arithmetic) declaration. Assume all mathematical operations use small integer arguments.
-heap-size NUMBER
Sets a fixed heap size of the generated executable to NUMBER bytes. The parameter may be followed by a M (m) or K (k) suffix which stand for mega- and kilobytes, respectively. The default heap size is 5 kilobytes. Note that only half of it is in use at every given time.
-heap-initial-size NUMBER
Sets the size that the heap of the compiled application should have at startup time.
-heap-growth PERCENTAGE
Sets the heap-growth rate for the compiled program at compile time (see: -:hg).
-heap-shrinkage PERCENTAGE
Sets the heap-shrinkage rate for the compiled program at compile time (see: -:hs).
-help
Print a summary of available options and the format of the command line parameters and exit the compiler.
-syntax
-hygienic
Load “syntax-case” macro package and enable high-level macros in compiled code. This option registers the hygienic-macros feature identifier.
-hygienic-at-run-time
Makes hygienic (“syntax-case”) macro system available at run-time. Note that this has a slight overhead, because the hygienic macro definitions have to be loaded. This will only install the standard R5RS macros, plus SRFI-0 (cond-expand). To load all further macros, execute:
     
     (require-extension chicken-more-macros)

-include-path PATHNAME
Specifies an additional search path for files included via the include special form. This option may be given multiple times. If the environment variable CHICKEN_INCLUDE_PATH is set, it should contain a list of alternative include pathnames separated by “;”. The environment variable CHICKEN_HOME is also considered as a search path.
-keyword-style STYLE
Enables alternative keyword syntax, where STYLE may be either prefix (as in Common Lisp), suffix (as in DSSSL) or none. Any other value is ignored. The default is suffix. If -strict or -strict-reader is specified, then the keyword style is set to none.
-lambda-lift
Enable the optimization known as lambda-lifting.
-no-trace
Disable generation of tracing information. If a compiled executable should halt due to a runtime error, then a list of the name and the line-number (if available) of the last procedure calls is printed, unless -no-trace is specified. With this option the generated code is slightly faster.
-no-feature SYMBOL
Unregisters feature identifier SYMBOL.
-no-warnings
Disable generation of compiler warnings.
-nursery NUMBER -stack-size NUMBER
Sets the size of the first heap-generation of the generated executable to NUMBER bytes. The parameter may be followed by a M (m) or K (k) suffix. The default stack-size depends on the target platform.
-optimize-leaf-routines
Enable leaf routine optimization.
-optimize-level LEVEL
Enables certain sets of optimization options. LEVEL should be an integer.
-output-file FILENAME
Specifies the pathname of the generated C file. Default is FILENAME.c.
-postlude EXPRESSIONS
Add EXPRESSIONS after all other toplevel expressions in the compiled file. This option may be given multiple times. Processing of this option takes place after processing of -epilogue.
-prelude EXPRESSIONS
Add EXPRESSIONS before all other toplevel expressions in the compiled file. This option may be given multiple times. Processing of this option takes place before processing of -prologue.
-profile
Instruments the source code to count procedure calls and execution times. After the program terminates (either via an explicit exit or implicitly), profiling statistics are written to a file named PROFILE. Each line of the generated file contains a list with the procedure name, the number of calls and the time spent executing it. Use the chicken-format-profile program to display the profiling information in a more user-friendly form. Enter chicken-format-profile with no arguments at the command line to get a list of available options.
-prologue FILENAME
Includes the file named FILENAME at the start of the compiled source file. The include-path is not searched. This option may be given multiple times.
-quiet
Disables output of compile information.
-require-for-syntax NAME
Loads the extension NAME before the compilation process commences.
-r5rs
Equivalent to -hygienic -strict.
-run-time-macros
Makes low-level macros (compiled without the -hygienic option) also available at run-time. By default low-level macros are not available at run-time. Note that highlevel-macros (syntax-case) defined in compiled code are never available at run-time.
-split NUMBER
Splits output into multiple C files that can be compiled seperately. The generated C files will be named filename0, ..., filename<NUMBER-1> with as many files as given in NUMBER.
-split-level NUMBER
Specifies how hard the partitioning algorithm should work:
-strict
Disable non-standard macros. This option registers the strict feature identifier. Implies -strict-letrec and -strict-reader.
-strict-srfi-0
Disable non-standard macros except cond-expand. This option registers the strict feature identifier. Implies -strict-letrec and -strict-reader.
-strict-reader
Disables non-standard read syntax. Implies -case-insensitive.
-strict-letrec
Enable fully R5RS compliant letrec. This generates slightly less efficient code but preserves standard semantics.
-to-stdout
Write compiled code to standard output instead of creating a .c file.
-unit NAME
Compile this file as a library unit. Equivalent to
     
     -prelude "(declare (unit NAME))"

-unsafe
Disable runtime safety checks.
-unsafe-libraries
Marks the generated file for being linked with the unsafe runtime system. This should be used when generating shared object files that are to be loaded dynamically. If the marker is present, any attempt to load code compiled with this option will signal an error.
-uses NAME
Use definitions from the library unit NAME. This is equivalent to
     
     -prelude "(declare (uses NAME))"

-usual-integrations
Specifies that standard procedures and certain internal procedures are never redefined, and can be inlined. This is equivalent to declaring (usual-integrations).
-version
Prints the version and some copyright information and exit the compiler.
-verbose
Prints progress information to standard output during compilation.

The environment variable CHICKEN_OPTIONS can be set to a string with default command-line options for the compiler.


Next: , Previous: Compiler command line format, Up: Using the compiler

3.2 Runtime options

After successful compilation a C source file is generated and can be compiled with a C compiler. Executables generated with CHICKEN (and the compiler itself) accept a small set of runtime options:

-:?
Shows a list of the available runtime options and exits the program.
-:c
Forces console mode. Currently this is only used in the interpreter (csi) to force output of the #;> prompt even if stdin is not a terminal (for example if running in an emacs buffer under Windows).
-:d
Prints some debug-information during startup.
-:hNUMBER
Specifies fixed heap size
-:hiNUMBER
Specifies the initial heap size
-:hgPERCENTAGE
Sets the growth rate of the heap in percent. If the heap is exhausted, then it will grow by PERCENTAGE. The default is 200.
-:hmNUMBER
Specifies a maximal heap size. The default is (2GB - 15).
-:hsPERCENTAGE
Sets the shrink rate of the heap in percent. If no more than a quarter of PERCENTAGE of the heap is used, then it will shrink to PERCENTAGE. The default is 50. Note: If you want to make sure that the heap never shrinks, specify a value of 0. (this can be useful in situations where an optimal heap-size is known in advance).
-:o
Disables detection of stack overflows at run-time
-:sNUMBER
Specifies stack size
-:tNUMBER
Specifies symbol table size
-:w
Enables garbage collection of unused symbols. By default unused and unbound symbols are not garbage collected.
-:r
Writes trace output to stdout. This option has no effect with in files compiled with the -no-trace or -debug-level 0 options.

The argument values may be given in bytes, in kilobytes (suffixed with K or k), in megabytes (suffixed with M or m), or in gigabytes (suffixed with G or g). Runtime options may be combined, like -:dc, but everything following a NUMBER argument is ignored. So -:wh64m is OK, but -:h64mw will not enable GC of unused symbols.


Next: , Previous: Runtime options, Up: Using the compiler

3.3 An example

To compile a Scheme program (assuming a UNIX-like environment) we perform the following steps:

If multiple bodies of Scheme code are to be combined into a single executable, then we have to compile each file and link the resulting object files together with the runtime system:

The declarations specify which of the compiled files is the main module, and which is the library module. An executable can only have one main module, since a program has only a single entry-point. In this case foo.scm is the main module, because it doesn't have a unit declaration.

Extensions to the basic CHICKEN runtime libraries are available in a separate utility library (libsrfi-chicken.[a|so] and libstuffed-chicken.[a|so] on UNIX-like platforms, libsrfi-chicken.lib and libstuffed-chicken.lib on Windows systems). Whenever you use one or more of the units format, srfi-1, srfi-4, srfi-13, srfi-14, srfi-18, srfi-25, srfi-37, posix, utils, lolevel, tinyclos or regex, then you should add these library to the command line of the C compiler or linker. The compiler driver csc and the helper script chicken-config will do this automatically.


Next: , Previous: An example, Up: Using the compiler

3.4 Extending the compiler

The compiler supplies a couple of hooks to add user-level passes to the compilation process. Before compilation commences any Scheme source files or compiled code specified using the -extend option are loaded and evaluated. The parameters user-options-pass, user-read-pass, user-preprocessor-pass, user-pass, user-pass-2 and user-post-analysis-pass can be set to procedures that are called to perform certain compilation passes instead of the usual processing (for more information about parameters see: Parameters.

— parameter: user-options-pass

Holds a procedure that will be called with a list of command-line arguments and should return two values: the source filename and the actual list of options, where compiler switches have their leading - (hyphen) removed and are converted to symbols. Note that this parameter is invoked before processing of the -extend option, and so can only be changed in compiled user passes.

— parameter: user-read-pass

Holds a procedure of three arguments. The first argument is a list of strings with the code passed to the compiler via -prelude options. The second argument is a list of source files including any files specified by -prologue and -epilogue. The third argument is a list of strings specified using -postlude options. The procedure should return a list of toplevel Scheme expressions.

— parameter: user-preprocessor-pass

Holds a procedure of one argument. This procedure is applied to each toplevel expression in the source file before macro-expansion. The result is macro-expanded and compiled in place of the original expression.

— parameter: user-pass

Holds a procedure of one argument. This procedure is applied to each toplevel expression after macro-expansion. The result of the procedure is then compiled in place of the original expression.

— parameter: user-pass-2

Holds a procedure of three arguments, which is called with the canonicalized node-graph and the analysis database accessors as arguments (see below for an explanation of the accessor arguments). The result is ignored, so this pass has to mutate the node-structure to cause any effect.

— parameter: user-post-analysis-pass

Holds a procedure that will be called after the last performed program analysis. The procedure (when defined) will be called with three arguments: the program database, a getter and a setter-procedure which can be used to access and manipulate the program database, which holds various information about the compiled program. The getter procedure should be called with two arguments: a symbol representing the binding for which information should be retrieved, and a symbol that specifies the database-entry. The current value of the database entry will be returned or #f, if no such entry is available. The setter procedure is called with three arguments: the symbol and key and the new value.

For information about the contents of the program database contact the author.

Loaded code (via the -extend option) has access to the library units extras, srfi-1, srfi-4, utils, regex and the pattern matching macros. The highlevel macro-system and multithreading is not available.

Note that the macroexpansion/canonicalization phase of the compiler adds certain forms to the source program. These extra expressions are not seen by user-preprocessor-pass but by user-pass.


Previous: Extending the compiler, Up: Using the compiler

3.5 Distributing compiled C files

It is relatively easy to create distributions of Scheme projects that have been compiled to C. The runtime system of CHICKEN consists of only two handcoded C files (runtime.c and chicken.h), plus the file chicken-config.h, which is generated by the build process. All other modules of the runtime system and the extension libraries are just compiled Scheme code. For more information, study the CHICKEN source code and/or get in contact with the author.


Next: , Previous: Using the compiler, Up: Top

4 Using the interpreter

CHICKEN provides an interpreter named csi for evaluating Scheme programs and expressions.


Next: , Up: Using the interpreter

4.1 Command line format

  csi {FILENAME|OPTION}

where FILENAME specifies a file with Scheme source-code. If the extension of the source file is .scm, it may be omitted. The runtime options described in Compiler command line format are also available for the interpreter. If the environment variable CSI_OPTIONS is set to a list of options, then these options are additionally passed to every direct or indirect invocation of csi. Please note that runtime options (like -:...) can not be passed using this method. The options recognized by the interpreter are:

--
Ignore everything on the command-line following this marker. Runtime options (“-:...”) are still recognized.
-case-insensitive
Enables the reader to read symbols case insensitive. The default is to read case sensitive (in violation of R5RS). This option registers the case-insensitive feature identifier.
-batch
Quit the interpreter after processing all command line options.
-eval EXPRESSIONS
Evaluate EXPRESSIONS.
-feature SYMBOL
Registers SYMBOL to be a valid feature identifier for cond-expand.
-help
Write a summary of the available command line options to standard output and exit.
-syntax
-hygienic
Load syntax-case macro package and enable high-level macros in interpreted code. This option registers the hygienic-macros feature identifier.
-include-path PATHNAME
Specifies an alternative search-path for files included via the include special form. This option may be given multiple times. If the environment variable CHICKEN_INCLUDE_PATH is set, it should contain a list of alternative include pathnames separated by “;”. The environment variable CHICKEN_HOME is also considered as a search path.
-keyword-style STYLE
Enables alternative keyword syntax, where STYLE may be either prefix (as in Common Lisp) or suffix (as in DSSSL). Any other value is ignored.
-no-feature SYMBOL
Unregisters feature identifier SYMBOL.
-no-init
Do not load initialization-file. If this option is not given and the file ~/.csirc exists, then it is loaded before the read-eval-print loop commences.
-no-warnings
Disables any warnings that might be issued by the reader or evaluated code.
-quiet
Do not print a startup message.
-r5rs
Equivalent to -hygienic -strict.
-script PATHNAME
This is equivalent to -batch -quiet -no-init PATHNAME. Arguments following PATHNAME are available by using command-line-arguments and are not processed as interpreter options.
-script-meta PATHNAME
This is similar to the -script option, but the file specified by PATHNAME is opened, the first line is treated as containing additional command line options and (after processing all the options) the code from the given file is loaded.
-strict
Disable non-standard macros. Implies -strict-letrec and -strict-reader.
-strict-srfi-0
Disable non-standard macros except cond-expand. Implies -strict-letrec and -strict-reader.
-strict-reader
Disables non-standard read syntax. Implies -case-insensitive.
-strict-letrec
Enable fully R5RS compliant letrec. This generates slightly less efficient code but preserves standard semantics.
-version
Write the banner with version information to standard output and exit.


Next: , Previous: Interpreter command line format, Up: Using the interpreter

4.2 Writing Scheme scripts


Next: , Previous: Writing Scheme scripts, Up: Using the interpreter

4.3 Toplevel commands

The toplevel loop understands a number of special commands:

,?
Show summary of available toplevel commands.
,NUM
Evaluate NUMth command in history-list
,h
List history of last commands
,l FILENAME
Load file with given FILENAME (may be a symbol or string).
,ln FILENAME
Load file and print result(s) of each top-level expression.
,p EXP
Pretty-print evaluated expression EXP.
,d EXP
Describe result of evaluated expression EXP.
,du EXP
Dump contents of the result of evaluated expression EXP.
,dur EXP N
Dump N bytes of the result of evaluated expression EXP.
,q
Quit the interpreter.
,r
Show system information.
,s STRING-OR-SYMBOL
Execute shell-command.
,t EXP
Evaluate form and print elapsed time.
,x EXP
Pretty-print macroexpanded expression EXP (the expression is not evaluated).


Previous: Toplevel commands, Up: Using the interpreter

4.4 Macros and procedures implemented in the interpreter

Additional macros and procedures available in the interpreter are:

— syntax: advise
          (advise NAME MODE PROC)
     

Modifies the behavior of the procedures named NAME, according to MODE:

before
Call the procedure PROC before every invocation of NAME, with the same arguments.
after
Call the procedure PROC with the result value(s) of NAME.
around
Call the procedure PROC with the arguments passed to NAME. Additionally the (original) value of NAME is passed as the first argument to PROC.

Only the PROC argument is evaluated. Note that multiple pieces of advice on the same procedure are allowed.

     
     #;> (define (fac n)
           (if (zero? n) 1 (* n (fac (sub1 n)))))
     #;> (define count 0)
     #;> (advise fac before (lambda _ (set! count (add1 count))))
     #;> (fac 10)                           ==> 3628800
     #;> count                              ==> 11
     #;> (advise fac around
           (let ((i 0))
             (define (indent)
               (do ((i i (sub1 i)))
                   ((zero? i))
                 (write-char #\space)))
           (lambda (f n)
             (indent)
             (print "fac: " n)
             (set! i (add1 i))
             (let ((x (f n)))
               (set! i (sub1 i))
               (indent)
               (print "-> " x)
               x))))
     #;> (fac 3)
     fac: 3
      fac: 2
       fac: 1
        fac: 0
        -> 1
       -> 1
      -> 2
     -> 6                                   ==> 6
     #;> count                              ==> 15
     #;> (set! count 0)
     #;> (unadvise fac)
     #;> (fac 10)                           ==> 3628800
     #;> count                              ==> 0
— syntax: unadvise
          (unadvise NAME ...)
     

Removes all pieces of advice from the procedures NAME ... and restores their original behavior.

— syntax: trace
          (trace NAME ...)
     

Switches tracing on for the procedures with the given names.

     
     #;> (fac 10)                       ==> 3628800
     #;> (trace fac)
     #;> (fac 3)
     |(fac 3)
     | (fac 2)
     |  (fac 1)
     |   (fac 0)
     |   fac -> 1 
     |  fac -> 1 
     | fac -> 2 
     |fac -> 6                          ==> 6
     #;> (untrace fac)
     #;> (fac 3)                        ==> 6
— syntax: untrace
          (untrace NAME ...)
     

Switches tracing of the given procedures off.

— procedure: $
          ($ [INDEX])
     

Returns entry number INDEX in the history list.

— procedure: &
          (& [INDEX])
     

Returns result(s) of entry number INDEX in the history list.


Next: , Previous: Using the interpreter, Up: Top

5 Supported language


Next: , Up: Supported language

5.1 Deviations from the standard

[2] Identifiers are by default case-sensitive.

[4.1.4] Extended DSSSL style lambda lists are supported. DSSSL formal argument lists are defined by the following grammar:

<formal-argument-list> ==> <required-formal-argument>*
                           [(#!optional <optional-formal-argument>*)]
                           [(#!rest <rest-formal-argument>)]
                           [(#!key <key-formal-argument>*)]
<required-formal-argument> ==> <ident>
<optional-formal-argument> ==> <ident>
                             | (<ident> <initializer>)
<rest-formal-argument> ==> <ident>
<key-formal-argument> ==> <ident>
                          | (<ident> <initializer>)
<initializer> ==> <expr>

When a procedure is applied to a list of actual arguments, the formal and actual arguments are processed from left to right as follows:

  1. Variables in required-formal-arguments are bound to successive actual arguments starting with the first actual argument. It shall be an error if there are fewer actual arguments than required-formal-arguments.
  2. Next, variables in optional-formal-arguments are bound to any remaining actual arguments. If there are fewer remaining actual arguments than optional-formal-arguments, then variables are bound to the result of the evaluation of initializer, if one was specified, and otherwise to #f. The initializer is evaluated in an environment in which all previous formal arguments have been bound.
  3. If there is a rest-formal-argument, then it is bound to a list of all remaining actual arguments. The remaining actual arguments are also eligible to be bound to keyword-formal-arguments. If there is no rest-formal-argument and there are no keyword-formal-arguments, the it shall be an error if there are any remaining actual arguments.
  4. If #!key was specified in the formal-argument-list, there shall be an even number of remaining actual arguments. These are interpreted as a series of pairs, where the first member of each pair is a keyword specifying the argument name, and the second is the corresponding value. It shall be an error if the first member of a pair is not a keyword. It shall be an error if the argument name is not the same as a variable in a keyword-formal-argument, unless there is a rest-formal-argument. If the same argument name occurs more than once in the list of actual arguments, then the first value is used. If there is no actual argument for a particular keyword-formal-argument, then the variable is bound to the result of evaluating initializer if one was specified, and otherwise #f. The initializer is evaluated in an environment in which all previous formal arguments have been bound.

It shall be an error for an <ident> to appear more than once in a formal-argument-list.

Example:

((lambda (x y) x) 3 4 5 6)   =>(3 4 5 6)
((lambda (x y #!rest z) z)
 3 4 5 6)                    => (5 6)
((lambda (x y #!optional z #!rest r #!key i (j 1)) 
    (list x y z i: i j: j))
 3 4 5 i: 6 i: 7)            => (3 4 5 i: 6 j: 1)

[4.1.6] set! for unbound toplevel variables is allowed.

[5.2] define with a single argument is allowed and initializes the toplevel or local binding to an unspecified value.

[6.2.4] The runtime system uses the numerical string-conversion routines of the underlying C library and so does only understand standard (C-library) syntax for floating-point constants.

[6.2.5] The routines complex?, real? and rational? are identical to the standard procedure number?. The procedures numerator, denominator and rationalize are not implemented. Also not implemented are all procedures related to complex numbers.

[6.2.6] The procedure string->number does not obey read/write invariance on inexact numbers.

[6.5] Code evaluated in scheme-report-environment or null-environment still sees non-standard syntax unless running under the interpreter (csi) invoked with the -strict option.

[6.6.2] The procedure char-ready? is handling terminal input ports only under DJGPP correctly. On other platforms it returns always #t. The procedure read does not obey read/write invariance on inexact numbers.

[6.6.3] The procedures write and display do not obey read/write invariance to inexact numbers.


Next: , Previous: Deviations from the standard, Up: Supported language

5.2 Extensions to the standard

[2.1] Identifiers may contain special characters if delimited with | ... |.

[2.3] The brackets [ ... ] are provided as an alternative syntax for ( ... ). A number of reader extensions is provided. See Non standard read syntax.

[4] Numerous non-standard macros are provided. See Non-standard macros and special forms for more information.

[4.2.2] It is allowed for initialization values of bindings in a letrec construct to refer to previous variables in the same set of bindings, so

(letrec ([foo 123]
         [bar foo] )
  bar)

is allowed and returns 123. This extension is not available when strict R5RS letrec semantics have been selected (by using the -strict, -strict-srfi-0 or -strict-letrec option).

[4.2.3] (begin) is allowed in non-toplevel contexts and evaluates to an unspecified value.

[4.2.5] Delayed expressions may return multiple values.

[5.2.2] CHICKEN extends standard semantics by allowing internal definitions everywhere, and not only at the beginning of a body. A set of internal definitions is equivalent to a letrec form enclosing all following expressions in the body:

(let ([foo 123])
  (bar)
  (define foo 456)
  (baz foo) )

expands into

(let ([foo 123])
  (bar)
  (letrec ([foo 456])
    (baz foo) ) )

This extension to the standard semantics is not available in combination with the hygienic (syntax-case) macro system. Under the hygienic macro system an error will be signalled when internal definitions occur at a position that is not at the beginning of a body.

[6] CHICKEN provides numerous non-standard procedures. See the manual sections on library units for more information.

[6.3.4] User defined character names are supported. See char-name in User-defined named characters.

[6.3.5] CHICKEN supports special characters preceded with a backslash “\” in quoted string constants. “\n” denotes the newline-character, “\r” carriage return, “\b” backspace, “\t” TAB and “\xXX” a character with the code XX in hex.

The third argument to substring is optional and defaults to the length of the string.

[6.4] force called with an argument that is not a promise returns that object unchanged. Captured continuations can be safely invoked inside before- and after-thunks of a dynamic-wind form and execute in the outer dynamic context of the dynamic-wind form.

[6.5] The second argument to eval is optional and defaults to the value of (interaction-environment). scheme-report-environment and null-environment accept an optional 2nd parameter: if not #f (which is the default), toplevel bindings to standard procedures are mutable and new toplevel bindings may be introduced.

[6.6.1] if the procedures current-input-port and current-output-port are called with an argument (which should be a port), then that argument is selected as the new current input- and output-port, respectively. The procedures open-input-file, open-output-file, with-input-from-file, with-output-to-file, call-with-input-file and call-with-output-file accept an optional second (or third) argument which should be one or more keywords, if supplied. These arguments specify the mode in which the file is opened. Possible values are the keywords #:text, #:binary or #:append.


Next: , Previous: Extensions to the standard, Up: Supported language

5.3 Non standard read syntax

— read syntax: #| ... |#

A multiline “block” comment. May be nested. Implements SRFI-30)

— read syntax: #;EXPRESSION

Treats EXPRESSION as a comment.

— read syntax: #,(CONSTRUCTORNAME DATUM ...)

Allows user-defined extension of external representations. (For more information see the documentation for SRFI-10)

— read syntax: #'EXPRESSION

An abbreviation for (syntax EXPRESSION).

— read syntax: #$EXPRESSION

An abbreviation for (location EXPRESSION).

— read syntax: #:SYMBOL

Syntax for keywords. Keywords are symbols that evaluate to themselves, and as such don't have to be quoted.

— read syntax: #<<TAG

Specifies a multiline string constant. Anything up to a line equal to TAG will be returned as a single string:

(define msg #<<END
"Hello, world!", she said. 
END
)

is equivalent to

     
     (define msg "\"Hello, world!\", she said.")

— read syntax: #<#TAG

Similar to #<<, but allows substitution of embedded Scheme expressions prefixed with # and optionally enclosed in { ... }. Two consecutive #s are translated to a single #:

     
     (define three 3)
     (display #<#EOF
     This is a simple string with an embedded `##' character
     and substituted expressions: (+ three 99) ==> #(+ three 99)
     (three is "#{three}")
     EOF
     )

prints

     
     This is a simple string with an embedded `#' character
     and substituted expressions: (+ three 99) ==> 102
     (three is "3")

— read syntax: #> ... <#

Abbreviation for (declare (foreign-declare " ... ")).

— read syntax: #>? ... <#

Abbreviation for (declare (foreign-parse " ... ")).

— read syntax: #>! ... <#

Abbreviation for

     
     (declare
       (foreign-declare " ... ")
       (foreign-parse " ... ") )

— read syntax: #%...

Reads like a normal symbol.

— read syntax: #!...

If occurring in the first line of an interpreted or compiled source file, then the #! and everything following it are ignored. If occurring anywhere else, reads as a normal symbol. The special (self-evaluating) symbol #!eof is read as the end-of-file object.


Next: , Previous: Non standard read syntax, Up: Supported language

5.4 Non-standard macros and special forms


Next: , Up: Non-standard macros and special forms

5.4.1 Making extra libraries and extensionsd available

— syntax: require-extension
— syntax: use
          (require-extension ID ...)
          (use ID ...)
     

This form does all necessary steps to make the libraries or extensions given in ID ... available. It loads syntactic extension, if needed and generates code for loading/linking with core library modules or separately installed extensions. use is just a shorter alias for require-extension.

During interpretation/evaluation require-extension performs one of the following:

During compilation one of the following happens instead:

To make long matters short - just use require-extension and it will normally figure everything out for dynamically loadable extensions and core library units.


Next: , Previous: Making extra libraries and extensionsd available, Up: Non-standard macros and special forms

5.4.2 Binding forms for optional arguments

— syntax: :optional
          (:optional ARGS DEFAULT)
     

Use this form for procedures that take a single optional argument. If ARGS is the empty list DEFAULT is evaluated and returned, otherwise the first element of the list ARGS. It is an error if ARGS contains more than one value.

     
     (define (incr x . i) (+ x (:optional i 1)))
     (incr 10)                                   ==> 11
     (incr 12 5)                                 ==> 17
— syntax: case-lambda
          (case-lambda (LAMBDA-LIST1 EXP1 ...) ...)
     

SRFI-16. Expands into a lambda that invokes the body following the first matching lambda-list.

     
     (define plus
       (case-lambda 
         (() 0)
         ((x) x)
         ((x y) (+ x y))
         ((x y z) (+ (+ x y) z))
         (args (apply + args))))
     
     (plus)                      ==> 9
     (plus 1)                    ==> 1
     (plus 1 2 3)                ==> 6

For more information see the documentation for SRFI-16

— syntax: let-optionals
— syntax: let-optionals*
          (let-optionals ARGS ((VAR1 DEFAULT1) ...) BODY ...)
          (let-optionals* ARGS ((VAR1 DEFAULT1) ... [RESTVAR]) BODY ...)
     

Binding constructs for optional procedure arguments. ARGS should be a rest-parameter taken from a lambda-list. let-optionals binds VAR1 ... to available arguments in parallel, or to DEFAULT1 ... if not enough arguments were provided. let-optionals* binds VAR1 ... sequentially, so every variable sees the previous ones. If a single variable RESTVAR is given, then it is bound to any remaining arguments, otherwise it is an error if any excess arguments are provided.

     
     (let-optionals '(one two) ((a 1) (b 2) (c 3))
       (list a b c) )                               ==> (one two 3)
     (let-optionals* '(one two) ((a 1) (b 2) (c a))
       (list a b c) )                               ==> (one two one)


Next: , Previous: Binding forms for optional arguments, Up: Non-standard macros and special forms

5.4.3 Other binding forms

— syntax: and-let*
          (and-let* (BINDING ...) EXP1 EXP2 ...)
     

SRFI-2. Bind sequentially and execute body. BINDING can be a list of a variable and an expression, a list with a single expression, or a single variable. If the value of an expression bound to a variable is #f, the and-let* form evaluates to #f (and the subsequent bindings and the body are not executed). Otherwise the next binding is performed. If all bindings/expressions evaluate to a true result, the body is executed normally and the result of the last expression is the result of the and-let* form. See also the documentation for SRFI-2.

— syntax: cut
— syntax: cute
          (cut SLOT ...)
          (cute SLOT ...)
     

Syntactic sugar for specializing parameters.

— syntax: define-values
          (define-values (NAME ...) EXP)
     

Defines several variables at once, with the result values of expression EXP.

— syntax: fluid-let
          (fluid-let ((VAR1 X1) ...) BODY ...)
     

Binds the variables VAR1 ... dynamically to the values X1 ... during execution of BODY ....

— syntax: let-values
          (let-values (((NAME ...) EXP) ...) BODY ...)
     

Binds multiple variables to the result values of EXP .... All variables are bound simultaneously.

— syntax: let*-values
          (let*-values (((NAME ...) EXP) ...) BODY ...)
     

Binds multiple variables to the result values of EXP .... The variables are bound sequentially.

     
     (let*-values (((a b) (values 2 3))
                   ((p) (+ a b)) )
       p)                               ==> 5
— syntax: letrec-values
          (letrec-values (((NAME ...) EXP) ...) BODY ...)
     

Binds the result values of EXP ... to multiple variables at once. All variables are mutually recursive.

     
     (letrec-values (((odd even)
                        (values 
                          (lambda (n) (if (zero? n) #f (even (sub1 n))))
                          (lambda (n) (if (zero? n) #t (odd (sub1 n)))) ) ) )
       (odd 17) )                           ==> #t
— syntax: parameterize
          (parameterize ((PARAMETER1 X1) ...) BODY ...)
     

Binds the parameters PARAMETER1 ... dynamically to the values X1 ... during execution of BODY .... (see also: make-parameter in Parameters). Note that PARAMETER may be any expression that evaluates to a parameter procedure.

— syntax: receive
          (receive (NAME1 ... [. NAMEn]) VALUEEXP BODY ...)
          (receive VALUEEXP)
     

SRFI-8. Syntactic sugar for call-with-values. Binds variables to the result values of VALUEEXP and evaluates BODY ....

The syntax

     
     (receive VALUEEXP)

is equivalent to

     
     (receive _ VALUEEXP _)
— syntax: set!-values
          (set!-values (NAME ...) EXP)
     

Assigns the result values of expression EXP to multiple variables.


Next: , Previous: Other binding forms, Up: Non-standard macros and special forms

5.4.4 Substitution forms and macros

— syntax: define-constant
          (define-constant NAME CONST)
     

Define a variable with a constant value, evaluated at compile-time. Any reference to such a constant should appear textually after its definition. This construct is equivalent to define when evaluated or interpreted. Constant definitions should only appear at toplevel. Note that constants are local to the current compilation unit and are not available outside of the source file in which they are defined. Names of constants still exist in the Scheme namespace and can be lexically shadowed. If the value is mutable, then the compiler is careful to preserve its identity. CONST may be any constant expression, and may also refer to constants defined via define-constant previously. This for should only be used at top-level.

— syntax: define-inline
          (define-inline (NAME VAR ... [. VAR]) BODY ...)
          (define-inline NAME EXP)
     

Defines an inline procedure. Any occurrence of NAME will be replaced by EXP or (lambda (VAR ... [. VAR]) BODY ...). This is similar to a macro, but variable-names and -scope will be correctly handled. Inline substitutions take place after macro-expansion. EXP should be a lambda-expression. Any reference to NAME should appear textually after its definition. Note that inline procedures are local to the current compilation unit and are not available outside of the source file in which they are defined. Names of inline procedures still exist in the Scheme namespace and can be lexically shadowed. This construct is equivalent to define when evaluated or interpreted. Inline definitions should only appear at toplevel.

— syntax: define-macro
          (define-macro (NAME VAR ... [. VAR]) EXP1 ...)
          (define-macro NAME (lambda (VAR ... [. VAR]) EXP1 ...))
          (define-macro NAME1 NAME2)
     

Define a globally visible macro special form. The macro is available as soon as it is defined, i.e. it is registered at compile-time. If the file containing this definition invokes eval and the declaration run-time-macros (or the command line option -run-time-macros) has been used, then the macro is visible in evaluated expressions during runtime. The second possible syntax for define-macro is allowed for portability purposes only. In this case the second argument must be a lambda-expression or a macro name. Only global macros can be defined using this form. (define-macro NAME1 NAME2) simply copies the macro definition from NAME2 to NAME1, creating an alias.

This form is also available with the syntax-case macro system.


Next: , Previous: Substitution forms and macros, Up: Non-standard macros and special forms

5.4.5 Conditional forms

— syntax: switch
          (switch EXP (KEY EXP1 ...) ... [(else EXPn ...)])
     

This is similar to case, but a) only a single key is allowed, and b) the key is evaluated.

— syntax: unless
          (unless TEST EXP1 EXP2 ...)
     

Equivalent to:

     
     (if (not TEST) (begin EXP1 EXP2 ...))
— syntax: when
          (when TEST EXP1 EXP2 ...)
     

Equivalent to:

     
     (if TEST (begin EXP1 EXP2 ...))


Next: , Previous: Conditional forms, Up: Non-standard macros and special forms

5.4.6 Record structures

— syntax: define-record
          (define-record NAME SLOTNAME ...)
     

Defines a record type. Call make-NAME to create an instance of the structure (with one initialization-argument for each slot). (NAME? STRUCT) tests any object for being an instance of this structure. Slots are accessed via (NAME-SLOTNAME STRUCT) and updated using (NAME-SLOTNAME-set! STRUCT VALUE).

     
     (define-record point x y)
     (define p1 (make-point 123 456))
     (point? p1)                      ==> #t
     (point-x p1)                     ==> 123
     (point-y-set! p1 99)
     (point-y p1)                     ==> 99
— syntax: define-record-printer
          (define-record-printer (NAME RECORDVAR PORTVAR) BODY ...)
          (define-record-printer NAME PROCEDURE)
     

Defines a printing method for record of the type NAME by associating a procedure with the record type. When a record of this type is written using display, write or print, then the procedure is called with two arguments: the record to be printed and an output-port.

     
     (define-record foo x y z)
     (define f (make-foo 1 2 3))
     (define-record-printer (foo x out)
       (fprintf out "#,(foo ~S ~S ~S)"
                (foo-x x) (foo-y x) (foo-z x)) )
     (define-reader-ctor 'foo make-foo)
     (define s (with-output-to-string
                   (lambda () (write f))))
     s                                   ==> "#,(foo 1 2 3)"
     (equal? f (with-input-from-string
                   s read)))             ==> #t

define-record-printer works also with SRFI-9 record types.

— syntax: define-record-type
          (define-record-type NAME (CONSTRUCTOR TAG ...) PREDICATE
                              (FIELD ACCESSOR [MODIFIER]) ...)
     

SRFI-9 record types. For more information see the documentation for SRFI-9


Previous: Record structures, Up: Non-standard macros and special forms

5.4.7 Other forms

— syntax: assert
          (assert EXP [STRING ARG ...])
     

Signal error if EXP evaluates to false. An optional message STRING and arguments ARG ... may be supplied to give a more informative error-message. If compiled in unsafe mode (either by specifying the -unsafe compiler option or by declaring (unsafe)), then this expression expands to an unspecified value.

— syntax: cond-expand
          (cond-expand FEATURE-CLAUSE ...)
     

SRFI-0. Expands by selecting feature clauses. Predefined feature-identifiers are srfi-0, srfi-2, srfi-6, srfi-8, srfi-9, srfi-10, and chicken. In strict-srfi-0 mode only srfi-0 and chicken are defined. If the source file containing this form is currently compiled, the feature compiling is defined. For further information, see the documentation for SRFI-0 This form is allowed to appear in non-toplevel expressions.

— syntax: critical-section
          (critical-section BODY ...)
     

Evaluate BODY ... with timer-interrupts temporarily disabled.

— syntax: ensure
          (ensure PREDICATE EXP [ARGUMENTS ...])
     

Evaluates the expression EXP and applies the one-argument procedure PREDICATE to the result. If the predicate returns #f an error is signaled, otherwise the result of EXP is returned. If compiled in unsafe mode (either by specifying the -unsafe compiler option or by declaring (unsafe)), then this expression expands to an unspecified value. If specified, the optional ARGUMENTS are used as arguments to the invocation of the error-signalling code, as in (error ARGUMENTS ...). If no ARGUMENTS are given, a generic error message is displayed with the offending value and PREDICATE expression.

— syntax: eval-when
          (eval-when (SITUATION ...) EXP ...)
     

Controls evaluation/compilation of subforms. SITUATION should be one of the symbols eval, compile or load. When encountered in the evaluator, and the situation specifier eval is not given, then this form is not evaluated and an unspecified value is returned. When encountered while compiling code, and the situation specifier compile is given, then this form is evaluated at compile-time. When encountered while compiling code, and the situation specifier load is not given, then this form is ignored and an expression resulting into an unspecified value is compiled instead.

The following table should make this clearer:

in compiled code In interpreted code
eval ignore evaluate
compile evaluate at compile time ignore
load compile as normal ignore

Note: It is currently not possible to use define-syntax or define inside eval-when forms when hygienic macros are enabled.

— syntax: include
          (include STRING)
     

Include toplevel-expressions from the given source file in the currently compiled/interpreted program. If the included file has the extension .scm, then it may be omitted. The file is searched in the current directory and, if not found, in all directories specified in the -include-path option.

— syntax: nth-value
          (nth-value N EXP)
     

Returns the Nth value (counting from zero) of the values returned by expression EXP.

— syntax: time
          (time EXP1 ...)
     

Evaluates EXP1 ... and print elapsed time and memory information. The result of the last expression is returned.


Next: , Previous: Non-standard macros and special forms, Up: Supported language

5.5 Declarations

— syntax: declare
          (declare DECLSPEC ...)
     

Process declaration specifiers. Declarations always override any command-line settings. Declarations are valid for the whole compilation-unit (source file), the position of the declaration in the source file can be arbitrary. DECLSPEC may be any of the following:

— declaration specifier: always-bound
               (always-bound SYMBOL ...)
          

Declares that the given variables are always bound and accesses to those have not to be checked.

— declaration specifier: block
               (block)
          

Assume global variables are never redefined. This is the same as specifying the -block option.

— declaration specifier: block-global
— declaration specifier: hide
               (block-global SYMBOL ...)
               (hide SYMBOL ...)
          

Declares that the toplevel bindings for SYMBOL ... should not be accessible from code in other compilation units or by eval. Access to toplevel bindings declared as block global is also more efficient.

— declaration specifier: bound-to-procedure
               (bound-to-procedure SYMBOL ...)
          

Declares that the given identifiers are always bound to procedure values.

— declaration specifier: compress-literals
               (compress-literals [THRESHOLD [INITIALIZER]])
          

The same as the -compress-literals compiler option. The threshold argument defaults to 50. If the optional argument INITIALIZER is given, then the literals will not be created at module startup, but when the procedure with this name will be called.

— declaration specifier: export
               (export SYMBOL ...)
          

The opposite of hide. All given identifiers will be exported and all toplevel variables not listed will be hidden and not be accessible outside of this compilation unit. When the hygienic (syntax-case) macro system is used, the exported identifier may also have the form (MODULE-NAME SYMBOL ...), which specifies identifiers to be exported from a module as (undecorated) toplevel variables.

— declaration specifier: foreign-declare
               (foreign-declare STRING ...)
          

Include given strings verbatim into header of generated file.

— declaration specifier: foreign-parse
               (foreign-parse STRING ...)
          

Parse given strings and generate foreign-interface bindings. See The Easy Foreign Function Interface for more information.

— declaration specifier: interrupts-enabled
               (interrupts-enabled)
          

Enable timer-interrupts checks in the compiled program (the default).

— declaration specifier: disable-interrupts
— declaration specifier: not
               (disable-interrupts)
               (not interrupts-enabled)
          

Disable timer-interrupts checks in the compiled program. Threads can not be preempted in main- or library-units that contain this declaration.

— declaration specifier: no-argc-checks
               (no-argc-checks)
          

Disables argument count checking.

— declaration specifier: no-bound-checks
               (no-bound-checks)
          

Disables the bound-checking of toplevel bindings.

— declaration specifier: no-procedure-checks
               (no-procedure-checks)
          

Disables checking of values in operator position for being of procedure type.

— declaration specifier: TYPE
— declaration specifier: fixnum-arithmetic
               ([number-type] TYPE)
               (fixnum-arithmetic)
          

Declares that only numbers of the given type are used. TYPE may be fixnum or generic (which is the default).

— declaration specifier: run-time-macros
               (run-time-macros)
          

Equivalent to the compiler option of the same name - low-level macros defined in the compiled code are also made available at runtime.

— declaration specifier: standard-bindings
               ([not] standard-bindings SYMBOL ...)
          

Declares that all given standard procedures (or all if no symbols are specified) are never globally redefined. If not is specified, then all but the given standard bindings are assumed to be never redefined.

— declaration specifier: extended-bindings
               ([not] extended-bindings SYMBOL ...)
          

Declares that all given non-standard and CHICKEN-specific procedures (or all if no symbols are specified) are never globally redefined. If not is specified, then all but the given extended bindings are assumed to be never redefined.

— declaration specifier: usual-integrations
               ([not] usual-integrations SYMBOL ...)
          

Declares that all given standard and extended bindings (or all if no symbols are specified) are never globally redefined. If not is specified, then all but the given standard and extended bindings are assumed to be never redefined.

— declaration specifier: unit
               (unit SYMBOL)
          

Specify compilation unit-name (if this is a library)

— declaration specifier: unsafe
— declaration specifier: not
               (unsafe)
               (not safe)
          

Do not generate safety-checks. This is the same as specifying the -unsafe option. Also implies

          
          (declare (no-bound-checks) (no-procedure-checks) (no-argc-checks))
     

— declaration specifier: uses
               (uses SYMBOL ...)
          

Gives a list of used library-units. Before the toplevel-expressions of the main-module are executed, all used units evaluate their toplevel-expressions in the order in which they appear in this declaration. If a library unit A uses another unit B, then B's toplevel expressions are evaluated before A's. Furthermore, the used symbols are registered as features during compile-time, so cond-expand knows about them.


Next: , Previous: Declarations, Up: Supported language

5.6 Parameters

Certain behavior of the interpreter and compiled programs can be customized via 'parameters', where a parameter is a procedure of zero or one arguments. To retrieve the value of a parameter call the parameter-procedure with zero arguments. To change the setting of the parameter, call the parameter-procedure with the new value as argument:

(define foo (make-parameter 123))
(foo)                             ==> 123
(foo 99)
(foo)                             ==> 99

Parameters are fully thread-local, each thread of execution owns a local copy of a parameters' value.

CHICKEN implements SRFI-39

— procedure: make-parameter
          (make-parameter VALUE [GUARD])
     

Returns a procedure that accepts zero or one argument. Invoking the procedure with zero arguments returns VALUE. Invoking the procedure with one argument changes its value to the value of that argument (subsequent invocations with zero parameters return the new value). GUARD should be a procedure of a single argument. Any new values of the parameter (even the initial value) are passed to this procedure. The guard procedure should check the value and/or convert it to an appropriate form.

— parameter: case-sensitive

If true, then read reads symbols and identifiers in case-sensitive mode and uppercase characters in symbols are printed escaped. Defaults to #t.

— parameter: dynamic-load-libraries

A list of strings containing shared libraries that should be checked for explicitly loaded library units (this facility is not available on all platforms). See load-library.

— parameter: command-line-arguments

Contains the list of arguments passed to this program, with the name of the program and any runtime options (all options starting with -:) removed.

— parameter: exit-handler

A procedure of a single optional argument. When exit is called, then this procedure will be invoked with the exit-code as argument. The default behavior is to terminate the program.

— parameter: eval-handler

A procedure of one or two arguments. When eval is invoked, it calls the value of this parameter with the same arguments. The default behavior is to evaluate the argument expression and to ignore the second parameter.

— parameter: force-finalizers

If true, force and execute all pending finalizers before exiting the program (either explicitly by exit or implicitly when the last toplevel expression has been executed). Default is #t.

— parameter: implicit-exit-handler

A procedure of no arguments. When the last toplevel expression of the program has executed, then the value of this parameter is called. The default behaviour is to do nothing, or, if one or more entry-points were defined (see: Entry points) to enter a loop that waits for callbacks from the host program.

— parameter: keyword-style

Enables alternative keyword syntax, where STYLE may be either #:prefix (as in Common Lisp) or #:suffix (as in DSSSL). Any other value disables the alternative syntaxes.

— parameter: load-verbose

A boolean indicating whether loading of source files, compiled code (if available) and compiled libraries should display a message.

— parameter: repl-prompt

A string that will be printed before reading interactive input from the user in a read-eval-print loop. Defaults to "#;> ".

— parameter: reset-handler

A procedure of zero arguments that is called via reset. The default behavior in compiled code is to invoke the value of (exit-handler). The default behavior in the interpreter is to abort the current computation and to restart the read-eval-print loop.

— parameter: strict-reader

If true, then most non-standard read syntax is disabled. Defaults to #f.


Next: , Previous: Parameters, Up: Supported language

5.7 Unit library

This unit contains basic Scheme definitions. This unit is used by default, unless the program is compiled with the -explicit-use option.


Next: , Up: Unit library

5.7.1 Arithmetic

— procedure: add1
— procedure: sub1
          (add1 N)
          (sub1 N)
     

Adds/subtracts 1 from N.

— procedure: bitwise-and
— procedure: bitwise-ior
— procedure: bitwise-xor
— procedure: bitwise-not
— procedure: arithmetic-shift
          (bitwise-and N1 ...)
          (bitwise-ior N1 ...)
          (bitwise-xor N1 ...)
          (bitwise-not N)
          (arithmetic-shift N1 N2)
     

Binary fixnum operations. arithmetic-shift shifts the argument N1 by N2 bits to the left. If N2 is negative, than N1 is shifted to the right.

— procedure: fixnum?
          (fixnum? X)
     

Returns #t if X is a fixnum, or #f otherwise.

— procedure: fx+
— procedure: fx-
— procedure: fx*
— procedure: fx/
— procedure: fxmod
— procedure: fxneg
— procedure: fxmin
— procedure: fxmax
— procedure: fx=
— procedure: fx>
— procedure: fx<
— procedure: fx>=
— procedure: fx<=
          (fx+ N1 N2)
          (fx- N1 N2)
          (fx* N1 N2)
          (fx/ N1 N2)
          (fxmod N1 N2)
          (fxneg N)
          (fxmin N1 N2)
          (fxmax N1 N2)
          (fx= N1 N2)
          (fx> N1 N2)
          (fx< N1 N2)
          (fx>= N1 N2)
          (fx<= N1 N2)
     

Arithmetic fixnum operations. These procedures do not check their arguments, so non-fixnum parameters will result in incorrect results. fxneg negates its argument.

On division by zero, fx/ and fxmod signal a condition of kind (exn arithmetic).

— procedure: signum
          (signum N)
     

Returns 1 if N is positive, -1 if N is negative or 0 if N is zero.


Next: , Previous: Arithmetic, Up: Unit library

5.7.2 File Input/Output

— procedure: current-error-port
          (current-error-port [PORT])
     

Returns default error output port. If PORT is given, then that port is selected as the new current error output port.

— procedure: end-of-file
          (end-of-file)
     

Returns the end-of-file object.

— procedure: flush-output
          (flush-output [PORT])
     

Write buffered output to the given output-port. PORT defaults to the value of (current-output-port).

— procedure: port-name
          (port-name PORT)
     

Fetch filename from PORT. This returns the filename that was used to open this file. Returns a special tag string, enclosed into parentheses for non-file ports.

— procedure: port-position
          (port-position PORT)
     

Returns the current position of PORT as two values: row and column number. If the port does not support such an operation an error is signaled. This procedure is currently only available for input ports.

— procedure: set-port-name!
          (set-port-name! PORT STRING)
     

Sets the name of PORT to STRING.


Next: , Previous: File Input/Output, Up: Unit library

5.7.3 Files

— procedure: delete-file
          (delete-file STRING)
     

Deletes the file with the pathname STRING. If the file does not exist, an error is signaled.

— procedure: file-exists?
          (file-exists? STRING)
     

Returns #t if a file with the given pathname exists, or #f otherwise.

— variable: pathname-directory-separator

Contains the directory-separator character for pathnames on this platform.

— variable: pathname-extension-separator

Contains the extension-separator character for pathnames on this platform.

— procedure: rename-file
          (rename-file OLD NEW)
     

Renames the file or directory with the pathname OLD to NEW. If the operation does not succeed, an error is signaled.


Next: , Previous: Files, Up: Unit library

5.7.4 String ports

— procedure: get-output-string
          (get-output-string PORT)
     

Returns accumulated output of a port created with (open-output-string).

— procedure: open-input-string
          (open-input-string STRING)
     

Returns a port for reading from STRING.

— procedure: open-output-string
          (open-output-string)
     

Returns a port for accumulating output in a string.


Next: , Previous: String ports, Up: Unit library

5.7.5 Feature identifiers

— procedure: features
          (features)
     

Returns a list of all registered features that will be accepted as valid feature-identifiers by cond-expand.

— procedure: register-feature!
          (register-feature! FEATURE ...)
     

Register one or more features that will be accepted as valid feature-identifiers by cond-expand. FEATURE ... may be a keyword, string or symbol.

— procedure: unregister-feature!
          (unregister-feature! FEATURE ...)
     

Unregisters the specified feature-identifiers. FEATURE ... may be a keyword, string or symbol.


Next: , Previous: Feature identifiers, Up: Unit library

5.7.6 Keywords

Keywords are special symbols prefixed with #: that evaluate to themselves. Procedures can use keywords to accept optional named parameters in addition to normal required parameters. Assignment to and bindings of keyword symbols is not allowed. The parameter keyword-style and the compiler/interpreter option -keyword-style can be used to allow an additional keyword syntax, either compatible to Common LISP, or to DSSSL.

— procedure: get-keyword
          (get-keyword KEYWORD ARGLIST [THUNK])
     

Returns the argument from ARGLIST specified under the keyword KEYWORD. If the keyword is not found, then the zero-argument procedure THUNK is invoked and the result value is returned. If THUNK is not given, #f is returned.

     
     (define (increase x . args)
       (+ x (get-keyword #:amount args (lambda () 1))) )
     (increase 123)                                      ==> 124
     (increase 123 #:amount 10)                          ==> 133

Note: the KEYWORD may actually be any kind of object.

— procedure: keyword?
          (keyword? X)
     

Returns #t if X is a keyword symbol, or #f otherwise.

— procedure: keyword->string
          (keyword->string KEYWORD)
     

Transforms KEYWORD into a string.

— procedure: string->keyword
          (string->keyword STRING)
     

Returns a keyword with the name STRING.


Next: , Previous: Keywords, Up: Unit library

5.7.7 Exceptions

CHICKEN implements the (currently withdrawn) SRFI-12 exception system. For more information, see the SRFI-12 document

— syntax: condition-case
          (condition-case EXPRESSION CLAUSE ...)
     

Evaluates EXPRESSION and handles any exceptions that are covered by CLAUSE ..., where CLAUSE should be of the following form:

     
     CLAUSE = ([VARIABLE] (KIND ...) BODY ...)

If provided, VARIABLE will be bound to the signalled exception object. BODY ... is executed when the exception is a property- or composite condition with the kinds given KIND ... (unevaluated). If no clause applies, the exception is re-signalled in the same dynamic context as the condition-case form.

     
     (define (check thunk)
       (condition-case (thunk)
         [(exn file) (print "file error")]
         [(exn) (print "other error")]
         [var () (print "something else")] ) )
     
     (check (lambda () (open-input-file "")))   ; -> "file error"
     (check (lambda () some-unbound-variable))  ; -> "othererror"
     (check (lambda () (signal 99)))            ; -> "something else"
     
     (condition-case some-unbound-variable
       [(exn file) (print "ignored)] )      ; -> signals error
     

All error-conditions signalled by the system are of kind exn. The following composite conditions are additionally defined:

(exn arity)
Signalled when a procedure is called with the wrong number of arguments.
(exn type)
Signalled on type-mismatch errors, for example when an argument of the wrong type is passed to a builtin procedure.
(exn arithmetic)
Signalled on arithmetic errors, like division by zero.
(exn i/o)
Signalled on input/output errors.
(exn i/o file)
Signalled on file-related errors.
(exn i/o net)
Signalled on network errors.
(exn bounds)
Signalled on errors caused by accessing non-existent elements of a collection.
(exn runtime)
Signalled on low-level runtime-system error-situations.
(exn runtime limit)
Signalled when an internal limit is exceeded (like running out of memory).
(exn match)
Signalled on errors raised by failed matches (see the section on match).
(exn syntax)
Signalled on syntax errors.

Notes:


Next: , Previous: Exceptions, Up: Unit library

5.7.8 Environment information and system interface

— procedure: argv
          (argv)
     

Return a list of all supplied command-line arguments. The first item in the list is a string containing the name of the executing program. The other items are the arguments passed to the application. This list is freshly created on every invocation of (argv). It depends on the host-shell whether arguments are expanded ('globbed') or not.

— procedure: exit
          (exit [CODE])
     

Exit the running process and return exit-code, which defaults to 0 (Invokes exit-handler).

— procedure: build-platform
          (build-platform)
     

Returns a symbol specifying the toolset which has been used for building the executing system, which is one of the following:

     
     djgpp
     cygwin
     msvc
     mingw32
     gnu
     metrowerks
     unknown
— procedure: chicken-version
          (chicken-version)
     

Returns a string containing the version number of the CHICKEN runtime system.

— procedure: errno
          (errno)
     

Returns the error code of the last system call.

— procedure: getenv
          (getenv STRING)
     

Returns the value of the environment variable STRING or #f if that variable is not defined.

— procedure: machine-type
          (machine-type)
     

Returns a symbol specifying the processor on which this process is currently running, which is one of the following:

     
     alpha
     mips
     hppa
     ultrasparc
     sparc
     ppc
     ia64
     x86
     x86-64
     unknown
— procedure: software-type
          (software-type)
     

Returns a symbol specifying the operating system on which this process is currently running, which is one of the following:

     
     msdos
     windows
     unix
     macos
     unknown
— procedure: software-version
          (software-version)
     

Returns a symbol specifying the operating system version on which this process is currently running, which is one of the following:

     
     linux
     freebsd
     netbsd
     openbsd
     macosx
     hpux
     solaris
     sunos
     unknown
— procedure: system
          (system STRING)
     

Execute shell command. The functionality offered by this procedure depends on the capabilities of the host shell.


Next: , Previous: Environment information and system interface, Up: Unit library

5.7.9 Execution time

— procedure: cpu-time
          (cpu-time)
     

Returns the used CPU time of the current process in milliseconds as two values: the time spent in user code, and the time spent in system code. On platforms where user and system time can not be differentiated, system time will be always be 0.

— procedure: current-milliseconds
          (current-milliseconds)
     

Returns the number of milliseconds since process- or machine startup.

— procedure: current-seconds
          (current-seconds)
     

Returns the number of seconds since midnight, Jan. 1, 1970.


Next: , Previous: Execution time, Up: Unit library

5.7.10 Interrupts and error-handling

— procedure: enable-interrupts
— procedure: disable-interrupts
          (enable-interrupts)
          (disable-interrupts)
     

Enables/disables processing of timer-interrupts and interrupts caused by signals.

     
     (disable-interrupts)
     (disable-interrupts)
     (enable-interrupts)
     ; <interrupts still disabled - call enable-interrupts once more>
— procedure: enable-warnings
          (enable-warnings [BOOL])
     

Enables or disables warnings, depending on wether BOOL is true or false. If called with no arguments, this procedure returns #t if warnings are currently enabled, or #f otherwise. Note that this is not a parameter. The current state (wether warnings are enabled or disabled) is global and not thread-local.

— procedure: error
          (error [LOCATION] STRING EXP ...)
     

Prints error message, writes all extra arguments to the value of (current-error-port) and invokes the current value of (error-handler). This conforms to SRFI-23. If LOCATION is given and a symbol, it specifies the “location” (the name of the procedure) where the error occurred.

— procedure: print-backtrace
          (print-backtrace [PORT])
     

Prints a backtrace of the procedure call history to PORT, which defaults to (current-error-port). Backtrace information is only generated in compiled code, with a -debug-level >= 1.

— procedure: print-error-message
          (print-error-message EXN [PORT [STRING]])
     

Prints an appropriate error message to PORT (which defaults to the value of (current-error-port) for the object EXN. EXN may be a condition, a string or any other object. If the optional argument STRING is given, it is printed before the error-message. STRING defaults to "Error:".

— procedure: reset
          (reset)
     

Reset program (Invokes reset-handler).


Next: , Previous: Interrupts and error-handling, Up: Unit library

5.7.11 Garbage collection

— procedure: gc
          (gc [FLAG])
     

Invokes a garbage-collection and returns the number of free bytes in the heap. The flag specifies whether a minor (#f) or major (#t) GC is to be triggered. If no argument is given, #t is assumed. When the argument is #t, all pending finalizers are executed.

— procedure: memory-statistics
               (memory-statistics)
          

Performs a major garbage collection and returns a three element vector containing the total heap size in bytes, the number of bytes currently used and the size of the nursery (the first heap generation). Note that the actual heap is actually twice the size given in the heap size, because CHICKEN uses a copying semi-space collector.

— procedure: set-finalizer!
          (set-finalizer! X PROC)
     

Registers a procedure of one argument PROC, that will be called as soon as the non-immediate data object X is about to be garbage-collected (with that object as its argument). Note that the finalizer will not be called when interrupts are disabled.

— procedure: set-gc-report!
          (set-gc-report! FLAG)
     

Print statistics after every GC, depending on FLAG. A value of #t shows statistics after every major GC. A true value different from #t shows statistics after every minor GC. #f switches statistics off.


Next: , Previous: Garbage collection, Up: Unit library

5.7.12 Other control structures

— procedure: andmap
          (andmap PROC LIST1 ...)
     

Repeatedly calls PROC with arguments taken from LIST1 .... If any invocation should return #f, the result of andmap is #f. If all invocations return a true result, then the result of andmap is #t.

— procedure: ormap
          (ormap PROC LIST1 ...)
     

Repeatedly calls PROC with arguments taken from LIST1 .... If any invocation should return a value different from #f, then this value is returned as the result of ormap. If all invocations return #f, then the result of ormap is #f.


Next: , Previous: Other control structures, Up: Unit library

5.7.13 String utilities

— procedure: reverse-list->string
          (reverse-list->string LIST)
     

Returns a string with the characters in LIST in reverse order. This is equivalent to (list->string (reverse LIST)), but much more efficient.


Next: , Previous: String utilities, Up: Unit library

5.7.14 Generating uninterned symbols

— procedure: gensym
          (gensym [STRING-OR-SYMBOL])
     

Returns a newly created uninterned symbol. If an argument is provided, the new symbol is prefixed with that argument.

— procedure: string->uninterned-symbol
          (string->uninterned-symbol STRING)
     

Returns a newly created, unique symbol with the name STRING.


Next: , Previous: Generating uninterned symbols, Up: Unit library

5.7.15 Standard Input/Output

— procedure: port?
          (port? X)
     

Returns #t if X is a port object or #f otherwise.

— procedure: print
          (print EXP1 EXP2 ...)
     

Outputs the arguments EXP1 EXP2 ... using display and writes a newline character to the port that is the value of (current-output-port). Returns its first argument.

— procedure: print*
          (print* EXP1 ...)
     

Similar to print, but does not output a terminating newline character.


Next: , Previous: Standard Input/Output, Up: Unit library

5.7.16 User-defined named characters

— procedure: char-name
          (char-name SYMBOL-OR-CHAR [CHAR])
     

This procedure can be used to inquire about character names or to define new ones. With a single argument the behavior is as follows: If SYMBOL-OR-CHAR is a symbol, then char-name returns the character with this name, or #f if no character is defined under this name. If SYMBOL-OR-CHAR is a character, then the name of the character is returned as a symbol, or #f if the character has no associated name.

If the optional argument CHAR is provided, then SYMBOL-OR-CHAR should be a symbol that will be the new name of the given character. If multiple names designate the same character, then the write will use the character name that was defined last.

     
     (char-name 'space)                  ==> #\space
     (char-name #\space)                 ==> space
     (char-name 'bell)                   ==> #f
     (char-name (integer->char 7))       ==> #f
     (char-name 'bell (integer->char 7))
     (char-name 'bell)                   ==> #\bell
     (char->integer (char-name 'bell))   ==> 7


Next: , Previous: User-defined named characters, Up: Unit library

5.7.17 Vectors

— procedure: vector-copy!
          (vector-copy! VECTOR1 VECTOR2 [COUNT])
     

Copies contents of VECTOR1 into VECTOR2. If the argument COUNT is given, it specifies the maximal number of elements to be copied. If not given, the minimum of the lengths of the argument vectors is copied.

Exceptions: (exn bounds)

— procedure: vector-resize
          (vector-resize VECTOR N [INIT])
     

Creates and returns a new vector with the contents of VECTOR and length N. If N is greater than the original length of VECTOR, then all additional items are initialized to INIT. If INIT is not specified, the contents are initialized to some unspecified value.


Next: , Previous: Vectors, Up: Unit library

5.7.18 The unspecified value

— procedure: void
          (void)
     

Returns an unspecified value.


Previous: The unspecified value, Up: Unit library

5.7.19 call/cc

— procedure: call/cc
          (call/cc PROCEDURE)
     

An alias for call-with-current-continuation.


Next: , Previous: Unit library, Up: Supported language

5.8 Unit eval

This unit has support for evaluation and macro-handling. This unit is used by default, unless the program is compiled with the -explicit-use option.


Next: , Up: Unit eval

5.8.1 Loading code

— procedure: load
          (load FILE [EVALPROC])
     

Loads and evaluates expressions from the given source file, which may be either a string or an input port. Each expression read is passed to EVALPROC (which defaults to eval). On platforms that support it (currently Linux ELF and Solaris), load can be used to load compiled programs:

     
     % cat x.scm
     (define (hello) (print "Hello!"))
     % chicken x.scm -quiet -dynamic
     % gcc x.c -shared -fPIC `chicken-config -cflags -shared -libs` -o x.so
     % csi -quiet
     #;> (load "x.so")
     ; loading x.so ...
     #;> (hello)
     Hello!
     #;>

The second argument to load is ignored when loading compiled code. The same compiled object file can not be loaded more than once. If source code is loaded from a port, then that port is closed after all expressions have been read.

— procedure: load-library
          (load-library UNIT [LIBRARYFILE])
     

On platforms that support dynamic loading, load-library loads the compiled library unit UNIT (which should be a symbol). If the string LIBRARYFILE is given, then the given shared library will be loaded and the toplevel code of the contained unit will be executed. If no LIBRARYFILE argument is given, then the following libraries are checked for the required unit:

If the unit is not found, an error is signaled. When the library unit can be successfully loaded, a feature-identifier named UNIT is registered. If the feature is already registered before loading, the load-library does nothing.

— procedure: load-noisily
          (load-noisily FILE [EVALPROC])
     

As load but the result(s) of each evaluated toplevel-expression is written to standard output.


Next: , Previous: Loading code, Up: Unit eval

5.8.2 Read-eval-print loop

— procedure: repl
          (repl)
     

Start a new read-eval-print loop. Sets the reset-handler so that any invocation of reset restarts the read-eval-print loop. Also changes the current error-handler to display a message, write any arguments to the value of (current-error-port) and reset.


Next: , Previous: Read-eval-print loop, Up: Unit eval

5.8.3 Macros

— procedure: get-line-number
          (get-line-number EXPR)
     

If EXPR is a pair with the car being a symbol, and line-number information is available for this expression, then this procedure returns the associated line number. If line-number information is not available, then #f is returned. Note that line-number information for expressions is only available in the compiler.

— procedure: macro?
          (macro? SYMBOL)
     

Returns #t if there exists a macro-definition for SYMBOL.

— procedure: macroexpand
          (macroexpand X)
     

If X is a macro-form, expand the macro (and repeat expansion until expression is a non-macro form). Returns the resulting expression.

— procedure: macroexpand-1
          (macroexpand-1 X)
     

If X is a macro-form, expand the macro. Returns the resulting expression.

— procedure: undefine-macro!
          (undefine-macro! SYMBOL)
     

Remove the current macro-definition of the macro named SYMBOL.

— procedure: syntax-error
          (syntax-error [LOCATION] MESSAGE ARGUMENT ...)
     

Signals an exception of the kind (exn syntax). Otherwise identical to error.


Next: , Previous: Macros, Up: Unit eval

5.8.4 Loading extension libraries

This functionality is only available on platforms that support dynamic loading of compiled code. Currently Linux, BSD, Solaris, Windows (with Cygwin) and HP/UX are supported.

— parameter: repository-path

Contains a string naming the path to the extension repository, which defaults to either the value of the environment variable CHICKEN_REPOSITORY, the value of the environment variable CHICKEN_HOME or the default library path (usually /usr/local/lib/chicken on UNIX systems.

— procedure: extension-info
          (extension-info ID)
     

If an extension with the name ID is installed and if it has a setup-information list registered in the extension repository, then the info-list is returned. Otherwise extension-info returns #f.

— procedure: provide
          (provide ID ...)
     

Registers the extension IDs ID ... as loaded. This is mainly intended to provide aliases for certain extension identifiers.

— procedure: provided?
          (provided? ID ...)
     

Returns #t if the extension with the IDs ID ... are currently loaded, or #f otherwise. Works also for feature-ids.

— procedure: require
          (require ID ...)
     

If the extension library ID is not already loaded into the system, then require will lookup the location of the shared extension library and load it. If ID names a library-unit of the base system, then it is loaded via load-library. If no extension library is available for the given ID, then an attempt is made to load the file ID.so or ID.scm (in that order) from one of the following locations:

  1. the current directory
  2. the current include path, which defaults to the pathnames given in CHICKEN_INCLUDE_PATH and CHICKEN_HOME. In case ID is a list, it is interpreted as a (relative) pathname.

ID may be a symbol, or a list of symbols. See also: require-for-syntax.


Next: , Previous: Loading extension libraries, Up: Unit eval

5.8.5 Reader extensions

— procedure: define-reader-ctor
          (define-reader-ctor SYMBOL PROC)
     

Define new read-time constructor for #, read syntax. For further information, see the documentation for SRFI-10.

— procedure: set-read-syntax!
          (set-read-syntax! CHAR PROC)
     

When the reader is encounting the non-whitespace character CHAR while reading an expression from a given port, then the procedure PROC will be called with that port as its argument. The procedure should return a value that will be returned to the reader:

          ; A simple RGB color syntax:
          
          (set-read-syntax! #\%
            (lambda (port)
              (apply vector
                (map (cut string->number <> 16)
                     (string-chop (read-string 6 port) 2) ) ) ) )
          
          (with-input-from-string "(1 2 %e0e0e0 3)" read)
          ; ==> (1 2 #(240 240 240) 3)
     


Previous: Reader extensions, Up: Unit eval

5.8.6 Eval

— procedure: eval
          (eval EXP [ENVIRONMENT])
     

Evaluates EXP and returns the result of the evaluation. The second argument is optional and defaults to the value of (interaction-environment).


Next: , Previous: Unit eval, Up: Supported language

5.9 Unit extras

This unit contains a collection of useful utility definitions. This unit is used by default, unless the program is compiled with the -explicit-use option.


Next: , Up: Unit extras

5.9.1 Lists

— procedure: alist-ref
          (alist-ref KEY ALIST [TEST [DEFAULT]])
     

Looks up KEY in ALIST using TEST as the comparison function (or eqv? if no test was given) and returns the cdr of the found pair, or DEFAULT (which defaults to #f).

— procedure: alist-update!
          (alist-update! KEY VALUE ALIST [TEST])
     

If the list ALIST contains a pair of the form (KEY . X), then this procedure replaces X with VALUE and returns ALIST. If ALIST contains no such item, then alist-update! returns ((KEY . VALUE) . ALIST). The optional argument TEST specifies the comparison procedure to search a matching pair in ALIST and defaults to eqv?.

— procedure: butlast
          (butlast LIST)
     

Returns a fresh list with all elements but the last of LIST.

— procedure: chop
          (chop LIST N)
     

Returns a new list of sublists, where each sublist contains N elements of LIST. If LIST has a length that is not a multiple of N, then the last sublist contains the remaining elements.

     
     (chop '(1 2 3 4 5 6) 2) ==> ((1 2) (3 4) (5 6))
     (chop '(a b c d) 3)     ==> ((a b c) (d))
— procedure: compress
          (compress BLIST LIST)
     

Returns a new list with elements taken from LIST with corresponding true values in the list BLIST.

     
     (define nums '(99 100 110 401 1234))
     (compress (map odd? nums) nums)      ==> (99 401)
— procedure: flatten
          (flatten LIST1 ...)
     

Returns LIST1 ... concatenated together, with nested lists removed (flattened).

— procedure: intersperse
          (intersperse LIST X)
     

Returns a new list with X placed between each element.

— procedure: join
          (join LISTOFLISTS [LIST])
     

Concatenates the lists in LISTOFLISTS with LIST placed between each sublist. LIST defaults to the empty list.

     
     (join '((a b) (c d) (e)) '(x y)) ==> (a b x y c d x y e)
     (join '((p q) () (r (s) t)) '(-))  ==> (p q - - r (s) t)
— procedure: shuffle
          (shuffle LIST)
     

Returns LIST with its elements sorted in a random order.

join could be implemented as follows:

     
     (define (join lstoflsts . lst)
       (apply append (intersperse lstoflists (:optional lst '()))) )
— procedure: tail?
          (tail? X LIST)
     

Returns true if X is one of the tails (cdr's) of LIST.


Next: , Previous: Lists, Up: Unit extras

5.9.2 String-port extensions

— procedure: call-with-input-string
          (call-with-input-string STRING PROC)
     

Calls the procedure PROC with a single argument that is a string-input-port with the contents of STRING.

— procedure: call-with-output-string
          (call-with-output-string PROC)
     

Calls the procedure PROC with a single argument that is a string-output-port. Returns the accumulated output-string.

— procedure: with-input-from-string
          (with-input-from-string STRING THUNK)
     

Call procedure THUNK with the current input-port temporarily bound to an input-string-port with the contents of STRING.

— procedure: with-output-to-string
          (with-output-to-string THUNK)
     

Call procedure THUNK with the current output-port temporarily bound to a string-output-port and return the accumulated output string.


Next: , Previous: String-port extensions, Up: Unit extras

5.9.3 Formatted output

— procedure: fprintf
— procedure: printf
— procedure: sprintf
          (fprintf PORT FORMATSTRING ARG ...)
          (printf FORMATSTRING ARG)
          (sprintf FORMATSTRING ARG ...)
     

Simple formatted output to a given port (fprintf), the value of (current-output-port) (printf) or a string (sprintf). The FORMATSTRING can contain any sequence of characters. The character `~' prefixes special formatting directives:

~%
write newline character
~S
write the next argument
~A
display the next argument
~\n
skip all whitespace in the format-string until the next non-whitespace character
~B
write the next argument as a binary number
~O
write the next argument as an octal number
~X
write the next argument as a hexadecimal number
~C
write the next argument as a character
~~
display `~'
~!
flush all pending output
~?
invoke formatted output routine recursively with the next two arguments as format-string and list of parameters

For more powerful output formatting, see the section about the format unit.


Next: , Previous: Formatted output, Up: Unit extras

5.9.4 Hash tables

— procedure: clear-hash-table!
          (clear-hash-table! HASH-TABLE)
     

Erases all entries in the hash-table HASH-TABLE.

— procedure: get
          (get HASH-TABLE KEY PROP)
     

Returns the value of property PROP of the item KEY in HASH-TABLE . This facility can be used as a kind of “disembodied” property-list. If no entry named KEY is stored in the hash-table or if no property PROP for that key exists, #f is returned.

— procedure: hash-table?
          (hash-table? X)
     

Returns #t if the argument is a hash-table.

— procedure: hash-table->list
          (hash-table->list HASH-TABLE)
     

Converts HASH-TABLE into an association-list.

— procedure: hash-table-count
          (hash-table-count HASH-TABLE)
     

Returns the number of entries in the given hash-table.

— procedure: hash-table-size
          (hash-table-size HASH-TABLE)
     

Returns the size of the hash-table.

— procedure: hash-table-for-each
          (hash-table-for-each PROC HASH-TABLE)
     

Calls PROC which should expect two arguments. This procedure is called for each entry in the hash-table with the key and the value as parameters.

— procedure: hash-table-ref
          (hash-table-ref HASH-TABLE KEY [DEFAULT])
     

Returns the entry in the given hash-table under KEY. If no entry is stored in the table, #f is returned.

— procedure: hash-table-remove!
          (hash-table-remove! HASH-TABLE KEY)
     

Removes an entry in the given hash-table.

— procedure: hash-table-set!
          (hash-table-set! HASH-TABLE KEY VALUE)
     

Adds or changes an entry in the given hash-table.

— procedure: make-hash-table
          (make-hash-table [PRED [SIZE]])
     

Creates and returns a hash-table with keys compared via PRED, which defaults to eq?. If SIZE is provided it specifies the initial size of the hash-table. If the hash-table fills above a certain size it is automatically resized to accommodate more entries.

— procedure: put!
          (put! HASH-TABLE KEY PROP VALUE)
     

Stores VALUE as property PROP under the item KEY in the given hash-table. Any previously existing value is overwritten.


Next: , Previous: Hash tables, Up: Unit extras

5.9.5 Queues

— procedure: list->queue
          (list->queue LIST)
     

Returns LIST converted into a queue, where the first element of the list is the same as the first element of the queue. The resulting queue may share memory with the list and the list should not be modified after this operation.

— procedure: make-queue
          (make-queue)
     

Returns a newly created queue.

— procedure: queue?
          (queue? X)
     

Returns #t if X is a queue, or #f otherwise.

— procedure: queue->list
          (queue->list QUEUE)
     

Returns QUEUE converted into a list, where the first element of the list is the same as the first element of the queue. The resulting list may share memory with the queue object and should not be modified.

— procedure: queue-add!
          (queue-add! QUEUE X)
     

Adds X to the rear of QUEUE.

— procedure: queue-empty?
          (queue-empty? QUEUE)
     

Returns #t if QUEUE is empty, or #f otherwise.

— procedure: queue-first
          (queue-first QUEUE)
     

Returns the first element of QUEUE. If QUEUE is empty an error is signaled

— procedure: queue-last
          (queue-last QUEUE)
     

Returns the last element of QUEUE. If QUEUE is empty an error is signaled

— procedure: queue-remove!
          (queue-remove! QUEUE)
     

Removes and returns the first element of QUEUE. If QUEUE is empty an error is signaled


Next: , Previous: Queues, Up: Unit extras

5.9.6 Sorting

— procedure: merge
— procedure: merge!
          (merge LIST1 LIST2 LESS?)
          (merge! LIST1 LIST2 LESS?)
     

Joins two lists in sorted order. merge! is the destructive version of merge. LESS? should be a procedure of two arguments, that returns true if the first argument is to be ordered before the second argument.

— procedure: sort
— procedure: sort!
          (sort SEQUENCE LESS?)
          (sort! SEQUENCE LESS?)
     

Sort SEQUENCE, which should be a list or a vector. sort! is the destructive version of sort.

— procedure: sorted?
          (sorted? SEQUENCE LESS?)
     

Returns true if the list or vector SEQUENCE is already sorted.


Next: , Previous: Sorting, Up: Unit extras

5.9.7 Random numbers

— procedure: random
          (random N)
     

Returns an exact random integer from 0 to N-1.

— procedure: randomize
          (randomize [X])
     

Set random-number seed. If X is not supplied, the current time is used. On startup (when the extras unit is initialized), the random number generator is initialized with the current time.


Next: , Previous: Random numbers, Up: Unit extras

5.9.8 Input/Output extensions

— procedure: make-input-port
          (make-input-port READ READY? CLOSE [PEEK])
     

Returns a custom input port. Common operations on this port are handled by the given parameters, which should be procedures of no arguments. READ is called when the next character is to be read and should return a character or the value of (end-of-file). READY? is called when char-ready? is called on this port and should return #t or #f. CLOSE is called when the port is closed. PEEK is called when peek-char is called on this port and should return a character or the value of (end-of-file). if the argument PEEK is not given, then READ is used instead and the created port object handles peeking automatically (by calling READ and buffering the character).

— procedure: make-output-port
          (make-output-port WRITE CLOSE [FLUSH])
     

Returns a custom output port. Common operations on this port are handled by the given parameters, which should be procedures. WRITE is called when output is sent to the port and receives a single argument, a string. CLOSE is called when the port is closed and should be a procedure of no arguments. FLUSH (if provided) is called for flushing the output port.

— procedure: pretty-print
— procedure: pp
          (pretty-print EXP [PORT])
          (pp EXP [PORT])
     

Print expression nicely formatted. PORT defaults to the value of (current-output-port).

— parameter: pretty-print-width

Specifies the maximal line-width for pretty printing, after which line wrap will occur.

— procedure: read-file
          (read-file [FILE-OR-PORT])
     

Returns a list containing all toplevel expressions read from the file or port FILE-OR-PORT. If no argument is given, input is read from the port that is the current value of (current-input-port). After all expressions are read, and if the argument is a port, then the port will not be closed.

— procedure: read-line
— procedure: write-line
          (read-line [PORT [LIMIT]])
          (write-line STRING [PORT])
     

Line-input and -output. PORT defaults to the value of (current-input-port) and (current-output-port), respectively. if the optional argument LIMIT is given and not #f, then read-line reads at most LIMIT characters per line.

— procedure: read-lines
          (read-lines [PORT [MAX]])
     

Read MAX or fewer lines from PORT. PORT defaults to the value of (current-input-port).

— procedure: read-string
— procedure: write-string
          (read-string [NUM [PORT]])
          (write-string STRING [NUM [PORT]]
     

Read or write NUM characters from/to PORT, which defaults to the value of (current-input-port) or (current-output-port), respectively. If NUM is #f or not given, then all data up to the end-of-file is read, or, in the case of write-string the whole string is written. If no more input is available, read-string returns the empty string.

— procedure: read-token
          (read-token PREDICATE [PORT])
     

Reads characters from PORT (which defaults to the value of (current-input-port)) and calls the procedure PREDICATE with each character until PREDICATE returns false.

— procedure: with-error-output-to-port
          (with-error-output-to-port PORT THUNK)
     

Call procedure THUNK with the current error output-port temporarily bound to PORT.

— procedure: with-input-from-port
          (with-input-from-port PORT THUNK)
     

Call procedure THUNK with the current input-port temporarily bound to PORT.

— procedure: with-output-to-port
          (with-output-to-port PORT THUNK)
     

Call procedure THUNK with the current output-port temporarily bound to PORT.


Next: , Previous: Input/Output extensions, Up: Unit extras

5.9.9 Strings

— procedure: conc
          (conc X ...)
     

Returns a string with the string-represenation of all arguments concatenated together. conc could be implemented as

     
     (define (conc . args)
       (apply string-append (map ->string args)) )

— procedure: ->string
          (->string X)
     

Returns a string-representation of X.

— procedure: string-chop
          (string-chop STRING LENGTH)
     

Returns a list of substrings taken by “chopping” STRING every LENGTH characters:

     
     (string-chop "one two three" 4)  ==>  ("one " "two " "thre" "e")
— procedure: string-compare3
          (string-compare3 STRING1 STRING2)
     
— procedure: string-compare3-ci
          (string-compare3-ci STRING1 STRING2)
     

Perform a three-way comparison between the STRING1 and STRING2, returning either -1 if STRING1 is lexicographically less than STRING2, 0 if it is equal, or 1 if it s greater. string-compare3-ci performs a case-insensitive comparison.

— procedure: string-intersperse
          (string-intersperse LIST STRING)
     

Returns a string that contains all strings in LIST concatenated together. STRING is placed between each concatenated string.

     
     (string-intersperse '("one" "two") "three")

is equivalent to

     
     (apply string-append (intersperse '("one" "two") "three"))
— procedure: string-split
          (string-split STRING [DELIMITER-STRING [KEEPEMPTY]])
     

Split string into substrings separated by the given delimiters. If no delimiters are specified, a string comprising the tab, newline and space characters is assumed. If the parameter KEEPEMPTY is given and not #f, then empty substrings are retained:

     
     (string-split "one  two  three") ==> ("one" "two" "three")
     (string-split "foo:bar::baz:" ":" #t) ==> ("foo" "bar" "" "baz" "")
— procedure: string-translate
          (string-translate STRING FROM [TO])
     

Returns a fresh copy of STRING with characters matching FROM translated to TO. If TO is omitted, then matching characters are removed. FROM and TO may be a character, a string or a list. If both FROM and TO are strings, then the character at the same position in TO as the matching character in FROM is substituted.

— procedure: string-translate*
          (string-translate* STRING SMAP)
     

Substitutes elements of STRING according to SMAP. SMAP should be an association-list where each element of the list is a pair of the form (MATCH \. REPLACEMENT). Every occurrence of the string MATCH in STRING will be replaced by the string REPLACEMENT:

     
     (string-translate*
       "<h1>this is a \"string\"</h1>"
       '(("<" . "&lt:") (">" . "&gt;") ("\"" . "&quot;")) )
     
     ==>  "&lt;h1&gt;this is a &quot;string&quot;&lt;/ht&gt;"
— procedure: substring=?
— procedure: substring-ci=?
          (substring=? STRING1 STRING2 [START1 [START2 [LENGTH]]])
          (substring-ci=? STRING1 STRING2 [START1 [START2 [LENGTH]]])
     

Returns #t if the strings STRING1 and STRING2 are equal, or #f otherwise. The comparison starts at the positions START1 and START2 (which default to 0), comparing LENGTH characters (which defaults to the minimum of the remaining length of both strings).

— procedure: substring-index
— procedure: substring-index-ci
          (substring-index WHICH WHERE [START])
          (substring-index-ci WHICH WHERE [START])
     

Searches for first index in string WHERE where string WHICH occurs. If the optional argument START is given, then the search starts at that index. substring-index-ci is a case-insensitive version of substring-index.


Next: , Previous: Strings, Up: Unit extras

5.9.10 Combinators

— procedure: constantly
          (constantly X ...)
     

Returns a procedure that always returns the values X ... regardless of the number and value of its arguments.

     
     (constantly X) <=> (lambda args X)
— procedure: complement
          (complement PROC)
     

Returns a procedure that returns the boolean inverse of PROC.

     
     (complement PROC) <=> (lambda (x) (not (PROC x)))
— procedure: compose
          (compose PROC1 PROC2 ...)
     

Returns a procedure that represents the composition of the argument-procedures PROC1 PROC2 ....

     
     (compose F G) <=> (lambda args
                           (call-with-values
                              (lambda () (apply G args))
                              F))
— procedure: conjoin
          (conjoin PRED ...)
     

Returns a procedure that returns #t if its argument satisfies the predicates PRED ....

     
     ((conjoin odd? positive?) 33)   ==>  #t
     ((conjoin odd? positive?) -33)  ==>  #f
— procedure: disjoin
          (disjoin PRED ...)
     

Returns a procedure that returns #t if its argument satisfies any predicate PRED ....

     
     ((disjoin odd? positive?) 32)    ==>  #t
     ((disjoin odd? positive?) -32)   ==>  #f
— procedure: flip
          (flip PROC)
     

Returns a two-argument procedure that calls PROC with its arguments swapped:

     
     (flip PROC) <=> (lambda (x y) (PROC y x))
— procedure: identity
          (identity X)
     

Returns its sole argument X.

— procedure: project
          (project N)
     

Returns a procedure that returns its Nth argument.

— procedure: list-of
          (list-of PRED)
     

Returns a procedure of one argument that returns #t when applied to a list of elements that all satisfy the predicate procedure PRED, or #f otherwise.

     
     ((list-of even?) '(1 2 3))   ==> #f
     ((list-of number?) '(1 2 3)) ==> #t


Previous: Combinators, Up: Unit extras

5.9.11 Binary searching

— procedure: binary-search
          (binary-search SEQUENCE PROC)
     

Performs a binary search in SEQUENCE, which should be a sorted list or vector. PROC is called to compare items in the sequence, should accept a single argument and return an exact integer: zero if the searched value is equal to the current item, negative if the searched value is “less” than the current item, and positive otherwise.


Next: , Previous: Unit extras, Up: Supported language

5.10 Unit srfi-1

List library, see the documentation for SRFI-1


Next: , Previous: Unit srfi-1, Up: Supported language

5.11 Unit srfi-4

Homogeneous numeric vectors, see the documentation for SRFI-4 In addition to that, the following procedures are also provided:

— procedure: u8vector->byte-vector
— procedure: s8vector->byte-vector
— procedure: u16vector->byte-vector
— procedure: s16vector->byte-vector
— procedure: u32vector->byte-vector
— procedure: s32vector->byte-vector
— procedure: f32vector->byte-vector
— procedure: f64vector->byte-vector
          (u8vector->byte-vector U8VECTOR)
          (s8vector->byte-vector S8VECTOR)
          (u16vector->byte-vector U16VECTOR)
          (s16vector->byte-vector S16VECTOR)
          (u32vector->byte-vector U32VECTOR)
          (s32vector->byte-vector S32VECTOR)
          (f32vector->byte-vector F32VECTOR)
          (f64vector->byte-vector F64VECTOR)
     

Each of these procedures return the contents of the given vector as a 'packed' byte-vector. The byte order in that vector is platform-dependent (for example little-endian on an Intel processor). The returned byte-vector shares memory with the contents of the vector.

— procedure: byte-vector->u8vector
— procedure: byte-vector->s8vector
— procedure: byte-vector->u16vector
— procedure: byte-vector->s16vector
— procedure: byte-vector->u32vector
— procedure: byte-vector->s32vector
— procedure: byte-vector->f32vector
— procedure: byte-vector->f64vector
          (byte-vector->u8vector BYTE-VECTOR)
          (byte-vector->s8vector BYTE-VECTOR)
          (byte-vector->u16vector BYTE-VECTOR)
          (byte-vector->s16vector BYTE-VECTOR)
          (byte-vector->u32vector BYTE-VECTOR)
          (byte-vector->s32vector BYTE-VECTOR)
          (byte-vector->f32vector BYTE-VECTOR)
          (byte-vector->f64vector BYTE-VECTOR)
     

Each of these procedures return a vector where the argument BYTE-VECTOR is taken as a 'packed' representation of the contents of the vector. The argument-byte-vector shares memory with the contents of the vector.

— procedure: subu8vector
— procedure: subu16vector
— procedure: subu32vector
— procedure: subs8vector
— procedure: subs16vector
— procedure: subs32vector
— procedure: subf32vector
— procedure: subf64vector
          (subu8vector U8VECTOR FROM TO)
          (subu16vector U16VECTOR FROM TO)
          (subu32vector U32VECTOR FROM TO)
          (subs8vector S8VECTOR FROM TO)
          (subs16vector S16VECTOR FROM TO)
          (subs32vector S32VECTOR FROM TO)
          (subf32vector F32VECTOR FROM TO)
          (subf64vector F64VECTOR FROM TO)
     

Creates a number vector of the same type as the argument vector with the elements at the positions FROM up to but not including TO.


Next: , Previous: Unit srfi-4, Up: Supported language

5.12 Unit srfi-13

String library, see the documentation for SRFI-13

The file srfi-13-syntax.scm contains macro definitions for let-string-start+end.

On systems that support dynamic loading, the srfi-13 unit can be made available in the interpreter (csi) by entering

(require-extension srfi-13)


Next: , Previous: Unit srfi-13, Up: Supported language

5.13 Unit srfi-14

Character set library, see the documentation for SRFI-14

On systems that support dynamic loading, the srfi-14 unit can be made available in the interpreter (csi) by entering

(require-extension srfi-14)


Next: , Previous: Unit srfi-14, Up: Supported language

5.14 Unit srfi-25

Multi-dimensional array, see the documentation for SRFI-25

On systems that support dynamic loading, the srfi-25 unit can be made available in the interpreter (csi) by entering

(require-extension srfi-25)


Next: , Previous: Unit srfi-25, Up: Supported language

5.15 Unit match

Andrew Wright's pattern matching package. Note that to use the macros in normal compiled code it is not required to declare this unit as used. Only if forms containing these macros are to be expanded at runtime, this is needed.

— syntax: match
— syntax: match-lambda
— syntax: match-lambda*
— syntax: match-let
— syntax: match-let*
— syntax: match-letrec
— syntax: match-define
          (match EXP CLAUSE ...)
          (match-lambda CLAUSE ...)
          (match-lambda* CLAUSE ...)
          (match-let ((PAT EXP) ...) BODY)
          (match-let* ((PAT EXP) ...) BODY)
          (match-letrec ((PAT EXP) ...) BODY)
          (match-define PAT EXP)
     

Match expression or procedure arguments with pattern and execute associated expressions. A Postscript manual is available .

— syntax: define-structure
— syntax: define-const-structure
          (define-structure (ID_0 ID_1 ... ID_N))
          (define-structure (ID_0 ID_1 ... ID_N)
                            ((ID_N+1 EXP_1) ...
                             (ID_N+M EXP_M)))
          (define-const-structure (ID_0 ARG_1 ... ARG_N))
          (define-const-structure (ID_0 ARG_1 ... ARG_N)
                                  ((ARG_N+1 EXP_1) ...
                                         (ARG_N+M EXP_M)))
     

Macros for defining record structures that can be decomposed by match.

— procedure: match-error-control
          (match-error-control [MODE])
     

Selects a mode that specifies how match... macro forms are to be expanded. With no argument this procedure returns the current mode. A single argument specifies the new mode that decides what should happen if no match-clause applies. The following modes are supported:

#:error
Signal an error. This is the default.
#:match
Signal an error and output the offending form.
#:fail
Omits pair? tests when the consequence is to fail in car or cdr rather than to signal an error.
#:unspecified
Non-matching expressions will either fail in car or cdr or return an unspecified value. This mode applies to files compiled with the unsafe option or declaration.

When an error is signalled, the raised exception will be of kind (exn match).

Note: match:structure-control is not available. Structures defined by the macros provided in this unit are always implemented as vectors. match:runtime-structures is also not available.

To use the pattern matching macros with the highlevel (syntax-case) macro system (require-extension match).


Next: , Previous: Unit match, Up: Supported language

5.16 Unit regex

This library unit provides support for regular expressions. The flavor depends on the particular installation platform:

— procedure: grep
          (grep REGEX LIST)
     

Returns all items of LIST that match the regular expression REGEX. This procedure could be defined as follows:

     
     (define (grep regex lst)
       (filter (lambda (x) (string-match regex x)) lst) )
— procedure: pattern->regexp
          (pattern->regexp PATTERN)
     

Converts the file-pattern PATTERN into a regular expression.

     
     (pattern->regexp "foo.*") ==> "foo\..*"
— procedure: regexp
          (regexp STRING)
     

Returns a precompiled regular expression object for string.

— procedure: regexp?
          (regexp? X)
     

Returns #t if X is a precompiled regular expression, or #f otherwise.

— procedure: string-match
— procedure: string-match-positions
          (string-match REGEXP STRING [START])
          (string-match-positions REGEXP STRING [START])
     

Matches the regular expression in REGEXP (a string or a precompiled regular expression) with STRING and returns either #f if the match failed, or a list of matching groups, where the first element is the complete match. If the optional argument START is supplied, it specifies the starting position in STRING. For each matching group the result-list contains either: #f for a non-matching but optional group; a list of start- and end-position of the match in STRING (in the case of string-match-positions); or the matching substring (in the case of string-match). Note that the exact string is matched. For searching a pattern inside a string, see below.

— procedure: string-search
— procedure: string-search-positions
          (string-search REGEXP STRING [START [RANGE]])
          (string-search-positions REGEXP STRING [START [RANGE]])
     

Searches for the first match of the regular expression in REGEXP with STRING. The search can be limited to RANGE characters.

— procedure: string-split-fields
          (string-split-fields REGEXP STRING [MODE [START]])
     

Splits STRING into a list of fields according to MODE, where MODE can be the keyword #:infix (REGEXP matches field separator), the keyword #:suffix (REGEXP matches field terminator) or #t (REGEXP matches field), which is the default.

— procedure: string-substitute
          (string-substitute REGEXP SUBST STRING [INDEX])
     

Searches substrings in STRING that match REGEXP and substitutes them with the string SUBST. The substitution can contain references to subexpressions in REGEXP with the \NUM notation, where NUM refers to the NUMth parenthesized expression. The optional argument INDEX defaults to 1 and specifies the number of the match to be substituted. Any non-numeric index specifies that all matches are to be substituted.

     
     (string-substitute "([0-9]+) (eggs|chicks)"
                        "2 (1)" "99 eggs or 99 chicks" 2)
      ==> "99 eggs or chicks (99)"
— procedure: string-substitute*
          (string-substitute* STRING SMAP)
     

Substitutes elements of STRING according to SMAP. SMAP should be an association-list where each element of the list is a pair of the form (MATCH . REPLACEMENT). Every occurrence of the regular expression MATCH in STRING will be replaced by the string REPLACEMENT

     
     (string-substitute* "<h1>Hello, world!</h1>"
                         '(("<[/A-Za-z0-9]+>" . ""))))
     
     ==>  "Hello, world!"


Next: , Previous: Unit regex, Up: Supported language

5.17 Unit syntax-case

Hieb's and Dybvig's hygienic macro package. Provides syntax-case and syntax-rules. A postscript manual can be found here: Technical Report #356

Notes:

Here is a small example that demonstrates separate compilation:

Let's say we have one file foo.scm that defines a module:

     (module foo (pr (bar baz))
       (define pr print)
       (define baz 99)
       (define-syntax bar
         (syntax-rules ()
           [(_ x) (list baz 'x)] ) ) )

and another that uses it (use-foo.scm):

     (load "foo.so")   ; (require 'foo) would also work
     
     (import foo)
     (pr (bar hello))

Compiling the files like this will lead to one dynamically loadable library and a plain executable:

$ csc -s -hygienic foo.scm
$ csc -hygienic use-foo.scm
$ use-foo
$ (99 hello)

The export declaration can be used to export identifiers exported from modules defined in a given source file. These exports will then be visible as normal toplevel variables in external code that loads or links with this file. The rationale behind this is to make it possible to create libraries and extensions that use modules internally, but still can be used in client code that doesn't use modules.

— procedure: visit
          (visit FILENAME)
     

Reads all toplevel expressions from the given file and expands all syntax, extracting module- and syntax-information to be subsequently used during the current compilation or interpretation of modules.


Next: , Previous: Unit syntax-case, Up: Supported language

5.18 Unit srfi-18

A simple multithreading package. This threading package follows largely the specification of SRFI-18. For more information see the documentation for SRFI-18

Notes:

The following procedures are provided, in addition to the procedures defined in SRFI-18:

— procedure: thread-deliver-signal!
          (thread-deliver-signal! THREAD X)
     

This will cause THREAD to signal the condition X once it is scheduled for execution. After signalling the condition, the thread continues with its normal execution.

— procedure: thread-quantum
          (thread-quantum THREAD)
     

Returns the quantum of THREAD, which is an exact integer specifying the approximate time-slice of the thread.

— procedure: thread-quantum-set!
          (thread-quantum-set! THREAD QUANTUM)
     

Sets the quantum of THREAD to QUANTUM.


Next: , Previous: Unit srfi-18, Up: Supported language

5.19 Unit format

— procedure: format
          (format DESTINATION FORMAT-STRING . ARGUMENTS)
     

An almost complete implementation of Common LISP format description according to the CL reference book Common LISP from Guy L. Steele, Digital Press. This code was originally part of SLIB. The author is Dirk Lutzebaeck.

Returns #t, #f or a string; has side effect of printing according to FORMAT-STRING. If DESTINATION is #t, the output is to the current output port and #t is returned. If DESTINATION is #f, a formatted string is returned as the result of the call. If DESTINATION is a string, DESTINATION is regarded as the format string; FORMAT-STRING is then the first argument and the output is returned as a string. If DESTINATION is a number, the output is to the value of (current-error-port). Otherwise DESTINATION must be an output port and #t is returned.

FORMAT-STRING must be a string. In case of a formatting error format returns #f and prints a message on the value of (current-error-port). Characters are output as if the string were output by the display function with the exception of those prefixed by a tilde (~). For a detailed description of the FORMAT-STRING syntax please consult a Common LISP format reference manual. A list of all supported, non-supported and extended directives can be found in format.txt.

This unit uses definitions from the extras unit.

format implements SRFI-28


Next: , Previous: Unit format, Up: Supported language

5.20 Unit posix

This unit provides services as used on many UNIX-like systems. Note that the following definitions are not available on non-UNIX systems like Windows or DOS.

This unit uses the regex and extras units.

All errors related to failing file-operations will signal a condition of kind (exn i/o file).


Next: , Up: Unit posix

5.20.1 Directories

— procedure: change-directory
          (change-directory NAME)
     

Changes the current working directory to NAME.

— procedure: current-directory
          (current-directory)
     

Returns the name of the current working directory.

— procedure: create-directory
          (create-directory NAME)
     

Creates a directory with the pathname NAME.

— procedure: delete-directory
          (delete-directory NAME)
     

Deletes the directory with the pathname NAME. The directory has to be empty.

— procedure: directory
          (directory PATHNAME)
     

Returns a list with all files that are contained in the directory with the name PATHNAME.

— procedure: directory?
          (directory? NAME)
     

Returns #t if there exists a file with the name NAME and if that file is a directory, or #f otherwise.

— procedure: glob
          (glob PATTERN1 ...)
     

Returns a list of the pathnames of all existing files matching PATTERN1 ..., which should be strings containing the usual file-patterns (with * matching zero or more characters and ? matching zero or one character).


Next: , Previous: Directories, Up: Unit posix

5.20.2 Pipes

— procedure: call-with-input-pipe
— procedure: call-with-output-pipe
          (call-with-input-pipe CMDLINE PROC [MODE])
          (call-with-output-pipe CMDLINE PROC [MODE])
     

Call PROC with a single argument: a input- or output port for a pipe connected to the subprocess named in CMDLINE. If PROC returns normally, the pipe is closed and any result values are returned.

— procedure: close-input-pipe
— procedure: close-output-pipe
          (close-input-pipe PORT)
          (close-output-pipe PORT)
     

Closes the pipe given in PORT and waits until the connected subprocess finishes.

— procedure: create-pipe
          (create-pipe)
     

The fundamental pipe-creation operator. Calls the C function pipe() and returns 2 values: the file-descriptors of the input- and output-ends of the pipe.

— procedure: open-input-pipe
          (open-input-pipe CMDLINE [MODE])
     

Spawns a subprocess with the command-line string CMDLINE and returns a port, from which the output of the process can be read. If MODE is specified, it should be the keyword #:text (the default) or #:binary.

— procedure: open-output-pipe
          (open-output-pipe CMDLINE [MODE])
     

Spawns a subprocess with the command-line string CMDLINE and returns a port. Anything written to that port is treated as the input for the process. If MODE is specified, it should be the keyword #:text (the default) or #:binary.

— limit: pipe/buf

This variable contains the maximal number of bytes that can be written atomically into a pipe or FIFO.

— procedure: with-input-from-pipe
— procedure: with-output-to-pipe
          (with-input-from-pipe CMDLINE THUNK [MODE])
          (with-output-to-pipe CMDLINE THUNK [MODE])
     

Temporarily set the value of current-input-port/current-output-port to a port for a pipe connected to the subprocess named in CMDLINE and call the procedure THUNK with no arguments. After THUNK returns normally the pipe is closed and the standard input-/output port is restored to its previous value and any result values are returned.

     
     (with-output-to-pipe 
       "gs -dNOPAUSE -sDEVICE=jpeg -dBATCH -sOutputFile=signballs.jpg -g600x600 -q -"
       (lambda ()
         (print #<<EOF
     %!IOPSC-1993 %%Creator: HAYAKAWA Takashi<xxxxxxxx@xx.xxxxxx.xx.xx>
     /C/neg/d/mul/R/rlineto/E/exp/H{{cvx def}repeat}def/T/dup/g/gt/r/roll/J/ifelse 8
     H/A/copy(z&v4QX&93r9AxYQOZomQalxS2w!!O&vMYa43d6r93rMYvx2dca!D&cjSnjSnjjS3o!v&6A
     X&55SAxM1CD7AjYxTTd62rmxCnTdSST0g&12wECST!&!J0g&D1!&xM0!J0g!l&544dC2Ac96ra!m&3A
     F&&vGoGSnCT0g&wDmlvGoS8wpn6wpS2wTCpS1Sd7ov7Uk7o4Qkdw!&Mvlx1S7oZES3w!J!J!Q&7185d
     Z&lx1CS9d9nE4!k&X&MY7!&1!J!x&jdnjdS3odS!N&mmx1C2wEc!G&150Nx4!n&2o!j&43r!U&0777d
     ]&2AY2A776ddT4oS3oSnMVC00VV0RRR45E42063rNz&v7UX&UOzF!F!J![&44ETCnVn!a&1CDN!Y&0M
     V1c&j2AYdjmMdjjd!o&1r!M){( )T 0 4 3 r put T(/)g{T(9)g{cvn}{cvi}J}{($)g[]J}J
     cvx}forall/moveto/p/floor/w/div/S/add 29 H[{[{]setgray fill}for Y}for showpage
     EOF
     ) ) )


Next: , Previous: Pipes, Up: Unit posix

5.20.3 Fifos

— procedure: create-fifo
          (create-fifo FILENAME [MODE])
     

Creates a FIFO with the name FILENAME and the permission bits MODE, which defaults to

     
     (+ perm/irwxu perm/irwxg perm/irwxo)
— procedure: fifo?
          (fifo? FILENAME)
     

Returns #t if the file with the name FILENAME names a FIFO.


Next: , Previous: Fifos, Up: Unit posix

5.20.4 File descriptors and low-level I/O

— procedure: duplicate-fileno
          (duplicate-fileno OLD [NEW])
     

If NEW is given, then the file-descriptor NEW is opened to access the file with the file-descriptor OLD. Otherwise a fresh file-descriptor accessing the same file as OLD is returned.

— procedure: file-close
          (file-close FILENO)
     

Closes the input/output file with the file-descriptor FILENO.

— procedure: file-open
          (file-open FILENAME FLAGS [MODE])
     

Opens the file specified with the string FILENAME and open-flags FLAGS using the C function open(). On success a file-descriptor for the opened file is returned. FLAGS should be a bitmask containing one or more of the open/... values ored together using bitwise-ior (or simply added together). The optional MODE should be a bitmask composed of one or more permission values like perm/irusr and is only relevant when a new file is created. The default mode is perm/irwxu | perm/irgrp | perm/iroth.

— procedure: file-mkstemp
          (file-mkstemp TEMPLATE-FILENAME)
     

Create a file based on the given TEMPLATE-FILENAME, in which the six last characters must be “XXXXXX”. These will be replaced with a string that makes the filename unique. The file descriptor of the created file and the generated filename is returned. See the mkstemp(3) manual page for details on how this function works. The template string given is not modified.

Example usage:

     
     (let-values (((fd temp-path) (file-mkstemp "/tmp/mytemporary.XXXXXX")))
       (let ((temp-port (open-output-file* fd)))
         (format temp-port "This file is ~A.~%" temp-path)
         (close-output-port temp-port)))
— procedure: file-read
          (file-read FILENO SIZE [BUFFER])
     

Reads SIZE bytes from the file with the file-descriptor FILENO. If a string or bytevector is passed in the optional argument BUFFER, then this string will be destructively modified to contain the read data. This procedure returns a list with two values: the buffer containing the data and the number of bytes read.

— procedure: file-select
          (file-select READFDLIST WRITEFDLIST [TIMEOUT])
     

Waits until any of the file-descriptors given in the lists READFDLIST and WRITEFDLIST is ready for input or output, respectively. If the optional argument TIMEOUT is given and not false, then it should specify the number of seconds after which the wait is to be aborted. This procedure returns two values: the lists of file-descriptors ready for input and output, respectively. READFDLIST and WRITEFDLIST may also by file-descriptors instead of lists. In this case the returned values are booleans indicating whether input/output is ready by #t or #f otherwise. You can also pass #f as READFDLIST or WRITEFDLIST argument, which is equivalent to ().

— procedure: file-write
          (file-write FILENO BUFFER [SIZE])
     

Writes the contents of the string or bytevector BUFFER into the file with the file-descriptor FILENO. If the optional argument SIZE is given, then only the specified number of bytes are written.

— file descriptor: fileno/stdin
— file descriptor: fileno/stdout
— file descriptor: fileno/stderr

These variables contain file-descriptors for the standard I/O files.

— flag: open/rdonly
— flag: open/wronly
— flag: open/rdwr
— flag: open/read
— flag: open/write
— flag: open/creat
— flag: open/append
— flag: open/excl
— flag: open/noctty
— flag: open/nonblock
— flag: open/trunc
— flag: open/sync
— flag: open/fsync
— flag: open/binary
— flag: open/text

Flags for use with file-open.

— procedure: open-input-file*
— procedure: open-output-file*
          (open-input-file* FILENO [OPENMODE])
          (open-output-file* FILENO [OPENMODE])
     

Opens file for the file-descriptor FILENO for input or output and returns a port. FILENO should be a positive exact integer. OPENMODE specifies an additional mode for opening the file (currently only the keyword #:append is supported, which opens an output-file for appending).

— procedure: port->fileno
          (port->fileno PORT)
     

If PORT is a file-port, then a file-descriptor is returned for this port. Otherwise an error is signaled.


Next: , Previous: File descriptors and low-level I/O, Up: Unit posix

5.20.5 Retrieving file attributes

— procedure: file-modification-time
          (file-modification-time FILE)
     

Returns time (in seconds) of the last modification of FILE. FILE may be a filename or a file-descriptor. If the file does not exist, an error is signaled.

— procedure: file-position
          (file-position FILE)
     

Returns the current file position of FILE, which should be a port or a file-descriptor.

— procedure: file-size
          (file-size FILENAME)
     

Returns the size of the file designated by FILE. FILE may be a filename or a file-descriptor. If the file does not exist, an error is signaled.


Next: , Previous: Retrieving file attributes, Up: Unit posix

5.20.6 Changing file attributes

— procedure: file-truncate
          (file-truncate FILE OFFSET)
     

Truncates the file FILE to the length OFFSET, which should be an integer. If the file-size is smaller or equal to OFFSET then nothing is done. FILE should be a filename or a file-descriptor.

— procedure: set-file-position!
          (set-file-position! FILE POSITION [WHENCE])
     

Sets the current read/write position of FILE to POSITION, which should be an exact integer. FILE should be a port or a file-descriptor. WHENCE specifies how the position is to interpreted and should be one of the values seek/set, seek/cur and seek/end. It defaults to seek/set.

Exceptions: (exn bounds), (exn i/o file)


Next: , Previous: Changing file attributes, Up: Unit posix

5.20.7 Processes

— procedure: current-process-id
          (current-process-id)
     

Returns the process ID of the current process.

— procedure: parent-process-id
          (parent-process-id)
     

Returns the process ID of the parent of the current process.

— procedure: process-execute
          (process-execute PATHNAME [LIST])
     

Creates a new child process and replaces the running process with it using the UNIX system call execv(). If the optional argument LIST is given, then it should contain a list of strings which are passed as arguments to the subprocess.

— procedure: process-fork
          (process-fork [THUNK])
     

Creates a new child process with the UNIX system call fork(). Returns either the PID of the child process or 0. If THUNK is given, then the child process calls it as a procedure with no arguments and terminates.

— procedure: process-run
          (process-run PATHNAME [LIST])
     

Creates a new child process using the UNIX system call fork() that executes the program given by the string PATHNAME using the UNIX system call execv(). The PID of the new process is returned. If LIST is not specified, then PATHNAME is passed to a program named by the environment variable SHELL (or /bin/sh, if the variable is not defined), so usual argument expansion can take place.

— procedure: process-signal
          (process-signal PID [SIGNAL])
     

Sends SIGNAL to the process with the id PID using the UNIX system call kill(). SIGNAL defaults to the value of the variable signal/term.

— procedure: process-wait
          (process-wait [PID [NOHANG]])
     

Suspends the current process until the child process with the id PID has terminated using the UNIX system call waitpid(). If PID is not given, then this procedure waits for any child process. If NOHANG is given and not #f then the current process is not suspended. This procedure returns three values:

— procedure: process
          (process COMMANDLINE)
     

Passes the string COMMANDLINE to the host-system's shell that is invoked as a subprocess and returns three values: an input port from which data written by the sub-process can be read, an output port from which any data written to will be received as input in the sub-process and the process-id of the started sub-process. All I/O from/to ports returned by process is fully nonblocking.

— procedure: sleep
          (sleep SECONDS)
     

Puts the process to sleep for SECONDS. Returns either 0 if the time has completely elapsed, or the number of remaining seconds, if a signal occurred.


Next: , Previous: Processes, Up: Unit posix

5.20.8 Symbolic links

— procedure: create-symbolic-link
          (create-symbolic-link OLDNAME NEWNAME)
     

Creates a symbolic link with the filename NEWNAME that points to the file named OLDNAME.

— procedure: read-symbolic-link
          (read-symbolic-link FILENAME)
     

Returns the filename to which the symbolic link FILENAME points.


Next: , Previous: Symbolic links, Up: Unit posix

5.20.9 Permissions, owners, users and groups

— procedure: file-owner
          (file-owner FILE)
     

Returns the user-id of FILE. FILE may be a filename or a file-descriptor.

— procedure: file-permissions
          (file-permissions FILE)
     

Returns the permission bits for FILE. You can test this value by performing bitwise operations on the result and the perm/... values. FILE may be a filename or a file-descriptor.

— procedure: file-read-access?
— procedure: file-write-access?
— procedure: file-execute-access?
          (file-read-access? FILENAME)
          (file-write-access? FILENAME)
          (file-execute-access? FILENAME)
     

These procedures return #t if the current user has read, write or execute permissions on the file named FILENAME.

— procedure: change-file-mode
          (change-file-mode FILENAME MODE)
     

Changes the current file mode of the file named FILENAME to MODE using the chmod() system call. The perm/... variables contain the various permission bits and can be combinded with the bitwise-ior procedure.

— procedure: change-file-owner
          (change-file-owner FILENAME UID GID)
     

Changes the owner information of the file named FILENAME to the user- and group-ids UID and GID (which should be exact integers) using the chown() system call.

— procedure: current-user-id
— procedure: current-group-id
— procedure: current-effective-user-id
— procedure: current-effective-group-id
          (current-user-id)
          (current-group-id)
          (current-effective-user-id)
          (current-effective-group-id)
     

Return the user- and group-ids of the current process.

— procedure: group-information
          (group-information GROUP)
     

If GROUP specifies a valid group-name or group-id, then this procedure returns four values: the group-name, the encrypted group password, the group ID and a list of the names of all group members. If no group with the given name or ID exists, then #f is returned.

— procedure: get-groups
          (get-groups)
     

Returns a list with the supplementary group IDs of the current user.

— procedure: set-groups!
          (set-groups! GIDLIST)
     

Sets the supplementrary group IDs of the current user to the IDs given in the list GIDLIST.

Only the superuser may invoke this procedure.

— procedure: initialize-groups
          (initialize-groups USERNAME BASEGID)
     

Sets the supplementrary group IDs of the current user to the IDs from the user with name USERNAME (a string), including BASEGID.

Only the superuser may invoke this procedure.

— permission bits: perm/irusr
— permission bits: perm/iwusr
— permission bits: perm/ixusr
— permission bits: perm/irgrp
— permission bits: perm/iwgrp
— permission bits: perm/ixgrp
— permission bits: perm/iroth
— permission bits: perm/iwoth
— permission bits: perm/ixoth
— permission bits: perm/irwxu
— permission bits: perm/irwxg
— permission bits: perm/irwxo
— permission bits: perm/isvtx
— permission bits: perm/isuid
— permission bits: perm/isgid

These variables contain permission bits as used in change-file-mode.

— procedure: set-user-id!
          (set-user-id! UID)
     

Sets the effective user id of the current process to UID, which should be a positive integer.

— procedure: user-information
          (user-information USER)
     

If USER specifes a valid username (as a string) or user ID, then the user database is consulted and a list of 7 values are returned: the user-name, the encrypted password, the user ID, the group ID, a user-specific string, the home directory and the default shell. If no user with this name or ID can be found, then #f is returned.


Next: , Previous: Permissions, Up: Unit posix

5.20.10 Record locking

— procedure: file-lock
          (file-lock PORT [START [LEN]])
     

Locks the file associated with PORT for reading or writing (according to whether PORT is an input- or output-port). START specifies the starting position in the file to be locked and defaults to 0. LEN specifies the length of the portion to be locked and defaults to #t, which means the complete file. file-lock returns a “lock”-object.

— procedure: file-lock/blocking
          (file-lock/blocking PORT [START [LEN]])
     

Similar to file-lock, but if a lock is held on the file, the current process blocks (including all threads) until the lock is released.

— procedure: file-test-lock
          (file-test-lock PORT [START [LEN]])
     

Tests whether the file associated with PORT is locked for reading or writing (according to whether PORT is an input- or output-port) and returns either #f or the process-id of the locking process.

— procedure: file-unlock
          (file-unlock LOCK)
     

Unlocks the previously locked portion of a file given in LOCK.


Next: , Previous: Record locking, Up: Unit posix

5.20.11 Signal handling

— procedure: set-alarm!
          (set-alarm! SECONDS)
     

Sets an internal timer to raise the signal/alrm after SECONDS are elapsed. You can use the set-signal-handler! procedure to write a handler for this signal.

— procedure: set-signal-handler!
          (set-signal-handler! SIGNUM PROC)
     

Establishes the procedure of one argument PROC as the handler for the signal with the code SIGNAL. PROC is called with the signal number as its sole argument. If the argument PROC is #f then this signal will be ignored.

— procedure: set-signal-mask!
          (set-signal-mask! SIGLIST)
     

Sets the signal mask of the current process to block all signals given in the list SIGLIST. Signals masked in that way will not be delivered to the current process.

— signal code: signal/term
— signal code: signal/kill
— signal code: signal/int
— signal code: signal/hup
— signal code: signal/fpe
— signal code: signal/ill
— signal code: signal/segv
— signal code: signal/abrt
— signal code: signal/trap
— signal code: signal/quit
— signal code: signal/alrm
— signal code: signal/vtalrm
— signal code: signal/prof
— signal code: signal/io
— signal code: signal/urg
— signal code: signal/chld
— signal code: signal/cont
— signal code: signal/stop
— signal code: signal/tstp
— signal code: signal/pipe
— signal code: signal/xcpu
— signal code: signal/xfsz
— signal code: signal/usr1
— signal code: signal/usr2
— signal code: signal/winch

These variables contain signal codes for use with process-signal or set-signal-handler!.


Next: , Previous: Signal handling, Up: Unit posix

5.20.12 Environment access

— procedure: current-environment
          (current-environment)
     

Returns a association list of the environment variables and their current values.

Note: Under Mac OS X, this procedure always returns the empty list.

— procedure: setenv
          (setenv VARIABLE VALUE)
     

Sets the environment variable named VARIABLE to VALUE. Both arguments should be strings. If the variable is not defined in the environment, a new definition is created.

— procedure: unsetenv
          (unsetenv VARIABLE)
     

Removes the definition of the environment variable VARIABLE from the environment of the current process. If the variable is not defined, nothing happens.


Next: , Previous: Environment access, Up: Unit posix

5.20.13 Memory mapped I/O

— procedure: map-file-to-memory
          (map-file-to-memory ADDRESS LEN PROTECTION FLAG FILENO [OFFSET])
     

Maps a section of a file to memory using the C function mmap(). ADDRESS should be a foreign pointer object or #f; LEN specifies the size of the section to be mapped; PROTECTION should be one or more of the flags prot/read, prot/write, prot/exec or prot/none bitwise-iored together; FLAG should be one or more of the flags map/fixed, map/shared, map/private, map/anonymous or map/file; FILENO should be the file-descriptor of the mapped file. The optional argument OFFSET gives the offset of the section of the file to be mapped and defaults to 0. This procedure returns an object representing the mapped file section. The procedure move-memory! can be used to access the mapped memory.

— procedure: memory-mapped-file-pointer
          (memory-mapped-file-pointer MMAP)
     

Returns a machine pointer to the start of the memory region to which the file is mapped.

— procedure: unmap-file-from-memory
          (unmap-file-from-memory MMAP [LEN])
     

Unmaps the section of a file mapped to memory using the C function munmap(). MMAP should be a mapped file as returned by the procedure map-file-to-memory. The optional argument LEN specifies the length of the section to be unmapped and defaults to the complete length given when the file was mapped.


Next: , Previous: Memory mapped I/O, Up: Unit posix

5.20.14 Time routines

— procedure: seconds->local-time
          (seconds->local-time SECONDS)
     

Breaks down the time value represented in SECONDS into a 10 element vector of the form #(seconds minutes hours mday month year wday yday dstflag timezone), in the following format:

— procedure: seconds->string
          (seconds->string SECONDS)
     

Converts the local time represented in SECONDS into a string of the form "Tue May 21 13:46:22 1991\n".

— procedure: seconds->utc-time
          (seconds->utc-time SECONDS)
     

Similar to seconds->local-time, but interpretes SECONDS as UTC time.

— procedure: time->string
          (time->string VECTOR)
     

Converts the broken down time represented in the 10 element vector VECTOR into a string of the form "Tue May 21 13:46:22 1991\n".


Next: , Previous: Time routines, Up: Unit posix

5.20.15 Raw exit

— procedure: _exit
          (_exit [CODE])
     

Exits the current process without flushing any buffered output (using the C function _exit). Note that the exit-handler is not called when this procedure is invoked. The optional return-code CODE defaults to 0.


Next: , Previous: Raw exit, Up: Unit posix

5.20.16 ERRNO values

— error code: errno/perm
— error code: errno/noent
— error code: errno/srch
— error code: errno/intr
— error code: errno/io
— error code: errno/noexec
— error code: errno/badf
— error code: errno/child
— error code: errno/nomem
— error code: errno/acces
— error code: errno/fault
— error code: errno/busy
— error code: errno/notdir
— error code: errno/isdir
— error code: errno/inval
— error code: errno/mfile
— error code: errno/nospc
— error code: errno/spipe
— error code: errno/pipe
— error code: errno/again
— error code: errno/rofs
— error code: errno/wouldblock

These variables contain error codes as returned by errno.


Next: , Previous: ERRNO values, Up: Unit posix

5.20.17 Finding files

— procedure: find-files
          (find-files DIRECTORY PREDICATE [ACTION [IDENTITY [LIMIT]]])
     

Recursively traverses the contents of DIRECTORY (which should be a string) and invokes the procedure ACTION for all files for which the procedure PREDICATE is true. PREDICATE may me a procedure of one argument or a regular-expression string. ACTION should be a procedure of two arguments: the currently encountered file and the result of the previous invocation of ACTION, or, if this is the first invocation, the value of IDENTITY. ACTION defaults to cons, IDENTITY defaults to (). LIMIT should a procedure of one argument that is called for each nested directory and which should return true, if that directory is to be traversed recursively. LIMIT may also be an exact integer that gives the maximum recursion depth. A depth of 0 means the files in the specified directory are traversed but not any nested directories. LIMIT may also be #f (the default), which is equivalent to (constantly #t).

Note that ACTION is called with the full pathname of each file, including the directory prefix.


Next: , Previous: Finding files, Up: Unit posix

5.20.18 Getting the hostname and system information

— procedure: get-host-name
          (get-host-name)
     

Returns the hostname of the machine that this process is running on.

— procedure: system-information
          (system-information)
     

Invokes the UNIX system call uname() and returns 5 values: system-name, node-name, OS release, OS version and machine.


Next: , Previous: Getting the hostname and system information, Up: Unit posix

5.20.19 Setting a files buffering mode

— procedure: set-buffering-mode!
          (set-buffering-mode! PORT MODE [BUFSIZE])
     

Sets the buffering-mode for the file associated with PORT to MODE, which should be one of the keywords #:full, #:line or #:none. If BUFSIZE is specified it determines the size of the buffer to be used (if any).


Next: , Previous: Setting a files buffering mode, Up: Unit posix

5.20.20 Terminal ports

— procedure: terminal-name
          (terminal-name PORT)
     

Returns the name of the terminal that is connected to PORT.

— procedure: terminal-port?
          (terminal-port? PORT)
     

Returns #t if PORT is connected to a terminal and #f otherwise.


Previous: Terminal ports, Up: Unit posix

5.20.21 How Scheme procedures relate to UNIX C functions

change-directory
chdir
change-file-mode
chmod
change-file-owner
chown
create-directory
mkdir
create-fifo
mkfifo
create-pipe
pipe
create-symbolic-link
link
current-directory
curdir
current-effective-groupd-id
getegid
current-effective-user-id
geteuid
current-group-id
getgid
current-parent-id
getppid
current-process-id
getpid
current-user-id
getuid
delete-directory
rmdir
duplicate-fileno
dup/dup2
_exit
_exit
file-close
close
file-execute-access?
access
file-open
open
file-lock
fcntl
file-position
ftell/lseek
file-read
read
file-read-access?
access
file-select
select
file-test-lock
fcntl
file-truncate
truncate/ftruncate
file-unlock
fcntl
file-write
write
file-write-access?
access
get-groups
getgroups
get-host-name
gethostname
initialize-groups
initgroups
map-file-to-memory
mmap
open-input-file*
fdopen
open-output-file*
fdopen
open-input-pipe
popen
open-output-pipe
popen
port->fileno
fileno
process-execute
execvp
process-fork
fork
process-signal
kill
process-wait
waitpid
close-input-pipe
pclose
close-output-pipe
pclose
read-symbolic-link
readlink
seconds->local-time
localtime
seconds->string
ctime
seconds->utc-time
gmtime
set-alarm!
alarm
set-buffering-mode!
setvbuf
set-file-position!
fseek/seek
set-groups!
setgroups
set-signal-mask!
sigprocmask
set-user-id!
setuid
setenv
setenv/putenv
sleep
sleep
system-information
uname
terminal-name
ttyname
terminal-port?
isatty
time->string
asctime
unsetenv
putenv
unmap-file-from-memory
munmap
user-information
getpwnam/getpwuid


Next: , Previous: Unit posix, Up: Supported language

5.21 Unit utils

This unit contains some utility procedures for Shell scripting and for some file operations.

This unit uses the extras and regex units.


Next: , Up: Unit utils

5.21.1 Pathname operations

— procedure: absolute-pathname?
          (absolute-pathname? PATHNAME)
     

Returns #t if the string PATHNAME names an absolute pathname, and returns #f otherwise.

— procedure: decompose-pathname
          (decompose-pathname PATHNAME)
     

Returns three values: the directory-, filename- and extension-components of the file named by the string PATHNAME. For any component that is not contained in PATHNAME, #f is returned.

— procedure: make-pathname
— procedure: make-absolute-pathname
          (make-pathname DIRECTORY FILENAME [EXTENSION])
          (make-absolute-pathname DIRECTORY FILENAME [EXTENSION])
     

Returns a string that names the file with the components DIRECTORY, FILENAME and (optionally) EXTENSION. DIRECTORY can be #f (meaning no directory component), a string or a list of strings. FILENAME and EXTENSION should be strings or #f. make-absolute-pathname returns always an absolute pathname.

— procedure: pathname-directory
          (pathname-directory PATHNAME)
     
— procedure: pathname-file
          (pathname-file PATHNAME)
     
— procedure: pathname-extension
          (pathname-extension PATHNAME)
     

Accessors for the components of PATHNAME. If the pathname does not contain the accessed component, then #f is returned.

— procedure: pathname-replace-directory
          (pathname-replace-directory PATHNAME DIRECTORY)
     
— procedure: pathname-replace-file
          (pathname-replace-file PATHNAME FILENAME)
     
— procedure: pathname-replace-extension
          (pathname-replace-extension PATHNAME EXTENSION)
     

Return a new pathname with the specified component of PATHNAME replaced by a new value.

— procedure: pathname-strip-directory
          (pathname-strip-directory PATHNAME)
     
— procedure: pathname-strip-extension
          (pathname-strip-extension PATHNAME)
     

Return a new pathname with the specified component of PATHNAME stripped.


Next: , Previous: Pathname operations, Up: Unit utils

5.21.2 Temporary files

— procedure: create-temporary-file
          (create-temporary-file [EXTENSION])
     

Creates an empty temporary file and returns its pathname. If EXTENSION is not given, then .tmp is used. If the environment variable TMPDIR, TEMP or TMP is set, then the pathname names a file in that directory.  


Next: , Previous: Temporary files, Up: Unit utils

5.21.3 Deleting a file without signalling an error

— procedure: delete-file*
          (delete-file* FILENAME)
     

If the file FILENAME exists, it is deleted and #t is returned. If the file does not exist, nothing happens and #f is returned.


Next: , Previous: Deleting a file without signalling an error, Up: Unit utils

5.21.4 Iterating over input lines and files

— procedure: for-each-line
          (for-each-line PROCEDURE [PORT])
     

Calls PROCEDURE for each line read from PORT (which defaults to the value of (current-input-port). The argument passed to PORCEDURE is a string with the contents of the line, excluding any line-terminators. When all input has been read from the port, for-each-line returns some unspecified value.

— procedure: for-each-argv-line
          (for-each-argv-line PROCEDURE)
     

Opens each file listed on the command line in order, passing one line at a time into PROCEDURE. The filename - is interpreted as (current-input-port). If no arguments are given on the command line it again uses the value of (current-input-port).

This code will act as a simple Unix cat(1) command:

     
     (for-each-argv-line print)


Next: , Previous: Iterating over input lines and files, Up: Unit utils

5.21.5 Executing shell commands with formatstring and error checking

— procedure: system*
          (system* FORMATSTRING ARGUMENT1 ...)
     

Similar to (system (sprintf FORMATSTRING ARGUMENT1 ...)), but signals an error if the invoked program should return a nonzero exit status.


Previous: Executing shell commands with formatstring and error checking, Up: Unit utils

5.21.6 Reading a file's contents

— procedure: read-all
          (read-all [FILE-OR-PORT])
     

If FILE-OR-PORT is a string, then this procedure returns the contents of the file as a string. If FILE-OR-PORT is a port, all remaining input is read and returned as a string. The port is not closed. If no argument is provided, input will be read from the port that is the current value of (current-input-port).


Next: , Previous: Unit utils, Up: Supported language

5.22 Unit tcp

This unit provides basic facilities for communicating over TCP sockets. The socket interface should be mostly compatible to the one found in PLT Scheme.

This unit uses the extras unit.

All errors related to failing network operations will raise a condition of kind (exn i/o network).

— procedure: tcp-listen
          (tcp-listen TCPPORT [BACKLOG [HOST]])
     

Creates and returns a TCP listener object that listens for connections on TCPPORT, which should be an exact integer. BACKLOG specifies the number of maximally pending connections (and defaults to 4). If the optional argument HOST is given and not #f, then only incoming connections for the given host (or IP) are accepted.

— procedure: tcp-listener?
          (tcp-listener? X)
     

Returns #t if X is a TCP listener object, or #f otherwise.

— procedure: tcp-close
          (tcp-close LISTENER)
     

Reclaims any resources associated with LISTENER.

— procedure: tcp-accept
          (tcp-accept LISTENER)
     

Waits until a connection is established on the port on which LISTENER is listening and returns two values: an input- and output-port that can be used to communicate with the remote process.

Note: this operation and any I/O on the ports returned will not block other running threads.

— procedure: tcp-accept-ready?
          (tcp-accept-ready? LISTENER)
     

Returns #t if there are any connections pending on LISTENER, or #f otherwise.

— procedure: tcp-listener-port
          (tcp-listener-port LISTENER)
     

Returns the port number assigned to LISTENER (If you pass 0 to tcp-listen, then the system will choose a port-number for you).

— procedure: tcp-connect
          (tcp-connect HOSTNAME [TCPPORT])
     

Establishes a client-side TCP connection to the machine with the name HOSTNAME (a string) at TCPPORT (an exact integer) and returns two values: an input- and output-port for communicating with the remote process.

Note: any I/O on the ports returned will not block other running threads.

— procedure: tcp-addresses
          (tcp-addresses PORT)
     

Returns two values for the input- or output-port PORT (which should be a port returned by either tcp-accept or tcp-connect): the IP address of the local and the remote machine that are connected over the socket associated with PORT. The returned addresses are strings in XXX.XXX.XXX.XXX notation.

— procedure: tcp-abandon-port
          (tcp-abandon-port PORT)
     

Marks the socket port PORT as abandoned. This is mainly useful to close down a port without breaking the connection.

A very simple example follows. Say we have the two files client.scm and server.scm:

; client.scm
(define-values (i o) (tcp-connect "localhost" 4242))
(write-line "Good Bye!" o)
(print (read-line i))
; server.scm
(define l (tcp-listen 4242))
(define-values (i o) (tcp-accept l))
(write-line "Hello!" o)
(print (read-line i))
(close-input-port i)
(close-output-port o)

% csi -script server.scm &
[1] 1409
% csi -script client.scm
Good Bye!
Hello!


Next: , Previous: Unit tcp, Up: Supported language

5.23 Unit srfi-37

Copyright (c) 2002 Anthony Carrico

A simple and flexible command-line option parsing facility. Options may be either short one-character options of the form -X[ARGUMENT] or long multicharacter ones of the form --XXX[=ARGUMENT]. Short options may be coalesced. An argument of the form -- stops option processing. For more information take a look at the SRFI-37 documentation.

An example:

#!/usr/local/bin/csi -script
;;;; secho - display command-line arguments

(define nl 1)

(define help 
  (option 
   '(#\h "help") #f #f
   (lambda _ 
     (print "Usage: secho [OPTION] ARG ...
  -h  --help          show this text
  -n  --newline N     add N newline characters (default: 1)")
     (exit) ) ) )

(define newlines
  (option
   '(#\n "newline") #t #f
   (lambda (o n x vals)
     (set! nl (string->number x))
     vals) ) )

(for-each 
 (lambda (x) (print* x #\space))
 (reverse
  (args-fold
   (command-line-arguments)
   (list help newlines)
   (lambda (o n x vals)
     (error "unrecognized option" n) )
   cons
   '() ) ) )

(display (make-string nl #\newline))


Next: , Previous: Unit srfi-37, Up: Supported language

5.24 Unit lolevel

This unit provides a number of handy low-level operations. Use at your own risk.

This unit uses the srfi-4 and extras units.


Next: , Up: Unit lolevel

5.24.1 Foreign pointers

— procedure: address->pointer
          (address->pointer ADDRESS)
     

Creates a new foreign pointer object initialized to point to the address given in the integer ADDRESS.

— procedure: allocate
          (allocate BYTES)
     

Returns a pointer to a freshly allocated region of static memory. This procedure could be defined as follows:

     
     (define allocate (foreign-lambda c-pointer "malloc" integer))
— procedure: free
          (free POINTER)
     

Frees the memory pointed to by POINTER. This procedure could be defined as follows:

     
     (define free (foreign-lambda c-pointer "free" integer))
— procedure: null-pointer
          (null-pointer)
     

Another way to say (address->pointer 0).

— procedure: null-pointer?
          (null-pointer? PTR)
     

Returns #t if PTR contains a NULL pointer, or #f otherwise.

— procedure: object->pointer
          (object->pointer X)
     

Returns a pointer pointing to the Scheme object X, which should be a non-immediate object. Note that data in the garbage collected heap moves during garbage collection.

— procedure: pointer?
          (pointer? X)
     

Returns #t if X is a foreign pointer object, and #f otherwise.

— procedure: pointer->address
          (pointer->address PTR)
     

Returns the address, to which the pointer PTR points.

— procedure: pointer->object
          (pointer->object PTR)
     

Returns the Scheme object pointed to by the pointer PTR.

— procedure: pointer-offset
          (pointer-offset PTR N)
     

Returns a new pointer representing the pointer PTR increased by N.

— procedure: pointer-u8-ref
          (pointer-u8-ref PTR)
     

Returns the unsigned byte at the address designated by PTR.

— procedure: pointer-s8-ref
          (pointer-s8-ref PTR)
     

Returns the signed byte at the address designated by PTR.

— procedure: pointer-u16-ref
          (pointer-u16-ref PTR)
     

Returns the unsigned 16-bit integer at the address designated by PTR.

— procedure: pointer-s16-ref
          (pointer-s16-ref PTR)
     

Returns the signed 16-bit integer at the address designated by PTR.

— procedure: pointer-u32-ref
          (pointer-u32-ref PTR)
     

Returns the unsigned 32-bit integer at the address designated by PTR.

— procedure: pointer-s32-ref
          (pointer-s32-ref PTR)
     

Returns the signed 32-bit integer at the address designated by PTR.

— procedure: pointer-f32-ref
          (pointer-f32-ref PTR)
     

Returns the 32-bit float at the address designated by PTR.

— procedure: pointer-f64-ref
          (pointer-f64-ref PTR)
     

Returns the 64-bit double at the address designated by PTR.

— procedure: pointer-u8-set!
          (pointer-u8-set! PTR N)
     

Stores the unsigned byte N at the address designated by PTR.

— procedure: pointer-s8-set!
          (pointer-s8-set! PTR N)
     

Stores the signed byte N at the address designated by PTR.

— procedure: pointer-u16-set!
          (pointer-u16-set! PTR N)
     

Stores the unsigned 16-bit integer N at the address designated by PTR.

— procedure: pointer-s16-set!
          (pointer-s16-set! PTR N)
     

Stores the signed 16-bit integer N at the address designated by PTR.

— procedure: pointer-u32-set!
          (pointer-u32-set! PTR N)
     

Stores the unsigned 32-bit integer N at the address designated by PTR.

— procedure: pointer-s32-set!
          (pointer-s32-set! PTR N)
     

Stores the 32-bit integer N at the address designated by PTR.

— procedure: pointer-f32-set!
          (pointer-f32-set! PTR N)
     

Stores the 32-bit floating-point number N at the address designated by PTR.

— procedure: pointer-f64-set!
          (pointer-f64-set! PTR N)
     

Stores the 64-bit floating-point number N at the address designated by PTR.

— procedure: align-to-word
          (align-to-word PTR-OR-INT)
     

Accepts either a machine pointer or an integer as argument and returns a new pointer or integer aligned to the native word size of the host platform.


Next: , Previous: Foreign pointers, Up: Unit lolevel

5.24.2 Tagged pointers

“Tagged” pointers are foreign pointer objects with an extra tag object.

— procedure: tag-pointer
          (tag-pointer PTR TAG)
     

Creates a new tagged pointer object from the foreign pointer PTR with the tag TAG, which may an arbitrary Scheme object.

— procedure: tagged-pointer?
          (tagged-pointer? X TAG)
     

Returns #t, if X is a tagged pointer object with the tag TAG (using an eq? comparison), or #f otherwise.

— procedure: pointer-tag
          (pointer-tag PTR)
     

If PTR is a tagged pointer object, its tag is returned. If PTR is a normal, untagged foreign pointer object #f is returned. Otherwise an error is signalled.


Next: , Previous: Tagged pointers, Up: Unit lolevel

5.24.3 Extending procedures with data

— procedure: extend-procedure
          (extend-procedure PROCEDURE X)
     

Returns a copy of the procedure PROCEDURE which contains an additional data slot initialized to X. If PROCEDURE is already an extended procedure, then its data slot is changed to contain X and the same procedure is returned.

— procedure: extended-procedure?
          (extended-procedure? PROCEDURE)
     

Returns #t if PROCEDURE is an extended procedure, or #f otherwise.

— procedure: procedure-data
          (procedure-data PROCEDURE)
     

Returns the data object contained in the extended procedure PROCEDURE.

— procedure: set-procedure-data!
          (set-procedure-data! PROCEDURE X)
     

Changes the data object contained in the extended procedure PROCEDURE to X.

     
     (define foo
       (letrec ((f (lambda () (procedure-data x)))
                (x #f) )
         (set! x (extend-procedure f 123))
         x) )
     (foo)                                         ==> 123
     (set-procedure-data! foo 'hello)
     (foo)                                         ==> hello


Next: , Previous: Extending procedures with data, Up: Unit lolevel

5.24.4 Bytevectors

— procedure: byte-vector
          (byte-vector FIXNUM ...)
     

Returns a freshly allocated byte-vector with FIXNUM ... as its initial contents.

— procedure: byte-vector?
          (byte-vector? X)
     

Returns #t if X is a byte-vector object, or #f otherwise.

— procedure: byte-vector-fill!
          (byte-vector-fill! BYTE-VECTOR N)
     

Sets each element of BYTE-VECTOR to N, which should be an exact integer.

— procedure: byte-vector->list
          (byte-vector->list BYTE-VECTOR)
     

Returns a list with elements taken from BYTE-VECTOR.

— procedure: byte-vector->string
          (byte-vector->string BYTE-VECTOR)
     

Returns a string with the contents of BYTE-VECTOR.

— procedure: byte-vector-length
          (byte-vector-length BYTE-VECTOR)
     

Returns the number of elements in BYTE-VECTOR.

— procedure: byte-vector-ref
          (byte-vector-ref BYTE-VECTOR INDEX)
     

Returns the byte at the INDEXth position of BYTE-VECTOR.

— procedure: byte-vector-set!
          (byte-vector-set! BYTE-VECTOR INDEX N)
     

Sets the byte at the INDEXth position of BYTE-VECTOR to the value of the exact integer n.

— procedure: executable-byte-vector->procedure
          (executable-byte-vector->procedure PBYTE-VECTOR)
     

Returns a procedure that on invocation will execute the code in PBYTE-VECTOR, which should have been allocated using make-executable-byte-vector. The procedure follows the native C calling convention, and will be called as if declared with the following prototype:

     
     void <procedure>(int argc, C_word closure, C_word k, C_word arg1, ...)

An example:

     
     (define x (make-executable-byte-vector 17))
     (move-memory! 
      '#u8(#x8b #x44 #x24 #x0c  ; movl 12(%esp), %eax - `k'
           #x8b #x5c #x24 #x10  ; movl 16(%esp), %ebx - `arg1'
           #x53                 ; pushl %ebx          - push result
           #x50                 ; pushl %eax          - push k
           #x6a #x02            ; pushl $2            - push argument count
           #x8b #x40 #x04       ; movl 4(%eax), %eax  - fetch code pointer
           #xff #xd0)           ; call *%eax
      x)
     (define y (executable-byte-vector->procedure x))
     (y 123)                                         ==> 123

The result of calling executable-byte-vector->procedure with a non-executable statically allocated byte-vector is undefined.

— procedure: invoke-executable-byte-vector
          (invoke-executable-byte-vector PBYTE-VECTOR ARG1 ...)
     

Invokes the machine code stored in the executable byte-vector PBYTE-VECTOR. The native C calling conventions are used, but the invoked code is passed a single argument containing a pointer to an array of the Scheme objects ARG1 ....

     
     (define v (make-executable-byte-vector 7))
     (move-memory!
      '#u8(#x8b #x44 #x24 #x04        ; movl 4(%esp), %eax
           #x8b #x00                  ; movl 0(%eax), %eax
           #xc3)                      ; ret
      v)
     (invoke-executable-byte-vector v "hello!") ==> "hello!"
— procedure: list->byte-vector
          (list->byte-vector LIST)
     

Returns a byte-vector with elements taken from LIST, where the elements of LIST should be exact integers.

— procedure: make-byte-vector
          (make-byte-vector SIZE [INIT])
     

Creates a new byte-vector of size SIZE. If INIT is given, then it should be an exact integer with which every element of the byte-vector is initialized.

— procedure: make-executable-byte-vector
          (make-executable-byte-vector SIZE [INIT])
     

As make-static-byte-vector, but the code is suitable for execution. Note: this feature is currently only available on x86 platforms.

Exceptions: (exn bounds), (exn runtime)

— procedure: make-static-byte-vector
          (make-static-byte-vector SIZE [INIT])
     

As make-byte-vector, but allocates the byte-vector in storage that is not subject to garbage collection. To free the allocated memory, one has to call object-release explicitly.

Exceptions: (exn bounds), (exn runtime)

— procedure: static-byte-vector->pointer
          (static-byte-vector->pointer PBYTE-VECTOR)
     

Returns a pointer object pointing to the data in the statically allocated byte-vector PBYTE-VECTOR.

— procedure: string->byte-vector
          (string->byte-vector STRING)
     

Returns a byte-vector with the contents of STRING.


Next: , Previous: Bytevectors, Up: Unit lolevel

5.24.5 Data in unmanaged memory

— procedure: object-evict
          (object-evict X [ALLOCATOR])
     

Copies the object X recursively into the memory pointed to by the foreign pointer object returned by ALLOCATOR, which should be a procedure of a single argument (the number of bytes to allocate). The freshly copied object is returned. This facility allows moving arbitrary objects into static memory, but care should be taken when mutating evicted data: setting slots in evicted vector-like objects to non-evicted data is not allowed. It is possible to set characters/bytes in evicted strings or byte-vectors, though. It is advisable not to evict ports, because they might be mutated by certain file-operations. object-evict is able to handle circular and shared structures, but evicted symbols are no longer unique: a fresh copy of the symbol is created, so

     
     (define x 'foo)
     (define y (object-evict 'foo))
     y                              ==> foo
     (eq? x y)                      ==> #f
     (define z (object-evict '(bar bar)))
     (eq? (car z) (cadr z))         ==> #t

The ALLOCATOR defaults to allocate.

— procedure: object-evict-to-location
          (object-evict-to-location X PTR [LIMIT])
     

As object-evict but moves the object at the address pointed to by the machine pointer PTR. If the number of copied bytes exceeds the optional LIMIT then an error is signalled. Two values are returned: the evicted object and a new pointer pointing to the first free address after the evicted object.

— procedure: object-evicted?
          (object-evicted? X)
     

Returns #t if X is a non-immediate evicted data object, or #f otherwise.

— procedure: object-size
          (object-size X)
     

Returns the number of bytes that would be needed to evict the data object X.

— procedure: object-release
          (object-release X [RELEASER])
     

Frees memory occupied by the evicted object X recursively. RELEASER should be a procedure of a single argument (a foreign pointer object to the static memory to be freed) and defaults to the C-library free().

— procedure: object-unevict
          (object-unevict X)
     

Copies the object X and nested objects back into the normal Scheme heap. Symbols are re-interned into the symbol table. Strings and byte-vectors are not copied.


Next: , Previous: Data in unmanaged memory, Up: Unit lolevel

5.24.6 Locatives

A locative is an object that points to an element of a containing object, much like a “pointer” in low-level, imperative programming languages like “C”. The element can be accessed and changed indirectly, by performing access or change operations on the locative. The container object can be computed by calling the location->object procedure.

Locatives may be passed to foreign procedures that expect pointer arguments. The effect of creating locatives for evicted data (see object-evict) is undefined.

— procedure: make-locative
          (make-locative EXP [INDEX])
     

Creates a locative that refers to the element of the non-immediate object EXP at position INDEX. EXP may be a vector, symbol, pair, string, byte-vector, SRFI-4 number-vector, SRFI-25 array, or record. INDEX should be a fixnum, or a valid index into a SRFI-25 array. INDEX defaults to 0.

— procedure: make-weak-locative
          (make-weak-locative EXP [INDEX])
     

Creates a “weak” locative. Even though the locative refers to an element of a container object, the container object will still be reclaimed by garbage collection if no other references to it exist.

— procedure: locative?
          (locative? X)
     

Returns #t if X is a locative, or #f otherwise.

— procedure: locative-ref
          (locative-ref LOC)
     

Returns the element to which the locative LOC refers. If the containing object has been reclaimed by garbage collection, an error is signalled.

— procedure: locative-set!
          (locative-set! LOC X)
     

Changes the element to which the locative LOC refers to X. If the containing object has been reclaimed by garbage collection, an error is signalled.

— procedure: locative->object
          (locative->object LOC)
     

Returns the object that contains the element referred to by LOC or #f if the container has been reclaimed by garbage collection.


Next: , Previous: Locatives, Up: Unit lolevel

5.24.7 Accessing toplevel variables

— procedure: global-bound?
          (global-bound? SYMBOL)
     

Returns #t, if the global (“toplevel”) variable with the name SYMBOL is bound to a value, or #f otherwise.

— procedure: global-ref
          (global-ref SYMBOL)
     

Returns the value of the global variable SYMBOL. If no variable under that name is bound, an error is signalled.

— procedure: global-set!
          (global-set! SYMBOL X)
     

Sets the global variable named SYMBOL to the value X.


Next: , Previous: Accessing toplevel variables, Up: Unit lolevel

5.24.8 Low-level data access

— procedure: block-ref
          (block-ref BLOCK INDEX)
     

Returns the contents of the INDEXth slot of the object BLOCK. BLOCK may be a vector, record structure, pair or symbol.

— procedure: block-set!
          (block-set! BLOCK INDEX X)
     

Sets the contents of the INDEXth slot of the object BLOCK to the value of X. BLOCK may be a vector, record structure, pair or symbol.

— procedure: object-copy
          (object-copy X)
     

Copies X recursively and returns the fresh copy. Objects allocated in static memory are copied back into garbage collected storage.

— procedure: make-record-instance
          (make-record-instance SYMBOL ARG1 ...)
     

Returns a new instance of the record type SYMBOL, with its slots initialized to ARG1 .... To illustrate:

     
     (define-record point x y)

expands into something quite similar to:

     
     (begin
       (define (make-point x y)
         (make-record-instance 'point x y) )
       (define (point? x)
         (and (record-instance? x)
              (eq? 'point (block-ref x 0)) ) )
       (define (point-x p) (block-ref p 1))
       (define (point-x-set! p x) (block-set! p 1 x))
       (define (point-y p) (block-ref p 2))
       (define (point-y-set! p y) (block-set! p 1 y)) )
— procedure: move-memory!
          (move-memory! FROM TO [BYTES])
     

Copies BYTES bytes of memory from FROM to TO. FROM and TO may be strings, primitive byte-vectors, SRFI-4 byte-vectors (see: Unit srfi-4), memory mapped files, foreign pointers (as obtained from a call to foreign-lambda, for example) or locatives. if BYTES is not given and the size of the source or destination operand is known then the maximal number of bytes will be copied. Moving memory to the storage returned by locatives will cause havoc, if the locative refers to containers of non-immediate data, like vectors or pairs.

— procedure: number-of-bytes
          (number-of-bytes BLOCK)
     

Returns the number of bytes that the object BLOCK contains. BLOCK may be any non-immediate value.

— procedure: number-of-slots
          (number-of-slots BLOCK)
     

Returns the number of slots that the object BLOCK contains. BLOCK may be a vector, record structure, pair or symbol.

— procedure: record-instance?
          (record-instance? X)
     

Returns #t if X is an instance of a record type. See also: make-record-instance.

— procedure: record->vector
          (record->vector BLOCK)
     

Returns a new vector with the type and the elements of the record BLOCK.


Next: , Previous: Low-level data access, Up: Unit lolevel

5.24.9 Procedure-call- and variable reference hooks

— procedure: invalid-procedure-call-handler
          (invalid-procedure-call-handler PROC)
     

Sets an internal hook that is invoked when a call to an object other than a procedure is executed at runtime. The procedure PROC will in that case be called with two arguments: the object being called and a list of the passed arguments.

     
     ;;; Access sequence-elements as in ARC:
     
     (invalid-procedure-call-handler
       (lambda (proc args)
         (cond [(string? proc) (apply string-ref proc args)]
               [(vector? proc) (apply vector-ref proc args)]
               [else (error "call of non-procedure" proc)] ) ) )
     
     ("hello" 4)    ==>  #\o

This facility does not work in code compiled with the “unsafe” setting.

— procedure: unbound-variable-value
          (unbound-variable-value [X])
     

Defines the value that is returned for unbound variables. Normally an error is signalled, use this procedure to override the check and return X instead. To set the default behavior (of signalling an error), call unbound-variable-value with no arguments.

This facility does not work in code compiled with the “unsafe” setting.


Previous: Procedure-call- and variable reference hooks, Up: Unit lolevel

5.24.10 Magic

— procedure: object-become!
          (object-become! ALIST)
     

Changes the identity of the value of the car of each pair in ALIST to the value of the cdr. Both values may not be immediate (i.e. exact integers, characters, booleans or the empty list).

     
     (define x "i used to be a string")
     (define y '#(and now i am a vector))
     (object-become! (list (cons x y)))
     x                                    ==> #(and now i am a vector)
     y                                    ==> #(and now i am a vector)
     (eq? x y)                            ==> #t

Note: this operation invokes a major garbage collection.

The effect of using object-become! on evicted data (see object-evict) is undefined.


Previous: Unit lolevel, Up: Supported language

5.25 Unit tinyclos

This unit is a port of Gregor Kiczales TinyCLOS with numerous modifications.

This unit uses the extras unit.


Next: , Up: Unit tinyclos

5.25.1 Defining forms

— syntax: define-class
          (define-class NAME (SUPERCLASS1 ...) (SLOTNAME1 ...) [METACLASS])
     

Sets the variable NAME to a new class (a new instance of the class <class>). SUPERCLASS1 ... is a list of superclasses of the newly created class. If no superclasses are given, then <object> is assumed. SLOTNAME1 ... are the names of the direct slots of the class. if METACLASS is provided, then the new class-instance is an instance of METACLASS instead of <class>.

     
     (define-class NAME (SUPER) (SLOT1 SLOT2) META)

is equivalent to

     
     (define NAME
       (make META 
         'name 'NAME
         'direct-supers (list SUPER)
         'direct-slots (list 'SLOT1 'SLOT2)) )

Note that slots-names are not required to be symbols, so the following is perfectly valid:

(define hidden-slot (list 'hidden))
(define <myclass>
  (make <class>
     'direct-supers (list <object>)
     'direct-slots (list hidden-slot) ) )
(define x1 (make <myclass>)
(slot-set! x1 hidden-slot 99)

     
— syntax: define-generic
          (define-generic NAME [CLASS])
     

Sets the variable NAME to contain a fresh generic function object without associated methods. If the optional argument CLASS is given, then the generic function will be an instance of that class.

— syntax: define-method
          (define-method (NAME (VARIABLE1 CLASS1) ... PARAMETERS ...) BODY ...)
     

Adds a new method with the code BODY ... to the generic function that was assigned to the variable name. CLASS1 ... is a list if classes that specialize this particular method. The method can have additional parameters PARAMETERS, which do not specialize the method any further. Inside the body of the method the identifier call-next-method names a procedure of zero arguments that can be invoked to call the next applicable method with the same arguments. If no generic function is defined under this name, then a fresh generic function object is created and assigned to NAME. Note that only define-generic expands into a valid definition, so for internal lexically scoped definitions or for definitions for module exports (see syntax-case) use define-generic.


Next: , Previous: Defining forms, Up: Unit tinyclos

5.25.2 Base language

— procedure: add-method
          (add-method GENERIC METHOD)
     

Adds the method object METHOD to the list of applicable methods for the generic function GENERIC.

— procedure: instance?
          (instance? X)
     

Returns #t if X is an instance of a non-primitive class.

— procedure: make
          (make CLASS INITARG ...)
     

Creates a new instance of CLASS and passes INITARG ... to the initialize method of this class.

— procedure: make-class
          (make-class SUPERCLASSES SLOTNAMES)
     

Creates a new class object, where SUPERCLASSES should be the list of direct superclass objects and SLOTNAMES should be a list of symbols naming the slots of this class.

— procedure: make-generic
          (make-generic [NAME])
     

Creates a new generic function object. If NAME is specified, then it should be a string.

— procedure: make-method
          (make-method SPECIALIZERS PROC)
     

Creates a new method object specialized to the list of classes in SPECIALIZERS.

     
     (define-method (foo (x <bar>)) 123)
        <=> (add-method foo
                        (make-method
                           (list <bar>)
                           (lambda (call-next-method x) 123)))
— procedure: slot-ref
          (slot-ref INSTANCE SLOTNAME)
     

Returns the value of the slot SLOTNAME of the object INSTANCE.

— procedure: slot-set!
          (slot-set! INSTANCE SLOTNAME VALUE)
     

Sets the value of the slot SLOTNAME of the object INSTANCE to VALUE.


Next: , Previous: Base language, Up: Unit tinyclos

5.25.3 Introspection

— procedure: class-cpl
          (class-cpl CLASS)
     

Returns the class-precedence-list of CLASS as a list of classes.

— procedure: class-direct-slots
          (class-direct-slots CLASS)
     

Returns the list of direct slots of CLASS as a list of lists, where each sublist contains the name of the slot.

— procedure: class-direct-supers
          (class-direct-supers CLASS)
     

Returns the list of direct superclasses of CLASS.

— procedure: class-of
          (class-of X)
     

Returns the class that the object X is an instance of.

— procedure: class-name
          (class-name CLASS)
     

Returns name of CLASS.

— procedure: class-slots
          (class-slots CLASS)
     

Returns the list of all slots of CLASS and its superclasses as a list of lists, where each sublist contains the name of the slot.

— procedure: generic-methods
          (generic-methods GENERIC)
     

Returns the list of all methods associated with the generic function GENERIC.

— procedure: method-specializers
          (method-specializers METHOD)
     

Returns the list of classes that specialize METHOD.

— procedure: method-procedure
          (method-procedure METHOD)
     

Returns the procedure that contains the body of METHOD.

— procedure: subclass?
          (subclass? CLASS1 CLASS2)
     

Returns #t is CLASS1 is a subclass of CLASS2, or #f otherwise. Note that the following holds:

     
     (subclass? X X) ==> #t


Next: , Previous: Introspection, Up: Unit tinyclos

5.25.4 Intercessory protocol

These definitions allow interfacing to the Meta Object Protocol of TinyCLOS. For serious use, it is recommended to consult the source code (tinyclos.scm).

— generic: allocate-instance
          (allocate-instance CLASS)
     

Allocates storage for an instance of CLASS and returns the instance.

— generic: compute-apply-generic
          (compute-apply-generic GENERIC)
     

Returns a procedure that will be called to apply the generic function methods to the arguments.

— generic: compute-apply-methods
          (compute-apply-methods GENERIC)
     

Returns a procedure of two arguments, a list of applicable methods and a list of arguments and applies the methods.

— generic: compute-methods
          (compute-methods GENERIC)
     

Returns a procedure of one argument. The procedure is called with the list of actual arguments passed to the generic function and should return a list of applicable methods, sorted by precedence.

— generic: compute-cpl
          (compute-cpl CLASS)
     

Computes and returns the class-precedence-list of CLASS.

— generic: compute-getter-and-setter
          (compute-getter-and-setter CLASS SLOT ALLOCATOR)
     

Returns two values, the procedures that get and set the contents of the slot SLOT. ALLOCATOR is a procedure of one argument (I currently don't know what it does).

— generic: compute-method-more-specific?
          (compute-method-more-specific? GENERIC)
     

Returns a procedure of three arguments (two methods and a list of arguments) that returns #t if the first method is more specific than the second one with respect to the list of arguments. Otherwise the returned predicate returns #f.

— generic: compute-slots
          (compute-slots CLASS)
     

Computes and returns the list of slots of CLASS.

— generic: initialize
          (initialize INSTANCE INITARGS)
     

Initializes the object INSTANCE. INITARGS is the list of initialization arguments that were passed to the make procedure.


Next: , Previous: Intercessory protocol, Up: Unit tinyclos

5.25.5 Additional protocol

— generic: describe-object
          (describe-object INSTANCE PORT)
     

Writes a description of INSTANCE to PORT. Execution of the interpreter command ,d will invoke this generic function.

— generic: print-object
          (print-object INSTANCE PORT)
     

Writes a textual representation of INSTANCE to PORT. Any output of an instance with display, write and print will invoke this generic function.


Next: , Previous: Additional protocol, Up: Unit tinyclos

5.25.6 Utility procedures

— procedure: initialize-slots
          (initialize-slots INSTANCE INITARGS)
     

This procedure takes a sequence of alternating slot-names and initialization values in INITARGS and initializes the corresponding slots in INSTANCE.

     
     (define-class <pos> () (x y))
     
     (define-method (initialize (pos <pos>) initargs)
       (call-next-method)
       (initialize-slots pos initargs))
     
     (define p1 (make <pos> 'x 1 'y 2))
     (define p2 (make <pos> 'x 3 'y 5))


Previous: Utility procedures, Up: Unit tinyclos

5.25.7 Builtin classes

The class hierarchy of builtin classes looks like this:

<top>
  <object>
    <class>
      <procedure-class>
        <procedure>
        <entity-class>
          <generic>
      <primitive-class>
    <c++-object>
  <primitive>
    <void>
    <boolean>
    <symbol>
    <char>
    <vector>
    <pair>
    <number>
      <integer>
        <exact>
      <inexact>
    <string>
    <port>
      <input-port>
      <output-port>
    <pointer>
      <tagged-pointer>
    <locative>
    <byte-vector>
      <u8vector>
      <s8vector>
      <u16vector>
      <s16vector>
      <u32vector>
      <s32vector>
      <f32vector>
      <f64vector>
    <structure>
      <array>
      <char-set>
      <condition>
      <environment>
      <hash-table>
      <lock>
      <mmap>
      <promise>
      <queue>
      <tcp-listener>
      <time>
    <end-of-file>

— class: <primitive> -> <top>

The parent class of the classes of all primitive Scheme objects.

— class: <boolean> -> <primitive>
— class: <symbol> -> <primitive>
— class: <char> -> <primitive>
— class: <vector> -> <primitive>
— class: <null> -> <primitive>
— class: <pair> -> <primitive>
— class: <number> -> <primitive>
— class: <integer> -> <primitive>
— class: <exact> -> <integer>
— class: <inexact> -> <number>
— class: <string> -> <primitive>
— class: <port> -> <primitive>
— class: <environment> -> <structure>
— class: <end-of-file> -> <primitive>
— class: <input-port> -> <port>
— class: <output-port> -> <port>
— class: <procedure> -> <procedure-class>

The classes of primitive Scheme objects.

— class: <byte-vector> -> <primitive>
— class: <structure> -> <primitive>
— class: <hash-table> -> <structure>
— class: <queue> -> <structure>

The classes of extended data types provided by the various library units.

— class: <class> -> <object>

The parent class of all class objects.

— class: <entity-class> -> <class>

The parent class of objects that can be invoked as a procedure and have slots.

— class: <generic> -> <entity-class>

The parent class of generic function objects.

— class: <method> -> <class>

The parent class of method objects.

— class: <object> -> <class>

The parent class of all objects.

— class: <procedure-class> -> <class>

The parent class of objects that can be invoked as a procedure.

— class: <condition> -> <structure>

Class of condition objects.

— class: <array> -> <structure>
— class: <char-set> -> <structure>
— class: <time> -> <structure>
— class: <u8vector> -> <byte-vector>
— class: <s8vector> -> <byte-vector>
— class: <u16vector> -> <byte-vector>
— class: <s16vector> -> <byte-vector>
— class: <u32vector> -> <byte-vector>
— class: <s32vector> -> <byte-vector>
— class: <f32vector> -> <byte-vector>
— class: <f64vector> -> <byte-vector>

The classes of data objects provided by the various supported SRFIs.

— class: <lock> -> <structure>
— class: <mmap> -> <structure>

Classes of objects used in the posix library unit.

— class: <pointer> -> <primitive>
— class: <tagged-pointer> -> <pointer>

A machine pointer (untagged, or tagged).

— class: <locative> -> <primitive>

A locative.

— class: <promise> -> <structure>

The class of objects returned by delay.

— class: <tcp-listener> -> <structure>

The class of an object returned by tcp-listen.

— class: <c++-class> -> <object>

The class of generated wrappers for C++ classes parsed by the “easy” Foreign Function interface.

The CHICKEN distribution provides several examples in the file tinyclos-examples.scm.


Next: , Previous: Supported language, Up: Top

6 Interface to external functions and variables


Next: , Up: Interface to external functions and variables

6.1 Accessing external objects

— syntax: foreign-code
          (foreign-code STRING)
     

Executes the embedded C/C++ code STRING, which should be a sequence of C statements, which are executed and return an unspecified result.

     
     (foreign-code "doSomeInitStuff();")     =>  #<unspecified>
— syntax: foreign-value
          (foreign-value STRING TYPE)
     

Evaluates the embedded C/C++ expression STRING, returning a value of type given in the foreign-type specifier TYPE.

     
     (print (foreign-value "my_version_string" c-string))
— syntax: define-foreign-type
          (define-foreign-type NAME TYPE [ARGCONVERT [RETCONVERT]])
     

Defines an alias for TYPE with the name NAME (a symbol). TYPE may be a type-specifier or a string naming a C type. The namespace of foreign type specifiers is separate from the normal Scheme namespace. The optional arguments ARGCONVERT and RETCONVERT should evaluate to procedures that map argument- and result-values to a value that can be transformed to TYPE:

     
     (require-extension extras)
     
     (define-foreign-type char-vector 
       nonnull-c-string
       (compose list->string vector->list)
       (compose list->vector string->list) )
     
     (define strlen
       (foreign-lambda int "strlen" char-vector) )
     
     (strlen '#(#\a #\b #\c))                      ==> 3
     
     (define memset
       (foreign-lambda char-vector "memset" char-vector char int) )
     
     (memset '#(#_ #_ #_) #\X 3)                ==> #(#\X #\X #\X)

Foreign type-definitions are only visible in the compilation-unit in which they are defined, so use include to use the same definitions in multiple files.

— syntax: define-foreign-variable
          (define-foreign-variable NAME TYPE [STRING])
     

Defines a foreign variable of name NAME (a symbol). STRING should be the real name of a foreign variable or parameterless macro. If STRING is not given, then the variable name NAME will be converted to a string and used instead. All references and assignments (via set!) are modified to correctly convert values between Scheme and C representation. This foreign variable can only be accessed in the current compilation unit, but the name can be lexically shadowed. Note that STRING can name an arbitrary C expression. If no assignments are performed, then STRING doesn't even have to specify an lvalue.

     
     #>
     enum { abc=3, def, ghi };
     <#
     
     (define-macro (define-foreign-enum . items)
       `(begin
          ,@(map (match-lambda 
                   [(name realname) `(define-foreign-variable ,name int ,realname)]
                   [name `(define-foreign-variable ,name int)] )
          items) ) )
     
     (define-foreign-enum abc def ghi)
     
     ghi                               ==> 5
— syntax: define-foreign-record
          (define-foreign-record NAME SLOT ...)
     

Defines accessor procedures for a C structure definition. NAME should either be a symbol or a list of the form (TYPENAME FOREIGNNAME). If NAME is a symbol, then a C declaration will be generated that defines a C struct named struct NAME. If NAME is a list, then no struct declaration will be generated. A foreign-type specifier named NAME (or TYPENAME) will be defined as a pointer to the given C structure. A SLOT definition should be a list of one of the following forms:

     
     (TYPE SLOTNAME)

or

     
     (TYPE SLOTNAME SIZE)

The latter form defines an array of SIZE elements of the type TYPE embedded in the structure. For every slot, the following accessor procedures will be generated:

— procedure: TYPENAME-SLOTNAME
               (TYPENAME-SLOTNAME FOREIGN-RECORD-POINTER [INDEX])
          

A procedure of one argument (a pointer to a C structure), that returns the slot value of the slot SLOTNAME. If a SIZE has been given in the slot definition, then an additional argument INDEX is required that specifies the index of an array-element.

— procedure: TYPENAME-SLOTNAME-set!
               (TYPENAME-SLOTNAME-set! FOREIGN-RECORD-POINTER [INXDEX] VALUE)
          

A procedure of two arguments (a pointer to a C structure) and a value, that sets the slot value of the slot SLOTNAME in the structure. If a SIZE has been given in the slot definition, then an additional argument INDEX is required for the array index.

If a slot type is of the form (const ...), then no setter procedure will be generated. Slots of the types (struct ...) or (union ...) are accessed as pointers to the embedded struct (or union) and no setter will be generated.

— syntax: foreign-callback-lambda
          (foreign-callback-lambda RETURNTYPE NAME ARGTYPE ...)
     

This is similar to foreign-lambda, but also allows the called function to call Scheme functions. See Callbacks.

— syntax: foreign-callback-lambda*
          (foreign-callback-lambda* RETURNTYPE ((ARGTYPE VARIABLE)...) STRING ...)
     

This is similar to foreign-lambda*, but also allows the called function to call Scheme functions. See Callbacks.

— syntax: foreign-lambda
          (foreign-lambda RETURNTYPE NAME ARGTYPE ...)
     

Represents a binding to an external routine. This form can be used in the position of an ordinary lambda expression. NAME specifies the name of the external procedure and should be a string or a symbol.

— syntax: foreign-lambda*
          (foreign-lambda* RETURNTYPE ((ARGTYPE VARIABLE) ...) STRING ...)
     

Similar to foreign-lambda, but instead of generating code to call an external function, the body of the C procedure is directly given in STRING ...:

     
     (define my-strlen
       (foreign-lambda* int ((c-string str))
         "int n = 0;
          while(*(str++)) ++n;
          return(n);") )
     
     (my-strlen "one two three")             ==> 13

For obscure technical reasons any use of the return statement should enclose the result value in parentheses. For the same reasons return without an argument is not allowed.


Next: , Previous: Accessing external objects, Up: Interface to external functions and variables

6.2 Foreign type specifiers

Here is a list of valid foreign type specifiers:

scheme-object
An arbitrary Scheme data object (immediate or non-immediate).
bool
As argument: any value (#f is false, anything else is true). As result: anything different from 0 and the NULL-pointer is #t.
byte unsigned-byte
A byte.
char unsigned-char
A character.
short unsigned-short
A short integer number.
int unsigned-int
An small integer number in fixnum range (at least 30 bit).
integer unsigned-integer
Either a fixnum or a flonum in the range of a (unsigned) machine “int”.
long unsigned-long
Either a fixnum or a flonum in the range of a (unsigned) machine “long”.
float double
A floating-point number. If an exact integer is passed as an argument, then it is automatically converted to a float.
pointer
An untyped pointer to the contents of a non-immediate Scheme object (not allowed as return type). The value #f is also allowed and is passed as a NULL pointer.
nonnull-pointer
As pointer, but guaranteed not to be #f.
c-pointer
An untyped operating-system pointer or a locative. The value #f is also allowed and is passed as a NULL pointer. If uses as the type of a return value, a NULL pointer will be returned as #f.
nonnull-c-pointer
As c-pointer, but guaranteed not to be #f/NULL.
[nonnull-] byte-vector
A byte-vector object, passed as a pointer to its contents. Arguments of type byte-vector may optionally be #f, which is passed as a NULL pointer. This is not allowed as a return type.
[nonnull-] u8vector
[nonnull-] u16vector
[nonnull-] u32vector
[nonnull-] s8vector
[nonnull-] s16vector
[nonnull-] s32vector
[nonnull-] f32vector
[nonnull-] f64vector
A SRFI-4 number-vector object, passed as a pointer to its contents. Arguments of type byte-vector may optionally be #f, which is passed as a NULL pointer. These are not allowed as return types.
c-string
A C string (zero-terminated). The value #f is also allowed and is passed as a NULL pointer. If uses as the type of a return value, a NULL pointer will be returned as #f. Note that the string is copied (with a zero-byte appended) when passed as an argument to a foreign function. Also a return value of this type is copied into garbage collected memory.
nonnull-c-string
As c-string, but guaranteed not to be #f/NULL.
[nonnull-] c-string*
Similar to [nonnull-]c-string, but if used as a result-type, the pointer returned by the foreign code will be freed (using the C-libraries free()) after copying.
void
Specifies an undefined return value. Not allowed as argument type.
(const TYPE)
The foreign type TYPE with an additional const specifier.
(enum NAME)
An enumeration type. Handled internally as an integer.
(pointer TYPE)
(c-pointer TYPE)
An operating-system pointer or a locative to an object of TYPE.
(nonnull-pointer TYPE)
(nonnull-c-pointer TYPE)
As (pointer TYPE), but guaranteed not to be #f/NULL.
(ref TYPE)
A C++ reference type. Reference types are handled the same way as pointers inside Scheme code.
(struct NAME)
A struct of the name NAME, which should be a string. Structs can not be directly passed as arguments to foreign function, neither can they be result values. Pointers to structs are allowed, though.
(template TYPE ARGTYPE ...)
A C++ template type. For example vector<int> would be specified as (template "vector" int). Template types can not be directly passed as arguments or returned as results.
(union NAME)
A union of the name NAME, which should be a string. Unions can not be directly passed as arguments to foreign function, neither can they be result values. Pointers to unions are allowed, though.
(instance CNAME SCHEMECLASS)
A pointer to a C++ class instance. CNAME should designate the name of the C++ class, and SCHEMECLASS should be the class that wraps the instance pointer. Normally SCHEMECLASS should be a subclass of <c++-object>.
(function RESULTTYPE (ARGUMENTTYPE1 ... [...]) [CALLCONV])
A function pointer. CALLCONV specifies an optional calling convention and should be a string. The meaning of this string is entirely platform dependent. The value #f is also allowed and is passed as a NULL pointer.

Foreign types are mapped to C types in the following manner:

bool
int
[unsigned-]char
[unsigned] char
[unsigned-]short
[unsigned] short
[unsigned-]int
[unsigned] int
[unsigned-]integer
[unsigned] int
[unsigned-]long
[unsigned] long
float
float
double
double
[nonnull-]pointer
void *
[nonnull-]c-pointer
void *
[nonnull-]byte-vector
unsigned char *
[nonnull-]u8vector
unsigned char *
[nonnull-]s8vector
char *
[nonnull-]u16vector
unsigned short *
[nonnull-]s16vector
short *
[nonnull-]u32vector
uint32_t *
[nonnull-]s32vector
int32_t *
[nonnull-]f32vector
float *
[nonnull-]f64vector
double *
[nonnull-]c-string
char *
void
void
([nonnull-]pointer TYPE)
TYPE *
(enum NAME)
enum NAME
(struct NAME)
struct NAME
(ref TYPE)
TYPE &
(template T1 T2 ...)
T1<T2, ...>
(union NAME)
union NAME
(function RTYPE (ATYPE ...) [CALLCONV])
[CALLCONV] RTYPE (*)(ATYPE, ...)


Next: , Previous: Foreign type specifiers, Up: Interface to external functions and variables

6.3 Entry points

To simplify embedding compiled Scheme code into arbitrary programs, one can define so called “entry points”, which provide a uniform interface and parameter conversion facilities. To use this facility, add

(include "chicken-entry-points")

to the beginning of your code.

— syntax: define-entry-point
          (define-entry-point INDEX ((VAR1 TYPE1) ...)
                                    (RTYPE1 ...)
                                    EXP1 EXP2 ...)
     

Defines a new entry-point with index INDEX which should evaluate to an exact integer. During execution of the body EXP1 EXP2 ... the variables VAR1 ... are bound to the parameters passed from the host program to the invoked entry point. The parameters passed are converted according to the foreign type specifiers TYPE1 .... The expressions should return as many values as foreign type specifiers are given in RTYPE1 ..., with the exception that if the list of return types is empty, a single value is expected to be returned from the body (there is no need to add a (values) form at the end). The results are then transformed into values that can be used in the host program.

Note: if one or more of the result types RTYPE ... specify the type c-string, then the parameter types at the same positions in TYPE1 ... have to be c-strings as well, because the result strings are copied into the same area in memory. You should also take care that the passed buffer is long enough to hold the result string or unpredictable things will happen.

If entry points were defined then the program will not terminate after execution of the last toplevel expression, but instead it will enter a loop that waits for the host to invoke one of the defined entry points.

— syntax: define-embedded
          (define-embedded [QUALIFIER ... ] ([CCONV] NAME (TYPE1 VAR1) ...) RTYPE BODY ...)
     

Defines a named entry-point that can be accessed from external code with a normal C/C++ function call. QUALIFIER may be a string for special function declarations (like __declspec(dllexport) on Windows, for example) and CCONV may be a string designating a calling convention (like __cdecl). During the execution of the BODY the variables VAR1 ... are bound to the arguments passed to the function, which should be of types compatible to the type specifiers TYPE1 .... RTYPE specifies the result type of the entry-point. The return type specifiers [nonnull-]c-string and [nonnull-]c-string* are handled specially: if the return type is c-string, a heap-allocated pointer to a zero-terminated string will be returned, which will be valid until the next invocation of an entry-point into the Scheme code. If the return type is c-string*, a buffer allocated with malloc will be returned, and freeing the string is the responsibility of the caller.

All entry-point definitions with define-embedded should be put into the same source file. define-embedded expands into a define-entry-point form and the entry-point index is provided by the macro (starting from 1). If define-embedded is used in multiple source files then the index is counted from 1 in each of the files, resulting in multiple entry-points with the same index.

The following C functions and data types are provided:

— C function: void CHICKEN_parse_command_line (int argc, char *argv[], int *heap, int *stack int *symbols)

Parse the programs command-line contained in argc and argv and return the heap-, stack- and symbol table limits given by runtime options of the form -:..., or choose default limits. The library procedure argv can access the command-line only if this function has been called by the containing application.

— C function: int CHICKEN_initialize (int heap, int stack, int symbols, void *toplevel)

Initializes the Scheme execution context and memory. heap holds the number of bytes that are to be allocated for the secondary heap. stack holds the number of bytes for the primary heap. symbols contains the size of the symbol table. Passing 0 to one or more of these parameters will select a default size. toplevel should be a pointer to the toplevel entry point procedure. You should pass C_toplevel here. In any subsequent call to CHICKEN_run or CHICKEN_invoke you can simply pass NULL. Calling this function more than once has no effect. If enough memory is available and initialization was successful, then 1 is returned, otherwise this function returns 0.

— C function: void CHICKEN_run (void **data, int *bytes, int *maxlen, void *toplevel)

Starts the Scheme program. data, bytes and maxlen contain invocation parameters in raw form. Pass NULL here. Call this function once to execute all toplevel expressions in your compiled Scheme program. If the runtime system was not initialized before, then CHICKEN_initialize is called with default sizes. toplevel is the toplevel entry-point procedure.

— C function: void CHICKEN_invoke (int index, C_parameter *params, int count, void *toplevel)

Invoke the entry point with index index. count should contain the maximum number of arguments or results (whatever is higher). params is a pointer to parameter data:

     
     typedef union
     {
       C_word x;           /* parameter type scheme-object */
       long i;             /* parameter type bool, [unsigned] int/short/long */
       long c;             /* parameter type [unsigned] char */
       double f;           /* parameter type float/double */
       void *p;            /* any pointer parameter type and C strings */
     } C_parameter;

This function calls CHICKEN_run if it was not called at least once before.

— C function: int CHICKEN_is_running ()

Returns 1, if called inside a dynamic context invoked via CHICKEN_run or CHICKEN_invoke (i.e. if running inside a call from Scheme to C). If no Scheme stack frame is currently active, then this function returns 0.

Here is a simple example (assuming a UNIX-like environment):

% cat foo.c
#include <stdio.h>
#include "chicken.h"

int main(void)
{
  C_parameter p[ 3 ];
  char str[ 32 ] = "hello!";  /* We need some space for the result string! */

  memset(p, 0, sizeof(p));
  p[ 0 ].i = -99;
  p[ 1 ].p = str;
  p[ 2 ].f = 3.14;
  CHICKEN_invoke(1, p, 3, C_toplevel);
  printf("->\n%d\n%s\n", p[ 0 ].i, p[ 1 ].p);
  return 0;
}

% cat bar.scm
(include "chicken-entry-points")

(define-entry-point 1
    ((a integer) (b c-string) (c double))
    (int c-string)
  (print (list a b c))
  (values 123 "good bye!") )

% chicken bar.scm -quiet
% gcc foo.c bar.c -o foo `chicken-config -cflags -libs -embedded`
% foo
(-99 "hello!" 3.14)
->
123
good bye!

Note the use of -embedded. We have to compile with additional compiler options, because the host program provides the main function.

Here another example that uses named entry-points defined with define-embedded:

% cat foo.c
extern int foo(int, char *);
extern unsigned int square(double);

int main() { foo(square(9), "yo!"); return 0; }

% cat bar.scm
(include "chicken-entry-points")

(define-embedded (foo (int x) (c-string y)) int
  (print x ": " y) 
  x)

(define-embedded (square (double x)) unsigned-int
  (* x x))

% chicken bar.scm
compiling `bar.scm' ...
% gcc foo.c bar.c `chicken-config -cflags -libs -embedded`
% a.out
81: yo!

CHICKEN also provides “boilerplate” entry points, that simplify invoking Scheme code embedded in a C or C++ application tremendously. The include file default-entry-points.scm will define entry-points for common usage patterns, like loading a file, evaluating an expression or calling a procedure.

— C macro: void CHICKEN_eval (C_word exp, C_word *result, int *status)

Evaluates the Scheme object passed in exp, writing the result value to result. status is set to 1 if the operation succeeded, or 0 if an error occurred. Call CHICKEN_get_error_message to obtain a description of the error.

— C macro: void CHICKEN_eval_string (char *str, C_word *result, int *status)

Evaluates the Scheme expression passed in the string str, writing the result value to result.

— C macro: void CHICKEN_eval_to_string (C_word exp, char *result, int size, int *status)

Evaluates the Scheme expression passed in exp, writing a textual representation of the result into result. size should specify the maximal size of the result string.

— C macro: void CHICKEN_eval_string_to_string (char *str, char *result, int size, int *status)

Evaluates the Scheme expression passed in the string str, writing a textual representation of the result into result. size should specify the maximal size of the result string.

— C macro: void CHICKEN_apply (C_word func, C_word args, C_word *result, int *status)

Applies the procedure passed in func to the list of arguments args, writing the result value to result.

— C macro: void CHICKEN_apply_to_string (C_word func, C_word args, char *result, int size, int *status)

Applies the procedure passed in func to the list of arguments args, writing a textual representation of the result into result.

— C macro: void CHICKEN_read (char *str, C_word *result, int *status)

Reads a Scheme object from the string str, writing the result value to result.

— C macro: void CHICKEN_load (char *filename, int *status)

Loads the Scheme file filename (either in source form or compiled).

— C macro: void CHICKEN_get_error_message (char *result, int size)

Returns a textual description, in case an error occurred while invoking embedded Scheme code.

— C macro: void CHICKEN_yield (int *status)

If threads have been spawned during earlier invocations of embedded Scheme code, then this function will run the next scheduled thread for one complete time-slice. This is useful, for example, inside an “idle” handler in a GUI application with background Scheme threads.

An example:

% cat x.scm
;;; x.scm

(include "chicken-default-entry-points")
(define (bar x) (gc) (* x x))

% cat y.c
/* y.c */

#include "chicken.h"
#include <assert.h>

int main() {
  char buffer[ 256 ];
  int status;
  C_word val = C_SCHEME_UNDEFINED;
  C_word *data[ 1 ];
  
  data[ 0 ] = &val;

  CHICKEN_read("(bar 99)", &val, &status);
  assert(status);

  C_gc_protect(data, 1);

  printf("data: %08x\n", val);

  CHICKEN_eval_string_to_string("(bar)", buffer, 255, &status);
  assert(!status);

  CHICKEN_get_error_message(buffer, 255);
  printf("ouch: %s\n", buffer);

  CHICKEN_eval_string_to_string("(bar 23)", buffer, 255, &status);
  assert(status);

  printf("-> %s\n", buffer);
  printf("data: %08x\n", val);

  CHICKEN_eval_to_string(val, buffer, 255, &status);
  assert(status);
  printf("-> %s\n", buffer);

  return 0;
}

% csc x.scm y.c -embedded

A simpler interface For handling GC-safe references to Scheme data are the so called “gc-roots”:

— C function: void* CHICKEN_new_gc_root ()

Returns a pointer to a “GC root”, which is an object that holds a reference to a Scheme value that will always be valid, even after a garbage collection. The content of the gc root is initialized to an unspecified value.

— C function: void CHICKEN_delete_gc_root (void *root)

Deletes the gc root.

— C macro: C_word CHICKEN_gc_root_ref (void *root)

Returns the value stored in the gc root.

— C macro: void CHICKEN_gc_root_set (void *root, C_word value)

Sets the content of the GC root to a new value.

Sometimes it is handy to access global variables from C code:

— C function: void* CHICKEN_global_lookup (char *name)

Returns a GC root that holds the global variable with the name name. If no such variable exists, NULL is returned.

— C function: C_word CHICKEN_global_ref (void *global)

Returns the value of the global variable referenced by the GC root global.

— C function: void CHICKEN_global_set (void *global, C_word value)

Sets the value of the global variable referenced by the GC root global to value.


Next: , Previous: Entry points, Up: Interface to external functions and variables

6.4 Callbacks

To enable an external C function to call back to Scheme, the form foreign-callback-lambda (or foreign-callback-lambda*) has to be used. This generates special code to save and restore important state information during execution of C code. There are two ways of calling Scheme procedures from C: the first is to invoke the runtime function C_callback with the closure to be called and the number of arguments. The second is to define an externally visible wrapper function around a Scheme procedure with the define-external or foreign-callback-wrapper forms.

Note: the names of all functions, variables and macros exported by the CHICKEN runtime system start with “C_”. It is advisable to use a different naming scheme for your own code to avoid name clashes. Callbacks (either defined by define-external or foreign-callback-wrapper do not capture the lexical environment.

— syntax: define-external
          (define-external [QUALIFIERS] (NAME (ARGUMENTTYPE1 VARIABLE1) ...) RETURNTYPE BODY ...)
          (define-external NAME TYPE [INIT])
     

The first form defines an externally callable Scheme procedure. NAME should be a symbol, which, when converted to a string, represents a legal C identifier. ARGUMENTTYPE1 ... and RETURNTYPE are foreign type specifiers for the argument variables VAR1 ... and the result, respectively. QUALIFIERS is an optional qualifier for the foreign procedure definition, like __stdcall.

     
     (define-external (foo (c-string x)) int (string-length x))

is equivalent to

     
     (define foo 
       (foreign-callback-wrapper int "foo" 
         (c-string) (lambda (x) (string-length x))))

The second form of define-external can be used to define variables that are accessible from foreign code. It declares a global variable named by the symbol NAME that has the type TYPE. INIT can be an arbitrary expression that is used to initialize the variable. NAME is accessible from Scheme just like any other foreign variable defined by define-foreign-variable.

     
     (define-external foo int 42)
     ((foreign-lambda* int ()
       "return(foo);"))           ==> 42

Note: don't be tempted to assign strings or bytevectors to external variables. Garbage collection moves those objects around, so it is very bad idea to assign pointers to heap-data. If you have to do so, then copy the data object into statically allocated memory (for example by using object-evict).

— syntax: foreign-callback-wrapper
          (foreign-callback-wrapper RETURNTYPE NAME [QUALIFIERS] (ARGUMENTTYPE1 ...) EXP)
     

Defines an externally callable wrapper around the procedure EXP. EXP must be a lambda expression of the form (lambda ...). The wrapper will have the name NAME and will have a signature as specified in the return- and argument-types given in RETURNTYPE and ARGUMENTTYPE1 .... QUALIFIERS is a qualifier string for the function definition (see define-external).

— C function: C_word C_callback (C_word closure, int argc)

This function can be used to invoke the Scheme procedure closure. argc should contain the number of arguments that are passed to the procedure on the temporary stack. Values are put onto the temporary stack with the C_save macro.


Next: , Previous: Callbacks, Up: Interface to external functions and variables

6.5 Locations

It is also possible to define variables containing unboxed C data, so called locations. It should be noted that locations may only contain simple data, that is: everything that fits into a machine word, and double-precision floating point values.

— syntax: define-location
          (define-location NAME TYPE [INIT])
     

Identical to (define-external NAME TYPE [INIT]), but the variable is not accessible from outside of the current compilation unit (it is declared static).

— syntax: let-location
          (let-location ((NAME TYPE [INIT]) ...) BODY ...)
     

Defines a lexically bound location.

— syntax: location
          (location NAME)
          (location X)
          NAME
          define-external
     

or let-location. This form returns a pointer object that contains the address of the variable NAME. If the argument to location is not a location defined by define-location, define-external or let-location, then

     
     (location X)

is essentially equivalent to

     
     (make-locative X)

(See the manual chapter or locatives for more information about locatives.

Note that (location X) may be abbreviated as #$X.

     
     (define-external foo int)
     ((foreign-lambda* void (((pointer int) ip)) "*ip = 123;") 
       (location foo))
     foo                                                                               ==> 123

This facility is especially useful in situations, where a C function returns more than one result value:

     
     #>
     #include <math.h>
     <#
     
     (define modf
       (foreign-lambda double "modf" double (pointer double)) )
     
     (let-location ([i double])
       (let ([f (modf 1.99 (location i))])
         (print "i=" i ", f=" f) ) )

location returns a value of type c-pointer, when given the name of a callback-procedure defined with define-external.


Next: , Previous: Locations, Up: Interface to external functions and variables

6.6 Other support procedures

— procedure: argc+argv
          (argc+argv)
     

Returns two values: an integer and a foreign-pointer object representing the argc and argv arguments passed to the current process.


Next: , Previous: Other support procedures, Up: Interface to external functions and variables

6.7 The Easy Foreign Function Interface

The compiler contains a builtin parser for a restricted subset of C and C++ that allows the easy generation of foreign variable declarations, procedure bindings and C++ class wrappers. The parser is invoked via the declaration-specifier foreign-parse, which extracts binding information and generates the necessary code. An example:

(declare 
  (foreign-declare "
#include <math.h>

#define my_pi 3.14
")
  (foreign-parse "extern double sin(double);") )

(print (sin 3.14))

The parser would generate code that is equivalent to

(declare 
  (foreign-declare "
#include <math.h>

#define my_pi 3.14
")

(define-foreign-variable my_pi float "my_pi")
(define sin (foreign-lambda double "sin" double))

Note that the read syntax #>[SPEC] ... <# provides a somewhat simpler way of using the parser. The example above could alternatively be expressed as

#>!
#define my_pi 3.14

extern double sin(double);
<#

(print (sin 3.14))

Another example, here using C++. Consider the following class:

// file: foo.h

class Foo {
 private:
  int x_;
 public:
  Foo(int x);
  void setX(int x);
  int getX();
};

To generate a wrapper class that provides generic functions for the constructor and the setX and getX methods, we can use the following class definition:

; file: test-foo.scm

#>!
#include "Foo.h"
<#

(define x (make <Foo> 99))
(print (getX x))              ; prints ``99''
(setX x 42)
(print (getX x))              ; prints ``42''
(destroy x)

Provided the file foo.o contains the implementation of the class Foo, the given example could be compiled like this (assuming a UNIX like environment):

% csc test-foo.scm foo.o -c++

Here is another example, a minimal “Hello world” application for QT. We can see the three different ways of embedding C/C++ code in Scheme:

; compile like this: 
; csc hello.scm -c++ -C -IQTDIR/include -L "-LQTDIR/lib -lqt"

; Include into generated code, but don't parse:
#>
#include <qapplication.h>
#include <qpushbutton.h>
<#

; Parse but don't embed: we only want wrappers for a few classes:
#>?
class QWidget 
{
public:
  void resize(int, int);
  void show();
};

class QApplication 
{
public:
  QApplication(int, char **);
  ~QApplication();
  void setMainWidget(QWidget *);
  void exec();
};

class QPushButton : public QWidget
{
public:
  QPushButton(char *, QWidget *);
  ~QPushButton();
}
<#

(define a (apply make <QApplication> (receive (argc+argv))))
(define hello (make <QPushButton> "hello world!" #f))
(resize hello 100 30)
(setMainWidget a hello)
(show hello)
(exec a)
(destroy hello)
(destroy a)


Next: , Up: The Easy Foreign Function Interface

6.7.1 #> ... <# Syntax

Occurrences of the special read syntax #>[SPEC ...] ...<# will be handled according to SPEC:

If SPEC is the ? character, the text following up to the next <# will be processed as a (declare (foreign-parse "...")) declaration (the code will be processed by the FFI parser described in this section).

If SPEC is the ! character, the text will be embedded as

(declare
  (foreign-declare "...")
  (foreign-parse "...") )

It will be both included verbatim in the declaration section of the generated C/C++ file and processed by the FFI parser.

If SPEC is the : character the text will be embedded as a (foreign-code "...") form, so it will be executed at the location where it appears.

If SPEC is a list of the form (TAG ...), then each TAG (which should be a symbol) specifies what should be done with the text:

declare
(declare (foreign-declare "..."))
parse
(declare (foreign-parse "..."))
execute
(foreign-code "...")

If any other character follows the #>, then the complete text will be included verbatim in the declaration part of the generated file (as in a foreign-declare declaration).


Next: , Previous: #> ... <# Syntax, Up: The Easy Foreign Function Interface

6.7.2 General operation

The parser will generally perform the following functions

1) Translate macro, enum-definitions and constants into define-foreign-variable or define-constant forms

2) Translate function prototypes into foreign-lambda forms

3) Translate variable declarations into accessor procedures

4) Handle basic preprocessor operations

5) Translate simple C++ class definitions into TinyCLOS wrapper classes and methods

Basic token-substitution of macros defined via #define is performed. The preprocessor commands #ifdef, #ifndef, #else, #endif, #undef and #error are handled. The preprocessor commands #if and #elif are not supported and will signal an error when encountered by the parser, because C expressions (even if constant) are not parsed. The preprocessor command #pragma is allowed but will be ignored.

During processing of foreign-parse declarations the macro CHICKEN is defined (similar to the C compiler option -DCHICKEN).

Macro- and type-definitions are available in subsequent foreign-parse forms. C variables declared generate a procedure with zero or one argument with the same name as the variable. When called with no arguments, the procedure returns the current value of the variable. When called with an argument, then the variable is set to the value of that argument. Structs are not supported. C and C++ style comments are supported. Variables declared as const will generate normal Scheme variables, bound to the initial value of the variable.

Function-, member-function and constructor/destructor definitions may be preceded by the __callback qualifier, which marks the function as performing a callback into Scheme. If a wrapped function calls back into Scheme code, and __callback has not been given very strange and hard to debug problems will occur. Member functions prefixed with __discard and a result type that maps to a Scheme string (c-string), will have their result type changed to c-string* instead.

Constants (as declared by #define or enum) are not visible outside of the current Compilation units unless the export_constants pseudo declaration has been used.

When given the option -ffi, CHICKEN will compile a C/C++ file in “Scheme” mode, that is, it wraps the C/C++ source inside #>! ... <# and compiles it while generating Scheme bindings for exported definitions.

Keep in mind that this is not a fully general C/C++ parser. Taking an arbitrary headerfile and feeding it to CHICKEN will in most cases not work or generate riduculuous amounts of code. This FFI facility is for carefully written headerfiles, and for declarations directly embedded into Scheme code.


Next: , Previous: General operation, Up: The Easy Foreign Function Interface

6.7.3 Pseudo declarations

Using the __declare(DECL, VALUE) form, pseudo declarations can be embedded into processed C/C++ code to provide additional control over the wrapper generation. Pseudo declarations will be ignored when processed by the system's C/C++ compiler.


Next: , Previous: Pseudo declarations, Up: The Easy Foreign Function Interface

6.7.4 Grammar

The parser understand the following grammar:

PROGRAM = PPCOMMAND
        | DECLARATION ";"

PPCOMMAND = "#define" ID [TOKEN ...]
          | "#ifdef" ID
          | "#ifndef" ID
          | "#else"
          | "#endif"
          | "#undef" ID
          | "#error" TOKEN ...
          | "#include" INCLUDEFILE
          | "#pragma" TOKEN ...

DECLARATION = FUNCTION
            | VARIABLE
            | ENUM
            | TYPEDEF
            | CLASS
            | CONSTANT
            | "struct" ID
            | "__declare" "(" PSEUDODECL "," <tokens> ")"

INCLUDEFILE = "\"" ... "\""
            | "<" ... ">"

FUNCTION = {"__callback" | "__specialize" | "__discard"} [STORAGE] TYPE ID "(" TYPE [ID] "," ... ")" [CODE]
         | {"__callback" | "__specialize" | "__discard"} [STORAGE] TYPE ID "(" "void" ")" [CODE]

VARIABLE = [STORAGE] TYPE ID ["=" INITDATA]

STORAGE = "extern" | "static" | "volatile" | "inline"

CONSTANT = "const" TYPE ID "=" INITDATA

PSEUDODECL = "export_constants"
           | "prefix"
           | "substitute"
           | "abstract"
           | "type"
           | "scheme"
           | "rename"
           | "transform"
           | "full_specialization"
           | "destructor_name"
           | "class_finalizers"
           | "exception_handler"

ENUM = "enum" "{" ID ["=" NUMBER] "," ... "}"

TYPEDEF = "typedef" TYPE ["*" ...] ID

TYPE = ["const"] BASICTYPE [("*" ... | "&" | "<" TYPE "," ... ">" | "(" "*" [ID] ")" "(" TYPE "," ... ")")]

BASICTYPE = ["unsigned" | "signed"] "int" 
          | ["unsigned" | "signed"] "char" 
          | ["unsigned" | "signed"] "short" 
          | ["unsigned" | "signed"] "long" 
          | ["unsigned" | "signed"] "__byte" 
          | "float"
          | "double"
          | "void"
          | "bool"
          | "__bool"
          | "__scheme_value"
          | "__fixnum"
          | "struct" ID
          | "union" ID
          | "enum" ID
          | ID

CLASS = ["__abstract"] "class" ID [":" [QUALIFIER] ID "," ...] "{" MEMBER ... "}"

MEMBER = [QUALIFIER ":"] ["virtual"] (MEMBERVARIABLE | CONSTRUCTOR | DESTRUCTOR | MEMBERFUNCTION)

MEMBERVARIABLE = TYPE ID ["=" INITDATA]

MEMBERFUNCTION = {"__callback" | "static" | "__specialize" | "__discard"} TYPE ID "(" TYPE [ID] "," ... ")" ["const"] ["=" "0"] [CODE]
               | {"__callback" | "static" | "__specialize" | "__discard"} TYPE ID "(" "void" ")" ["const"] ["=" "0"] [CODE]

CONSTRUCTOR = ["__callback"] ["explicit"] ID "(" TYPE [ID] "," ... ")" [BASECONSTRUCTORS] [CODE]

DESTRUCTOR = ["__callback"] "~" ID "(" ["void"] ")" [CODE]

QUALIFIER = ("public" | "private" | "protected")

NUMBER = <a C integer or floating-point number, in decimal, octal or hexadecimal notation>

INITDATA = <everything up to end of chunk>

BASECONSTRUCTORS = <everything up to end of chunk>

CODE = <everything up to end of chunk>

The following table shows how argument-types are translated:

[unsigned] char
char
[unsigned] short
[unsigned-]short
[unsigned] int
[unsigned-]integer
[unsigned] long
[unsigned-]long
float
float
double
double
bool
int
__bool
int
__fixnum
int
__scheme_value
scheme-object
char *
c-string
signed char *
s8vector
[signed] short *
s16vector
[signed] int *
s32vector
[signed] long *
s32vector
unsigned char *
u8vector
unsigned short *
u16vector
unsigned int *
u32vector
unsigned long *
u32vector
float *
f32vector
double *
f64vector
CLASS *
(instance CLASS <CLASS>)
TYPE *
(pointer TYPE)
TYPE
&(ref TYPE)
TYPE<T1, ...>
(template TYPE T1 ...)
TYPE1 (*)(TYPE2, ...)
(function TYPE1 (TYPE2 ...))

The following table shows how result-types are translated:

void
void
[unsigned] char
char
[unsigned] short
[unsigned-]short
[unsigned] int
[unsigned-]integer
[unsigned] long
[unsigned-]long
float
float
double
double
bool
bool
__bool
bool
__fixnum
int
__scheme_value
scheme-object
char *
c-string
CLASS *
(instance CLASS <CLASS>)
TYPE *
(pointer TYPE)
TYPE
&(ref TYPE)
TYPE<T1, ...>
(template TYPE T1 ...)
TYPE1 (*)(TYPE2, ...)
(function TYPE1 (TYPE2 ...))


Next: , Previous: Grammar, Up: The Easy Foreign Function Interface

6.7.5 C notes

Foreign variable definitions for macros are not exported from the current compilation unit, but definitions for C variables and functions are.

foreign-parse does not embed the text into the generated C file, use foreign-declare for that (or even better, use the #>! ... <# syntax which does both).

Functions with variable number of arguments are not supported.


Previous: C notes, Up: The Easy Foreign Function Interface

6.7.6 C++ notes

Each C++ class defines a TinyCLOS class, which is a subclass of <c++-object>. Instances of this class contain a single slot named this, which holds a pointer to a heap-allocated C++ instance. The name of the TinyCLOS class is obtained by putting the C++ classname between angled brackets (<...>). TinyCLOS classes are not seen by C++ code.

The C++ constructor is invoked by the initialize generic, which accepts as many arguments as the constructor. If no constructor is defined, a default-constructor will be provided taking no arguments. To allow creating class instances from pointers created in foreign code, the initialize generic will optionally accept an arguments list of the form 'this POINTER, where POINTER is a foreign pointer object. This will create a TinyCLOS instance for the given C++ object.

To release the storage allocated for a C++ instance invoke the destroy generic (the name can be changed by using the destructor_name pseudo declaration).

Static member functions are wrapped in a Scheme procedure named <class>::<member>.

Member variables and non-public member functions are ignored.

Virtual member functions are not seen by C++ code. Overriding a virtual member function with a TinyCLOS method will not work when the member function is called by C++.

Operator functions and default arguments are not supported.

Exceptions must be explicitly handled by user code and may not be thrown beyond an invocation of C++ by Scheme code.


Previous: The Easy Foreign Function Interface, Up: Interface to external functions and variables

6.8 C interface

The following functions and macros are available for C code that invokes Scheme:

— C function: void C_save (C_word x)

Saves the Scheme data object x on the temporary stack.

— C macro: C_word C_fix (int integer)
— C macro: C_word C_make_character (int char_code)
— C macro: C_word C_SCHEME_END_OF_LIST
— C macro: C_word C_SCHEME_END_OF_FILE
— C macro: C_word C_SCHEME_FALSE
— C macro: C_word C_SCHEME_TRUE

These macros return immediate Scheme data objects.

— C function: C_word C_string (C_word **ptr, int length, char *string)
— C function: C_word C_string2 (C_word **ptr, char *zero_terminated_string)
— C function: C_word C_intern2 (C_word **ptr, char *zero_terminated_string)
— C function: C_word C_intern3 (C_word **ptr, char *zero_terminated_string, C_word initial_value)
— C function: C_word C_pair (C_word **ptr, C_word car, C_word cdr)
— C function: C_word C_flonum (C_word **ptr, double number)
— C function: C_word C_int_to_num (C_word **ptr, int integer)
— C function: C_word C_mpointer (C_word **ptr, void *pointer)
— C function: C_word C_vector (C_word **ptr, int length, ...)
— C function: C_word C_list (C_word **ptr, int length, ...)

These functions allocate memory from ptr and initialize a fresh data object. The new data object is returned. ptr should be the address of an allocation pointer created with C_alloc.

— C macro: C_word* C_alloc (int words)

Allocates memory from the C stack (C_alloc) and returns a pointer to it. words should be the number of words needed for all data objects that are to be created in this function. Note that stack-allocated data objects have to be passed to Scheme callback functions, or they will not be seen by the garbage collector. This is really only usable for callback procedure invocations, make sure not to use it in normal code, because the allocated memory will be re-used after the foreign procedure returns. When invoking Scheme callback procedures a minor garbage collection is performed, so data allocated with C_alloc will already have moved to a safe place.

— C macro: int C_SIZEOF_LIST (int length)
— C macro: int C_SIZEOF_STRING (int length)
— C macro: int C_SIZEOF_VECTOR (int length)
— C macro: int C_SIZEOF_INTERNED_SYMBOL (int length)
— C macro: int C_SIZEOF_PAIR
— C macro: int C_SIZEOF_FLONUM
— C macro: int C_SIZEOF_POINTER
— C macro: int C_SIZEOF_LOCATIVE
— C macro: int C_SIZEOF_TAGGED_POINTER

These are macros that return the size in words needed for a data object of a given type.

— C macro: int C_character_code (C_word character)
— C macro: int C_unfix (C_word fixnum)
— C macro: double C_flonum_magnitude (C_word flonum)
— C function: char* C_c_string (C_word string)
— C function: int C_num_to_int (C_word fixnum_or_flonum)
— C function: void* C_pointer_address (C_word pointer)

These macros and functions can be used to convert Scheme data objects back to C data.

— C macro: int C_header_size (C_word x)
— C macro: int C_header_bits (C_word x)

Return the number of elements and the type-bits of the non-immediate Scheme data object x.

— C macro: C_word C_block_item (C_word x, int index)

This macro can be used to access slots of the non-immediate Scheme data object x. index specifies the index of the slot to be fetched, starting at 0. Pairs have 2 slots, one for the car and one for the cdr. Vectors have one slot for each element.

— C macro: void* C_data_pointer (C_word x)

Returns a pointer to the data-section of a non-immediate Scheme object.

— C macro: C_word C_make_header (C_word bits, C_word size)

A macro to build a Scheme object header from its bits and size parts.

— C function: C_word C_mutate (C_word *slot, C_word val)

Assign the Scheme value val to the location specified by slot. If the value points to data inside the nursery (the first heap-generation), then the garbage collector will remember to handle the data appropriately. Assigning nursery-pointers directly will otherwise result in lost data.

— C macro: C_word C_symbol_value (C_word symbol)

Returns the global value of the variable with the name symbol.

— C function: void C_gc_protect (C_word *ptrs[], int n)

Registers n variables at address ptrs to be garbage collection roots. The locations should not contain pointers to data allocated in the nursery, only immediate values or pointers to heap-data are valid. Any assignment of potential nursery data into a root-array should be done via C_mutate(). The variables have to be initialized to sensible values before the next garbage collection starts (when in doubt, set all locations in ptrs to C_SCHEME_UNDEFINED) C_gc_protect may not called before the runtime system has been iniitalized (either by CHICKEN_initialize, CHICKEN_run or CHICKEN_invoke.

— C function: void C_gc_unprotect (int n)

Removes the last n registered variables from the set of root variables.

An example:

% cat foo.scm
#>
extern int callout(int, int, int);
<#

(define callout (foreign-callback-lambda int "callout" int int int))

(define-external (callin (scheme-object xyz)) int
  (print "This is 'callin': " xyz)
  123)

(print (callout 1 2 3))

% cat bar.c
#include <stdio.h>
#include "chicken.h"

extern int callout(int, int, int);
extern int callin(C_word x);

int callout(int x, int y, int z)
{
  C_word *ptr = C_alloc(C_SIZEOF_LIST(3));
  C_word lst;

  printf("This is 'callout': %d, %d, %d\n", x, y, z);
  lst = C_list(&ptr, 3, C_fix(x), C_fix(y), C_fix(z));
  return callin(lst);  /* Note: `callin' will have GC'd the data in `ptr' */
}

% chicken foo.scm -quiet
% gcc foo.c bar.c -o foo `chicken-config -cflags -libs`
% foo
This is 'callout': 1, 2, 3
This is 'callin': (1 2 3)
123

Notes:


Next: , Previous: Interface to external functions and variables, Up: Top

7 chicken-setup


Next: , Up: chicken-setup

7.1 Extension libraries

Extension libraries are extensions to the core functionality provided by the basic CHICKEN system, to be built and installed separately. The mechanism for loading compiled extensions is based on dynamically loadable code and as such is only available on systems on which loading compiled code at runtime is supported. Currently this are most UNIX-compatible platforms that provide the libdl functionality like Linux, Solaris, BSD or Mac OS X. Windows with Cygwin is supported as well.

Note: Extension may also be normal applications or shell scripts.


Next: , Previous: Extension libraries, Up: chicken-setup

7.2 Installing extensions

To install an extension library, run the chicken-setup program with the extension name as argument. If the extension consists of a single Scheme file, then it is compiled and installed in the extension repository. If it is an archive containing addition files, then the files are extracted and the contained setup script is executed. This setup script is a normal Scheme source file, which will be interpreted by chicken-setup. The complete language supported by csi is available, and the library units srfi-1 regex utils posix tcp are loaded. Additional libraries can of course be loaded at run-time.

The setup script should perform all necessary steps to build the new library (or application). After a successful build, the extension can be installed by invoking one of the procedures install-extension, install-program or install-script. These procedures will copy a number of given files into the extension repository or in the path where the CHICKEN executables are located (in the case of executable programs or scripts). Additionally the list of installed files, and user-defined metadata is stored in the repository.


Next: , Previous: Installing extensions, Up: chicken-setup

7.3 Creating extensions

Extensions can be created by creating an archive named EXTENSION.egg containing all needed files plus a .setup script in the root directory. After chicken-setup has extracted the files, the setup script will be invoked. There are no additional constraints on the structure of the archive, but the setup script has to be in the root path of the archive.


Next: , Previous: Creating extensions, Up: chicken-setup

7.4 Procedures and macros available in setup scripts

— procedure: install-extension
          (install-extension ID FILELIST [INFOLIST])
     

Installs extension library with the name ID. All files given in the list of strings FILELIST will be copied to the extension repository. The optional argument INFOLIST should be an association list that maps symbols to values, this list will be stored as ID.setup at the same location as the extension code. Currently the following properties are used:

— property: syntax
               (syntax)
          

Marks the extension as syntax-only. No code is compiled, the extension is intended as a file containing macros to be loaded at compile/macro-expansion time.

— property: require-at-runtime
               (require-at-runtime ID ...)
          

Specifies extensions that should be loaded (via require) at runtime. This is mostly useful for syntax extensions that need additional support code at runtime.

— property: version
               (version STRING)
          

Specifies version string.

All other properties are currently ignored. The FILELIST argument may also be a single string.

— procedure: install-program
          (install-program ID FILELIST [INFOLIST])
     

Similar to install-extension, but installs an executable program in the executable path (usually /usr/local/bin.

— procedure: install-script
          (install-program ID FILELIST [INFOLIST])
     

Similar to install-program, but additionally changes the file permissions of all files in FILELIST to executable (for installing shell-scripts).

— syntax: run
          (run FORM ...)
     

Runs the shell command FORM, which is wrapped in an implicit quasiquote.

— syntax: make
          (make ((TARGET (DEPENDENT ...) COMMAND ...) ...) ARGUMENTS)
     

A “make” macro that executes the expressions COMMAND ..., when any of the dependents DEPENDENT ... have changed, to build TARGET. This is the same as the make extension, which is available separately. For more information, see make.

— procedure: patch
          (patch WHICH REGEX SUBST)
     

Replaces all occurrences of the regular expression REGEX with the string SUBST, in the file given in WHICH. If WHICH is a string, the file will be patched and overwritten. If WHICH is a list of the form OLD NEW, then a different file named NEW will be generated.


Next: , Previous: Procedures and macros available in setup scripts, Up: chicken-setup

7.5 Examples for extensions

The simplest case is a single file that does not export any syntax. For example

     ;;;; hello.scm
     
     (define (hello name)
       (print "Hello, " name " !") )

After entering

$ chicken-setup hello

at the shell prompt, the file hello.scm will be compiled into a dynamically loadable library, with the default compiler options -optimize-level 2 -debug-level 0 -shared. If the compilation succeeds, hello.so will be stored in the repository, together with a file named hello.setup (not to be confused with a setup script - this .setup file just contains an a-list with metadata).

Use it like any other CHICKEN extension:

$ csi -quiet
#;> (require-extension hello)
; loading /usr/local/lib/chicken/hello.so ...
#;> (hello "me")
Hello, me!
#;>

For more elaborate build operations, when installing applications or scripts, or when additional metadata should be stored for an extension, a setup script is required and the script and all additional files should be packaged in a gzipped tar archive.

Here we create a simple application:

     ;;;; hello2.scm
     
     (require-extension extras)  ; needed for `printf'
     
     (print "Hello, ")
     (for-each (lambda (x) (printf "~A " x)) (command-line-arguments))
     (print "!")

We also need a setup script:

     ;;;; hello2.setup
     
     (run (csc hello2.scm))  ; compile `hello2'
     (install-program 'hello2 "hello2") ; name of the extension and files to be installed

To use it, just run chicken-setup in the same directory:

$ chicken-setup hello2

Now the program hello2 will be installed in the same location as the other CHICKEN tools (like chicken, csi, etc.), which will normally be /usr/local/bin. Note that you need write-permissions for those locations.

Uninstallation is just as easy:

$ chicken-setup -uninstall hello2

chicken-setup provides a make tool, so building operations can be of arbitrary complexity. When running chicken-setup with an argument NAME, for which no associated file NAME.setup, NAME.egg or NAME.scm exists will ask you to download the extension via HTTP from the default URL http://www.call-with-current-continuation.org/eggs. You can use the -host option to specify an alternative source location.

Finally a somewhat more complex example: We want to package a syntax extension with additional support code that is to be loaded at run-time of any Scheme code that uses that extension. We create a “glass” lambda, a procedure with free variables that can be manipulated from outside:

     ;;;; glass.scm
     
     (define-macro (glass-lambda llist vars . body)
       ;; Low-level macros are fun!
       (let ([lvar (gensym)]
     	[svar (gensym)]
     	[x (gensym)]
     	[y (gensym)]
     	[yn (gensym)] )
         `(let ,(map (lambda (v) (list v #f)) vars)
            (define (,svar ,x . ,y)
     	 (let* ([,yn (pair? ,y)]
     		[,y (and ,yn (car ,y))] )
     	   (case ,x
     	     ,@(map (lambda (v)
     		      `([,v] (if ,yn
     				 (set! ,v ,y)
     				 ,v) ) )
     		    vars)
     	     (else (error "variable not found" ,x)) ) ) )
            (define ,lvar (lambda ,llist ,@body))
            (extend-procedure ,lvar ,svar) ) ) )

Here some support code that needs to be loaded at runtime:

     ;;;; glass-support.scm
     
     (require-extension lolevel)
     
     (define glass-lambda-accessor procedure-data)
     (define (glass-lambda-ref gl v) ((procedure-data gl) v))
     (define (glass-lambda-set! gl v x) ((procedure-data gl) v x))

The setup script looks like this:

     (run (csc -s -O2 -d0 glass-support.scm))
     
     (install-extension
       'glass
       '("glass.scm" "glass-support.so")
       '((syntax) (require-at-runtime glass-support)) )

The invocation of install-extension provides the files that are to be copied into the extension repository, and a metadata list that specifies that the extension glass is a syntax extension and that, if it is declared to be used by other code (either with the require-extension or require-for-syntax form), then client code should perform an implicit (require 'glass-support) at startup.

This can be conveniently packaged as an “egg”:

$ tar cfz glass.egg glass.setup glass.scm glass-support.scm

And now we use it:

$ csi -quiet
#;> (require-extension glass)
; loading /usr/local/lib/chicken/glass.scm ...
; loading /usr/local/lib/chicken/glass-support.so ...
#;> (define foo (glass-lambda (x) (y) (+ x y)))
#;> (glass-lambda-set! foo 'y 99)
#;> (foo 33)
132


Previous: Examples for extensions, Up: chicken-setup

7.6 chicken-setup reference

Available options:

Note that the options are processed exactly in the order in which they appear in the command-line.


Next: , Previous: chicken-setup, Up: Top

8 Additional files

In addition to library units the following files are provided. Use them by including the file in your code with the include special form.


Next: , Up: Additional files

8.1 srfi-13-syntax.scm

This file provides the let-string-start+end syntax defined in SRFI-13. See Unit srfi-13.


Next: , Previous: srfi-13-syntax.scm, Up: Additional files

8.2 chicken-highlevel-macros.scm

This file contains highlevel (syntax-case) macro definitions for all non-standard macros CHICKEN provides. Normally you don't directly use this file, since it is loaded automatically by the compiler or interpreter, when the -hygienic option is used.

If you intend to make highlevel macros (R5RS and/or syntax-case) available at run-time (for example, for evaluating code at runtime that contains highlevel macros), declare the syntax-case unit as used and install the macro system by hand, as in this example:

(require-extension syntax-case)

(install-highlevel-macro-system)

(This is basically what the -hygienic-at-run-time option does, but this procedure gives you a little more control.)

— procedure: install-highlevel-macro-system
          (install-highlevel-macro-system FEATURE ...)
     

Installs the highlevel (R5RS / syntax-case) macro system in the running program. FEATURE should be a symbol and selects the amount of non-standard macros being provided. Possible features are:

  • r5rs

    The R5RS standard derived syntax, and only those.

  • srfi-0

    SRFI-0 (cond-expand)

  • extensions

    R5RS standard derived syntax and all non-standard extension provided in CHICKEN

If no feature is give, 'r5rs is assumed.


Next: , Previous: chicken-highlevel-macros.scm, Up: Additional files

8.3 chicken-more-macros.scm

This file contains the definitions of all non-standard syntax forms. You normally don't use this file directly.


Next: , Previous: chicken-more-macros.scm, Up: Additional files

8.4 chicken-ffi-macros.scm

This file contains the definitions of macros for interfacing to foreign code, and the definitions contained in this file are automatically made available in compiled code.


Next: , Previous: chicken-ffi-macros.scm, Up: Additional files

8.5 chicken-entry-points.scm

This file contains the definition of the macros define-entry-point and define-embedded. See the section “Entry points” earlier in this manual.


Next: , Previous: chicken-entry-points.scm, Up: Additional files

8.6 chicken-default-entry-points.scm

This file contains boilerplate entry point definitions. See the section “Entry points”. This file automatically includes entry-points.scm.


Previous: chicken-default-entry-points.scm, Up: Additional files

8.7 test-infrastructure.scm

This file provides a macro based unit testing facility based upon expectations concerning evaluations of expressions. These functions return tagged lists which contain the results of the test package, test case, or expectations evaluated(there are a few other types of results dealing with the gloss, todo, and skip macros detailed below). These result lists are wired together during evaluation to form a large hierarchical tree in memory. This result tree is then passed to either user defined functions which traverse the tree manipulating it in any way desired, or passed to a supplied (read: defined already in test-infrastructure.scm) function which manipulates it in a simple way usually producing human readable or html generated output. API functions to deal with the result types are supplied and the representation of the result is black boxed to the user. It is a violation of encapsulation to inspect the representation directly, and it may change unpredictably in the future.


Next: , Up: test-infrastructure.scm

8.7.1 The Test Package Macro API

This macro will evaluate in a left to right fashion the clauses inside it. Clauses can only be certain things, detailed below. All of the clauses are executed, except of course if you bail out of the test package with the escape procedure mechanism. Test packages may nest indefinitely.

— macro: test-package
          (test-package MESSAGE DESTNAME TERMNAME CLAUSES)
          (test-package MESSAGE DESTNAME TERMNAME (BINDINGS) CLAUSES)
          (test-package MESSAGE DESTNAME TERMNAME (warn MESSAGE) CLAUSES)
          (test-package MESSAGE DESTNAME TERMNAME (warn MESSAGE) (BINDINGS) CLAUSES)
     
  • MESSAGE can be any scheme object, though usually it is a string.
  • DESTNAME is an unquoted symbol for an automatic destructor object that gets called when the test package completes for any reason. This symbol is bound to a destructor object and is available to you in the CLAUSES section of the test package. See below for the description of the destructor object interface.
  • TERMNAME is an unquoted symbol for an escape procedure available in the body of the test package, usually, this escape procedure is passed to (terminate ...) which calls it for you and performs other tasks. It is not recommended to call the escape procedure directly.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the test package. The warn function name is actually a syntax reserved word in the macro.
  • BINDINGS are let-style bindings that you may create and exist in the lexical scope of the test package.
  • CLAUSES are uses of (test-case ...) macros along with (gloss ...), (todo ...), (skip ...), and (terminate ...) macros. While you may use the (expect-* ...) style macros directly in a test package, doing so is not recommended. If the expectation fails, the test package macro will continue evaluating until all clauses are evaluated or the escape procedure mechanism is activated. This is different than a test-case macro where upon discovery of a failed expectation, evaluation stops immediately.


Next: , Previous: The Test Package Macro API, Up: test-infrastructure.scm

8.7.2 The Test Case Macro API

This macro will evaluate in a left to right fashion the clauses inside it stopping at the first failed expectation. Clauses can only be certain things as detailed below. You may also stop the execution of expectations if you bail out of the test case with the escape procedure mechanism. Test cases may NOT nest.

— macro: test-case
          (test-case MESSAGE DESTNAME TERMNAME CLAUSES)
          (test-case MESSAGE DESTNAME TERMNAME (BINDINGS) CLAUSES)
          (test-case MESSAGE DESTNAME TERMNAME (warn MESSAGE) CLAUSES)
          (test-case MESSAGE DESTNAME TERMNAME (warn MESSAGE) (BINDINGS) CLAUSES)
     
  • MESSAGE can be any scheme object, though usually it is a string.
  • DESTNAME is an unquoted symbol for an automatic destructor object that gets called when the test case completes for any reason. This symbol is bound to a destructor object and is available to you in the CLAUSES section of the test package. See below for the description of the destructor object interface.
  • TERMNAME is an unquoted symbol for an escape procedure available in the body of the test case, usually, this escape procedure is passed to (terminate ...) which calls it for you and performs other tasks. It is not recommended to call the escape procedure directly.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the test case. The warn function name is actually a syntax reserved word in the macro.
  • BINDINGS are let-style bindings that you may create and exist in the lexical scope of the test case.
  • CLAUSES are uses of (expect-* ...) macros along with (gloss ...), (todo ...), (skip ...), and (terminate ...) macros. It is important to note that upon discovery of a failed expectation, the test case stops its evaluation and returns with the previous successful, and including the failed, expectations.


Next: , Previous: The Test Case Macro API, Up: test-infrastructure.scm

8.7.3 The Expectation Macro API

An expectation at its core simply evaluates its arguments and check to see if it matches the expectation. The positive or negative result is encapsulated, along with other things such as the unevaluated expressions being checked and some messages supplied with each expectation into a particular type of black box object that one can query with the appropriate API calls(detailed below).

Expectations all have a descriptive message that can be bound to them, along with an optional warning syntax detailed below. A design decision was made to supply expectation macros for the usual types of expectations a user needs because this reduced the abstractness of an expectation into something more manageable. In a future release however, I will supply a special expectation macro where you may supply any predicate you wish along with a “type tag” of the predicate.

— macro: expect-zero
          (expect-zero MESSAGE CLAUSE)
          (expect-zero MESSAGE (warn MESSAGE) CLAUSE)
     
  • This expectation checks to see if the evaluated expression passed to it is numerically equal to the exact integer zero.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • CLAUSE is a single expression which should return a exact or inexact integer.

— macro: expect-nonzero
          (expect-nonzero MESSAGE CLAUSE)
          (expect-nonzero MESSAGE (warn MESSAGE) CLAUSE)
     
  • This expectation checks to see if the evaluated expression passed to it is numerically not equal to the exact integer zero.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • CLAUSE is a single expression which should return an exact or inexact integer.

— macro: expect-true
          (expect-true MESSAGE CLAUSE)
          (expect-true MESSAGE (warn MESSAGE) CLAUSE)
     
  • This expectation checks to see if the evaluated expression passed to it is the value #t.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • CLAUSE is a single expression which should return #t.

— macro: expect-false
          (expect-false MESSAGE CLAUSE)
          (expect-false MESSAGE (warn MESSAGE) CLAUSE)
     
  • This expectation checks to see if the evaluated expression passed to it is the value #f.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • CLAUSE is a single expression which should return #t.

— macro: expect-eq?
          (expect-eq? MESSAGE EXPECTED CLAUSE)
          (expect-eq? MESSAGE (warn MESSAGE) EXPECTED CLAUSE)
     
  • This expectation checks to see if (eq? EXPECTED CLAUSE) is true.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • EXPECTED is a single expression which is evaluated and represents the value the CLAUSE must be eq? to in order for this expectation to return a positive result.
  • CLAUSE is a single expression which, when evaluated must return an object where an eq? of this result and the EXPECTED expression is #t.
  • The result object this macro produce shall contain the unevaluated CLAUSE expression as a field, but not an unevaluated EXPECTED expression.

— macro: expect-eqv?
          (expect-eqv? MESSAGE EXPECTED CLAUSE)
          (expect-eqv? MESSAGE (warn MESSAGE) EXPECTED CLAUSE)
     
  • This expectation checks to see if (eqv? EXPECTED CLAUSE) is true.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • EXPECTED is a single expression which is evaluated and represents the value the CLAUSE must be eqv? to in order for this expectation to return a positive result.
  • CLAUSE is a single expression which, when evaluated must return an object where an eqv? of this result and the EXPECTED expression is #t.
  • The result object this macro produce shall contain the unevaluated CLAUSE expression as a field, but not an unevaluated EXPECTED expression.

— macro: expect-equal?
          (expect-equal? MESSAGE EXPECTED CLAUSE)
          (expect-equal? MESSAGE (warn MESSAGE) EXPECTED CLAUSE)
     
  • This expectation checks to see if (equal? EXPECTED CLAUSE) is true.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • EXPECTED is a single expression which is evaluated and represents the value the CLAUSE must be equal? to in order for this expectation to return a positive result.
  • CLAUSE is a single expression which, when evaluated must return an object where an equal? of this result and the EXPECTED expression is #t.
  • The result object this macro produce shall contain the unevaluated CLAUSE expression as a field, but not an unevaluated EXPECTED expression.

— macro: expect-near?
          (expect-near? MESSAGE EXPECTED TOL CLAUSE)
          (expect-near? MESSAGE (warn MESSAGE) EXPECTED TOL CLAUSE)
     
  • This expectation checks to see if (< (abs (- EXPECTED CLAUSE)) TOL))) is true.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • EXPECTED is a single expression which is evaluated and represents the value the CLAUSE must be “near” to in order for this expectation to return a positive result.
  • CLAUSE is a single expression which should return an inexact or exact number.
  • TOL is a single expression which, when evaluated must return a tolerance value(usually a small inexact number like .0001).
  • The result object this macro produce shall contain the unevaluated CLAUSE expression as a field, but not the unevaluated EXPECTED or TOL expression.

— macro: expect-positive
          (expect-positive MESSAGE CLAUSE)
          (expect-positive MESSAGE (warn MESSAGE) CLAUSE)
     
  • This expectation checks to see if the evaluated expression passed to it is a positive value greater than zero.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • CLAUSE is a single expression which should return an inexact or exact number.

— macro: expect-negative
          (expect-negative MESSAGE CLAUSE)
          (expect-negative MESSAGE (warn MESSAGE) CLAUSE)
     
  • This expectation checks to see if the evaluated expression passed to it is a negative value less than zero.
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn MESSAGE) allows you to specify a warning object, usually a string, that gets associated with the expectation. The warn function name is actually a syntax reserved word in the macro.
  • CLAUSE is a single expression which should return an inexact or exact number.


Next: , Previous: The Expectation Macro API, Up: test-infrastructure.scm

8.7.4 Result Object API

Expectations, test cases, test packages, and helper macros(gloss, todo, etc) all return an object that contains the results and other various aspects of the action performed which ultimately get wired together to form the result tree. This collection of functions forming the rest of the test infrastructure API allows manipulation of these results in an abstracted way as to allow changing of the representation in the future.


Next: , Previous: Result Object API, Up: test-infrastructure.scm

8.7.5 Test Package Result Object API

If any of these API functions, except test-package-result?, are passed something that isn't a test package result object, they will return 'not-a-test-package-result.

— procedure: test-package-result?
          (test-package-result? RESULT)
     

If RESULT is a result object from the invocation of a test package macro, then this function will return #t. Otherwise, it will return #f.

— procedure: test-package-result-result-ref
          (test-package-result-result-ref RESULT)
     

Returns the boolean result associated with the test package RESULT object.

— procedure: test-package-result-message-ref
          (test-package-result-message-ref RESULT)
     

Returns the message object associated with the test package RESULT object.

— procedure: test-package-result-exps-ref
          (test-package-result-exps-ref RESULT)
     

Returns the list of result objects associated with the test package RESULT object.

— procedure: test-package-result-warning?
          (test-package-result-warning? RESULT)
     

If a warning had been attached to this test package, this function will return #t, otherwise it will be #f.

— procedure: test-package-result-warning-ref
          (test-package-result-warning-ref RESULT)
     

If a warning had been attached to this test package, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Test Package Result Object API, Up: test-infrastructure.scm

8.7.6 Test Case Result Object API

If any of these API functions, except test-case-result?, are passed something that isn't a test case result object, they will return 'not-a-test-case-result.

— procedure: test-case-result?
          (test-case-result? RESULT)
     

If RESULT is a result object from the invocation of a test case macro, then this function will return #t. Otherwise, it will return #f.

— procedure: test-case-result-result-ref
          (test-case-result-result-ref RESULT)
     

Returns the boolean result associated with the test case RESULT object.

— procedure: test-case-result-message-ref
          (test-case-result-message-ref RESULT)
     

Returns the message object associated with the test case RESULT object.

— procedure: test-case-result-expectations-ref
          (test-case-result-expectations-ref RESULT)
     

Returns the list of expctation result objects associated with the test case RESULT object.

— procedure: test-case-result-warning?
          (test-case-result-warning? RESULT)
     

If a warning had been attached to this test case, this function will return #t, otherwise it will be #f.

— procedure: test-case-result-warning-ref
          (test-case-result-warning-ref RESULT)
     

If a warning had been attached to this test case, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Test Case Result Object API, Up: test-infrastructure.scm

8.7.7 Expect Result Object API: Single Clause Style Expectation

These expectations all take the form of passing a single expression to them to see if they match some a priori expectation. If any of these API functions, except expect-result?, are passed something that isn't a single clause style expectation result object, they will return 'not-an-expect-result.

— procedure: expect-result?
          (expect-result? RESULT)
     

If RESULT is a single clause style result object from the invocation of an expectation macro, then this function will return #t. Otherwise, it will return #f.

— procedure: expect-result-result-ref
          (expect-result-result-ref RESULT)
     

Returns the boolean result associated with the single clause style expectation RESULT object.

— procedure: expect-result-specific-ref
          (expect-result-specific-ref RESULT)
     

This retrieves the “specific” field of a particular single clause style expectation. For example, if you had a result object from an invocation of a (expect-zero? "foobar" (- 1 1)) expectation, then the “specific” field of the expectation result object will be the string "zero". Here is a table describing what the “specific” fields are for each kind of single clause style expectation:

Single Clause Style Expectation Associated Specific String
expect-zero "zero"
expect-nonzero "nonzero"
expect-true "true"
expect-false "false"
expect-positive "positive"
expect-negative "negative"

— procedure: expect-result-message-ref
          (expect-result-message-ref RESULT)
     

Returns the message object associated with the single clause style expectation RESULT object.

— procedure: expect-result-unevaled-ref
          (expect-result-unevaled-ref RESULT)
     

Returns the unevaluated expression supplied to a single clause style expectation macro.

— procedure: expect-result-evaled-ref
          (expect-result-evaled-ref RESULT)
     

Returns the evaluated expression supplied to a single clause style expectation macro.

— procedure: expect-result-warning?
          (expect-result-warning? RESULT)
     

If a warning had been attached to this expectation, this function will return #t, otherwise it will be #f.

— procedure: expect-result-warning-ref
          (expect-result-warning-ref RESULT)
     

If a warning had been attached to this expectation, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Single Clause Style Expectation, Up: test-infrastructure.scm

8.7.8 Expect Result Object API: Equivalence Style Expectation

These expectations all take the form of passing a two expressions, the “left hand side” and the “right hand side” to them to see if they match some a priori equivalence. The left hand side is that which you expect the right hand side to be equivalent. If any of these API functions, except expect-equivalence-result?, are passed something that isn't a single clause style expectation result object, they will return 'not-an-expect-equivalence-result.

— procedure: expect-equivalence-result?
          (expect-equivalence-result? RESULT)
     

If RESULT is a comparison style result object from the invocation of an expectation macro, then this function will return #t. Otherwise, it will return #f.

— procedure: expect-equivalence-result-result-ref
          (expect-equivalence-result-result-ref RESULT)
     

Returns the boolean result associated with the comparison style expectation RESULT object.

— procedure: expect-equivalence-result-specific-ref
          (expect-equivalence-result-specific-ref RESULT)
     

This retrieves the “specific” field of a particular equivalence style expectation. For example, if you had a result object from an invocation of a (expect-equal? "foobar" 0 (- 1 1)) expectation, then the “specific” field of the expectation result object will be the string "equal". Here is a table describing what the “specific” fields are for each kind of equivalence style expectation:

Equivalence Style Expectation Associated Specific String
expect-eq "eq"
expect-eqv "eqv"
expect-equal "equal"

— procedure: expect-equivalence-result-message-ref
          (expect-equivalence-result-message-ref RESULT)
     

Returns the message object associated with the equivalence style expectation RESULT object.

— procedure: expect-equivalence-result-lhs-evaled-ref
          (expect-equivalence-result-lhs-evaled-ref RESULT)
     

Returns the evaluated “left hand side” expression supplied to an equivalence style expectation.

— procedure: expect-equivalence-result-rhs-unevaled-ref
          (expect-equivalence-result-rhs-unevaled-ref RESULT)
     

Returns the unevaluated “right hand side” expression supplied to an equivalence style expectation.

— procedure: expect-equivalence-result-rhs-evaled-ref
          (expect-equivalence-result-rhs-evaled-ref RESULT)
     

Returns the evaluated “right hand side” expression supplied to an equivalence style expectation.

— procedure: expect-equivalence-result-warning?
          (expect-equivalence-result-warning? RESULT)
     

If a warning had been attached to this expectation, this function will return #t, otherwise it will be #f.

— procedure: expect-equivalence-result-warning-ref
          (expect-equivalence-result-warning-ref RESULT)
     

If a warning had been attached to this expectation, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Equivalence Style Expectation, Up: test-infrastructure.scm

8.7.9 Expect Result Object API: Tolerance Style Expectation

This is a specialized expectation which accepts three expressions and checks to see if the “right hand side” is within a “tolerance” of the “left hand side”. There is only one expectation in the tolerance style currently. If any of these API functions, except expect-tolerance-result?, are passed something that isn't a tolerance style expectation result object, they will return 'not-an-expect-tolerance-result.

— procedure: expect-tolerance-result?
          (expect-tolerance-result? RESULT)
     

If RESULT is a tolerance style result object from the invocation of an expectation macro, then this function will return #t. Otherwise, it will return #f.

— procedure: expect-tolerance-result-result-ref
          (expect-tolerance-result-result-ref RESULT)
     

Returns the boolean result associated with the tolerance style expectation RESULT object.

— procedure: expect-tolerance-result-specific-ref
          (expect-tolerance-result-specific-ref RESULT)
     

This retrieves the “specific” field of a particular tolerance style expectation. For example, if you had a result object from an invocation of a (expect-near? "foobar" 100 .01 100.001) expectation, then the “specific” field of the expectation result object will be the string "near". Here is a table describing what the “specific” fields are for each kind of tolerance style expectation:

Tolerance Style Expectation Associated Specific String
expect-near "near"

— procedure: expect-tolerance-result-message-ref
          (expect-tolerance-result-message-ref RESULT)
     

Returns the message object associated with a tolerance style expectation RESULT object.

— procedure: expect-tolerance-result-lhs-evaled-ref
          (expect-tolerance-result-lhs-evaled-ref RESULT)
     

Returns the evaluated “left hand side” expression supplied to a tolerance style expectation.

— procedure: expect-tolerance-result-lhs-tol-evaled-ref
          (expect-tolerance-result-lhs-tol-evaled-ref RESULT)
     

Returns the evaluated “tolerance” expression supplied to a tolerance style expectation.

— procedure: expect-tolerance-result-rhs-unevaled-ref
          (expect-tolerance-result-rhs-unevaled-ref RESULT)
     

Returns the unevaluated “right hand side” expression supplied to a tolerance style expectation.

— procedure: expect-tolerance-result-rhs-evaled-ref
          (expect-tolerance-result-rhs-evaled-ref RESULT)
     

Returns the evaluated “right hand side” expression supplied to a tolerance style expectation.

— procedure: expect-tolerance-result-warning?
          (expect-tolerance-result-warning? RESULT)
     

If a warning had been attached to this expectation, this function will return #t, otherwise it will be #f.

— procedure: expect-tolerance-result-warning-ref
          (expect-tolerance-result-warning-ref RESULT)
     

If a warning had been attached to this expectation, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Tolerance Style Expectation, Up: test-infrastructure.scm

8.7.10 Various Helper API

These upcoming macros and functions allow the author of the test suite to better control both the execution flow of the test suite and “decoration” of the test suite with important information like things yet to do, or just plain documentation.


Next: , Previous: Various Helper API, Up: test-infrastructure.scm

8.7.11 Termination API

When executing in a test package or a test case, one might discover some catastrophic failure of such proportions that it is utterly impossible to continue executing the test case or test package. When that happens you can use the termination facility to exit the test case or test package. Of course, no more expressions will be evaluated in the scope of the termination. It is recommended that you use this method of terminating the test case or test package evaluation since it wraps some contextual information up into the termination result so you can figure out what happened(and where) later when analyzing the result tree.

When using the manipulation API for a terminate result, if you pass a result to one of these function that is not a terminate result, it will return 'not-a-terminate-result.

— procedure: terminate
          (terminate TERMFUNC MESSAGE)
     
  • This is the recommended termination method for a test case or a test package.
  • TERMFUNC is the name of the termination procedure that you specified in a test case or test package. You may pass any test package or test case termination function available to you in the lexical scope in which you call this function. The termination will take effect in the scope of the created termination function.
  • MESSAGE can be any scheme object, though usually it is a string.

— procedure: terminate-result?
          (terminate-result? RESULT)
     

If RESULT is a termination result object from the invocation of a termination function, then this function will return #t. Otherwise, it will return #f.

— procedure: terminate-result-result-ref
          (terminate-result-result-ref RESULT)
     

Returns the boolean result associated with the termination function RESULT object. This is currently hard coded to be #f.

— procedure: terminate-result-scope-ref
          (terminate-result-scope-ref RESULT)
     

The “scope” of the termination result is exactly the MESSAGE parameter supplied to the test case or test package associated with the TERMFUNC.

— procedure: terminate-result-container-ref
          (terminate-result-container-ref RESULT)
     

The “container” of the termination result is going to be either 'test-package or 'test-case depending upon which the TERMFUNC was associated.

— procedure: terminate-result-message-ref
          (terminate-result-message-ref RESULT)
     

Returns the message object associated with the termination RESULT object.


Next: , Previous: Termination API, Up: test-infrastructure.scm

8.7.12 Destructor Object API

The destructor object allows for you to create helper functions which clean up for you usually in case of aborting of a test case or package. For example, suppose you are testing whether or not file writing to a file works correctly in a test case, so, you'd perform an expectation to open the file, and then queue a function in the destructor to remove the file, and then perform the expectation of the write. If the write(or subsequent) expectation fails, then the test case will automatically invoke the helper cleanup function specified in the destructor object that removes the file.

NOTE: This API is still a little experimental in the sense that eventually he destructor object should return a typed result that contains the success of the individual destructor calls. But for now, it is functional for what it does. Also, be VERY CAREFUL that you specify the arguments to these API calls correctly since due to lambda functions not being comparable, this API cannot garuantee that a true destructor object name had been passed to it. So if you call one of the following API calls incorrectly, the behavior will be undefined.

— procedure: destructor-atexit!
          (destructor-atexit! DESTNAME FUNC ARGS ...)
     

This will insert a promise to calculate the FUNC with the supplied ARGS ... into a queue in the DESTNAME destructor object. Multiple invocations of this API call will continue to queue up (FUNC ARGS ...) promises indefinitely. This function returns a special ignore type that is ignored by the test infrastructure system.

— procedure: destructor-activate!
          (destructor-activate! DESTNAME)
     

This function will call, in order of queueing, all the promises embedded into this destructor object, and then delete the queue. This function is ALWAYS called at the completion of a test package or test case; so be careful that the destructor object doesn't contain anything harmful. However, you may call it yourself and if you do, it will execute all of the queued promises and then clear itself. This function returns a special ignore type that is ignored by the test infrastructure system.

— procedure: destructor-clear!
          (destructor-clear! DESTNAME)
     

This function completely removes all of the promises associated with the destructor object DESTNAME. This function returns a special ignore type that is ignored by the test infrastructure system.

— procedure: destructor-dump
          (destructor-dump DESTNAME)
     

This function, mostly used for debugging purposes, prints out a simple representation of the queued atexit functions to the current port. This function returns a special ignore type that is ignored by the test infrastructure system.


Next: , Previous: Destructor Object API, Up: test-infrastructure.scm

8.7.13 Todo API

The purpose of the todo API is to allow the author of a test suite the ability to record into the result tree for later analysis that something still needs to be done. This way you can count/manipulate this information at a later date. Todo macro invocations can occur inside of test cases or test packages.

— macro: todo
          (todo MESSAGE)
          (todo (warn WARNING) MESSAGE)
     
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn WARNING) allows you to specify a warning object, usually a string, that gets associated with the todo. The warn function name is actually a syntax reserved word in the macro.

— procedure: todo-result?
          (todo-result? RESULT)
     

If RESULT is a todo result object from the invocation of a todo macro, then this function will return #t. Otherwise, it will return #f.

— procedure: todo-result-message-ref
          (todo-result-message-ref RESULT)
     

Returns the message object associated with the todo RESULT object.

— procedure: todo-result-warning?
          (todo-result-warning? RESULT)
     

If a warning had been attached to this todo, this function will return #t, otherwise it will be #f.

— procedure: todo-result-warning-ref
          (todo-result-warning-ref RESULT)
     

If a warning had been attached to this todo, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Todo API, Up: test-infrastructure.scm

8.7.14 Gloss API

The purpose of the gloss API is to allow the author of a test suite the ability to record messages into the result tree purely for documentation purposes. Gloss macro invocations can occur inside of test cases or test packages.

— macro: gloss
          (gloss MESSAGE)
          (gloss (warn WARNING) MESSAGE)
     
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn WARNING) allows you to specify a warning object, usually a string, that gets associated with the gloss. The warn function name is actually a syntax reserved word in the macro.

— procedure: gloss-result?
          (gloss-result? RESULT)
     

If RESULT is a gloss result object from the invocation of the gloss macro, then this function will return #t. Otherwise, it will return #f.

— procedure: gloss-result-message-ref
          (gloss-result-message-ref RESULT)
     

Returns the message object associated with the gloss RESULT object.

— procedure: gloss-result-warning?
          (gloss-result-warning? RESULT)
     

If a warning had been attached to this gloss, this function will return #t, otherwise it will be #f.

— procedure: gloss-result-warning-ref
          (gloss-result-warning-ref RESULT)
     

If a warning had been attached to this gloss, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Gloss API, Up: test-infrastructure.scm

8.7.15 Skip API

The purpose of the skip API is to allow the author of a test suite to completely skip evaluation of a set of expresssions. Skip macro invocations can occur inside of test cases or test packages.

— macro: skip
          (skip MESSAGE CLAUSES)
          (skip (warn WARNING) MESSAGE CLAUSES)
     
  • MESSAGE can be any scheme object, though usually it is a string.
  • (warn WARNING) allows you to specify a warning object, usually a string, that gets associated with the gloss. The warn function name is actually a syntax reserved word in the macro.
  • CLAUSES can be more than one expression(as in a lambda form) that does NOT get evaluated at any time.

— procedure: skip-result?
          (skip-result? RESULT)
     

If RESULT is a skip result object from the invocation of the skip macro, then this function will return #t. Otherwise, it will return #f.

— procedure: skip-result-message-ref
          (skip-result-message-ref RESULT)
     

Returns the message object associated with the skip RESULT object. Hopefully, it was stated why this set of clauses had been skipped.

— procedure: skip-result-warning?
          (skip-result-warning? RESULT)
     

If a warning had been attached to this skip, this function will return #t, otherwise it will be #f.

— procedure: skip-result-warning-ref
          (skip-result-warning-ref RESULT)
     

If a warning had been attached to this skip, this function will return the warning object supplied by the user, otherwise it shall return '().


Next: , Previous: Skip API, Up: test-infrastructure.scm

8.7.16 Side Effect API

This section of the API just contains a single macro currently since it is considered a little experimental for now. The side effecting evaluates all of its arguments as in a (begin ...) form, it returns a result that is completely ignored by the system and unavailable to the output analysis code.

— macro: side-effect
          (side-effect CLAUSES)
     

This macro expands into a begin form the clauses in order and when it finishes evaluating them, returns a result that is silently ignored by the testing infrastructure system. Usually this is used in conjunction with (set! ...) or with complicated situations where a lot of setup work must happen for an expectation to be performed.


Next: , Previous: Side Effect API, Up: test-infrastructure.scm

8.7.17 Miscellaneous API

This section contains a few functions whose purpose is to simplify certain kinds of manipulations of result objects.

— procedure: *-restult?
          (*-result? RESULTOBJ)
     

This function will return #t of RESULTOBJ is any kind of an evaluated result object. Otherwise, it shall return #f.

— procedure: *-result-ref
          (*-result-ref RESULTOBJ)
     

This function will return the result of any kind of RESULTOBJ passed to it if it is indeed a true result object. Otherwise, it shall emit an error to the current port, and return #f.

— procedure: all-testpackage-results-true?
          (all-testpackage-results-true? RESULTLIST)
     

This function takes the result of a call to (test-package-result-result-ref PACKAGERESULTOBJ) and returns #t if every single contained result in that list had been true, or #f otherwise.

— procedure: all-testcase-expectations-true?
          (all-testcase-expectations-true? RESULTLIST)
     

This function takes the result of a call to (test-case-result-result-ref CASERESULTOBJ) and returns #t if every single contained result in that list had been true, or #f otherwise.


Next: , Previous: Miscellaneous API, Up: test-infrastructure.scm

8.7.18 Analysis of the Result Tree

Once a result tree has been evaluated and constructed in memory, what do you do with it? Well, you can do anything you want with it if you choose to write the analysis or output generation functions and pass it the evaluated result tree, and this is, in fact, encouraged. However, usually you just want to print out the tree in a human readable format so you can check things before the serious analysis code gets written. So, to save work for the test suite designer, a tiny API has been written to produce human readable output given a tree of results rooted in a single test package. Of course the single test package may have many other test packages and test cases embedded within it.


Next: , Previous: Analysis of the Result Tree, Up: test-infrastructure.scm

8.7.19 Output Generation API

— procedure: printnl
          (printnl CLAUSES)
     

This function will print all of the evaluated CLAUSES in order with a newline at the end of it.

— procedure: printinl
          (printinl INDENT CLAUSES)
     

This function will INDENT a number of spaces and then print all of the evaluated CLAUSES in order with a newline at the end of it.

— procedure: output-style-human
          (output-style-human RESULTTREE)
     

This function will print out a human readable rendering of the RESULTTREE and return the toplevel package result.


Previous: Output Generation API, Up: test-infrastructure.scm

8.7.20 Example Usages of the Test Suite Infrastructure

This section contains some simple examples of how to author test suites.

Here is a simple example:

(let ((result
        ;; output-style-human function requires a single test package
        ;; to encapsulate everything.
        (test-package "Arithmetic Operators" pd pe

                (test-case "Testing '+'" d e
                        (expect-equal "Adding two positive numbers" 2 (+ 1 1))
                        (expect-equal "Adding two negative numbers" -2 (+ -1 -1))
                        (expect-zero "Adding positive and negative" (+ -1 1)))

                (test-case "Testing '-'" d e
                        (expect-zero "Subtracting two positive numbers" (- 1 1))
                        (expect-zero "Subtracting two negative numbers" (- -1 -1))
                        (expect-equal "Subtracting positive and negative" -2 (- -1 1))))))
        (output-style-human result))

The above example, when evaluated, will produce some human readable output and the a #t value as the result of the top level package. The result variable contains the tree of evaluated expectations, test cases, and the package arranged in a hierarchy extremely similar to the nesting of the above macros. If you desire to manipulate the result tree yourself, you may use the various APIs to manipulate the various results. Please see the implementation of (output-style-human ...) (it isn't large) in the test-infrastructure.scm file to see an example of this.

The variables: pe, e are the escape functions you may use with the (terminate ...) function call if you wish to abort the above code somewhere.

The variables: pd, d allow use of a "destructor" object which allows you to run side effecting functions at the "finishing" point of the evaluation of the test case or test package. These functions are run no matter if the code succeeded correctly or not, so be careful to manage the destructor object carefully so you don't perform unwanted side effects. The names of the destructor objects you supply are lexically scoped in the bodies of the test case or test package.

Now, here is some example output that (output-style-human ...) might generate with the above testing code:

Begin Package: Arithmetic Operators
  Begin Test Case: Testing '+'
    Begin Expectation: Adding two positive numbers
    Expect equal
      Expected Value: 
      2
      Unevaluated: 
      (+ 1 1)
      Evaluated: 
      2
    Result: #t
    End Expectation: Adding two positive numbers

    Begin Expectation: Adding two negative numbers
    Expect equal
      Expected Value: 
      -2
      Unevaluated: 
      (+ -1 -1)
      Evaluated: 
      -2
    Result: #t
    End Expectation: Adding two negative numbers

    Begin Expectation: Adding positive and negative
    Expect zero
      Unevaluated: 
      (+ -1 1)
      Evaluated: 
      0
    Result: #t
    End Expectation: Adding positive and negative

  Result: #t
  End Test Case: Testing '+'

  Begin Test Case: Testing '-'
    Begin Expectation: Subtracting two positive numbers
    Expect zero
      Unevaluated: 
      (- 1 1)
      Evaluated: 
      0
    Result: #t
    End Expectation: Subtracting two positive numbers

    Begin Expectation: Subtracting two negative numbers
    Expect zero
      Unevaluated: 
      (- -1 -1)
      Evaluated: 
      0
    Result: #t
    End Expectation: Subtracting two negative numbers

    Begin Expectation: Subtracting positive and negative
    Expect equal
      Expected Value: 
      -2
      Unevaluated: 
      (- -1 1)
      Evaluated: 
      -2
    Result: #t
    End Expectation: Subtracting positive and negative

  Result: #t
  End Test Case: Testing '-'

Result: #t
End Package: Arithmetic Operators

#t


Next: , Previous: Additional files, Up: Top

9 Data Representation

There exist two different kinds of data objects in the CHICKEN system: immediate and non-immediate objects. Immediate objects are represented by a tagged machine word, which is usually of 32 bits length (64 bits on 64-bit architectures). The immediate objects come in four different flavors:

Non-immediate objects are blocks of data represented by a pointer into the heap. The first word of the data block contains a header, which gives information about the type of the object. The header has the size of a machine word, usually 32 bits (64 bits on 64 bit architectures).

The actual data follows immediately after the header. Note that block-addresses are always aligned to the native machine-word boundary. Scheme data objects map to blocks in the following manner:

Data objects may be allocated outside of the garbage collected heap, as long as their layout follows the above mentioned scheme. But care has to be taken not to mutate these objects with heap-data (i.e. non-immediate objects), because this will confuse the garbage collector.

For more information see the header file chicken.h.


Next: , Previous: Data Representation, Up: Top

10 Bugs and limitations


Next: , Previous: Bugs and limitations, Up: Top

11 Acknowledgements

Thanks also to:

William Annis, Marc Baily, Peter Barabas, Peter Bex, Dave Bodenstab, T. Kurt Bond, Terence Brannon, Roy Bryant, Taylor Campbell, Franklin Chen, Grzegorz Chrupala, James Crippen, Alejandro Forero Cuervo, Brian Denheyer, Chris Double, Petter Egesund, Daniel B. Faken, Fizzie, Kimura Fuyuki, Martin Gasbichler, Joey Gibson, Johannes Groedem, Andreas Gustafsson, Jun-ichiro itojun Hagino, Matthias Heiler, Karl M. Hegbloom, William P. Heinemann, Dale Jordan, Valentin Kamyshenko, Ron Kneusel, Matthias Koeppe, Todd R. Kueny Sr, Charles Martin, Alain Mellan, Perry Metzger, Scott G. Miller, Mikael, Bruce Mitchener, Eric Merrit, Eric E. Moore, o.t., David Rush, Lars Rustemeier, Oskar Schirmer, Burton Samograd, Ronald Schroder, Spencer Schumann, Shmul, Jeffrey B. Siegal, Robert Skeels, Jason Songhurst, Clifford Stein, Christian Tismer, Vladimir Tsichevsky, Neil van Dyke, Sander Vesik, Shawn Wagner, Matthew Welland, Richard Zidlicky and Houman Zolfaghari for bug-fixes, tips, suggestions and moral support.


Next: , Previous: Acknowledgements, Up: Top

Bibliography

Henry Baker: CONS Should Not CONS Its Arguments, Part II: Cheney on the M.T.A.
http://home.pipeline.com/\~hbaker1/CheneyMTA.html
Revised^5 Report on the Algorithmic Language Scheme
http://www.schemers.org/Documents/Standards/R5RS


Previous: Bibliography, Up: Top

Index