1;;; -*- Mode: Lisp; Package: Maxima; Syntax: Common-Lisp; Base: 10 -*- ;;;; 2;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; 3;;; The data in this file contains enhancments. ;;;;; 4;;; ;;;;; 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;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; 10 11(in-package :maxima) 12 13(macsyma-module lesfac) 14 15(load-macsyma-macros rzmac ratmac) 16 17(defun getunhack (gen) 18 (or (get gen 'unhacked) (pget gen))) 19 20(defun frpoly? (r) 21 (equal 1 (cdr r))) 22 23(defmacro setcall (fn a b &rest args) 24 `(prog1 25 (first (setq ,a (,fn ,a ,b ,@args))) 26 (setq ,b (third ,a) 27 ,a (second ,a)))) 28 29(defun pquocof (p q) 30 (let ((qq (testdivide p q))) 31 (if qq 32 (values q qq 1) 33 (values 1 p q)))) 34 35(defun polyst (a) 36 (if (pcoefp a) 37 (list a) 38 (cons (cons (car a) (cadr a)) (polyst (caddr a))))) 39 40(defun cdinf (a b both) 41 (cond ((or (pcoefp a) (pcoefp b)) 42 (values 1 a b)) 43 (t 44 (setq a (ncons (copy-tree a)) 45 b (ncons (if both (copy-tree b) b))) 46 (values (cd1 a b both) (car a) (car b))))) 47 48(defun cd1 (a b both) 49 (cond ((or (pcoefp (car a)) (pcoefp (car b))) 1) 50 ((eq (caar a) (caar b)) 51 (ptimes (pexpt (pget (caar a)) ;CHECK FOR ALG. TRUNC. 52 (prog1 (cond (both (+ (cadar a) (cadar b))) (t (cadar a))) 53 (rplaca a (caddar a)) 54 (cond (both (rplaca b (caddar b))) 55 (t (setq b (cddar b)))))) 56 (cd1 a b both))) 57 ((pointergp (caar a) (caar b)) (cd1 (cddar a) b both)) 58 (t (cd1 a (cddar b) both)))) 59 60(defun lmake (p l) 61 (cond ((pcoefp p) 62 (cons p l)) 63 ((get (car p) 'unhacked) 64 (lmake (caddr p) (cons (cons (car p) (cadr p)) l))) 65 (t 66 (setq l (lmake (caddr p) l)) 67 (rplaca l (list (car p) (cadr p) (car l)))))) 68 69(defun lmake2 (p l) 70 (setq l (lmake p l)) 71 (mapc #'(lambda (x) (rplaca x (getunhack (car x)))) (cdr l)) 72 (if (equal (car l) 1) 73 (cdr l) 74 (rplaca l (cons (car l) 1)))) 75 76(defun pmake (l) 77 (cond ((null l) 78 1) 79 ((zerop (cdar l)) 80 (pmake (cdr l))) 81 ((numberp (caar l)) ;CLAUSE SHOULD BE ELIMINATED ASAP 82 (ptimes (cexpt (caar l) (cdar l)) (pmake (cdr l)))) 83 (t 84 (ptimes (list (caar l) (cdar l) 1) (pmake (cdr l)))))) 85 86(defun fpgcdco (p q) 87 (let (($ratfac nil) 88 (gcdl nil)) ;FACTORED PGCDCOFACTS 89 (if (or (pcoefp p) (pcoefp q)) 90 (values-list (pgcdcofacts p q)) 91 (values (ptimeschk (setcall pgcdcofacts p q) 92 (car (setq p (lmake p nil) 93 q (lmake q nil) 94 gcdl (mapcar #'pmake (lgcd1 (cdr p) (cdr q)))))) 95 (ptimeschk (car p) (cadr gcdl)) 96 (ptimeschk (car q) (caddr gcdl)))))) 97 98(defun facmgcd (pl) ;GCD OF POLY LIST FOR EZGCD WITH RATFAC 99 (do ((l (cdr pl) (cdr l)) 100 (ans nil) 101 (gcd (car pl))) 102 ((null l) (cons gcd (nreverse ans))) 103 (multiple-value-bind (g x y) (fpgcdco gcd (car l)) 104 (cond ((equal g 1) 105 (return (cons 1 pl))) 106 ((null ans) 107 (setq ans (list x))) 108 ((not (equal x 1)) 109 (do ((l2 ans (cdr l2))) 110 ((null l2)) 111 (rplaca l2 (ptimes x (car l2)))))) 112 (push y ans) 113 (setq gcd g)))) 114 115;;; NOTE: ITEMS ON VARLIST ARE POS. NORMAL 116;;; INTEGER COEF GCD=1 AND LEADCOEF. IS POS. 117 118(defun lgcd1 (a b) 119 (prog (ptlist g bj c t1 d1 d2 dummy) 120 (setq ptlist (mapcar #'(lambda (ig) (declare (ignore ig)) b) a)) 121 (do ((a a (cdr a)) 122 (ptlist ptlist (cdr ptlist))) 123 ((null a)) 124 (do ((ai (getunhack (caar a))) 125 (b (car ptlist) (cdr b))) 126 ((null b)) 127 (and (zerop (cdar b)) (go nextb)) 128 (setq d1 1 d2 1) 129 (setq bj (getunhack (caar b))) 130 (setq c (cond ((pirredp (caar a)) 131 (if (pirredp (caar b)) 132 1 133 (multiple-value-setq (dummy bj ai) (pquocof bj ai)))) 134 ((pirredp (caar b)) 135 (multiple-value-setq (dummy ai bj) (pquocof ai bj))) 136 (t 137 (setcall pgcdcofacts ai bj)))) 138 (cond ((equal c 1) (go nextb)) 139 ((equal ai 1) (go bloop))) 140 aloop 141 (when (setq t1 (testdivide ai c)) 142 (setq ai t1) 143 (incf d1) 144 (go aloop)) 145 bloop 146 (and (= d1 1) 147 (not (equal bj 1)) 148 (do ((t1 149 (testdivide bj c) 150 (testdivide bj c))) 151 ((null t1)) 152 (setq bj t1 d2 (1+ d2)))) 153 (setq g (cons (cons (makprodg c t) 154 (min (setq d1 (* d1 (cdar a))) 155 (setq d2 (* d2 (cdar b))))) 156 g)) 157 (cond ((> d1 (cdar g)) 158 (rplacd (last a) 159 (ncons (cons (caar g) (- d1 (cdar g))))) 160 (rplacd (last ptlist) (ncons (cdr b))))) 161 (cond ((> d2 (cdar g)) 162 (rplacd (last b) 163 (ncons (cons (caar g) (- d2 (cdar g))))))) 164 (rplaca (car a) (makprodg ai t)) 165 (rplaca (car b) (makprodg bj t)) 166 (and (equal bj 1) (rplacd (car b) 0)) 167 (and (equal ai 1) (rplacd (car a) 0) (return nil)) 168 nextb)) 169 (return (list g a b)))) 170 171(defun makprodg (p sw) 172 (if (pcoefp p) 173 p 174 (car (makprod p sw)))) 175 176(defun dopgcdcofacts (x y) 177 (let (($gcd $gcd) 178 ($ratfac nil)) 179 (unless (member $gcd *gcdl* :test #'eq) 180 (setq $gcd '$ez)) 181 (values-list (pgcdcofacts x y)))) 182 183(defun facrplus (x y) 184 (let ((a (car x)) 185 (b (cdr x)) 186 (c (car y)) 187 (d (cdr y)) 188 dummy) 189 (multiple-value-setq (x a c) (dopgcdcofacts a c)) 190 (multiple-value-setq (y b d) (fpgcdco b d)) 191 (setq a (makprod (pplus (pflatten (ptimeschk a d)) 192 (pflatten (ptimeschk b c))) nil)) 193 (setq b (ptimeschk b d)) 194 (cond ($algebraic 195 (setq y (ptimeschk y b)) 196 (multiple-value-setq (dummy y a) (fpgcdco y a)) ;for unexpected gcd 197 (cons (ptimes x a) y)) 198 (t 199 (multiple-value-setq (c y b) (cdinf y b nil)) 200 (multiple-value-setq (dummy y a) (fpgcdco y a)) 201 (cons (ptimes x a) (ptimeschk y (ptimeschk c b))))))) 202 203(defun mfacpplus (l) 204 (let (($gcd (or $gcd '$ez)) 205 ($ratfac nil) 206 (g nil)) 207 (setq g (oldcontent2 (sort (copy-list l) 'contodr) 0)) 208 (cond ((pzerop g) g) 209 ((do ((a (pflatten (pquotient (car l) g)) 210 (pplus a (pflatten (pquotient (car ll) g)))) 211 (ll (cdr l) (cdr ll))) 212 ((null ll) (ptimes g (makprod a nil)))))))) 213 214(defun facrtimes (x y gcdsw) 215 (if (not gcdsw) 216 (cons (ptimes (car x) (car y)) (ptimeschk (cdr x) (cdr y))) 217 ;; gcdsw = true 218 (multiple-value-bind (g1 g2 g3) (cdinf (car x) (car y) t) 219 (multiple-value-bind (h1 h2 h3) (cdinf (cdr x) (cdr y) t) 220 (multiple-value-bind (x1 x2 x3) (fpgcdco g2 h3) 221 (declare (ignore x1)) 222 (multiple-value-bind (y1 y2 y3) (fpgcdco g3 h2) 223 (declare (ignore y1)) 224 (cons (ptimes g1 (ptimes x2 y2)) 225 (ptimeschk h1 (ptimeschk x3 y3))))))))) 226 227(defun pfacprod (poly) ;FOR RAT3D 228 (if (pcoefp poly) 229 (cfactor poly) 230 (nconc (pfacprod (caddr poly)) (list (pget (car poly)) (cadr poly))))) 231 232(defun fpcontent (poly) 233 (let (($ratfac nil)) ;algebraic uses 234 (setq poly (oldcontent poly)) ;rattimes? 235 (let ((a (lowdeg (cdadr poly)))) ;main var. content 236 (when (plusp a) 237 (setq a (list (caadr poly) a 1)) 238 (setq poly (list (ptimes (car poly) a) 239 (pquotient (cadr poly) a))))) 240 (if (pminusp (cadr poly)) 241 (list (pminus (car poly)) (pminus (cadr poly))) 242 poly))) 243 244;; LOWDEG written to compute the lowest degree of a polynomial. - RZ 245 246(defun lowdeg (p) 247 (do ((l p (cddr l))) 248 ((null (cddr l)) (car l)))) 249 250(defun makprod (poly contswitch) 251 (cond ((pureprod poly) poly) 252 ((null (cdddr poly)) 253 (ptimes (list (car poly) (cadr poly) 1) 254 (makprod (caddr poly) contswitch))) 255 (contswitch 256 (makprod1 poly)) 257 (t (setq poly (fpcontent poly)) 258 (ptimes (makprod (car poly) contswitch) (makprod1 (cadr poly)))))) 259 260(defun makprod1 (poly) 261 (do ((v varlist (cdr v)) 262 (g genvar (cdr g)) 263 (p (pdis poly))) 264 ((null v) (maksymp poly)) 265 (and (alike1 p (car v)) (return (pget (car g)))))) 266 267(defun maksym (p) 268 (let ((g (gensym)) 269 (e (pdis p))) 270 (putprop g e 'disrep) 271 (valput g (1- (valget (car genvar)))) 272 (push g genvar) 273 (push e varlist) 274 (putprop g p 'unhacked) 275 g)) 276 277(defun maksymp (p) 278 (if (atom p) 279 p 280 (pget (maksym p)))) 281 282(defun pflatten (h) 283 (do ((m (listovars h) (cdr m))) 284 ((null m) (return-from pflatten h)) 285 (unless (let ((p (getunhack (car m)))) 286 (or (null p) (eq (car m) (car p)))) 287 (return-from pflatten (let (($ratfac nil)) (pflat1 h)))))) 288 289(defun pflat1 (p) 290 (cond ((pcoefp p) p) 291 ((null (cdddr p)) 292 (ptimes (pexpt (getunhack (car p)) (cadr p)) (pflat1 (caddr p)))) 293 (t (do ((val (getunhack (car p))) 294 (ld (cadr p) (car a)) 295 (a (cdddr p) (cddr a)) 296 (ans (pflat1 (caddr p)))) 297 ((null a) (ptimes ans (pexpt val ld))) 298 (setq ans (pplus (ptimes ans (pexpt val (- ld (car a)))) 299 (pflat1 (cadr a)))))))) 300 301(defun pirredp (x) 302 (and (setq x (get x 'disrep)) 303 (or (atom x) (member 'irreducible (cdar x) :test #'eq)))) 304