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 1982 Massachusetts Institute of Technology ;;;
9 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
13 (macsyma-module matcom
)
15 ;; This is the Match Compiler.
17 (declare-top (special $rules $props boundlist reflist topreflist program
))
19 (defmvar $announce_rules_firing nil
)
21 (defmspec $matchdeclare
(form)
22 (let ((meta-prop-p nil
))
23 (proc-$matchdeclare
(cdr form
))))
25 (defun proc-$matchdeclare
(x)
27 (merror (intl:gettext
"matchdeclare: must be an even number of arguments.")))
28 (do ((x x
(cddr x
))) ((null x
))
29 (cond ((symbolp (car x
))
30 (cond ((and (not (symbolp (cadr x
)))
31 (or (numberp (cadr x
))
32 (member (caaadr x
) '(mand mor mnot mcond mprog
) :test
#'eq
)))
33 (improper-arg-err (cadr x
) '$matchdeclare
)))
34 (meta-add2lnc (car x
) '$props
)
35 (meta-mputprop (car x
) (ncons (cadr x
)) 'matchdeclare
))
36 ((not ($listp
(car x
)))
37 (improper-arg-err (car x
) '$matchdeclare
))
38 (t (do ((l (cdar x
) (cdr l
))) ((null l
))
39 (proc-$matchdeclare
(list (car l
) (cadr x
)))))))
42 (defun compileatom (e p
)
45 (return (cond ((null d
)
52 ((member p boundlist
:test
#'eq
)
54 (list (list 'not
(list 'equal e p
))
56 (t (setq boundlist
(cons p boundlist
)) (emit d
))))))
58 (defun emit (x) (setq program
(nconc program
(list x
))))
61 (cond ((or (numberp x
) (member x boundlist
:test
#'eq
)) x
)
62 ((and (symbolp x
) (get x
'operators
)) `(quote ,x
))
63 ;; ((NULL BOUNDLIST) (LIST 'SIMPLIFYA (LIST 'QUOTE X) NIL))
64 (t `(meval (quote ,x
)))))
66 (defun makepreds (l gg
)
68 (t (cons (cond ((atom (car l
))
69 (list 'lambda
(list (setq gg
(gensym)))
70 `(declare (special ,gg
))
72 (t (defmatch1 (car l
) (gensym))))
73 (makepreds (cdr l
) nil
)))))
75 (defun defmatch1 (pt e
)
76 (prog (topreflist program prog-variables
)
77 (setq topreflist
(list e
))
78 (cond ((atom (errset (compilematch e pt
)))
79 (merror (intl:gettext
"defmatch: failed to compile match for pattern ~M") pt
))
81 ;; NOTE TO TRANSLATORS: MEANING OF FOLLOWING TEXT IS UNKNOWN
82 (mtell "defmatch: ~M will be matched uniquely since sub-parts would otherwise be ambiguous.~%" pt
)
85 `(declare (special ,e
))
88 (list (setq prog-variables
(cdr (reverse topreflist
))))
89 `((declare (special ,@ prog-variables
)))
91 (list (list 'return t
))))))))))
93 (defun compileplus (e p
)
94 (prog (reflist f g h flag leftover
)
97 (cond ((null leftover
)
98 (return (emit (list 'cond
99 (list (list 'not
(list 'equal e
0.
))
101 ((null (cdr leftover
)) (return (compilematch e
(car leftover
))))
102 ((setq f
(intersection leftover boundlist
:test
#'equal
))
109 (list '(mminus) (car f
)))))))
110 (setq leftover
(delete (car f
) leftover
:test
#'equal
))
113 ;; Almost nobody knows what this means. Just suppress the noise.
114 ;; (mtell "COMPILEPLUS: ~M partitions '+'
115 ;; expression.~%" (cons '(mplus) leftover))
116 (setq boundlist
(append boundlist
(remove-if-not #'atom leftover
)))
117 (return (emit (list 'cond
120 (list 'quote leftover
)
122 (makepreds leftover nil
))))
123 '(t (matcherr))))))))
124 ((fixedmatchp (car p
))
131 (list '(mminus) (car p
))))))))
133 (cond ((cdr p
) (setq leftover
(cons (car p
) leftover
)) (setq p
(cdr p
)) (go a1
))
134 (leftover (setq leftover
(cons (car p
) leftover
)) (setq p nil
) (go a1
)))
135 (setq boundlist
(cons (car p
) boundlist
))
136 (emit (getdec (car p
) e
))
137 (cond ((null (cdr p
)) (return nil
)) (t (go a
))))
138 ((eq (caaar p
) 'mtimes
)
139 (cond ((and (not (or (numberp (cadar p
))
140 (and (not (atom (cadar p
)))
141 (eq (caar (cadar p
)) 'rat
))))
142 (fixedmatchp (cadar p
)))
144 (emit `(setq ,(genref)
146 (ratcoef ,e
,(memqargs (cadar p
))))))
147 (compiletimes (car reflist
) (cons '(mtimes) (cddar p
)))
148 (emit `(setq ,e
(meval
152 ((mtimes) -
1 ,(car reflist
)
155 (setq flag t
) (rplacd (car p
) (reverse (cdar p
))) (go a1
))
156 (t (setq leftover
(cons (car p
) leftover
)) (go a
))))
157 ((eq (caaar p
) 'mexpt
)
158 (cond ((fixedmatchp (cadar p
))
162 ((fixedmatchp (caddar p
))
166 (t (go functionmatch
)))
169 (list f e
(setq g
(memqargs g
)) ''mplus
)))
177 (cond ((eq f
'findexpon
)
184 (compilematch (car reflist
) h
))
185 ((not (fixedmatchp (caaar p
)))
187 (setq leftover
(cons (car p
) leftover
))
190 (leftover (setq leftover
(cons (car p
) leftover
)) (setq p nil
) (go a1
)))
191 (setq boundlist
(cons (caaar p
) boundlist
))
196 (t (go functionmatch
)))
201 (list 'findfun e
(memqargs (caaar p
)) ''mplus
)))
202 (cond ((eq (caaar p
) 'mplus
)
203 (mtell (intl:gettext
"COMPILEPLUS: warning: '+' within '+' in: ~M~%") (car p
))
204 (compileplus (car reflist
) (car p
)))
205 (t (emit (list 'setq
(genref) (list 'kdr
(cadr reflist
))))
206 (compileeach (car reflist
) (cdar p
))))
211 (list '(mplus) e
(list '(mminus) (car p
)))))))
214 (defun compiletimes (e p
)
215 (prog (reflist f g h leftover
)
218 (cond ((null leftover
)
219 (return (emit (list 'cond
220 (list (list 'not
(list 'equal e
1.
))
222 ((null (cdr leftover
)) (return (compilematch e
(car leftover
))))
223 ((setq f
(intersection leftover boundlist
:test
#'equal
))
228 (list '(mquotient) e
(car f
))))))
229 (setq leftover
(delete (car f
) leftover
:test
#'equal
))
232 ;; Almost nobody knows what this means. Just suppress the noise.
233 ;; (mtell "COMPILETIMES: ~M partitions '*' expression.~%" (cons '(mtimes) leftover))
234 (setq boundlist
(append boundlist
(remove-if-not #'atom leftover
)))
235 (return (emit (list 'cond
238 (list 'quote leftover
)
240 (makepreds leftover nil
))))
241 '(t (matcherr))))))))
242 ((fixedmatchp (car p
))
246 (list 'quote
(list '(mquotient) e
(car p
)))))))
248 (cond ((cdr p
) (setq leftover
(cons (car p
) leftover
)) (setq p
(cdr p
)) (go a1
))
249 (leftover (setq leftover
(cons (car p
) leftover
)) (setq p nil
) (go a1
)))
250 (setq boundlist
(cons (car p
) boundlist
))
251 (emit (getdec (car p
) e
))
252 (cond ((null (cdr p
)) (return nil
)) (t (go a
))))
253 ((eq (caaar p
) 'mexpt
)
254 (cond ((fixedmatchp (cadar p
))
258 ((fixedmatchp (caddar p
))
262 (t (go functionmatch
)))
265 (list f e
(setq g
(memqargs g
)) ''mtimes
)))
266 (cond ((eq f
'findbase
)
268 (list (list 'equal
(car reflist
) 0)
276 (cond ((eq f
'findexpon
)
277 (list '(mexpt) g
(car reflist
)))
281 (compilematch (car reflist
) h
))
282 ((not (fixedmatchp (caaar p
)))
284 (setq leftover
(cons (car p
) leftover
))
287 (leftover (setq leftover
(cons (car p
) leftover
)) (setq p nil
) (go a1
)))
288 (setq boundlist
(cons (caaar p
) boundlist
))
293 (t (go functionmatch
)))
298 (list 'findfun e
(memqargs (caaar p
)) ''mtimes
)))
299 (cond ((eq (caaar p
) 'mtimes
)
300 (mtell (intl:gettext
"COMPILETIMES: warning: '*' within '*' in: ~M~%") (car p
))
301 (compiletimes (car reflist
) (car p
)))
302 (t (emit (list 'setq
(genref) (list 'kdr
(cadr reflist
))))
303 (compileeach (car reflist
) (cdar p
))))
307 (list 'quote
(list '(mquotient) e
(car p
))))))
311 (defmspec $defmatch
(form)
312 (let ((meta-prop-p nil
))
313 (proc-$defmatch
(cdr form
))))
315 (defun proc-$defmatch
(l)
316 (prog (pt pt
* args a boundlist reflist topreflist program name tem
)
318 (setq pt
(copy-tree (setq pt
* (simplify (cadr l
)))))
320 (setq pt
(copy-tree (setq pt
* (meval pt
))))
321 (mtell (intl:gettext
"defmatch: evaluation of atomic pattern yields: ~M~%") pt
)))
323 (cond ((null (allatoms args
)) (mtell (intl:gettext
"defmatch: some pattern variables are not atoms."))
325 (setq boundlist args
)
327 (cond ((atom (errset (compilematch a pt
)))
328 (merror (intl:gettext
"defmatch: failed to compile match for pattern ~M") pt
))
332 `(declare (special ,a
,@ boundlist
))
335 (list (setq tem
(cdr (reverse topreflist
))))
336 `((declare (special ,@ tem
)))
339 (cond (boundlist (cons 'retlist
342 (meta-add2lnc name
'$rules
)
343 (meta-mputprop name
(list '(mlist) pt
* (cons '(mlist) args
)) '$rule
)
346 (defmspec $tellsimp
(form)
347 (let ((meta-prop-p nil
))
348 (proc-$tellsimp
(cdr form
))))
350 (defmfun $clear_rules
()
351 (mapc 'kill1
(cdr $rules
))
352 (loop for v in
'(mexpt mplus mtimes
)
353 do
(setf (mget v
'rulenum
) nil
)))
355 (defun proc-$tellsimp
(l)
356 (prog (pt rhs boundlist reflist topreflist a program name tem
357 oldstuff pgname oname rulenum
)
358 (setq pt
(copy-tree (simplifya (car l
) nil
)))
360 (setq rhs
(copy-tree (simplifya (cadr l
) nil
)))
361 (cond ((alike1 pt rhs
) (merror (intl:gettext
"tellsimp: circular rule attempted.")))
362 ((atom pt
) (merror (intl:gettext
"tellsimp: pattern must not be an atom; found: ~A") (fullstrip1 (getop name
))))
363 ((mget (setq name
(caar pt
)) 'matchdeclare
)
364 (merror (intl:gettext
"tellsimp: main operator of pattern must not be match variable; found: ~A") (fullstrip1 (getop name
))))
365 ((member name
'(mplus mtimes
) :test
#'eq
)
366 (mtell (intl:gettext
"tellsimp: warning: rule will treat '~M' as noncommutative and nonassociative.~%") name
)))
368 (cond ((atom (errset (compileeach a
(cdr pt
))))
369 (merror (intl:gettext
"tellsimp: failed to compile match for pattern ~M") (cdr pt
))))
370 (setq oldstuff
(get name
'operators
))
371 (setq rulenum
(mget name
'rulenum
))
372 (cond ((null rulenum
) (setq rulenum
1.
)))
373 (setq oname
(getop name
))
374 (setq pgname
(implode (append (%to$
(explodec oname
))
376 (mexploden rulenum
))))
377 (meta-mputprop pgname name
'ruleof
)
378 (meta-add2lnc pgname
'$rules
)
379 (meta-mputprop name
(f1+ rulenum
) 'rulenum
)
381 (list 'lambda
'(x a2 a3
)
382 `(declare (special x a2 a3
))
384 (list 'ans a
'rule-hit
)
385 `(declare (special ans
,a
))
393 (t (mapcar #'(lambda (h) (simplifya h a3
))
400 (list (setq tem
(nconc boundlist
401 (cdr (reverse topreflist
)))))
402 `((declare (special ,@ tem
)))
405 (list 'values
(memqargs rhs
) t
))))))
406 (cond ((not (member name
'(mtimes mplus
) :test
#'eq
))
409 '(rule-hit ans
) '((and (not dosimp
) (member 'simp
(cdar x
) :test
#'eq
))x
)
411 (cond (oldstuff (cons oldstuff
413 (t '(eqtest x x
)))))))
417 (list '(and (equal 1. a2
) rule-hit
) 'ans
)
418 '(rule-hit (meval '((mexpt) ans a2
)))
420 (cond (oldstuff (cons oldstuff
422 (t '(eqtest x x
)))))))
426 (list '(and (equal 1. a2
) rule-hit
) 'ans
)
427 '(rule-hit (meval '((mtimes) ans a2
)))
429 (cond (oldstuff (cons oldstuff
431 (t '(eqtest x x
)))))))))))
432 (meta-mputprop pgname
(list '(mequal) pt rhs
) '$rule
)
433 (cond ((null (mget name
'oldrules
))
435 (list (get name
'operators
))
437 (meta-putprop name pgname
'operators
)
438 (return (cons '(mlist)
440 (cons pgname
(mget name
'oldrules
))
443 (defun %to$
(l) (cond ((eq (car l
) '%
) (rplaca l
'$
)) (l)))
446 (defmspec $tellsimpafter
(form)
447 (let ((meta-prop-p nil
))
448 (proc-$tellsimpafter
(cdr form
))))
450 (defun proc-$tellsimpafter
(l)
451 (prog (pt rhs boundlist reflist topreflist a program name oldstuff plustimes pgname oname tem
452 rulenum my
*afterflag
)
453 (setq pt
(copy-tree (simplifya (car l
) nil
)))
455 (setq rhs
(copy-tree (simplifya (cadr l
) nil
)))
456 (cond ((alike1 pt rhs
) (merror (intl:gettext
"tellsimpafter: circular rule attempted.")))
457 ((atom pt
) (merror (intl:gettext
"tellsimpafter: pattern must not be an atom; found: ~A") (fullstrip1 (getop name
))))
458 ((mget (setq name
(caar pt
)) 'matchdeclare
)
459 (merror (intl:gettext
"tellsimpafter: main operator of pattern must not be match variable; found: ~A") (fullstrip1 (getop name
)))))
461 (setq plustimes
(member name
'(mplus mtimes
) :test
#'eq
))
462 (if (atom (if plustimes
(errset (compilematch a pt
))
463 (errset (compileeach a
(cdr pt
)))))
464 (merror (intl:gettext
"tellsimpafter: failed to compile match for pattern ~M") (cdr pt
)))
465 (setq oldstuff
(get name
'operators
))
466 (setq rulenum
(mget name
'rulenum
))
467 (if (null rulenum
) (setq rulenum
1))
468 (setq oname
(getop name
))
469 (setq pgname
(implode (append (%to$
(explodec oname
))
470 '(|r| |u| |l| |e|
) (mexploden rulenum
))))
471 (setq my
*afterflag
(gensym "*AFTERFLAG-"))
472 (proclaim `(special ,my
*afterflag
))
473 (setf (symbol-value my
*afterflag
) nil
)
474 (meta-mputprop pgname name
'ruleof
)
475 (meta-add2lnc pgname
'$rules
)
476 (meta-mputprop name
(f1+ rulenum
) 'rulenum
)
483 (list 'setq
'x
(list oldstuff
'x
'ans
'a3
))
484 (list 'setq
'x
(list 'simpargs1
'x
'ans
'a3
)))
490 (list (cons a
`(,my
*afterflag rule-hit
)))
491 `((declare (special ,a
,my
*afterflag
)))
492 (list `(setq ,my
*afterflag t
))
493 (cond (oldstuff (subst (list 'quote name
)
495 '((cond ((or (atom x
) (not (eq (caar x
) name
)))
499 (cond (plustimes 'x
) (t '(cdr x
)))))
500 (list (list 'multiple-value-setq
504 (list (setq tem
(nconc boundlist
505 (cdr (reverse topreflist
)))))
506 `((declare (special ,@ tem
)))
509 ($announce_rules_firing
510 (list (list 'return
(list 'values
(list 'announce-rule-firing
`',pgname
'x
(memqargs rhs
)) t
))))
512 (list (list 'return
(list 'values
(memqargs rhs
) t
)))))))))
513 (list '(return (if rule-hit ans
(eqtest x x
)))))))))
514 (meta-mputprop pgname
(list '(mequal) pt rhs
) '$rule
)
515 (cond ((null (mget name
'oldrules
))
516 (meta-mputprop name
(list (get name
'operators
)) 'oldrules
)))
517 (meta-putprop name pgname
'operators
)
518 (return (cons '(mlist)
520 (cons pgname
(mget name
'oldrules
))
523 (defun announce-rule-firing (rulename expr simplified-expr
)
524 (let (($display2d nil
) ($stringdisp nil
))
525 ($print
"By" rulename
"," expr
"-->" simplified-expr
))
528 (defmspec $defrule
(form)
529 (let ((meta-prop-p nil
))
530 (proc-$defrule
(cdr form
))))
532 ;;(defvar *match-specials* nil);;Hell lets declare them all special, its safer--wfs
533 (defun proc-$defrule
(l)
534 (prog (pt rhs boundlist reflist topreflist name a program lhs
* rhs
* tem
)
535 (if (not (= (length l
) 3)) (wna-err '$defrule
))
537 (if (or (not (symbolp name
)) (mopp name
) (member name
'($all $%
) :test
#'eq
))
538 (merror (intl:gettext
"defrule: rule name must be a symbol, and not an operator or 'all' or '%'; found: ~M") name
))
539 (setq pt
(copy-tree (setq lhs
* (simplify (cadr l
)))))
540 (setq rhs
(copy-tree (setq rhs
* (simplify (caddr l
)))))
542 (cond ((atom (errset (compilematch a pt
)))
543 (merror (intl:gettext
"defrule: failed to compile match for pattern ~M") pt
))
547 `(declare (special ,a
))
550 (list (setq tem
(nconc boundlist
551 (cdr (reverse topreflist
)))))
552 `((declare (special ,@ tem
)))
555 (list 'values
(memqargs rhs
) t
)))))))
556 (meta-add2lnc name
'$rules
)
557 (meta-mputprop name
(setq l
(list '(mequal) lhs
* rhs
*)) '$rule
)
558 (meta-mputprop name
'$defrule
'$ruletype
)
559 (return (list '(msetq) name
(cons '(marrow) (cdr l
))))))))
561 ; GETDEC constructs an expression of the form ``if <match> then <assign value> else <match failed>''.
563 ; matchdeclare (aa, true);
564 ; :lisp (symbol-plist '$aa) => (MPROPS (NIL MATCHDECLARE (T)))
565 ; tellsimpafter (fa(aa), ga(aa));
566 ; getdec => (MSETQ $AA TR-GENSYM~1)
568 ; matchdeclare (bb, integerp);
569 ; :lisp (symbol-plist '$bb) => (MPROPS (NIL MATCHDECLARE ($INTEGERP)))
570 ; tellsimpafter (fb(bb), gb(bb));
571 ; getdec => (COND ((IS '(($INTEGERP) TR-GENSYM~3)) (MSETQ $BB TR-GENSYM~3)) ((MATCHERR)))
573 ; my_p(x) := integerp(x) and x>100;
574 ; matchdeclare (cc, my_p);
575 ; :lisp (symbol-plist '$cc) => (MPROPS (NIL MATCHDECLARE ($MY_P)))
576 ; tellsimpafter (fc(cc), gc(cc));
577 ; getdec => (COND ((IS '(($MY_P) TR-GENSYM~5)) (MSETQ $CC TR-GENSYM~5)) ((MATCHERR)))
579 ; :lisp (defmfun $my_p2 (y x) (is `((mgeqp) ,x ,y)))
580 ; matchdeclare (dd, my_p2 (200));
581 ; :lisp (symbol-plist '$dd) => (MPROPS (NIL MATCHDECLARE ((($MY_P2) 200))))
582 ; tellsimpafter (fd(dd), gd(dd));
583 ; getdec => (COND ((IS '(($MY_P2) 200 TR-GENSYM~7)) (MSETQ $DD TR-GENSYM~7)) ((MATCHERR)))
585 ; my_p3 (y, x) := is (x > y);
586 ; matchdeclare (ee, my_p3 (300));
587 ; :lisp (symbol-plist '$ee) => (MPROPS (NIL MATCHDECLARE ((($MY_P3) 300))))
588 ; tellsimpafter (fe(ee), ge(ee));
589 ; getdec => (COND ((IS '(($MY_P3) 300 TR-GENSYM~9)) (MSETQ $EE TR-GENSYM~9)) ((MATCHERR)))
591 ; matchdeclare (ff, lambda ([x], x > 400));
592 ; :lisp (symbol-plist '$ff) => (MPROPS (NIL MATCHDECLARE (((LAMBDA) ((MLIST) $X) ((MGREATERP) $X 400)))))
593 ; tellsimpafter (fff(ff), ggg(ff));
594 ; getdec => (COND ((IS (MAPPLY1 '((LAMBDA) ((MLIST) $X) ((MGREATERP) $X 400)) (LIST TR-GENSYM~11) T NIL)) (MSETQ $FF TR-GENSYM~11)) ((MATCHERR)))
596 ; matchdeclare (gg, lambda ([y, x], x > y) (500));
597 ; :lisp (symbol-plist '$gg) => (MPROPS (NIL MATCHDECLARE (((MQAPPLY) ((LAMBDA) ((MLIST) $Y $X) ((MGREATERP) $X $Y)) 500))))
598 ; tellsimpafter (fg(gg), gg(gg));
599 ; getdec => (COND ((IS (MEVAL '((MQAPPLY) ((LAMBDA) ((MLIST) $Y $X) ((MGREATERP) $X $Y)) 500 TR-GENSYM~13))) (MSETQ $GG TR-GENSYM~13)) ((MATCHERR)))
601 ; pattern-variable is the pattern variable (as declared by matchdeclare)
602 ; match-against is the expression to match against
604 ; Return T if $MAYBE returns T, otherwise NIL.
605 ; That makes all non-T values (e.g. $UNKNOWN or noun expressions) act like NIL.
607 (defun definitely-so (e)
608 (eq (mfuncall '$maybe e
) t
))
610 (defun getdec (pattern-variable match-against
)
612 (if (setq p
(mget pattern-variable
'matchdeclare
))
613 ; P is (<foo>) where <foo> is the matchdeclare predicate
614 ; If <foo> is an atom, it is T or the name of a Lisp or Maxima function
615 ; Otherwise, <foo> is ((<op>) <args>)
617 ; If <foo> is $TRUE, T, or $ALL, generated code always assigns gensym value to pattern variable
618 (if (and (atom (car p
)) (member (car p
) '($true t $all
) :test
#'eq
))
619 `(msetq ,pattern-variable
,match-against
)
621 ; Otherwise, we have some work to do.
623 (let ((p-op (car p
)) (p-args) (test-expr))
626 ; P-OP is the name of a function. Try to generate a Lisp function call.
627 (if (and (fboundp p-op
) (not (get p-op
'translated
))) ; WHY THE TEST FOR TRANSLATED PROPERTY ??
628 `(eq t
(,p-op
,@(ncons match-against
)))
629 `(definitely-so '((,p-op
) ,@(ncons match-against
))))
631 ; Otherwise P-OP is something like ((<op>) <args>).
633 (setq p-args
(cdr p-op
))
635 ((eq (caar p-op
) 'lambda
)
636 `(definitely-so (mapply1 ',p-op
(list ,match-against
) t nil
)))
637 ((eq (caar p-op
) 'mqapply
)
638 `(definitely-so (meval ',(append p-op
(ncons match-against
)))))
639 ; Otherwise P-OP must be a function call with the last arg missing.
641 (if (and (consp (car p-op
)) (mget (caar p-op
) 'mmacro
))
642 `(definitely-so (cons ',(car p-op
) ,(append '(list) (mapcar 'memqargs p-args
) (ncons match-against
))))
643 `(definitely-so (cons ',(car p-op
) ',(append (mapcar 'memqargs p-args
) (ncons match-against
))))))))))
646 (,test-expr
(msetq ,pattern-variable
,match-against
))
649 (defun compilematch (e p
)
651 (cond ((fixedmatchp p
)
656 (list 'meval
(list 'quote
659 ((atom p
) (compileatom e p
))
660 ((eq (caar p
) 'mplus
) (compileplus e p
))
661 ((eq (caar p
) 'mtimes
) (compiletimes e p
))
662 (t (compileatom (list 'kaar e
)
667 (compileeach (car reflist
) (cdr p
))))
673 (setq topreflist
(cons a topreflist
))
674 (return (car (setq reflist
(cons a reflist
))))))
675 (defun compileeach (elist plist
)
676 (prog (reflist count
)
678 (setq reflist
(cons elist reflist
))
679 a
(setq count
(f1+ count
))
681 (return (emit (list 'cond
682 (list (list 'nthkdr elist
(f1- count
))
684 (emit (list 'setq
(genref) (list 'kar
(cadr reflist
))))
685 (compilematch (car reflist
) (car plist
))
686 (setq plist
(cdr plist
))
687 (setq reflist
(cons (list 'kdr
(cadr reflist
)) reflist
))
690 (defun fixedmatchp (x)
691 (cond ((numberp x
) t
)
693 (if (or (member x boundlist
:test
#'eq
) (null (mget x
'matchdeclare
))) t
))
694 (t (and (or (member (caar x
) boundlist
:test
#'eq
)
695 (null (mget (caar x
) 'matchdeclare
)))
699 (if (null x
) t
(and (fixedmatchp (car x
)) (fmp1 (cdr x
)))))