;;; -*- Mode: Lisp; Package: Maxima; Syntax: Common-Lisp; Base: 10 -*- ;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; The data in this file contains enhancments. ;;;;; ;;; ;;;;; ;;; Copyright (c) 1984,1987 by William Schelter,University of Texas ;;;;; ;;; All rights reserved ;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (in-package :maxima) ;; ** (c) Copyright 1982 Massachusetts Institute of Technology ** (macsyma-module comm2) ;;;; DIFF2 (declare-top (special $props $dotdistrib)) (defmfun diffint (e x) (let (a) (cond ((null (cdddr e)) (cond ((alike1 x (caddr e)) (cadr e)) ((and (not (atom (caddr e))) (atom x) (not (free (caddr e) x))) (mul2 (cadr e) (sdiff (caddr e) x))) ((or ($constantp (setq a (sdiff (cadr e) x))) (and (atom (caddr e)) (free a (caddr e)))) (mul2 a (caddr e))) (t (simplifya (list '(%integrate) a (caddr e)) t)))) ((alike1 x (caddr e)) (addn (diffint1 (cdr e) x x) t)) (t (addn (cons (if (equal (setq a (sdiff (cadr e) x)) 0) 0 (simplifya (list '(%integrate) a (caddr e) (cadddr e) (car (cddddr e))) t)) (diffint1 (cdr e) x (caddr e))) t))))) (defun diffint1 (e x y) (let ((u (sdiff (cadddr e) x)) (v (sdiff (caddr e) x))) (list (if (pzerop u) 0 (mul2 u (maxima-substitute (cadddr e) y (car e)))) (if (pzerop v) 0 (mul3 v (maxima-substitute (caddr e) y (car e)) -1))))) (defmfun diffsumprod (e x) (cond ((or (not (atom x)) (not (free (cadddr e) x)) (not (free (car (cddddr e)) x))) (diff%deriv (list e x 1))) ((eq (caddr e) x) 0) (t (let ((u (sdiff (cadr e) x))) (setq u (simplifya (list '(%sum) (if (eq (caar e) '%sum) u (div u (cadr e))) (caddr e) (cadddr e) (car (cddddr e))) t)) (if (eq (caar e) '%sum) u (mul2 e u)))))) (defmfun difflaplace (e x) (cond ((or (not (atom x)) (eq (cadddr e) x)) (diff%deriv (list e x 1))) ((eq (caddr e) x) 0) (t ($laplace (sdiff (cadr e) x) (caddr e) (cadddr e))))) (defmfun diff-%at (e x) (cond ((freeof x e) 0) ((not (freeofl x (hand-side (caddr e) 'r))) (diff%deriv (list e x 1))) (t ($at (sdiff (cadr e) x) (caddr e))))) (defmfun diffncexpt (e x) ((lambda (base* pow) (cond ((and (mnump pow) (or (not (eq (ml-typep pow) 'fixnum)) (< pow 0))) ; POW cannot be 0 (diff%deriv (list e x 1))) ((and (atom base*) (eq base* x) (free pow base*)) (mul2* pow (list '(mncexpt) base* (add2 pow -1)))) ((ml-typep pow 'fixnum) ((lambda (deriv ans) (do ((i 0 (1+ i))) ((= i pow)) (setq ans (cons (list '(mnctimes) (list '(mncexpt) base* i) (list '(mnctimes) deriv (list '(mncexpt) base* (- pow 1 i)))) ans))) (addn ans nil)) (sdiff base* x) nil)) ((and (not (depends pow x)) (or (atom pow) (and (atom base*) (free pow base*)))) ((lambda (deriv index) (simplifya (list '(%sum) (list '(mnctimes) (list '(mncexpt) base* index) (list '(mnctimes) deriv (list '(mncexpt) base* (list '(mplus) pow -1 (list '(mtimes) -1 index))))) index 0 (list '(mplus) pow -1)) nil)) (sdiff base* x) (gensumindex))) (t (diff%deriv (list e x 1))))) (cadr e) (caddr e))) (defmfun stotaldiff (e) (cond ((or (mnump e) (constant e)) 0) ((or (atom e) (member 'array (cdar e) :test #'eq)) (let ((w (mget (if (atom e) e (caar e)) 'depends))) (if w (cons '(mplus) (mapcar #'(lambda (x) (list '(mtimes) (chainrule e x) (list '(%del) x))) w)) (list '(%del) e)))) ((specrepp e) (stotaldiff (specdisrep e))) ((eq (caar e) 'mnctimes) (let (($dotdistrib t)) (add2 (ncmuln (cons (stotaldiff (cadr e)) (cddr e)) t) (ncmul2 (cadr e) (stotaldiff (ncmuln (cddr e) t)))))) ((eq (caar e) 'mncexpt) (if (and (ml-typep (caddr e) 'fixnum) (> (caddr e) 0)) (stotaldiff (list '(mnctimes) (cadr e) (ncpower (cadr e) (f1- (caddr e))))) (list '(%derivative) e))) (t (addn (cons 0 (mapcar #'(lambda (x) (mul2 (sdiff e x) (list '(%del simp) x))) (extractvars (margs e)))) t)))) (defun extractvars (e) (cond ((null e) nil) ((atom (car e)) (if (not (maxima-constantp (car e))) (union* (ncons (car e)) (extractvars (cdr e))) (extractvars (cdr e)))) ((member 'array (cdaar e) :test #'eq) (union* (ncons (car e)) (extractvars (cdr e)))) (t (union* (extractvars (cdar e)) (extractvars (cdr e)))))) ;;;; AT ;;dummy-variable-operators is defined in COMM, which uses it inside of SUBST1. (declare-top (special atvars ateqs atp munbound dummy-variable-operators)) (defmfun $atvalue (exp eqs val) (let (dl vl fun) (cond ((notloreq eqs) (improper-arg-err eqs '$atvalue)) ((or (atom exp) (and (eq (caar exp) '%derivative) (atom (cadr exp)))) (improper-arg-err exp '$atvalue))) (cond ((not (eq (caar exp) '%derivative)) (setq fun (caar exp) vl (cdr exp) dl (listof0s vl))) (t (setq fun (caaadr exp) vl (cdadr exp)) (dolist (v vl) (setq dl (nconc dl (ncons (or (getf (cddr exp) v) 0))))))) (if (or (mopp fun) (eq fun 'mqapply)) (improper-arg-err exp '$atvalue)) (atvarschk vl) (do ((vl1 vl (cdr vl1)) (l atvars (cdr l))) ((null vl1)) (if (and (symbolp (car vl1)) (not (mget (car vl1) '$constant))) (setq val (maxima-substitute (car l) (car vl1) val)) (improper-arg-err (cons '(mlist) vl) '$atvalue))) (setq eqs (if (eq (caar eqs) 'mequal) (list eqs) (cdr eqs))) (setq eqs (do ((eqs eqs (cdr eqs)) (l)) ((null eqs) l) (if (not (member (cadar eqs) vl :test #'eq)) (improper-arg-err (car eqs) '$atvalue)) (setq l (nconc l (ncons (cons (cadar eqs) (caddar eqs))))))) (setq vl (do ((vl vl (cdr vl)) (l)) ((null vl) l) (setq l (nconc l (ncons (cdr (or (assoc (car vl) eqs :test #'eq) (cons nil munbound)))))))) (do ((atvalues (mget fun 'atvalues) (cdr atvalues))) ((null atvalues) (mputprop fun (cons (list dl vl val) (mget fun 'atvalues)) 'atvalues)) (when (and (equal (caar atvalues) dl) (equal (cadar atvalues) vl)) (rplaca (cddar atvalues) val) (return nil))) (add2lnc fun $props) val)) (defmfun $at (exp ateqs) (if (notloreq ateqs) (improper-arg-err ateqs '$at)) (atscan (let ((atp t)) ($substitute ateqs exp)))) (defun atscan (exp) (cond ((or (atom exp) (member (caar exp) '(%at mrat) :test #'eq)) exp) ((eq (caar exp) '%derivative) (or (and (not (atom (cadr exp))) (let ((vl (cdadr exp)) dl) (dolist (v vl) (setq dl (nconc dl (ncons (or (getf (cddr exp) v) 0))))) (atfind (caaadr exp) (cdr ($substitute ateqs (cons '(mlist) vl))) dl))) (list '(%at) exp ateqs))) ((member (caar exp) dummy-variable-operators :test #'eq) (list '(%at) exp ateqs)) ((at1 exp)) (t (recur-apply #'atscan exp)))) (defun at1 (exp) (atfind (caar exp) (cdr exp) (listof0s (cdr exp)))) (defun atfind (fun vl dl) (do ((atvalues (mget fun 'atvalues) (cdr atvalues))) ((null atvalues)) (and (equal (caar atvalues) dl) (do ((l (cadar atvalues) (cdr l)) (vl vl (cdr vl))) ((null l) t) (if (and (not (equal (car l) (car vl))) (not (eq (car l) munbound))) (return nil))) (return (prog2 (atvarschk vl) (substitutel vl atvars (caddar atvalues))))))) (defun listof0s (llist) (do ((llist llist (cdr llist)) (l nil (cons 0 l))) ((null llist) l))) (declare-top (special $ratfac genvar varlist $keepfloat *e*)) (defmvar $logconcoeffp nil) (defmvar superlogcon t) (defmvar $superlogcon t) (defmfun $logcontract (e) (lgccheck (logcon e))) ; E is assumed to be simplified. (defun logcon (e) (cond ((atom e) e) ((member (caar e) '(mplus mtimes) :test #'eq) (if (and $superlogcon (not (lgcsimplep e))) (setq e (lgcsort e))) (cond ((mplusp e) (lgcplus e)) ((mtimesp e) (lgctimes e)) (t (logcon e)))) (t (recur-apply #'logcon e)))) (defun lgcplus (e) (do ((x (cdr e) (cdr x)) (log) (notlogs) (y)) ((null x) (cond ((null log) (subst0 (cons '(mplus) (nreverse notlogs)) e)) (t (setq log (sratsimp (muln log t))) (addn (cons (lgcsimp log) notlogs) t)))) (cond ((atom (car x)) (setq notlogs (cons (car x) notlogs))) ((eq (caaar x) '%log) (setq log (cons (logcon (cadar x)) log))) ((eq (caaar x) 'mtimes) (setq y (lgctimes (car x))) (cond ((or (atom y) (not (eq (caar y) '%log))) (setq notlogs (cons y notlogs))) (t (setq log (cons (cadr y) log))))) (t (setq notlogs (cons (logcon (car x)) notlogs)))))) (defun lgctimes (e) (setq e (subst0 (cons '(mtimes) (mapcar 'logcon (cdr e))) e)) (cond ((not (mtimesp e)) e) (t (do ((x (cdr e) (cdr x)) (log) (notlogs) (decints)) ((null x) (cond ((or (null log) (null decints)) e) (t (muln (cons (lgcsimp (power log (muln decints t))) notlogs) t)))) (cond ((and (null log) (not (atom (car x))) (eq (caaar x) '%log) (not (equal (cadar x) -1))) (setq log (cadar x))) ((logconcoeffp (car x)) (setq decints (cons (car x) decints))) (t (setq notlogs (cons (car x) notlogs)))))))) (defun lgcsimp (e) (cond ((atom e) (simpln (list '(%log) e) 1 t)) (t (list '(%log simp) e)))) (defun lgcsimplep (e) (and (eq (caar e) 'mplus) (not (do ((l (cdr e) (cdr l))) ((null l)) (cond ((not (or (atom (car l)) (not (isinop (car l) '%log)) (eq (caaar l) '%log) (and (eq (caaar l) 'mtimes) (null (cdddar l)) (mnump (cadar l)) (not (atom (caddar l))) (eq (caar (caddar l)) '%log)))) (return t))))))) (defun lgcsort (e) (let (genvar varlist ($keepfloat t) vl e1) (newvar e) (setq vl (do ((vl varlist (cdr vl)) (logs) (notlogs) (decints)) ((null vl) (setq logs (sort logs #'great)) (nreconc decints (nconc logs (nreverse notlogs)))) (cond ((and (not (atom (car vl))) (eq (caaar vl) '%log)) (setq logs (cons (car vl) logs))) ((logconcoeffp (car vl)) (setq decints (cons (car vl) decints))) (t (setq notlogs (cons (car vl) notlogs)))))) (setq e1 (ratdisrep (ratrep e vl))) (if (alike1 e e1) e e1))) (defun lgccheck (e) (let (num denom) (cond ((atom e) e) ((and (eq (caar e) '%log) (setq num (member ($num (cadr e)) '(1 -1) :test #'equal)) (not (equal (setq denom ($denom (cadr e))) 1))) (list '(mtimes simp) -1 (list '(%log simp) (if (= (car num) 1) denom (neg denom))))) (t (recur-apply #'lgccheck e))))) (defun logconcoeffp (e) (if $logconcoeffp (let ((*e* e)) (is '(($logconcoeffp) *e*))) (maxima-integerp e))) ;;;; RTCON (declare-top (special $radexpand $domain radpe)) (defmvar $rootsconmode t) (defun $rootscontract (e) ; E is assumed to be simplified ((lambda (radpe $radexpand) (rtcon e)) (and $radexpand (not (eq $radexpand '$all)) (eq $domain '$real)) nil)) (defun rtcon (e) (cond ((atom e) e) ((eq (caar e) 'mtimes) (if (and (not (free e '$%i)) (let ((num ($num e))) (and (not (alike1 e num)) (or (eq num '$%i) (and (not (atom num)) (member '$%i num :test #'eq) (member '$%i (rtcon num) :test #'eq)))))) (setq e (list* (car e) -1 '((mexpt) -1 ((rat simp) -1 2)) (delete '$%i (copy-list (cdr e)) :count 1 :test #'eq)))) (do ((x (cdr e) (cdr x)) (roots) (notroots) (y)) ((null x) (cond ((null roots) (subst0 (cons '(mtimes) (nreverse notroots)) e)) (t (if $rootsconmode (destructuring-let (((min gcd lcm) (rtc-getinfo roots))) (cond ((and (= min gcd) (not (= gcd 1)) (not (= min lcm)) (not (eq $rootsconmode '$all))) (setq roots (rt-separ (list gcd (rtcon (rtc-fixitup (rtc-divide-by-gcd roots gcd) nil)) 1) nil))) ((eq $rootsconmode '$all) (setq roots (rt-separ (simp-roots lcm roots) nil)))))) (rtc-fixitup roots notroots)))) (cond ((atom (car x)) (cond ((eq (car x) '$%i) (setq roots (rt-separ (list 2 -1) roots))) (t (setq notroots (cons (car x) notroots))))) ((and (eq (caaar x) 'mexpt) (ratnump (setq y (caddar x)))) (setq roots (rt-separ (list (caddr y) (list '(mexpt) (rtcon (cadar x)) (cadr y))) roots))) ((and radpe (eq (caaar x) 'mabs)) (setq roots (rt-separ (list 2 `((mexpt) ,(rtcon (cadar x)) 2) 1) roots))) (t (setq notroots (cons (rtcon (car x)) notroots)))))) ((and radpe (eq (caar e) 'mabs)) (power (power (rtcon (cadr e)) 2) '((rat simp) 1 2))) (t (recur-apply #'rtcon e)))) ;; RT-SEPAR separates like roots into their appropriate "buckets", ;; where a bucket looks like: ;; (( ( ) ;; ( )) etc) (defun rt-separ (a roots) (let ((u (assoc (car a) roots :test #'equal))) (cond (u (nconc u (cdr a))) (t (setq roots (cons a roots))))) roots) (defun simp-roots (lcm root-list) (let (root1) (do ((x root-list (cdr x))) ((null x) (push lcm root1)) (push (list '(mexpt) (muln (cdar x) nil) (quotient lcm (caar x))) root1)))) (defun rtc-getinfo (llist) (let ((m (caar llist)) (g (caar llist)) (l (caar llist))) (do ((x (cdr llist) (cdr x))) ((null x) (list m g l)) (setq m (min m (caar x)) g (gcd g (caar x)) l (lcm l (caar x)))))) (defun rtc-fixitup (roots notroots) (mapcar #'(lambda (x) (rplacd x (list (sratsimp (muln (cdr x) (not $rootsconmode)))))) roots) (muln (nconc (mapcar #'(lambda (x) (power* (cadr x) `((rat) 1 ,(car x)))) roots) notroots) (not $rootsconmode))) (defun rtc-divide-by-gcd (llist gcd) (mapcar #'(lambda (x) (rplaca x (quotient (car x) gcd))) llist) llist) (defmfun $nterms (e) (cond ((zerop1 e) 0) ((atom e) 1) ((eq (caar e) 'mtimes) (if (equal -1 (cadr e)) (setq e (cdr e))) (do ((l (cdr e) (cdr l)) (c 1 (* c ($nterms (car l))))) ((null l) c))) ((eq (caar e) 'mplus) (do ((l (cdr e) (cdr l)) (c 0 (+ c ($nterms (car l))))) ((null l) c))) ((and (eq (caar e) 'mexpt) (integerp (caddr e)) (plusp (caddr e))) ($binomial (+ (caddr e) ($nterms (cadr e)) -1) (caddr e))) ((specrepp e) ($nterms (specdisrep e))) (t 1))) ;;;; ATAN2 (declare-top (special $numer $%piargs $logarc $trigsign half%pi fourth%pi)) (defun simpatan2 (e vestigial z) ; atan2(y,x) ~ atan(y/x) (declare (ignore vestigial)) (twoargcheck e) (let (y x signy signx) (setq y (simpcheck (cadr e) z) x (simpcheck (caddr e) z)) (cond ((and (zerop1 y) (zerop1 x)) (merror "atan2(0,0) has been generated.")) ( ;; float contagion (and (or (numberp x) (ratnump x)) ; both numbers (or (numberp y) (ratnump y)) ; ...but not bigfloats (or $numer (floatp x) (floatp y))) ;at least one float (atan2 ($float y) ($float x))) ( ;; bfloat contagion (and (mnump x) (mnump y) (or ($bfloatp x) ($bfloatp y))) ;at least one bfloat (setq x ($bfloat x) y ($bfloat y)) (*fpatan y (list x))) ((and $%piargs (free x '$%i) (free y '$%i) ;; Only use asksign if %piargs is on. (cond ((zerop1 y) (if (atan2negp x) (simplify '$%pi) 0)) ((zerop1 x) (if (atan2negp y) (mul2* -1 half%pi) (simplify half%pi))) ((alike1 y x) ;; Should we check if ($sign x) is $zero here? (if (atan2negp x) (mul2* -3 fourth%pi) (simplify fourth%pi))) ((alike1 y (mul2 -1 x)) (if (atan2negp x) (mul2* 3 fourth%pi) (mul2* -1 fourth%pi))) ;; Why is atan2(1,sqrt(3)) super-special-cased here?!?! ;; It doesn't even handle atan2(1,-sqrt(3)); ;; *Atan* should handle sqrt(3) etc., so all cases will work ((and (equal y 1) (alike1 x '((mexpt simp) 3 ((rat simp) 1 2)))) (mul2* '((rat simp) 1 6) '$%pi))))) ($logarc (logarc '%atan (div y x))) ((and $trigsign (mminusp* y)) (neg (simplifya (list '($atan2) (neg y) x) t))) ; atan2(y,x) = atan(y/x) + pi sign(y) (1-sign(x))/2 ((and (free x '$%i) (eq (setq signx ($sign x)) '$pos)) (simplifya (list '(%atan) (div y x)) t)) ((and (eq signx '$neg) (free y '$%i) (member (setq signy ($sign y)) '($pos $neg) :test #'eq)) (add2 (simplifya (list '(%atan) (div y x)) t) (porm (eq signy '$pos) (simplify '$%pi)))) ((and (eq signx '$zero) (eq signy '$zero)) ;; Unfortunately, we'll rarely get here. For example, ;; assume(equal(x,0)) atan2(x,x) simplifies via the alike1 case above (merror "atan2(0,0) has been generated.")) (t (eqtest (list '($atan2) y x) e))))) (defun atan2negp (e) (eq (asksign-p-or-n e) '$neg)) ;;;; ARITHF (declare-top (special lnorecurse)) (defmfun $fibtophi (e) (cond ((atom e) e) ((eq (caar e) '$fib) (setq e (cond (lnorecurse (cadr e)) (t ($fibtophi (cadr e))))) (let ((phi (meval '$%phi))) (div (add2 (power phi e) (neg (power (add2 1 (neg phi)) e))) (add2 -1 (mul2 2 phi))))) (t (recur-apply #'$fibtophi e)))) (defmspec $numerval (l) (setq l (cdr l)) (do ((l l (cddr l)) (x (ncons '(mlist simp)))) ((null l) x) (cond ((null (cdr l)) (merror "`numerval' takes an even number of args")) ((not (symbolp (car l))) (merror "~M must be atomic - `numerval'" (car l))) ((boundp (car l)) (merror "~M is bound - `numerval'" (car l)))) (mputprop (car l) (cadr l) '$numer) (add2lnc (car l) $props) (nconc x (ncons (car l))))) (declare-top (special powers var depvar)) (defmfun $derivdegree (e depvar var) (let (powers) (derivdeg1 e) (if (null powers) 0 (maximin powers '$max)))) (defun derivdeg1 (e) (cond ((or (atom e) (specrepp e))) ((eq (caar e) '%derivative) (cond ((alike1 (cadr e) depvar) (do ((l (cddr e) (cddr l))) ((null l)) (cond ((alike1 (car l) var) (return (setq powers (cons (cadr l) powers))))))))) (t (mapc 'derivdeg1 (cdr e))))) (declare-top (unspecial powers var depvar)) ;;;; BOX (defmfun $dpart (&rest args) (mpart args nil t nil '$dpart)) (defmfun $lpart (e &rest args) (mpart args nil (list e) nil '$lpart)) (defmfun $box (e &optional (l nil l?)) (if l? (list '(mlabox) e (box-label l)) (list '(mbox) e))) (defmfun box (e label) (if (eq label t) (list '(mbox) e) ($box e (car label)))) (defun box-label (x) (if (atom x) x (implode (cons #\& (mstring x))))) (declare-top (special label)) (defmfun $rembox (e &optional (l nil l?)) (let ((label (if l? (box-label l) '(nil)))) (rembox1 e))) (defun rembox1 (e) (cond ((atom e) e) ((or (and (eq (caar e) 'mbox) (or (equal label '(nil)) (member label '($unlabelled $unlabeled) :test #'eq))) (and (eq (caar e) 'mlabox) (or (equal label '(nil)) (equal label (caddr e))))) (rembox1 (cadr e))) (t (recur-apply #'rembox1 e)))) (declare-top (unspecial label)) ;;;; MAPF (declare-top (special scanmapp)) (defmspec $scanmap (l) (let ((scanmapp t)) (resimplify (apply #'scanmap1 (mmapev l))))) (defmfun scanmap1 (func e &optional (flag nil flag?)) (let ((arg2 (specrepcheck e)) newarg2) (cond ((eq func '$rat) (merror "`scanmap' results must be in general representation.")) (flag? (unless (eq flag '$bottomup) (merror "Only `bottomup' is an acceptable 3rd arg to `scanmap'.")) (if (mapatom arg2) (funcer func (ncons arg2)) (subst0 (funcer func (ncons (mcons-op-args (mop arg2) (mapcar #'(lambda (u) (scanmap1 func u '$bottomup)) (margs arg2))))) arg2))) ((mapatom arg2) (funcer func (ncons arg2))) (t (setq newarg2 (specrepcheck (funcer func (ncons arg2)))) (cond ((mapatom newarg2) newarg2) ((and (alike1 (cadr newarg2) arg2) (null (cddr newarg2))) (subst0 (cons (ncons (caar newarg2)) (ncons (subst0 (mcons-op-args (mop arg2) (mapcar #'(lambda (u) (scanmap1 func u)) (margs arg2))) arg2))) newarg2)) (t (subst0 (mcons-op-args (mop newarg2) (mapcar #'(lambda (u) (scanmap1 func u)) (margs newarg2))) newarg2))))))) (defun subgen (form) ; This function does mapping of subscripts. (do ((ds (if (eq (caar form) 'mqapply) (list (car form) (cadr form)) (ncons (car form))) (outermap1 #'dsfunc1 (simplify (car sub)) ds)) (sub (reverse (or (and (eq 'mqapply (caar form)) (cddr form)) (cdr form))) (cdr sub))) ((null sub) ds))) (defun dsfunc1 (dsn dso) (cond ((or (atom dso) (atom (car dso))) dso) ((member 'array (car dso) :test #'eq) (cond ((eq 'mqapply (caar dso)) (nconc (list (car dso) (cadr dso) dsn) (cddr dso))) (t (nconc (list (car dso) dsn) (cdr dso))))) (t (mapcar #'(lambda (d) (dsfunc1 dsn d)) dso)))) ;;;; GENMAT (defmfun $genmatrix (a i2 &optional (j2 i2) (i1 1) (j1 i1)) (unless (or (symbolp a) (hash-table-p a) (and (not (atom a)) (eq (caar a) 'lambda))) (improper-arg-err a '$genmatrix)) (when (notevery #'fixnump (list i2 j2 i1 j1)) (merror "Invalid arguments to `genmatrix':~%~M" (list '(mlist) a i2 j2 i1 j1))) (let ((header (list a 'array)) (l (ncons '($matrix)))) (cond ((and (or (zerop i2) (zerop j2)) (= i1 1) (= j1 1))) ((or (> i1 i2) (> j1 j2)) (merror "Invalid arguments to `genmatrix':~%~M" (list '(mlist) a i2 j2 i1 j1)))) (dotimes (i (1+ (- i2 i1))) (nconc l (ncons (ncons '(mlist))))) (do ((i i1 (1+ i)) (l (cdr l) (cdr l))) ((> i i2)) (do ((j j1 (1+ j))) ((> j j2)) (nconc (car l) (ncons (meval (list header i j)))))) l)) ; Execute deep copy for copymatrix and copylist. ; Resolves SF bug report [ 1224960 ] sideeffect with copylist. ; An optimization would be to call COPY-TREE only on mutable expressions. (defmfun $copymatrix (x) (unless ($matrixp x) (merror "Argument not a matrix - `copymatrix':~%~M" x)) (copy-tree x)) (defmfun $copylist (x) (unless ($listp x) (merror "Argument not a list - `copylist':~%~M" x)) (copy-tree x)) (defmfun $copy (x) (copy-tree x)) ;;;; ADDROW (defmfun $addrow (m &rest rows) (declare (dynamic-extent rows)) (cond ((not ($matrixp m)) (merror "First argument to `addrow' must be a matrix")) ((null rows) m) (t (dolist (r rows m) (setq m (addrow m r)))))) (defmfun $addcol (m &rest cols) (declare (dynamic-extent cols)) (cond ((not ($matrixp m)) (merror "First argument to `addcol' must be a matrix")) ((null cols) m) (t (let ((m ($transpose m))) (dolist (c cols ($transpose m)) (setq m (addrow m ($transpose c)))))))) (defun addrow (m r) (cond ((not (mxorlistp r)) (merror "Illegal argument to `addrow' or `addcol'")) ((and (cdr m) (or (and (eq (caar r) 'mlist) (not (= (length (cadr m)) (length r)))) (and (eq (caar r) '$matrix) (not (= (length (cadr m)) (length (cadr r)))) (prog2 (setq r ($transpose r)) (not (= (length (cadr m)) (length (cadr r)))))))) (merror "Incompatible structure - `addrow'//`addcol'"))) (append m (if (eq (caar r) '$matrix) (cdr r) (ncons r)))) ;;;; ARRAYF (defmfun $arraymake (ary subs) (cond ((or (not ($listp subs)) (null (cdr subs))) (merror "Wrong type argument to `arraymake':~%~M" subs)) ((eq (ml-typep ary) 'symbol) (cons (cons (getopr ary) '(array)) (cdr subs))) (t (cons '(mqapply array) (cons ary (cdr subs)))))) (defmspec $arrayinfo (ary) (setq ary (cdr ary)) (arrayinfo-aux (car ary) (getvalue (car ary)))) (defun arrayinfo-aux (sym val) (prog (arra ary) (setq arra val) (setq ary sym) (if arra (cond ((hash-table-p arra) (let ((dim1 (gethash 'dim1 arra))) (return (list* '(mlist) '$hash_table (if dim1 1 t) (loop for u being the hash-keys in arra using (hash-value v) unless (eq u 'dim1) collect (if (progn v dim1) ;;ignore v u (cons '(mlist simp) u))))))) ((arrayp arra) (return (let ((dims (array-dimensions arra))) (list '(mlist) '$declared ;; they don't want more info (array-type arra) (length dims) (cons '(mlist) (mapcar #'1- dims))))))) (let ((gen (mgetl sym '(hashar array))) ary1) (cond ((null gen) (merror "Not an array - `arrayinfo':~%~M" ary)) ((mfilep (cadr gen)) (i-$unstore (ncons ary)) (setq gen (mgetl ary '(hashar array))))) (setq ary1 (cadr gen)) (cond ((eq (car gen) 'hashar) (setq ary1 (symbol-array ary1)) (return (append '((mlist simp) $hashed) (cons (aref ary1 2) (do ((i 3 (1+ i)) (l) (n (cadr (arraydims ary1)))) ((= i n) (sort l #'(lambda (x y) (great y x)))) (do ((l1 (aref ary1 i) (cdr l1))) ((null l1)) (push (cons '(mlist simp) (caar l1)) l))))))) (t (setq ary1 (arraydims ary1)) (return (list '(mlist simp) (cond ((safe-get ary 'array) (cdr (assoc (car ary1) '((t . $complete) (fixnum . $integer) (flonum . $float)) :test #'eq))) (t '$declared)) (length (cdr ary1)) (cons '(mlist simp) (mapcar #'1- (cdr ary1))))))))))) ;;;; ALIAS (declare-top (special aliaslist aliascntr greatorder lessorder)) (defmspec $makeatomic (l) (setq l (cdr l)) (do ((l l (cdr l)) (bas) (x)) ((null l) '$done) (if (or (atom (car l)) (not (or (setq x (member (caaar l) '(mexpt mncexpt) :test #'eq)) (member 'array (cdaar l) :test #'eq)))) (improper-arg-err (car l) '$makeatomic)) (if x (setq bas (cadar l) x (and (atom (caddar l)) (caddar l))) (setq bas (caaar l) x (and (atom (cadar l)) (cadar l)))) (unless (atom bas) (improper-arg-err (car l) '$makeatomic)) (setq aliaslist (cons (cons (car l) (implode (nconc (exploden bas) (or (and x (exploden x)) (ncons '| |)) (cons '$ (mexploden (incf aliascntr)))))) aliaslist)))) (defmspec $ordergreat (l) (if greatorder (merror "Reordering is not allowed.")) (makorder (setq greatorder (reverse (cdr l))) '_)) (defmspec $orderless (l) (if lessorder (merror "Reordering is not allowed.")) (makorder (setq lessorder (cdr l)) '|#|)) (defun makorder (l char) (do ((l l (cdr l)) (n 101 (1+ n))) ((null l) '$done) (alias (car l) (implode (nconc (ncons char) (mexploden n) (exploden (stripdollar (car l)))))))) (defmfun $unorder () (let ((l (delete nil (cons '(mlist simp) (nconc (mapcar #'(lambda (x) (remalias (getalias x))) lessorder) (mapcar #'(lambda (x) (remalias (getalias x))) greatorder))) :test #'eq))) (setq lessorder nil greatorder nil) l)) ;;;; CONCAT (defmfun $concat (&rest l) (when (null l) (merror "`concat' needs at least one argument.")) (getalias (implode (cons (cond ((not (atom (car l)))) ((or (numberp (car l)) (char= (char (symbol-name (car l)) 0) #\&)) #\&) (t #\$)) (mapcan #'(lambda (x) (unless (atom x) (merror "Argument to `concat' not an atom: ~M" x)) (string* x)) l)))))