[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
Re: Two functions for everything?
- To: SANDRA <LOOSEMORE@UTAH-20.ARPA>, DCP@QUABBIN.SCRC.Symbolics.COM
- Subject: Re: Two functions for everything?
- From: David C. Plummer <DCP@QUABBIN.SCRC.Symbolics.COM>
- Date: Mon, 30 Jun 86 13:53 EDT
- Cc: common-lisp@SU-AI.ARPA
- In-reply-to: <12218981356.18.LOOSEMORE@UTAH-20.ARPA>
Date: Mon 30 Jun 86 11:09:16-MDT
From: SANDRA <LOOSEMORE@UTAH-20.ARPA>
Setting the safety switch only seems like half the story to me. You still
have to provide some way to specify what error checking a given function
does for each value of the safety setting. The reason why I would prefer
to see this mechanism part of the language spec, rather than left up to
each implementation, is that there is no real distinction between "system"
code and "user" code in Lisp. Since most CL implementations are, in fact,
written *in* CL, chances are they would have some internal, high-level
hooks to do this anyway. (Although I suppose there is at least one
masochistic implementor out there who actually *likes* writing code in
assembly language! :-)) So why not make these hooks available to the
user?
I'm still not sure I understand you, but I have thought up one scenario.
I was thinking that what safety affected were things that "are an error"
as defined by the language. This includes such things as AREF out of
bounds, calling a function with the wrong number of arguments, etc.
Under this view, if I compile
(defun foo (a) (car a))
with safety 0, the implementation may encode CAR inline. If I compile
it with safety 1, speed fast, it might compile as if I had written
(defun foo (a) (check-type a 'list) (and a (car a)))
If I compile it with safety 3, speed slow, space small, it might compile
as
(defun foo (a) (call-car's-full-checking-entrypoint a))
Some other settings might make it compile as
(defun foo (a) (call-car's-no-checking-entrypoint a))
[Seems silly for CAR, but consider a slightly larger function, such as
AREF.]
Maybe what you are saying is that the user might write
(defun foo (a) (check-type a 'list) (and a (car a)))
but might want to allow the check-type to be avoided if a caller of FOO
was compiled with safety 0. I don't know what to do about this. One
theory, which I think I believe, is that if a programmer puts in
explicit checking, that checking is always run.
The sort of thing I have in mind is like a declarative form that contains
the assertions about the arguments to a particular function, and the
"safety level" at which to test each assertion. It would be up to the
implementation to figure out how to wrap the tests around the body of the
function.
Personally, I think having to write error checking code at all is a
nuisance, and trying to manage multiple levels of error checking sounds
like a real nightmare. Unless we can come up with some abstractions for
dealing with the complexity, I suspect that a lot of implementors are
going to avoid messing with it entirely.
If I write a function that I expect other programmers to call, I usually
put in some error checking. I don't find it a nuisance, I find it a
benefit to make sure (a) they get the error messages they deserve at a
reasonable time, and (b) to make sure assumptions about the data I make
are valid (because they passed the tests).
Isn't this is a tug-of-war between speed and robustness on stock
hardware? Special hardware is usually designed to do both. CLtL tries
to give users the option of speed or robustness. I'm under the
impression this is a compile-time-only choice, and that you would like a
run-time choice as well?