26 KiB
- ECL compiler
- old notes
- si:cmp-env-register-macrolet should be part of cmpenv-api
- extract type propagation pass, see 7e8d0985155
- cmpdata, cmpdata-bk
- pass1 extract 1st attempt:
- be explicit in dispatch symbol packages (i.e cl:progn)
- 'UNWIND-PROTECT tag should be made a keyword
- use package agnostic marks '(CB LB CLB CCB UNWIND-PROTECT CLOSURE)
- declared-inline-p, inline-possible and declared-notinline-p should have one common interface
- cmpdata should be merged with cmpwt (which has only data accessors)
- wt-structure-ref doesn't exist!
- some compiler macros belong to the backend!
- generic function potential optimizations
- ecl has one dispatcher and one cache for all generic functions - many misses
- each generic function needs to have its own cache and dispatcher (for instance if there is one method it may be way faster)
- effective method may be compiled into one function unless one of the methods is a closure (or has eql specializer)
- Lambdas
- cmp-env- interface
- cmppolicy.lsp:cmp-env-policy
- cmppolicy.lsp:cmp-env-add-optimizations
- cmppolicy.lsp:cmp-env-optimization
- cmppolicy.lsp:add-default-optimizations
- cmpenv-api.lsp:cmp-env-root
- cmpenv-api.lsp:cmp-env-copy
- cmpenv-api.lsp:cmp-env-cleanups
- cmpenv-api.lsp:cmp-env-register-var
- cmpenv-api.lsp:cmp-env-declare-special
- cmpenv-api.lsp:cmp-env-add-declaration
- cmpenv-api.lsp:cmp-env-extend-declaration
- cmpenv-api.lsp:cmp-env-register-function
- cmpenv-api.lsp:cmp-env-register-global-macro
- cmpenv-api.lsp:cmp-env-register-macro
- cmpenv-api.lsp:cmp-env-register-ftype
- cmpenv-api.lsp:cmp-env-register-symbol-macro
- cmpenv-api.lsp:cmp-env-register-block
- cmpenv-api.lsp:cmp-env-register-tag
- cmpenv-api.lsp:cmp-env-register-cleanup
- cmpenv-api.lsp:cmp-env-search-function
- cmpenv-api.lsp:cmp-env-search-variables
- cmpenv-api.lsp:cmp-env-search-block
- cmpenv-api.lsp:cmp-env-search-tag
- cmpenv-api.lsp:cmp-env-search-symbol-macro
- cmpenv-api.lsp:cmp-env-search-var
- cmpenv-api.lsp:cmp-env-search-macro
- cmpenv-api.lsp:cmp-env-search-ftype
- cmpenv-api.lsp:cmp-env-mark
- cmpenv-api.lsp:cmp-env-new-variables
- cmpenv-api.lsp:cmp-env-search-declaration
- cmpenv-fun.lsp
- cmpform
- c1form-local-type
- c1form-local-vars
- c1form-sp-change
- c1form-volatile
- c1form-name
- c1form-parents
- c1form-env
- c1form-args
- c1form-side-effects
- c1form-form
- c1form-toplevel-form
- c1form-file
- c1form-file-position
- print-c1form
- make-c1form
- make-c1form*
- c1form-arg
- c1form-volatile*
- c1form-primary-type
- location-primary-type (same as above)
- find-form-in-node-list
- add-form-to-node-list
- delete-form-from-node-list
- traverse-c1form-tree
- c1form-movable-p
- c1form-values-number
- c1form-single-valued-p
- with-c1form-env
- relocate-parents-list
- c1form-replace-with
- delete-c1forms
- c1form-constant-p
- khm
- links
ECL compiler
ECL's comipler source code may be little hard to read. It relies heavily on global variables and the code has grown over many years of fixes and improvements. These notes are meant to serve the purpose of a guide (not a reference manual or a documentation). If you notice that they are not up to date then please submit a patch with corrections.
Abstract syntax tree
Syntax tree nodes are represented as instances of the c1form
structure. Each node has a slot name which is a symbol denoting the
operator and information about the file and position in it.
Operators are dispatched to functions with appropriate tables
associated with a functionality (i.e *p1-dispatch-table* is a
dispatch table for type propagators associated with c1form's).
References
Object references are used for numerous optimizations. They are
represented as instances of ref structure descendants:
- var
- variable reference
- fun
- function reference
- blk
- block reference (block/return)
- tag
- tag reference (tagbody/go)
Each reference has an identifier, number of references, flags for cross-closure-boundary and cross-local-function-boundary references and a list of nodes (c1forms) which refer to the object.
Compilation algorithm (simlified)
When compiling a file (simplified ovierview):
First pass:
- Check if the file exists and that we can perform the compilation
- Estabilish the compilation environment
- Load
cmpinit.lspif present in the same directory - Initialize a data section and construct the AST (
compiler-pass1)
Second pass:
- Compute initialization function name (entry point of the object)
- Propagate types through the AST
- Compile AST to a C source files
.cand.eclh(compiler-pass2) - Dump a data segment in a
.datafile (symbolcompiler_data_text) - Compile artifacts with the C compiler (
compiler-ccandbundle-cc))
compiler-pass1
- Initialize a data section
Data section contains permanent and temporary objects which are later
serialized in a data segment of the complaition artifacts after the
second pass. Objects put in data section are constants, symbols,
load-time-value's etc. The same object may be added few times, then
it is stored as a location. Not object types can be dumped to C file.
- Construct the AST
Each form which is read is passed to t1expr creates a c1form which
are stored in *top-level-forms* which are later used in the second
pass. c1form is created as follows (simplified pseudocode):
(defun t1expr* (form)
(setq form (maybe-expand-symbol-macro form))
(when (atom form)
;; ignore the form
(return))
(destructuring-bind (op args) form
(typecase op
(cons
(t1ordinary form))
((not symbol)
(error "illegal function"))
((eq quote)
;; should we ignore the form(?)
(t1ordinary 'NIL))
(t1-dispatch-entry
(top-level-dispatch form))
(c1-dispatch-entry
(not-top-level-dispatch form))
(expandable-compiler-macro
(add 'macroexpand *current-toplevel-form*)
(t1expr* (expand-macro form)))
(expandable-macro
(add 'macroexpand *current-toplevel-form*)
(t1expr* (expand-macro form)))
(otherwise
(t1ordinary form)))))
Forms are processed recursively with appropriate operator
handlers. Funcations named t1xxx are a top level form handlers while
c1xxx are handlers for the rest. When operator is not special it is
processed according to normal rules i.e with c1call.
Function t1ordinary handles top-level forms which do not have
special semantics associated with them by binding top-levelness flag
to NIL and adding a c1form with a name ordinary and storing result
of (c1expr form) in load-time values. Top level forms may have side
effects (i.e registering a macro in a global compiler environment).
Function c1expr is used to handle forms which are not
top-level. Dispatched operator handler may eliminate dead code,
perform constant folding and propagate constants and rewrite the form
which is processed again. Handler may modify the compiler environment
(i.e register a local function or a local variable) and add new
objects to a data section. Already created c1forms may be updated i.e
to note that there is a cross-closure reference.
compiler-pass2
Second pass is responsible for producing files which are then compiled
by the C compiler. For top level forms we have t2xxx handlers and
for the rest c2xxx handlers. Additionally there are other helper
tables (p1xxx for type propagation and location dispatch tables
set-loc and wt-loc with varying handler names).
(defun pass2 ()
(produce-headers)
(eclh/produce-data-section-declarations)
(with-initialization-code () ; this is put at the end of c file
(include-data-file)
(produce-initialize-cblock)
(produce-setf-function-definitions)
(do-type-propagation *top-level-forms*)
;; compiler-phase "t2" starts now
;;
;; This part is tricky. When we emit top-level form part of it
;; lands in the c-file before the initialization code (C function
;; definitions) and part is put in the initialization code.
(emit-top-level-forms *make-forms*)
(emit-top-level-forms *top-level-forms*))
(eclh/produce-data-segment-declarations)
(eclh/produce-setf-function-definers) ; should be inlined in c file?
(eclh/add-static-constants) ; CHECKME never triggered?
(eclh/declare-c-funciton-table) ; static table with function data
;; compiler-phase "t3" starts now
(eclh/declare-callback-functions) ; calls t3-callback
(data/dump-data-section))
(defun emit-top-level-form (form)
(with-init ()
(emit (t2expr form)))
(do-local-funs (fun)
;; t3local-fun may add new local funs to process.
(emit (t3local-fun fun))))
Example output
Example output in pseudocode follows. I've put some comments to indicate potential issues and improvement opportunities.
- <file-name>.eclh
- static data, declarations and symbol mappings
static cl_object *VV; /* declare data section */
static cl_object Cblock;
#define VM size_of_data_permanent_storage;
#define VMtemp size_of_data_temporary_storage;
/* Declare functions in this file. They are declared static and hold
in Cblock to assure that we can recompile the fasl and load it. */
static cl_object L1ordinary_function(cl_object , cl_object );
static cl_object LC2foobar(cl_object , cl_object );
static cl_object LC3__g0(cl_object , cl_object );
/* In safe code we always go through ecl_fdefinition and then this
macro definition expands to nothing. */
#define ECL_DEFINE_SETF_FUNCTIONS \\
VV[10]=ecl_setf_definition(VV[11],ECL_T); \\
VV[12]=ecl_setf_definition(VV[13],ECL_T);
/* Statically defined constants.
XXX I'm not sure how to trigger constant builders. Needs
investigation if it is not a dead code, and if so whether we should
resurrect it or remove. */
/* exported lisp functions -- installed in Cblock */
#define compiler_cfuns_size 1
static const struct ecl_cfunfixed compiler_cfuns[] = {
/*t,m,narg,padding,name=function-location,block=name-location,entry,entry_fixed,file,file_position*/
{0,0,2,0,ecl_make_fixnum(6),ecl_make_fixnum(0),(cl_objectfn)L1ordinary_function,NULL,ECL_NIL,ecl_make_fixnum(23)},
};
/* callback declarations (functions defined with defcallback). */
#include <ecl/internal.h>
static int ecl_callback_0(int var0,int var1);
- <file-name>.data
- data segment
static const struct ecl_base_string compiler_data_text1[] = {
(int8_t)t_base_string, 0, ecl_aet_bc, 0,
ECL_NIL, (cl_index)1065, (cl_index)1065,
(ecl_base_char*)
"common-lisp-user::make-closure common-lisp-user::ordinary-function common-lisp-u"
"ser::+ordinary_constant+ common-lisp-user::*foobar* common-lisp-user::foobar :de"
"lete-methods clossy-package::bam 0 0 si::dodefpackage clos::install-method clos:"
":associate-methods-to-gfun \"CL-USER\" ((optimize (debug 1))) (defun common-lisp-u"
"ser::make-closure) (#1=#P\"/home/jack/test/foobar.lisp\" . 55) (defun common-lisp-"
"user::ordinary-function) (#1# . 132) (common-lisp-user::a common-lisp-user::b) 4"
"2.32 (defconstant common-lisp-user::+ordinary_constant+) (#1# . 175) (defvar com"
"mon-lisp-user::*foobar*) (#1# . 216) (defun common-lisp-user::foobar) (#1# . 237"
") \"CLOSSY-PACKAGE\" (\"CL\") (\"BAM\" \"GENERIC-FUNCTION\") (defgeneric generic-functio"
"n) (#1# . 451) (clossy-package::a clossy-package::b) (defmethod generic-function"
" (clossy-package::a real) (clossy-package::b real)) (real real) (defmethod gener"
"ic-function (clossy-package::a integer) (clossy-package::b integer)) (integer in"
"teger) (defclass clossy-package::bam) (#1# . 582) ((:initform 42 :initargs (:a) "
":name clossy-package::a))" };
static const cl_object compiler_data_text[] = {
(cl_object)compiler_data_text1,
NULL};
- <file-name>.c
- function definitions and the initialization code
#include <ecl/ecl-cmp.h>
#include "/absolute/path/to/<file-name>.eclh"
/* Normal functions are defined with DEFUN. Local functions may be
lambdas, closures, methods, callbacks etc.
XXX callback function implementations should be inlined to avoid
indirection.
XXX method function names are named like LCn__g0 and on lisp side
they have names like g0 -- gensymed part of the name should be
produced from the generic function name for easier debugging. */
/* normal function definitions */
static cl_object L1fun (cl_object v1a, cl_object v2b) { /*...*/ }
/* local function definitions */
static cl_object LC2__g0 (cl_object v1a) { /* method */ }
static cl_object LC3__g0 (cl_narg narg, ...) { /* closure */ }
static cl_object LC4foobar (cl_object v1a, cl_object v2b) { /* callback */ }
/* callbacks */
static int ecl_callback_0 (int var0, int var1) { /* calls LC2foobar */ }
#include "/absolute/path/to/<file-name>.data"
ECL_DLLEXPORT void init_fas_CODE(cl_object flag) {
/* Function is designed to work in two passes. */
if (flag != OBJNULL) {
/* The loader passes a cblock as flag for us to initialize. */
Cblock = flag->cblock;
flag->cblock.data = VV;
flag->cblock.data_text = compiler_data_text;
/* ... */
return;
}
/* The loader initializes the module (calls READ on data segment
elements and initializes cblock.data with results, then installs
functions and their source information. */
/* 2. Execute top level code. */
VVtemp = Cblock->cblock.temp_data;
ECL_DEFINE_SETF_FUNCTIONS;
/* Note that mere annotation in a simple file requires plenty of
function calls so that impacts FASL load time. We should make
annotations part of the objects themself (instead of keeping a
central registry), then maybe we could keep this data static. */
si_select_package(VVtemp[0]);
(cl_env_copy->function=(ECL_SYM("MAPC",545)->symbol.gfdef))->cfun.entry(2, ECL_SYM("PROCLAIM",668), VVtemp[1]) /* MAPC */;
ecl_function_dispatch(cl_env_copy,ECL_SYM("ANNOTATE",1823))(4, VV[0], ECL_SYM("LOCATION",1829), VVtemp[2], VVtemp[3]) /* ANNOTATE */;
ecl_function_dispatch(cl_env_copy,ECL_SYM("ANNOTATE",1823))(4, VV[0], ECL_SYM("LAMBDA-LIST",1000), ECL_NIL, ECL_NIL) /* ANNOTATE */;
ecl_cmp_defun(VV[7]); /* MAKE-CLOSURE */
/* ... */
si_select_package(VVtemp[14]);
/* XXX defgeneric should be compiled. */
(cl_env_copy->function=(ECL_SYM("ENSURE-GENERIC-FUNCTION",944)->symbol.gfdef))->cfun.entry(5, ECL_SYM("GENERIC-FUNCTION",947), VV[5], ECL_T, ECL_SYM("LAMBDA-LIST",1000), VVtemp[19]) /* ENSURE-GENERIC-FUNCTION */;
clos_load_defclass(VV[6], ECL_NIL, VVtemp[26], ECL_NIL);
/* ... */
}
Generic functions are not compiled.
Representation types
Compilation target machine is described in terms of types supported by
the target compiler. +representation-types+ cover primitives types
which are representable in C (:byte, :fixnum, :float-sse-pack, :bool,
:pointer-void etc.). Each type has a corresponding Lisp type, C type
and ways to convert between Lisp and C types (a separate column shows
how to perform an unsafe convertion on unboxed values). List is
ordered from the most specific to the least specific.
To describe a concreete machine two variables are used:
+all-machines-c-types+ containing common types for all C compilers
(without integers) and +this-machine-c-types+ adding integers and
types which vary between C compilers (i.e long long int). Optionally
each type has information about number of bits used (for bit
fiddling), that information should be kept separate (imo). Variable
*default-machine* use constructed from these both
tables. Alternative machine representations may be created for cross
compilation.
Each representation type is represented by an instance of a structure
rep-type. That information is used when the C code is generated to
manipulate data of certain type.
Environments
Compilation environment
The Global environment
Dynamic environments
Lexical environments
Debug Lexical Environment
Environment objects
http://www.lispworks.com/documentation/HyperSpec/Body/03_aa.htm
Loading FASL files
Cross compilation
old notes
si:cmp-env-register-macrolet should be part of cmpenv-api
extract type propagation pass, see 7e8d0985155
cmpdata, cmpdata-bk
Frontend
vv structure is a single data section entry. We have two data stores – permanent and temporary.
- vv-location
- index in data-store (a number)
- vv-permanent-p
- store flag (t -> permanent, nil -> temporary)
- vv-used-p
- flag indicating if entry is referenced, if not it gets optimized away (same goes for load-objects). To keep indexing and data size intact we put 0 in place of optimized objects.
- vv-value
- holds the entry actual value
- load-objects
- collection of all objects which are created by a lisp form – we don't include them in datasection. We need to keep track of them to filter them out.
- data-size
- size of data stores combined
- data-init
- initalizes data stores. If filename parameter is present, then it objects are read from the file. Otherwise store is empty.
- data-dump-array
- dumps data stores
Backend
- add-static-constant
- called from data frontend.
- data-c-dump
- called from cmpmain, creates data section in a separate C file
- wt-vv
- used by cmploc, accesses element in data section
- set-vv
- used in cmploc, modifies element in data section
pass1 extract 1st attempt:
(defpackage ecl-cmp/int
(:use #:ffi #:ext #:cl)
(:export
;; BACKEND REFERENCES IN FRONTEND!!!
#:lisp-type->rep-type #:c1make-var #:check-vref #:lisp-type-p
#:rep-type->lisp-type #:expand-type-assertion #:machine-c-type-p
;; opts (SHOULDN'T BE)
#:constant-value-p
;; things which should be local to the module
#:*compile-toplevel*
#:*top-level-forms* ; referenced in cmp1top, bound in cmptop (and not used?)
#:*load-time-values* ; referenced in cmp1top, bound in cmpmain (and not used?)
#:clos-compiler-macro-expand ; used only in pass1
#:*optimizable-constants* ; used only in pass1 and cmpopt-constant
#:*setjmps* ; local to pass1
#:*use-c-global* ; local to pass1
#:*clines-string-list* ; shared by ffi of both passses (and 1ct)
#:c1body ; should be brought back to cmpenv-declaim!
#:*next-cfun* ; used only in cmp1lam, but part of cmpenv
#:lisp-to-c-name ; used in cmpvar, cmp1lam
;; common utilities
#:make-dispatch-table #:check-args-number #:print-current-form
;; cmputil (conditions)
#:cmpck #:cmpassert #:cmperr #:cmpdebug #:cmpnote
;; types (arith and propagation)
#:object-type #:trivial-type-p #:values-type-and #:values-type-primary-type
#:type-and #:type-or #:values-type-or #:valid-type-specifier
#:propagate-types
;; locations
#:add-object #:add-symbol #:loc-in-c1form-movable-p
#:*make-forms*
;; internal representation
#:call-global #:ordinary #:var #:fmla-and #:fmla-or #:fmla-not
#:locals #:stack-push-values #:with-stack #:call-local
;;
#:make-c1form* #:*current-toplevel-form*
#:c1form-p #:c1form-type
#:c1form-primary-type
#:c1form-name
#:c1form-constant-p
#:c1form-arg
#:c1form-args
#:c1form-replace-with
#:c1form-side-effects
#:c1form-volatile
#:delete-c1forms
#:and-form-type ; not sure if it belogns here
#:local-function-ref ; XXX: defined in env
#:*current-function*
#:make-fun
#:fun-name
#:fun-parent
#:fun-lambda-expression
#:fun-cmp-env
#:fun-global
#:fun-cfun
#:fun-exported
#:fun-closure
#:fun-minarg
#:fun-maxarg
#:fun-description
#:fun-no-entry
#:fun-referenced-funs
#:fun-child-funs
#:fun-lambda
#:fun-var
#:fun-ref
#:fun-referenced-vars
#:fun-referencing-funs
#:add-to-fun-referenced-vars
#:add-to-fun-referenced-funs
#:update-fun-closure-type
#:get-arg-types
#:make-var
#:make-global-variable
#:var-type
#:var-ignorable
#:var-p
#:var-ref
#:var-ref-ccb
#:var-ref-clb
#:var-kind
#:var-name
#:var-loc
#:var-set-nodes
#:var-read-nodes
#:var-functions-reading
#:var-functions-setting
#:var-read-forms
#:var-changed-in-form-list
#:update-variable-type ; ref only in 1let
#:global-var-p
#:add-to-set-nodes
#:add-to-set-nodes-of-var-list
#:add-to-read-nodes
#:add-to-read-nodes-of-var-list
#:delete-from-set-nodes
#:delete-from-read-nodes
#:make-blk
#:blk-ref-ccb
#:blk-ref-clb
#:blk-ref
#:blk-type
#:make-tag
#:tag-name
#:tag-p
#:tag-var
#:tag-ref
;; environment
#:*global-funs* ; in cmpglobals
#:*cmp-env* #:cmp-env-root #:cmp-env-copy #:cmp-env-mark
#:cmp-env-search-macro
#:cmp-env-search-block
#:cmp-env-register-function
#:cmp-env-register-global-macro
#:cmp-env-register-symbol-macro
#:cmp-env-search-symbol-macro
#:cmp-env-register-block
#:cmp-env-search-var
#:cmp-env-declare-special
#:cmp-env-new-variables
#:cmp-env-register-tag
#:cmp-env-search-tag
#:get-return-type
#:inline-possible ; queries for notinline decl
#:declared-inline-p
#:function-may-change-sp
#:function-may-have-side-effects
#:special-variable-p
#:push-vars
#:add-one-declaration
#:check-arguments-type
#:variable-type-in-env
#:alien-declaration-p
#:get-local-return-type
#:get-local-arg-types
#:policy-check-arguments-type #:policy-type-assertions #:policy-evaluate-forms
#:policy-declaration-name-p #:policy-debug-ihs-frame
;; first pass interface
#:t1expr #:c1expr #:c1args* #:cmp-eval))
(defpackage ecl-cmp/pass1
(:use #:ffi #:ext #:cl #:c #:ecl-cmp/int))
(defpackage "C"
(:nicknames "COMPILER")
(:use "FFI" "EXT" "CL" #:ecl-cmp/int)
(:shadow #:disassemble
#:compile
#:compile-file
#:compile-file-pathname
;;; These functions will be common in our frontend
;; #:proclaim #:declaim #:with-compilation-unit
)
(:export "*COMPILER-BREAK-ENABLE*"
"*COMPILE-PRINT*"
"*COMPILE-TO-LINKING-CALL*"
"*COMPILE-VERBOSE*"
"*COMPILER-FEATURES*"
"*CC*"
"*CC-OPTIMIZE*"
"*USER-CC-FLAGS*"
"*USER-LD-FLAGS*"
"*SUPPRESS-COMPILER-MESSAGES*"
"BUILD-ECL"
"BUILD-PROGRAM"
"BUILD-FASL"
"BUILD-STATIC-LIBRARY"
"BUILD-SHARED-LIBRARY"
"COMPILER-WARNING"
"COMPILER-NOTE"
"COMPILER-MESSAGE"
"COMPILER-ERROR"
"COMPILER-FATAL-ERROR"
"COMPILER-INTERNAL-ERROR"
"COMPILER-UNDEFINED-VARIABLE"
"COMPILER-MESSAGE-FILE"
"COMPILER-MESSAGE-FILE-POSITION"
"COMPILER-MESSAGE-FORM"
"*SUPPRESS-COMPILER-MESSAGES*"
"INSTALL-C-COMPILER"
"UPDATE-COMPILER-FEATURES")
(:import-from "SI" "GET-SYSPROP" "PUT-SYSPROP" "REM-SYSPROP" "MACRO"
"*COMPILER-CONSTANTS*" "COMPILER-LET"))
TODO be explicit in dispatch symbol packages (i.e cl:progn)
TODO 'UNWIND-PROTECT tag should be made a keyword
TODO use package agnostic marks '(CB LB CLB CCB UNWIND-PROTECT CLOSURE)
TODO declared-inline-p, inline-possible and declared-notinline-p should have one common interface
cmpdata should be merged with cmpwt (which has only data accessors)
TODO wt-structure-ref doesn't exist!
This is a removal from CLOS merge probably, fixme!
TODO some compiler macros belong to the backend!
generic function potential optimizations
ecl has one dispatcher and one cache for all generic functions - many misses
each generic function needs to have its own cache and dispatcher (for instance if there is one method it may be way faster)
effective method may be compiled into one function unless one of the methods is a closure (or has eql specializer)
Lambdas
;;; lambda expression
;;; During Pass1, a lambda-list
;;;
;;; ( { var }*
;;; [ &optional { var | ( var [ initform [ svar ] ] ) }* ]
;;; [ &rest var ]
;;; [ &key { var | ( { var | ( kwd var ) } [initform [ svar ]])}*
;;; [&allow-other-keys]]
;;; [ &aux {var | (var [initform])}*]
;;; )
;;;
;;; is transformed into
;;;
;;; ( ( { var }* ) ; required
;;; ( { var initform svar }* ) ; optional
;;; { var | nil } ; rest
;;; allow-other-keys-flag
;;; ( { kwd-vv-index var initform svar }* ) ; key
;;; )
;;;
;;; where
;;; svar: NIL ; means svar is not supplied
;;; | var
;;;
;;; &aux parameters will be embedded into LET*.
;;;
;;; c1lambda-expr receives
;;; ( lambda-list { doc | decl }* . body )
;;; and returns
;;; ( lambda info-object lambda-list' doc body' )
;;;
;;; Doc is NIL if no doc string is supplied.
;;; Body' is body possibly surrounded by a LET* (if &aux parameters are
;;; supplied) and an implicit block.
cmp-env- interface
cmppolicy.lsp:cmp-env-policy local
cmppolicy.lsp:cmp-env-add-optimizations internal
cmppolicy.lsp:cmp-env-optimization external
cmppolicy.lsp:add-default-optimizations internal
cmpenv-api.lsp:cmp-env-root external
cmpenv-api.lsp:cmp-env-copy external
cmpenv-api.lsp:cmp-env-cleanups unused
cmpenv-api.lsp:cmp-env-register-var used
cmpenv-api.lsp:cmp-env-declare-special used
cmpenv-api.lsp:cmp-env-add-declaration internal
cmpenv-api.lsp:cmp-env-extend-declaration internal
cmpenv-api.lsp:cmp-env-register-function used
cmpenv-api.lsp:cmp-env-register-global-macro used
cmpenv-api.lsp:cmp-env-register-macro used
cmpenv-api.lsp:cmp-env-register-ftype internal
cmpenv-api.lsp:cmp-env-register-symbol-macro external
cmpenv-api.lsp:cmp-env-register-block used
cmpenv-api.lsp:cmp-env-register-tag used
cmpenv-api.lsp:cmp-env-register-cleanup unused
cmpenv-api.lsp:cmp-env-search-function external
cmpenv-api.lsp:cmp-env-search-variables local
cmpenv-api.lsp:cmp-env-search-block used
cmpenv-api.lsp:cmp-env-search-tag used
cmpenv-api.lsp:cmp-env-search-symbol-macro external
cmpenv-api.lsp:cmp-env-search-var external
cmpenv-api.lsp:cmp-env-search-macro used
cmpenv-api.lsp:cmp-env-search-ftype internal
cmpenv-api.lsp:cmp-env-mark external
cmpenv-api.lsp:cmp-env-new-variables used
cmpenv-api.lsp:cmp-env-search-declaration internal
cmpenv-fun.lsp
proclaim-function external
add-function-declaration internal
get-arg-types external
get-return-type external
get-local-arg-types used
get-local-return-type used
get-proclaimed-narg external
declare-inline internal
declare-notinline internal
proclaim-inline internal
proclaim-notinline internal
declared-inline-p external
declared-notinline-p local
inline-possible external
maybe-install-inline-function hook
cmpform
c1form-local-type info unused
c1form-local-vars info
c1form-sp-change info
c1form-volatile info
c1form-name
c1form-parents local
c1form-env
c1form-args
c1form-side-effects
c1form-form
c1form-toplevel-form
c1form-file
c1form-file-position
print-c1form
make-c1form
make-c1form*
c1form-arg
c1form-volatile* backend
c1form-primary-type
location-primary-type (same as above)
find-form-in-node-list
add-form-to-node-list
delete-form-from-node-list
used only in cmpvar