[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
common lisp- &environment objects
- To: Scott E. Fahlman <Fahlman@CMU-CS-C.ARPA>, David C. Plummer in disguise <DCP@SCRC-QUABBIN.ARPA>
- Subject: common lisp- &environment objects
- From: David C. Plummer in disguise <DCP@SCRC-QUABBIN.ARPA>
- Date: Mon, 10 Jun 85 06:56 EDT
- Cc: common-lisp@SU-AI.ARPA
- In-reply-to: <FAHLMAN.12117904408.BABYL@CMU-CS-C.ARPA>
Date: Sun, 9 Jun 1985 23:17 EDT
From: "Scott E. Fahlman" <Fahlman@CMU-CS-C.ARPA>
One false move here and we have not Lisp but Conniver. How about
requiring "dynamic" extent for environment objects, and no more? Will
that take care of the needs for macro-expansion, steppers, and so on?
True, you could imagine cute ways of using environments that outlive
their invocations, but I REALLY don't want to pay for those cute uses
with inefficiency or severe constraints on the implementor. If anyone
wants to implement Conniver in Common Lisp, I can supply the old Maclisp
code, but let's not drag down Common Lisp itself.
Moon very carefully solved this problem in Symbolics' implementation of
Common Lisp by having an &ENVIRONMENT keyword to defmacro and making
sure all possible expanders of macros took an optional environment
argument, to which they can pass to MACROEXPAND(-1). This isn't cute;
it is functional. It isn't inefficient and it doesn't pose any
constraints on the implementor, unless I misunderstand what "the
implementor" is suposed to mean.
I don't understand what you are saying here. What does this have to do
with the extent of environment objects? What, exactly, are you arguing
for?
I'm arguing that CLtL says much too little about environments. It does
not say how to get your hands on one, but specifies them as optional
arguments to functions, for example EVALHOOK, APPLYHOOK, MACROEXPAND(-1)
(and in our implementation EVAL). Dynamic extent isn't good enough; how
can you do upward funargs? This is far from cute:
(defun adder (val)
#'(lambda (x) (+ x val)))
it is instead at the heart of lexical scoping.
As I said in an immediate follow-up to the above message, you only get a
lot of additional expense if you actually let users do all of the
"obvious" things with an immortal environment object, such as going to
the tags
Tags have dynamic extent. Going with tags outside their dynamic extent
should generate errors.
and diddling the lexical variables.
What good is lexical scoping if you can't diddle lexical variables?
I really don't care if
environment objects have indefinite or dynamic extent, as long as these
expensive kinds of uses are clearly forbidden once you exit the dynamic
extent, but the politics of this group are such that we must be very
careful about introducing features that, if taken to their logical
conclusion, lead us to somewhere we don't want to go; there are always a
lot of people who demand that everything be taken to its logical
conclusion, and damn the expense, so it is sometimes better not to take
the first step.
CLtL'84, on page 39 pretty clearly gives the scope and extent of just
about everything. Maybe you are concerned that I want the users to
diddle with the environment itself? Not at all. The structure of the
environment is peculiar to the implementation. I am, perhaps
incorrectly, assuming that creating a lexical closure captures the
current environment.
I do note that the original query suggested that Symbolics was
stack-allocating environment objects, and other implementations might
want to do similarly tense things in order to make the interpreter
reasonably cons-free. That suggests to me that it would be a good idea
to allow environment objects to have dynamic extent only, unless someone
can explain why this is unworkable or can demonstrate some important
thing that cannot be done if we take this position. I am always
nervous about disappearing objects, such as the disappearing &rest args
in Zetalisp (not legal in Common Lisp), but in the case of environment
arguments we might choose to go with a dangerous but efficient solution.
I'll repeat the classic lexical scoping example:
(defun adder (val)
#'(lambda (x) (+ x val)))
Environments must have indefinite extent (at least the environment of
non-special variables and local functions (FLET, LABELS)). The way this
works is that the environment has a slot which is called ENV-EVACUATION
which is the heap version created as necessary. Therefore, internal
interpreter functions (e.g., the definition of the FUNCTION special
form) explicitly evacuate the environment since it is being stored in
permanent storage.