1 ;;; -*- Mode: Lisp; Package: Maxima; Syntax: Common-Lisp; Base: 10 -*- ;;;;
2 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
3 ;;; The data in this file contains enhancments. ;;;;;
5 ;;; Copyright (c) 1984,1987 by William Schelter,University of Texas ;;;;;
6 ;;; All rights reserved ;;;;;
7 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
8 ;;; (c) Copyright 1981 Massachusetts Institute of Technology ;;;
9 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
13 ;;; Run-time support for translated code.
15 ;;; GJC: Experimental macsyma array lisp level support for translated code.
16 ;;; To quickly handle the array reference and setting syntax in macsyma,
18 ;;; In macsyma arrays go by an atomic name. Lists and matrices
19 ;;; may be hacked with the array syntax, which is convient.
21 ;;; additions for handling arrays in value cell on cl --wfs
23 (macsyma-module acall
)
25 (defun interval-error (fun low high
)
26 (merror (intl:gettext
"~@:M: lower bound ~M is greater than upper bound ~M") fun low high
))
28 (defun mfuncall (f &rest l
)
31 ((and (symbolp f
) (or (macro-function f
) (special-operator-p f
)))
36 ;;; ((MQAPPLY ARRAY) X Y) is a strange form, meaning (X)[Y].
38 (defun marrayref (aarray ind1
&rest inds
)
39 (declare (special fixunbound flounbound
))
42 (case (array-element-type aarray
)
44 (apply #'aref aarray ind1 inds
))
46 (merror (intl:gettext
"MARRAYREF: encountered array ~M of unknown type.") aarray
))))
48 (gethash (if inds
(cons ind1 inds
) ind1
) aarray
))
51 (let ((tem (and (boundp aarray
) (symbol-value aarray
))))
52 (simplify (cond ((arrayp tem
)
53 (apply #'aref tem ind1 inds
))
55 (gethash (if inds
(cons ind1 inds
) ind1
) tem
))
57 (apply #'marrayref ind1 inds
))
58 ((mget aarray
'hashar
)
59 (harrfind `((,aarray array
) ,ind1
,@inds
)))
61 `((,tem array
) ,ind1
,@inds
))
63 (error "unknown type of array for use_fast_arrays. ~
64 the value cell should have the array or hash table")))))
65 (let (ap) ; no fast arrays
66 (simplify (cond ((setq ap
(get aarray
'array
))
67 (let ((val (if (null inds
)
69 (apply #'aref
(append (list ap ind1
) inds
)))))
70 ;; Check for KLUDGING array function implementation.
71 (if (case (array-element-type ap
)
72 ((flonum) (= val flounbound
))
73 ((fixnum) (= val fixunbound
))
74 ((t) (eq val munbound
))
75 (t (merror (intl:gettext
"MARRAYREF: encountered array pointer ~S of unknown type.") ap
)))
76 (arrfind `((,aarray array
) ,ind1
,@inds
))
78 ((setq ap
(mget aarray
'array
))
79 (arrfind `((,aarray array
) ,ind1
,@inds
)))
80 ((setq ap
(mget aarray
'hashar
))
81 (harrfind `((,aarray array
) ,ind1
,@inds
)))
83 (apply #'marrayref ind1 inds
))
85 `((,aarray array
) ,ind1
,@inds
)))))))
87 (simplify (if (member (caar aarray
) '(mlist $matrix
) :test
#'eq
)
88 (list-ref aarray
(cons ind1 inds
))
89 `((mqapply array
) ,aarray
,ind1
,@inds
))))
91 (merror (intl:gettext
"MARRAYREF: cannot retrieve an element of ~M") aarray
))))
93 (defmfun $arrayapply
(ar inds
)
95 (merror (intl:gettext
"arrayapply: second argument must be a list; found ~M") inds
))
96 (apply #'marrayref ar
(cdr inds
)))
98 (defmfun $arraysetapply
(ar inds val
)
100 (merror (intl:gettext
"arraysetapply: second argument must be a list; found ~M") inds
))
101 (apply #'marrayset val ar
(cdr inds
)))
103 (defun marrayset (val aarray
&rest all-inds
)
104 (let ((ind1 (first all-inds
))
105 (inds (rest all-inds
)))
108 (case (array-element-type aarray
)
110 (setf (apply #'aref aarray ind1 inds
) val
))
112 (merror (intl:gettext
"MARRAYSET: encountered array ~M of unknown type.") aarray
))))
114 (setf (gethash (if (cdr all-inds
)
120 (cond ((setq ap
(get aarray
'array
))
122 (setf (aref ap ind1
) val
)
123 (setf (apply #'aref ap all-inds
) val
)))
124 ((setq ap
(mget aarray
'array
))
125 ;; the macsyma ARRAY frob is NOT an array pointer, it
126 ;; is a GENSYM with a lisp array property, don't
129 (setf (aref (symbol-array ap
) ind1
) val
)
130 (setf (apply #'aref
(symbol-array ap
) all-inds
) val
)))
131 ((setq ap
(mget aarray
'hashar
))
132 (arrstore `((,aarray
,'array
)
133 ,@(mapcar #'(lambda (u) `((mquote simp
) ,u
)) all-inds
))
135 ((eq aarray
'mqapply
)
136 (apply #'marrayset val ind1 inds
))
138 (arrstore `((,aarray
,'array
)
139 ,@(mapcar #'(lambda (u) `((mquote simp
) ,u
)) all-inds
))
141 (cl:list
(if (member (caar aarray
) '(mlist $matrix
) :test
#'eq
)
142 (list-ref aarray all-inds t val
)
143 (merror (intl:gettext
"MARRAYSET: cannot assign to an element of ~M") aarray
)))
145 (merror (intl:gettext
"MARRAYSET: ~M is not an array.") aarray
))) )
148 ;;; Note that all these have HEADERS on the list. The CAR of a list I
149 ;;; will call element 0. So [1,2][1] => 1
151 (defun list-ref (l indexl
&optional set-flag val
)
153 (merror (intl:gettext
"LIST-REF: argument must be a list; found ~M") l
))
155 (let ((n (car indexl
)))
156 (cond ((and (integerp n
) (plusp n
)
157 (or (eq (caar l
) 'mlist
)
158 (eq (caar l
) '$matrix
)))
159 (let ((ret (do ((j 1 (1+ j
))
161 ((or (null l
) (= j n
))
163 (merror (intl:gettext
"LIST-REF: invalid subscript: ~M") n
))
168 (if set-flag l ret
)))
170 (merror (intl:gettext
"LIST-REF: invalid subscript: ~M") n
)))))
172 (list-ref (list-ref l
`(,(car indexl
))) (cdr indexl
) set-flag val
)
175 (list-ref (list-ref l
`(,(car indexl
))) (cdr indexl
)))))
177 (declare-top (special $dispflag
))
179 (defun display-for-tr (labelsp equationsp
&rest argl
)
180 (declare (special *linelabel
*))
181 (do ((argl argl
(cdr argl
))
184 ((null argl
) (if labelsp
`((mlist) ,@(nreverse lablist
)) '$done
))
185 (let ((ans (car argl
)))
186 (cond ((and equationsp
187 ;; ((MEQUAL) FOO BAR)
188 (not (atom (caddr ans
)))
189 (eq (caar (caddr ans
)) 'mequal
))
190 ;; if the ANS evaluats to something with an "="
191 ;; already then of course he really meant to use
192 ;; DISP, but we might as well do what he means right?
193 (setq ans
(caddr ans
))))
195 (unless (checklabel $linechar
)
197 (makelabel $linechar
)
198 ;; setqs the free variable *LINELABEL*, what a win,
199 ;; how convenient, now I don't need to use LET !
200 (push *linelabel
* lablist
)
202 (setf (symbol-value *linelabel
*) ans
)))
203 (setq tim
(get-internal-run-time))
204 (let ((*display-labels-p
* (not (null lablist
))))
205 (declare (special *display-labels-p
*))
206 (displa `((mlabel) ,(cond (labelsp *linelabel
*)) ,ans
)))
211 (defun insure-array-props (fnname ignore-mode number-of-args
&aux ary
)
212 (declare (ignore ignore-mode
))
213 ;; called during load or eval time by the defining forms
214 ;; for translated array-functions.
215 ;; this duplicates code in JPG;MLISP (however, the code in MLISP
216 ;; is not callable because it is in a big piece of so-called
217 ;; multi-purpose code).
219 ;; This code is incredibly kludgy. For example, what if
220 ;; the function FOO[J] had a lisp array property gotten
221 ;; by ARRAY(FOO,FIXNUM,33), how is *THAT* detected by this code?
222 ;; Well, it is because that will also put an MPROP ARRAY of $FOO,
223 ;; and (ARRAYDIMS '$FOO) works! (Also checks the array property).
224 ;; Isn't that something. Shit, I never knew that ARRAYDIMS worked
225 ;; on symbols. What a crock.
227 (add2lnc fnname $arrays
)
228 (setq ary
(mgetl fnname
'(hashar array
))))
229 (unless (= (if (eq (car ary
) 'hashar
)
230 (aref (symbol-array (cadr ary
)) 2)
231 (length (cdr (arraydims (cadr ary
)))))
233 (merror (intl:gettext
"INSURE-ARRAY-PROPS: array ~:@M already defined with different dimensions.") fnname
)))
236 (mputprop fnname ary
'hashar
)
237 (setf (symbol-array ary
) (make-array 7 :initial-element nil
))
238 (setf (aref (symbol-array ary
) 0) 4)
239 (setf (aref (symbol-array ary
) 1) 0)
240 (setf (aref (symbol-array ary
) 2) number-of-args
))))
242 ;;; An entry point to $APPLY for translated code.
244 (defun mapply-tr (fun list
)
245 (unless ($listp list
)
246 (merror (intl:gettext
"apply: second argument must be a list; found ~M") list
))
247 (mapply1 fun
(cdr list
) '|the first arg to a translated
`apply
'| list
))
249 (defun assign-check (var val
)
250 (let ((a (get var
'assign
)))
251 (if a
(funcall a var val
))))
253 (declare-top (special maplp
))
255 (defun maplist_tr (fun l1
&rest l
)
256 (setq l
(cons l1
(copy-list l
)))
257 (simplify (let ((maplp t
) res
)
258 (setq res
(apply #'map1
(getopr fun
) l
))
259 (cond ((atom res
) (list '(mlist) res
))
260 ((eq (caar res
) 'mlist
) res
)
261 (t (cons '(mlist) (margs res
)))))))
263 ;;; Entry point into DB for translated code. The main point here
264 ;;; is that evaluation of a form takes place first, (using the lisp
265 ;;; evaluator), and then the trueness is checked. It is not correct
266 ;;; to call the function IS because double-evaluation will then
267 ;;; result, which is wrong, not to mention being incompatible with
270 ;;; This code is taken from the COMPAR module, and altered such that calls to
271 ;;; the macsyma evaluator do not take place. It would be a lot
272 ;;; better to simply modify the code in COMPAR! However, mumble...
273 ;;; Anyway, be careful of changes to COMPAR that break this code.
275 (defun is-boole-check (form)
276 (cond ((null form
) nil
)
279 ;; We check for T and NIL quickly, otherwise go for the database.
280 (mevalp_tr form $prederror nil
))))
282 (defun maybe-boole-check (form)
283 (mevalp_tr form nil nil
))
285 (defun mevalp_tr (pat error? meval?
)
286 (let (patevalled ans
)
287 (declare (special patevalled
))
288 (setq ans
(mevalp1_tr pat error? meval?
))
289 (cond ((member ans
'(t nil
) :test
#'eq
) ans
)
291 (pre-err patevalled
))
294 (defun mevalp1_tr (pat error? meval?
)
295 (declare (special patevalled
))
296 (cond ((and (not (atom pat
)) (member (caar pat
) '(mnot mand mor
) :test
#'eq
))
297 (cond ((eq 'mnot
(caar pat
)) (is-mnot_tr (cadr pat
) error? meval?
))
298 ((eq 'mand
(caar pat
)) (is-mand_tr (cdr pat
) error? meval?
))
299 (t (is-mor_tr (cdr pat
) error? meval?
))))
300 ((atom (setq patevalled
(if meval?
(meval pat
) pat
))) patevalled
)
301 ((member (caar patevalled
) '(mnot mand mor
) :test
#'eq
) (mevalp1_tr patevalled
304 (t (mevalp2 patevalled
(caar patevalled
) (cadr patevalled
) (caddr patevalled
)))))
306 (defun is-mnot_tr (pred error? meval?
)
307 (setq pred
(mevalp_tr pred error? meval?
))
308 (cond ((eq t pred
) nil
)
310 (t (pred-reverse pred
))))
312 (defun is-mand_tr (pl error? meval?
)
314 ((null pl
) (cond ((null npl
))
315 ((null (cdr npl
)) (car npl
))
316 (t (cons '(mand) (nreverse npl
)))))
317 (setq dummy
(mevalp_tr (car pl
) error? meval?
)
320 ((null dummy
) (return nil
))
321 (t (setq npl
(cons dummy npl
))))))
323 (defun is-mor_tr (pl error? meval?
)
325 ((null pl
) (cond ((null npl
) nil
)
326 ((null (cdr npl
)) (car npl
))
327 (t (cons '(mor) (nreverse npl
)))))
328 (setq dummy
(mevalp_tr (car pl
) error? meval?
)
330 (cond ((eq t dummy
) (return t
))
332 (t (setq npl
(cons dummy npl
))))))
334 ;; Some functions for even faster calling of arrays.
336 (defun marrayref1$
(aarray index
)
339 (case (array-element-type aarray
)
340 ((flonum) (aref aarray index
))
341 (t (merror (intl:gettext
"MARRAYREF1$: array must be an array of floats; found ~M") aarray
))))
343 (marrayref aarray index
))))
345 (defun marrayset1$
(value aarray index
)
348 (case (array-element-type aarray
)
349 ((flonum) (setf (aref aarray index
) value
))
350 (t (merror (intl:gettext
"MARRAYSET1$: array must be an array of floats; found ~M") aarray
))))
352 (marrayset value aarray index
))))
355 (defun application-operator (form &rest ign
)
356 (declare (ignore ign
))
357 (apply (caar form
) (cdr form
)))
359 ;; more efficient operators calls.
364 (simplify (list '(mminus) x
))))
366 (defun retlist_tr (&rest args
)
367 (do ((j (- (length args
) 2) (- j
2))
368 (l () (cons (list '(mequal simp
) (nth j args
) (nth (1+ j
) args
)) l
)))
369 ((< j
0) (cons '(mlist simp
) l
))))