mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-06-13 07:10:20 +02:00
use scm_is_true instead of SCM_NFALSEP in the examples.
This commit is contained in:
parent
abe1308cb9
commit
ac9a5a03e5
1 changed files with 0 additions and 762 deletions
|
@ -1,762 +0,0 @@
|
||||||
@c -*-texinfo-*-
|
|
||||||
@c This is part of the GNU Guile Reference Manual.
|
|
||||||
@c Copyright (C) 1996, 1997, 2000, 2001, 2002, 2003, 2004
|
|
||||||
@c Free Software Foundation, Inc.
|
|
||||||
@c See the file guile.texi for copying conditions.
|
|
||||||
|
|
||||||
@page
|
|
||||||
@node Utility Functions
|
|
||||||
@section General Utility Functions
|
|
||||||
|
|
||||||
@c FIXME::martin: Review me!
|
|
||||||
|
|
||||||
This chapter contains information about procedures which are not cleanly
|
|
||||||
tied to a specific data type. Because of their wide range of
|
|
||||||
applications, they are collected in a @dfn{utility} chapter.
|
|
||||||
|
|
||||||
@menu
|
|
||||||
* Equality:: When are two values `the same'?
|
|
||||||
* Object Properties:: A modern interface to object properties.
|
|
||||||
* Sorting:: Sort utility procedures.
|
|
||||||
* Copying:: Copying deep structures.
|
|
||||||
* General Conversion:: Converting objects to strings.
|
|
||||||
* Hooks:: User-customizable event lists.
|
|
||||||
@end menu
|
|
||||||
|
|
||||||
|
|
||||||
@node Equality
|
|
||||||
@subsection Equality
|
|
||||||
|
|
||||||
@c FIXME::martin: Review me!
|
|
||||||
|
|
||||||
@cindex sameness
|
|
||||||
@cindex equality
|
|
||||||
|
|
||||||
Three different kinds of @dfn{sameness} are defined in Scheme.
|
|
||||||
|
|
||||||
@itemize @bullet
|
|
||||||
@item
|
|
||||||
Two values can refer to exactly the same object.
|
|
||||||
|
|
||||||
@item
|
|
||||||
Two objects can have the same @dfn{value}.
|
|
||||||
|
|
||||||
@item
|
|
||||||
Two objects can be structurally equivalent.
|
|
||||||
@end itemize
|
|
||||||
|
|
||||||
The differentiation between these three kinds is important, because
|
|
||||||
determining whether two values are the same objects is very efficient,
|
|
||||||
while determining structural equivalence can be quite expensive
|
|
||||||
(consider comparing two very long lists). Therefore, three different
|
|
||||||
procedures for testing for equality are provided, which correspond to
|
|
||||||
the three kinds of @dfn{sameness} defined above.
|
|
||||||
|
|
||||||
@rnindex eq?
|
|
||||||
@deffn {Scheme Procedure} eq? x y
|
|
||||||
@deffnx {C Function} scm_eq_p (x, y)
|
|
||||||
Return @code{#t} iff @var{x} references the same object as @var{y}.
|
|
||||||
@code{eq?} is similar to @code{eqv?} except that in some cases it is
|
|
||||||
capable of discerning distinctions finer than those detectable by
|
|
||||||
@code{eqv?}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deftypefn {C Function} int scm_is_eq (SCM x, SCM y)
|
|
||||||
Return @code{1} when @var{x} and @var{y} are equal in the sense of
|
|
||||||
@code{eq?}, else return @code{0}.
|
|
||||||
@end deftypefn
|
|
||||||
|
|
||||||
@rnindex eqv?
|
|
||||||
@deffn {Scheme Procedure} eqv? x y
|
|
||||||
@deffnx {C Function} scm_eqv_p (x, y)
|
|
||||||
The @code{eqv?} procedure defines a useful equivalence relation on objects.
|
|
||||||
Briefly, it returns @code{#t} if @var{x} and @var{y} should normally be
|
|
||||||
regarded as the same object. This relation is left slightly open to
|
|
||||||
interpretation, but works for comparing immediate integers, characters,
|
|
||||||
and inexact numbers.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@rnindex equal?
|
|
||||||
@deffn {Scheme Procedure} equal? x y
|
|
||||||
@deffnx {C Function} scm_equal_p (x, y)
|
|
||||||
Return @code{#t} iff @var{x} and @var{y} are recursively @code{eqv?}
|
|
||||||
equivalent. @code{equal?} recursively compares the contents of pairs,
|
|
||||||
vectors, and strings, applying @code{eqv?} on other objects such as
|
|
||||||
numbers and symbols. A rule of thumb is that objects are generally
|
|
||||||
@code{equal?} if they print the same. @code{equal?} may fail to
|
|
||||||
terminate if its arguments are circular data structures.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
|
|
||||||
@node Object Properties
|
|
||||||
@subsection Object Properties
|
|
||||||
|
|
||||||
It's often useful to associate a piece of additional information with a
|
|
||||||
Scheme object even though that object does not have a dedicated slot
|
|
||||||
available in which the additional information could be stored. Object
|
|
||||||
properties allow you to do just that.
|
|
||||||
|
|
||||||
An object property is most commonly used to associate one kind of
|
|
||||||
additional information with each instance of a class of similar Scheme
|
|
||||||
objects. For example, all procedures have a `name' property, which
|
|
||||||
stores the name of the variable in which the procedure was stored by a
|
|
||||||
@code{define} expression, or @code{#f} if the procedure wasn't created
|
|
||||||
by that kind of expression.
|
|
||||||
|
|
||||||
Guile's representation of an object property is a procedure-with-setter
|
|
||||||
(@pxref{Procedures with Setters}) that can be used with the generalized
|
|
||||||
form of @code{set!} (REFFIXME) to set and retrieve that property for any
|
|
||||||
Scheme object. So, setting a property looks like this:
|
|
||||||
|
|
||||||
@lisp
|
|
||||||
(set! (my-property obj1) value-for-obj1)
|
|
||||||
(set! (my-property obj2) value-for-obj2)
|
|
||||||
@end lisp
|
|
||||||
|
|
||||||
@noindent
|
|
||||||
And retrieving values of the same property looks like this:
|
|
||||||
|
|
||||||
@lisp
|
|
||||||
(my-property obj1)
|
|
||||||
@result{}
|
|
||||||
value-for-obj1
|
|
||||||
|
|
||||||
(my-property obj2)
|
|
||||||
@result{}
|
|
||||||
value-for-obj2
|
|
||||||
@end lisp
|
|
||||||
|
|
||||||
To create an object property in the first place, use the
|
|
||||||
@code{make-object-property} procedure:
|
|
||||||
|
|
||||||
@lisp
|
|
||||||
(define my-property (make-object-property))
|
|
||||||
@end lisp
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} make-object-property
|
|
||||||
Create and return an object property. An object property is a
|
|
||||||
procedure-with-setter that can be called in two ways. @code{(set!
|
|
||||||
(@var{property} @var{obj}) @var{val})} sets @var{obj}'s @var{property}
|
|
||||||
to @var{val}. @code{(@var{property} @var{obj})} returns the current
|
|
||||||
setting of @var{obj}'s @var{property}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
A single object property created by @code{make-object-property} can
|
|
||||||
associate distinct property values with all Scheme values that are
|
|
||||||
distinguishable by @code{eq?} (including, for example, integers).
|
|
||||||
|
|
||||||
Internally, object properties are implemented using a weak key hash
|
|
||||||
table. This means that, as long as a Scheme value with property values
|
|
||||||
is protected from garbage collection, its property values are also
|
|
||||||
protected. When the Scheme value is collected, its entry in the
|
|
||||||
property table is removed and so the (ex-) property values are no longer
|
|
||||||
protected by the table.
|
|
||||||
|
|
||||||
@menu
|
|
||||||
* Property Primitives:: Low level property implementation.
|
|
||||||
* Old-fashioned Properties:: An older approach to properties.
|
|
||||||
@end menu
|
|
||||||
|
|
||||||
|
|
||||||
@node Property Primitives
|
|
||||||
@subsubsection Low Level Property Implementation.
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} primitive-make-property not-found-proc
|
|
||||||
@deffnx {C Function} scm_primitive_make_property (not_found_proc)
|
|
||||||
Create a @dfn{property token} that can be used with
|
|
||||||
@code{primitive-property-ref} and @code{primitive-property-set!}.
|
|
||||||
See @code{primitive-property-ref} for the significance of
|
|
||||||
@var{not-found-proc}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} primitive-property-ref prop obj
|
|
||||||
@deffnx {C Function} scm_primitive_property_ref (prop, obj)
|
|
||||||
Return the property @var{prop} of @var{obj}.
|
|
||||||
|
|
||||||
When no value has yet been associated with @var{prop} and @var{obj},
|
|
||||||
the @var{not-found-proc} from @var{prop} is used. A call
|
|
||||||
@code{(@var{not-found-proc} @var{prop} @var{obj})} is made and the
|
|
||||||
result set as the property value. If @var{not-found-proc} is
|
|
||||||
@code{#f} then @code{#f} is the property value.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} primitive-property-set! prop obj val
|
|
||||||
@deffnx {C Function} scm_primitive_property_set_x (prop, obj, val)
|
|
||||||
Set the property @var{prop} of @var{obj} to @var{val}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} primitive-property-del! prop obj
|
|
||||||
@deffnx {C Function} scm_primitive_property_del_x (prop, obj)
|
|
||||||
Remove any value associated with @var{prop} and @var{obj}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
|
|
||||||
@node Old-fashioned Properties
|
|
||||||
@subsubsection An Older Approach to Properties
|
|
||||||
|
|
||||||
Traditionally, Lisp systems provide a different object property
|
|
||||||
interface to that provided by @code{make-object-property}, in which the
|
|
||||||
object property that is being set or retrieved is indicated by a symbol.
|
|
||||||
|
|
||||||
Guile includes this older kind of interface as well, but it may well be
|
|
||||||
removed in a future release, as it is less powerful than
|
|
||||||
@code{make-object-property} and so increases the size of the Guile
|
|
||||||
library for no benefit. (And it is trivial to write a compatibility
|
|
||||||
layer in Scheme.)
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} object-properties obj
|
|
||||||
@deffnx {C Function} scm_object_properties (obj)
|
|
||||||
Return @var{obj}'s property list.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} set-object-properties! obj alist
|
|
||||||
@deffnx {C Function} scm_set_object_properties_x (obj, alist)
|
|
||||||
Set @var{obj}'s property list to @var{alist}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} object-property obj key
|
|
||||||
@deffnx {C Function} scm_object_property (obj, key)
|
|
||||||
Return the property of @var{obj} with name @var{key}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} set-object-property! obj key value
|
|
||||||
@deffnx {C Function} scm_set_object_property_x (obj, key, value)
|
|
||||||
In @var{obj}'s property list, set the property named @var{key}
|
|
||||||
to @var{value}.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
|
|
||||||
@node Sorting
|
|
||||||
@subsection Sorting
|
|
||||||
|
|
||||||
@c FIXME::martin: Review me!
|
|
||||||
|
|
||||||
@cindex sorting
|
|
||||||
@cindex sorting lists
|
|
||||||
@cindex sorting vectors
|
|
||||||
|
|
||||||
Sorting is very important in computer programs. Therefore, Guile comes
|
|
||||||
with several sorting procedures built-in. As always, procedures with
|
|
||||||
names ending in @code{!} are side-effecting, that means that they may
|
|
||||||
modify their parameters in order to produce their results.
|
|
||||||
|
|
||||||
The first group of procedures can be used to merge two lists (which must
|
|
||||||
be already sorted on their own) and produce sorted lists containing
|
|
||||||
all elements of the input lists.
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} merge alist blist less
|
|
||||||
@deffnx {C Function} scm_merge (alist, blist, less)
|
|
||||||
Merge two already sorted lists into one.
|
|
||||||
Given two lists @var{alist} and @var{blist}, such that
|
|
||||||
@code{(sorted? alist less?)} and @code{(sorted? blist less?)},
|
|
||||||
return a new list in which the elements of @var{alist} and
|
|
||||||
@var{blist} have been stably interleaved so that
|
|
||||||
@code{(sorted? (merge alist blist less?) less?)}.
|
|
||||||
Note: this does _not_ accept vectors.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} merge! alist blist less
|
|
||||||
@deffnx {C Function} scm_merge_x (alist, blist, less)
|
|
||||||
Takes two lists @var{alist} and @var{blist} such that
|
|
||||||
@code{(sorted? alist less?)} and @code{(sorted? blist less?)} and
|
|
||||||
returns a new list in which the elements of @var{alist} and
|
|
||||||
@var{blist} have been stably interleaved so that
|
|
||||||
@code{(sorted? (merge alist blist less?) less?)}.
|
|
||||||
This is the destructive variant of @code{merge}
|
|
||||||
Note: this does _not_ accept vectors.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
The following procedures can operate on sequences which are either
|
|
||||||
vectors or list. According to the given arguments, they return sorted
|
|
||||||
vectors or lists, respectively. The first of the following procedures
|
|
||||||
determines whether a sequence is already sorted, the other sort a given
|
|
||||||
sequence. The variants with names starting with @code{stable-} are
|
|
||||||
special in that they maintain a special property of the input sequences:
|
|
||||||
If two or more elements are the same according to the comparison
|
|
||||||
predicate, they are left in the same order as they appeared in the
|
|
||||||
input.
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} sorted? items less
|
|
||||||
@deffnx {C Function} scm_sorted_p (items, less)
|
|
||||||
Return @code{#t} iff @var{items} is a list or a vector such that
|
|
||||||
for all 1 <= i <= m, the predicate @var{less} returns true when
|
|
||||||
applied to all elements i - 1 and i
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} sort items less
|
|
||||||
@deffnx {C Function} scm_sort (items, less)
|
|
||||||
Sort the sequence @var{items}, which may be a list or a
|
|
||||||
vector. @var{less} is used for comparing the sequence
|
|
||||||
elements. This is not a stable sort.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} sort! items less
|
|
||||||
@deffnx {C Function} scm_sort_x (items, less)
|
|
||||||
Sort the sequence @var{items}, which may be a list or a
|
|
||||||
vector. @var{less} is used for comparing the sequence
|
|
||||||
elements. The sorting is destructive, that means that the
|
|
||||||
input sequence is modified to produce the sorted result.
|
|
||||||
This is not a stable sort.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} stable-sort items less
|
|
||||||
@deffnx {C Function} scm_stable_sort (items, less)
|
|
||||||
Sort the sequence @var{items}, which may be a list or a
|
|
||||||
vector. @var{less} is used for comparing the sequence elements.
|
|
||||||
This is a stable sort.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} stable-sort! items less
|
|
||||||
@deffnx {C Function} scm_stable_sort_x (items, less)
|
|
||||||
Sort the sequence @var{items}, which may be a list or a
|
|
||||||
vector. @var{less} is used for comparing the sequence elements.
|
|
||||||
The sorting is destructive, that means that the input sequence
|
|
||||||
is modified to produce the sorted result.
|
|
||||||
This is a stable sort.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
The procedures in the last group only accept lists or vectors as input,
|
|
||||||
as their names indicate.
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} sort-list items less
|
|
||||||
@deffnx {C Function} scm_sort_list (items, less)
|
|
||||||
Sort the list @var{items}, using @var{less} for comparing the
|
|
||||||
list elements. This is a stable sort.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} sort-list! items less
|
|
||||||
@deffnx {C Function} scm_sort_list_x (items, less)
|
|
||||||
Sort the list @var{items}, using @var{less} for comparing the
|
|
||||||
list elements. The sorting is destructive, that means that the
|
|
||||||
input list is modified to produce the sorted result.
|
|
||||||
This is a stable sort.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} restricted-vector-sort! vec less startpos endpos
|
|
||||||
@deffnx {C Function} scm_restricted_vector_sort_x (vec, less, startpos, endpos)
|
|
||||||
Sort the vector @var{vec}, using @var{less} for comparing
|
|
||||||
the vector elements. @var{startpos} and @var{endpos} delimit
|
|
||||||
the range of the vector which gets sorted. The return value
|
|
||||||
is not specified.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
|
|
||||||
@node Copying
|
|
||||||
@subsection Copying Deep Structures
|
|
||||||
|
|
||||||
@c FIXME::martin: Review me!
|
|
||||||
|
|
||||||
The procedures for copying lists (@pxref{Lists}) only produce a flat
|
|
||||||
copy of the input list, and currently Guile does not even contain
|
|
||||||
procedures for copying vectors. @code{copy-tree} can be used for these
|
|
||||||
application, as it does not only copy the spine of a list, but also
|
|
||||||
copies any pairs in the cars of the input lists.
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} copy-tree obj
|
|
||||||
@deffnx {C Function} scm_copy_tree (obj)
|
|
||||||
Recursively copy the data tree that is bound to @var{obj}, and return a
|
|
||||||
pointer to the new data structure. @code{copy-tree} recurses down the
|
|
||||||
contents of both pairs and vectors (since both cons cells and vector
|
|
||||||
cells may point to arbitrary objects), and stops recursing when it hits
|
|
||||||
any other object.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
|
|
||||||
@node General Conversion
|
|
||||||
@subsection General String Conversion
|
|
||||||
|
|
||||||
@c FIXME::martin: Review me!
|
|
||||||
|
|
||||||
When debugging Scheme programs, but also for providing a human-friendly
|
|
||||||
interface, a procedure for converting any Scheme object into string
|
|
||||||
format is very useful. Conversion from/to strings can of course be done
|
|
||||||
with specialized procedures when the data type of the object to convert
|
|
||||||
is known, but with this procedure, it is often more comfortable.
|
|
||||||
|
|
||||||
@code{object->string} converts an object by using a print procedure for
|
|
||||||
writing to a string port, and then returning the resulting string.
|
|
||||||
Converting an object back from the string is only possible if the object
|
|
||||||
type has a read syntax and the read syntax is preserved by the printing
|
|
||||||
procedure.
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} object->string obj [printer]
|
|
||||||
@deffnx {C Function} scm_object_to_string (obj, printer)
|
|
||||||
Return a Scheme string obtained by printing @var{obj}.
|
|
||||||
Printing function can be specified by the optional second
|
|
||||||
argument @var{printer} (default: @code{write}).
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
|
|
||||||
@node Hooks
|
|
||||||
@subsection Hooks
|
|
||||||
@tpindex Hooks
|
|
||||||
|
|
||||||
A hook is a list of procedures to be called at well defined points in
|
|
||||||
time. Typically, an application provides a hook @var{h} and promises
|
|
||||||
its users that it will call all of the procedures in @var{h} at a
|
|
||||||
defined point in the application's processing. By adding its own
|
|
||||||
procedure to @var{h}, an application user can tap into or even influence
|
|
||||||
the progress of the application.
|
|
||||||
|
|
||||||
Guile itself provides several such hooks for debugging and customization
|
|
||||||
purposes: these are listed in a subsection below.
|
|
||||||
|
|
||||||
When an application first creates a hook, it needs to know how many
|
|
||||||
arguments will be passed to the hook's procedures when the hook is run.
|
|
||||||
The chosen number of arguments (which may be none) is declared when the
|
|
||||||
hook is created, and all the procedures that are added to that hook must
|
|
||||||
be capable of accepting that number of arguments.
|
|
||||||
|
|
||||||
A hook is created using @code{make-hook}. A procedure can be added to
|
|
||||||
or removed from a hook using @code{add-hook!} or @code{remove-hook!},
|
|
||||||
and all of a hook's procedures can be removed together using
|
|
||||||
@code{reset-hook!}. When an application wants to run a hook, it does so
|
|
||||||
using @code{run-hook}.
|
|
||||||
|
|
||||||
@menu
|
|
||||||
* Hook Example:: Hook usage by example.
|
|
||||||
* Hook Reference:: Reference of all hook procedures.
|
|
||||||
* C Hooks:: Hooks for use from C code.
|
|
||||||
* GC Hooks:: Garbage collection hooks.
|
|
||||||
* REPL Hooks:: Hooks into the Guile REPL.
|
|
||||||
@end menu
|
|
||||||
|
|
||||||
|
|
||||||
@node Hook Example
|
|
||||||
@subsubsection Hook Usage by Example
|
|
||||||
|
|
||||||
Hook usage is shown by some examples in this section. First, we will
|
|
||||||
define a hook of arity 2 --- that is, the procedures stored in the hook
|
|
||||||
will have to accept two arguments.
|
|
||||||
|
|
||||||
@lisp
|
|
||||||
(define hook (make-hook 2))
|
|
||||||
hook
|
|
||||||
@result{} #<hook 2 40286c90>
|
|
||||||
@end lisp
|
|
||||||
|
|
||||||
Now we are ready to add some procedures to the newly created hook with
|
|
||||||
@code{add-hook!}. In the following example, two procedures are added,
|
|
||||||
which print different messages and do different things with their
|
|
||||||
arguments.
|
|
||||||
|
|
||||||
@lisp
|
|
||||||
(add-hook! hook (lambda (x y)
|
|
||||||
(display "Foo: ")
|
|
||||||
(display (+ x y))
|
|
||||||
(newline)))
|
|
||||||
(add-hook! hook (lambda (x y)
|
|
||||||
(display "Bar: ")
|
|
||||||
(display (* x y))
|
|
||||||
(newline)))
|
|
||||||
@end lisp
|
|
||||||
|
|
||||||
Once the procedures have been added, we can invoke the hook using
|
|
||||||
@code{run-hook}.
|
|
||||||
|
|
||||||
@lisp
|
|
||||||
(run-hook hook 3 4)
|
|
||||||
@print{} Bar: 12
|
|
||||||
@print{} Foo: 7
|
|
||||||
@end lisp
|
|
||||||
|
|
||||||
Note that the procedures are called in the reverse of the order with
|
|
||||||
which they were added. This is because the default behaviour of
|
|
||||||
@code{add-hook!} is to add its procedure to the @emph{front} of the
|
|
||||||
hook's procedure list. You can force @code{add-hook!} to add its
|
|
||||||
procedure to the @emph{end} of the list instead by providing a third
|
|
||||||
@code{#t} argument on the second call to @code{add-hook!}.
|
|
||||||
|
|
||||||
@lisp
|
|
||||||
(add-hook! hook (lambda (x y)
|
|
||||||
(display "Foo: ")
|
|
||||||
(display (+ x y))
|
|
||||||
(newline)))
|
|
||||||
(add-hook! hook (lambda (x y)
|
|
||||||
(display "Bar: ")
|
|
||||||
(display (* x y))
|
|
||||||
(newline))
|
|
||||||
#t) ; @r{<- Change here!}
|
|
||||||
|
|
||||||
(run-hook hook 3 4)
|
|
||||||
@print{} Foo: 7
|
|
||||||
@print{} Bar: 12
|
|
||||||
@end lisp
|
|
||||||
|
|
||||||
|
|
||||||
@node Hook Reference
|
|
||||||
@subsubsection Hook Reference
|
|
||||||
|
|
||||||
When you create a hook with @code{make-hook}, you must specify the arity
|
|
||||||
of the procedures which can be added to the hook. If the arity is not
|
|
||||||
given explicitly as an argument to @code{make-hook}, it defaults to
|
|
||||||
zero. All procedures of a given hook must have the same arity, and when
|
|
||||||
the procedures are invoked using @code{run-hook}, the number of
|
|
||||||
arguments passed must match the arity specified at hook creation time.
|
|
||||||
|
|
||||||
The order in which procedures are added to a hook matters. If the third
|
|
||||||
parameter to @code{add-hook!} is omitted or is equal to @code{#f}, the
|
|
||||||
procedure is added in front of the procedures which might already be on
|
|
||||||
that hook, otherwise the procedure is added at the end. The procedures
|
|
||||||
are always called from the front to the end of the list when they are
|
|
||||||
invoked via @code{run-hook}.
|
|
||||||
|
|
||||||
The ordering of the list of procedures returned by @code{hook->list}
|
|
||||||
matches the order in which those procedures would be called if the hook
|
|
||||||
was run using @code{run-hook}.
|
|
||||||
|
|
||||||
Note that the C functions in the following entries are for handling
|
|
||||||
@dfn{Scheme-level} hooks in C. There are also @dfn{C-level} hooks which
|
|
||||||
have their own interface (@pxref{C Hooks}).
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} make-hook [n_args]
|
|
||||||
@deffnx {C Function} scm_make_hook (n_args)
|
|
||||||
Create a hook for storing procedure of arity @var{n_args}.
|
|
||||||
@var{n_args} defaults to zero. The returned value is a hook
|
|
||||||
object to be used with the other hook procedures.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} hook? x
|
|
||||||
@deffnx {C Function} scm_hook_p (x)
|
|
||||||
Return @code{#t} if @var{x} is a hook, @code{#f} otherwise.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} hook-empty? hook
|
|
||||||
@deffnx {C Function} scm_hook_empty_p (hook)
|
|
||||||
Return @code{#t} if @var{hook} is an empty hook, @code{#f}
|
|
||||||
otherwise.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} add-hook! hook proc [append_p]
|
|
||||||
@deffnx {C Function} scm_add_hook_x (hook, proc, append_p)
|
|
||||||
Add the procedure @var{proc} to the hook @var{hook}. The
|
|
||||||
procedure is added to the end if @var{append_p} is true,
|
|
||||||
otherwise it is added to the front. The return value of this
|
|
||||||
procedure is not specified.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} remove-hook! hook proc
|
|
||||||
@deffnx {C Function} scm_remove_hook_x (hook, proc)
|
|
||||||
Remove the procedure @var{proc} from the hook @var{hook}. The
|
|
||||||
return value of this procedure is not specified.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} reset-hook! hook
|
|
||||||
@deffnx {C Function} scm_reset_hook_x (hook)
|
|
||||||
Remove all procedures from the hook @var{hook}. The return
|
|
||||||
value of this procedure is not specified.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} hook->list hook
|
|
||||||
@deffnx {C Function} scm_hook_to_list (hook)
|
|
||||||
Convert the procedure list of @var{hook} to a list.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
@deffn {Scheme Procedure} run-hook hook . args
|
|
||||||
@deffnx {C Function} scm_run_hook (hook, args)
|
|
||||||
Apply all procedures from the hook @var{hook} to the arguments
|
|
||||||
@var{args}. The order of the procedure application is first to
|
|
||||||
last. The return value of this procedure is not specified.
|
|
||||||
@end deffn
|
|
||||||
|
|
||||||
If, in C code, you are certain that you have a hook object and well
|
|
||||||
formed argument list for that hook, you can also use
|
|
||||||
@code{scm_c_run_hook}, which is identical to @code{scm_run_hook} but
|
|
||||||
does no type checking.
|
|
||||||
|
|
||||||
@deftypefn {C Function} void scm_c_run_hook (SCM hook, SCM args)
|
|
||||||
The same as @code{scm_run_hook} but without any type checking to confirm
|
|
||||||
that @var{hook} is actually a hook object and that @var{args} is a
|
|
||||||
well-formed list matching the arity of the hook.
|
|
||||||
@end deftypefn
|
|
||||||
|
|
||||||
For C code, @code{SCM_HOOKP} is a faster alternative to
|
|
||||||
@code{scm_hook_p}:
|
|
||||||
|
|
||||||
@deftypefn {C Macro} int SCM_HOOKP (x)
|
|
||||||
Return 1 if @var{x} is a Scheme-level hook, 0 otherwise.
|
|
||||||
@end deftypefn
|
|
||||||
|
|
||||||
|
|
||||||
@subsubsection Handling Scheme-level hooks from C code
|
|
||||||
|
|
||||||
Here is an example of how to handle Scheme-level hooks from C code using
|
|
||||||
the above functions.
|
|
||||||
|
|
||||||
@example
|
|
||||||
if (SCM_NFALSEP (scm_hook_p (obj)))
|
|
||||||
/* handle Scheme-level hook using C functions */
|
|
||||||
scm_reset_hook_x (obj);
|
|
||||||
else
|
|
||||||
/* do something else (obj is not a hook) */
|
|
||||||
@end example
|
|
||||||
|
|
||||||
|
|
||||||
@node C Hooks
|
|
||||||
@subsubsection Hooks For C Code.
|
|
||||||
|
|
||||||
The hooks already described are intended to be populated by Scheme-level
|
|
||||||
procedures. In addition to this, the Guile library provides an
|
|
||||||
independent set of interfaces for the creation and manipulation of hooks
|
|
||||||
that are designed to be populated by functions implemented in C.
|
|
||||||
|
|
||||||
The original motivation here was to provide a kind of hook that could
|
|
||||||
safely be invoked at various points during garbage collection.
|
|
||||||
Scheme-level hooks are unsuitable for this purpose as running them could
|
|
||||||
itself require memory allocation, which would then invoke garbage
|
|
||||||
collection recursively @dots{} However, it is also the case that these
|
|
||||||
hooks are easier to work with than the Scheme-level ones if you only
|
|
||||||
want to register C functions with them. So if that is mainly what your
|
|
||||||
code needs to do, you may prefer to use this interface.
|
|
||||||
|
|
||||||
To create a C hook, you should allocate storage for a structure of type
|
|
||||||
@code{scm_t_c_hook} and then initialize it using @code{scm_c_hook_init}.
|
|
||||||
|
|
||||||
@deftp {C Type} scm_t_c_hook
|
|
||||||
Data type for a C hook. The internals of this type should be treated as
|
|
||||||
opaque.
|
|
||||||
@end deftp
|
|
||||||
|
|
||||||
@deftp {C Enum} scm_t_c_hook_type
|
|
||||||
Enumeration of possible hook types, which are:
|
|
||||||
|
|
||||||
@table @code
|
|
||||||
@item SCM_C_HOOK_NORMAL
|
|
||||||
@vindex SCM_C_HOOK_NORMAL
|
|
||||||
Type of hook for which all the registered functions will always be called.
|
|
||||||
@item SCM_C_HOOK_OR
|
|
||||||
@vindex SCM_C_HOOK_OR
|
|
||||||
Type of hook for which the sequence of registered functions will be
|
|
||||||
called only until one of them returns C true (a non-NULL pointer).
|
|
||||||
@item SCM_C_HOOK_AND
|
|
||||||
@vindex SCM_C_HOOK_AND
|
|
||||||
Type of hook for which the sequence of registered functions will be
|
|
||||||
called only until one of them returns C false (a NULL pointer).
|
|
||||||
@end table
|
|
||||||
@end deftp
|
|
||||||
|
|
||||||
@deftypefn {C Function} void scm_c_hook_init (scm_t_c_hook *hook, void *hook_data, scm_t_c_hook_type type)
|
|
||||||
Initialize the C hook at memory pointed to by @var{hook}. @var{type}
|
|
||||||
should be one of the values of the @code{scm_t_c_hook_type} enumeration,
|
|
||||||
and controls how the hook functions will be called. @var{hook_data} is
|
|
||||||
a closure parameter that will be passed to all registered hook functions
|
|
||||||
when they are called.
|
|
||||||
@end deftypefn
|
|
||||||
|
|
||||||
To add or remove a C function from a C hook, use @code{scm_c_hook_add}
|
|
||||||
or @code{scm_c_hook_remove}. A hook function must expect three
|
|
||||||
@code{void *} parameters which are, respectively:
|
|
||||||
|
|
||||||
@table @var
|
|
||||||
@item hook_data
|
|
||||||
The hook closure data that was specified at the time the hook was
|
|
||||||
initialized by @code{scm_c_hook_init}.
|
|
||||||
|
|
||||||
@item func_data
|
|
||||||
The function closure data that was specified at the time that that
|
|
||||||
function was registered with the hook by @code{scm_c_hook_add}.
|
|
||||||
|
|
||||||
@item data
|
|
||||||
The call closure data specified by the @code{scm_c_hook_run} call that
|
|
||||||
runs the hook.
|
|
||||||
@end table
|
|
||||||
|
|
||||||
@deftp {C Type} scm_t_c_hook_function
|
|
||||||
Function type for a C hook function: takes three @code{void *}
|
|
||||||
parameters and returns a @code{void *} result.
|
|
||||||
@end deftp
|
|
||||||
|
|
||||||
@deftypefn {C Function} void scm_c_hook_add (scm_t_c_hook *hook, scm_t_c_hook_function func, void *func_data, int appendp)
|
|
||||||
Add function @var{func}, with function closure data @var{func_data}, to
|
|
||||||
the C hook @var{hook}. The new function is appended to the hook's list
|
|
||||||
of functions if @var{appendp} is non-zero, otherwise prepended.
|
|
||||||
@end deftypefn
|
|
||||||
|
|
||||||
@deftypefn {C Function} void scm_c_hook_remove (scm_t_c_hook *hook, scm_t_c_hook_function func, void *func_data)
|
|
||||||
Remove function @var{func}, with function closure data @var{func_data},
|
|
||||||
from the C hook @var{hook}. @code{scm_c_hook_remove} checks both
|
|
||||||
@var{func} and @var{func_data} so as to allow for the same @var{func}
|
|
||||||
being registered multiple times with different closure data.
|
|
||||||
@end deftypefn
|
|
||||||
|
|
||||||
Finally, to invoke a C hook, call the @code{scm_c_hook_run} function
|
|
||||||
specifying the hook and the call closure data for this run:
|
|
||||||
|
|
||||||
@deftypefn {C Function} {void *} scm_c_hook_run (scm_t_c_hook *hook, void *data)
|
|
||||||
Run the C hook @var{hook} will call closure data @var{data}. Subject to
|
|
||||||
the variations for hook types @code{SCM_C_HOOK_OR} and
|
|
||||||
@code{SCM_C_HOOK_AND}, @code{scm_c_hook_run} calls @var{hook}'s
|
|
||||||
registered functions in turn, passing them the hook's closure data, each
|
|
||||||
function's closure data, and the call closure data.
|
|
||||||
|
|
||||||
@code{scm_c_hook_run}'s return value is the return value of the last
|
|
||||||
function to be called.
|
|
||||||
@end deftypefn
|
|
||||||
|
|
||||||
|
|
||||||
@node GC Hooks
|
|
||||||
@subsubsection Hooks for Garbage Collection
|
|
||||||
|
|
||||||
Whenever Guile performs a garbage collection, it calls the following
|
|
||||||
hooks in the order shown.
|
|
||||||
|
|
||||||
@defvr {C Hook} scm_before_gc_c_hook
|
|
||||||
C hook called at the very start of a garbage collection, after setting
|
|
||||||
@code{scm_gc_running_p} to 1, but before entering the GC critical
|
|
||||||
section.
|
|
||||||
|
|
||||||
If garbage collection is blocked because @code{scm_block_gc} is
|
|
||||||
non-zero, GC exits early soon after calling this hook, and no further
|
|
||||||
hooks will be called.
|
|
||||||
@end defvr
|
|
||||||
|
|
||||||
@defvr {C Hook} scm_before_mark_c_hook
|
|
||||||
C hook called before beginning the mark phase of garbage collection,
|
|
||||||
after the GC thread has entered a critical section.
|
|
||||||
@end defvr
|
|
||||||
|
|
||||||
@defvr {C Hook} scm_before_sweep_c_hook
|
|
||||||
C hook called before beginning the sweep phase of garbage collection.
|
|
||||||
This is the same as at the end of the mark phase, since nothing else
|
|
||||||
happens between marking and sweeping.
|
|
||||||
@end defvr
|
|
||||||
|
|
||||||
@defvr {C Hook} scm_after_sweep_c_hook
|
|
||||||
C hook called after the end of the sweep phase of garbage collection,
|
|
||||||
but while the GC thread is still inside its critical section.
|
|
||||||
@end defvr
|
|
||||||
|
|
||||||
@defvr {C Hook} scm_after_gc_c_hook
|
|
||||||
C hook called at the very end of a garbage collection, after the GC
|
|
||||||
thread has left its critical section.
|
|
||||||
@end defvr
|
|
||||||
|
|
||||||
@defvr {Scheme Hook} after-gc-hook
|
|
||||||
@vindex scm_after_gc_hook
|
|
||||||
Scheme hook with arity 0. This hook is run asynchronously
|
|
||||||
(@pxref{Asyncs}) soon after the GC has completed and any other events
|
|
||||||
that were deferred during garbage collection have been processed. (Also
|
|
||||||
accessible from C with the name @code{scm_after_gc_hook}.)
|
|
||||||
@end defvr
|
|
||||||
|
|
||||||
All the C hooks listed here have type @code{SCM_C_HOOK_NORMAL}, are
|
|
||||||
initialized with hook closure data NULL, are are invoked by
|
|
||||||
@code{scm_c_hook_run} with call closure data NULL.
|
|
||||||
|
|
||||||
@cindex guardians, testing for GC'd objects
|
|
||||||
The Scheme hook @code{after-gc-hook} is particularly useful in
|
|
||||||
conjunction with guardians (@pxref{Guardians}). Typically, if you are
|
|
||||||
using a guardian, you want to call the guardian after garbage collection
|
|
||||||
to see if any of the objects added to the guardian have been collected.
|
|
||||||
By adding a thunk that performs this call to @code{after-gc-hook}, you
|
|
||||||
can ensure that your guardian is tested after every garbage collection
|
|
||||||
cycle.
|
|
||||||
|
|
||||||
|
|
||||||
@node REPL Hooks
|
|
||||||
@subsubsection Hooks into the Guile REPL
|
|
||||||
|
|
||||||
|
|
||||||
@c Local Variables:
|
|
||||||
@c TeX-master: "guile.texi"
|
|
||||||
@c End:
|
|
Loading…
Add table
Add a link
Reference in a new issue