1 ;;; calc-fin.el --- financial functions for Calc
3 ;; Copyright (C) 1990, 1991, 1992, 1993, 2001 Free Software Foundation, Inc.
5 ;; Author: David Gillespie <daveg@synaptics.com>
6 ;; Maintainers: D. Goel <deego@gnufans.org>
7 ;; Colin Walters <walters@debian.org>
9 ;; This file is part of GNU Emacs.
11 ;; GNU Emacs is distributed in the hope that it will be useful,
12 ;; but WITHOUT ANY WARRANTY. No author or distributor
13 ;; accepts responsibility to anyone for the consequences of using it
14 ;; or for whether it serves any particular purpose or works at all,
15 ;; unless he says so in writing. Refer to the GNU Emacs General Public
16 ;; License for full details.
18 ;; Everyone is granted permission to copy, modify and redistribute
19 ;; GNU Emacs, but only under the conditions described in the
20 ;; GNU Emacs General Public License. A copy of this license is
21 ;; supposed to have been given to you along with GNU Emacs so you
22 ;; can know your rights and responsibilities. It should be in a
23 ;; file named COPYING. Among other things, the copyright notice
24 ;; and this notice must be preserved on all copies.
30 ;; This file is autoloaded from calc-ext.el.
35 (defun calc-Need-calc-fin () nil
)
38 ;;; Financial functions.
43 (if (calc-is-hyperbolic)
44 (calc-enter-result 3 "pvl" (cons 'calcFunc-pvl
(calc-top-list-n 3)))
46 (calc-enter-result 3 "pvb" (cons 'calcFunc-pvb
(calc-top-list-n 3)))
47 (calc-enter-result 3 "pv" (cons 'calcFunc-pv
(calc-top-list-n 3)))))))
49 (defun calc-fin-npv (arg)
53 (calc-vector-op "npvb" 'calcFunc-npvb
(1+ arg
))
54 (calc-vector-op "npv" 'calcFunc-npv
(1+ arg
)))))
59 (if (calc-is-hyperbolic)
60 (calc-enter-result 3 "fvl" (cons 'calcFunc-fvl
(calc-top-list-n 3)))
62 (calc-enter-result 3 "fvb" (cons 'calcFunc-fvb
(calc-top-list-n 3)))
63 (calc-enter-result 3 "fv" (cons 'calcFunc-fv
(calc-top-list-n 3)))))))
65 (defun calc-fin-pmt ()
68 (if (calc-is-hyperbolic)
69 (calc-enter-result 3 "fvl" (cons 'calcFunc-fvl
(calc-top-list-n 3)))
71 (calc-enter-result 3 "pmtb" (cons 'calcFunc-pmtb
(calc-top-list-n 3)))
72 (calc-enter-result 3 "pmt" (cons 'calcFunc-pmt
(calc-top-list-n 3)))))))
74 (defun calc-fin-nper ()
77 (if (calc-is-hyperbolic)
78 (calc-enter-result 3 "nprl" (cons 'calcFunc-nperl
(calc-top-list-n 3)))
80 (calc-enter-result 3 "nprb" (cons 'calcFunc-nperb
82 (calc-enter-result 3 "nper" (cons 'calcFunc-nper
83 (calc-top-list-n 3)))))))
85 (defun calc-fin-rate ()
88 (calc-pop-push-record 3
89 (if (calc-is-hyperbolic) "ratl"
90 (if (calc-is-inverse) "ratb" "rate"))
93 (cons (if (calc-is-hyperbolic) 'calcFunc-ratel
94 (if (calc-is-hyperbolic) 'calcFunc-rateb
96 (calc-top-list-n 3)))))))
98 (defun calc-fin-irr (arg)
101 (if (calc-is-inverse)
102 (calc-vector-op "irrb" 'calcFunc-irrb arg
)
103 (calc-vector-op "irr" 'calcFunc-irr arg
))))
105 (defun calc-fin-sln ()
108 (calc-enter-result 3 "sln" (cons 'calcFunc-sln
(calc-top-list-n 3)))))
110 (defun calc-fin-syd ()
113 (calc-enter-result 4 "syd" (cons 'calcFunc-syd
(calc-top-list-n 4)))))
115 (defun calc-fin-ddb ()
118 (calc-enter-result 4 "ddb" (cons 'calcFunc-ddb
(calc-top-list-n 4)))))
121 (defun calc-to-percentage (x)
122 (cond ((Math-objectp x
)
123 (setq x
(math-mul x
100))
124 (if (Math-num-integerp x
)
125 (setq x
(math-trunc x
)))
126 (list 'calcFunc-percent x
))
128 (cons 'vec
(mapcar 'calc-to-percentage
(cdr x
))))
131 (defun calc-convert-percent ()
134 (calc-pop-push-record 1 "c%" (calc-to-percentage (calc-top-n 1)))))
136 (defun calc-percent-change ()
139 (let ((res (calc-normalize (cons 'calcFunc-relch
(calc-top-list 2)))))
140 (calc-pop-push-record 2 "%ch" (calc-to-percentage res
)))))
143 ;;; Financial functions.
145 (defun calcFunc-pv (rate num amount
&optional lump
)
146 (math-check-financial rate num
)
147 (math-with-extra-prec 2
148 (let ((p (math-pow (math-add 1 rate
) num
)))
149 (math-add (math-mul amount
150 (math-div (math-sub 1 (math-div 1 p
))
152 (math-div (or lump
0) p
)))))
153 (put 'calcFunc-pv
'math-expandable t
)
155 (defun calcFunc-pvl (rate num amount
)
156 (calcFunc-pv rate num
0 amount
))
157 (put 'calcFunc-pvl
'math-expandable t
)
159 (defun calcFunc-pvb (rate num amount
&optional lump
)
160 (math-check-financial rate num
)
161 (math-with-extra-prec 2
162 (let* ((p (math-pow (math-add 1 rate
) num
)))
163 (math-add (math-mul amount
164 (math-div (math-mul (math-sub 1 (math-div 1 p
))
167 (math-div (or lump
0) p
)))))
168 (put 'calcFunc-pvb
'math-expandable t
)
170 (defun calcFunc-npv (rate &rest flows
)
171 (math-check-financial rate
1)
172 (math-with-extra-prec 2
173 (let* ((flat (math-flatten-many-vecs flows
))
174 (pp (math-add 1 rate
))
177 (while (setq flat
(cdr flat
))
178 (setq accum
(math-add accum
(math-div (car flat
) p
))
181 (put 'calcFunc-npv
'math-expandable t
)
183 (defun calcFunc-npvb (rate &rest flows
)
184 (math-check-financial rate
1)
185 (math-with-extra-prec 2
186 (let* ((flat (math-flatten-many-vecs flows
))
187 (pp (math-add 1 rate
))
190 (while (setq flat
(cdr flat
))
191 (setq accum
(math-add accum
(math-div (car flat
) p
))
194 (put 'calcFunc-npvb
'math-expandable t
)
196 (defun calcFunc-fv (rate num amount
&optional initial
)
197 (math-check-financial rate num
)
198 (math-with-extra-prec 2
199 (let ((p (math-pow (math-add 1 rate
) num
)))
200 (math-add (math-mul amount
201 (math-div (math-sub p
1)
203 (math-mul (or initial
0) p
)))))
204 (put 'calcFunc-fv
'math-expandable t
)
206 (defun calcFunc-fvl (rate num amount
)
207 (calcFunc-fv rate num
0 amount
))
208 (put 'calcFunc-fvl
'math-expandable t
)
210 (defun calcFunc-fvb (rate num amount
&optional initial
)
211 (math-check-financial rate num
)
212 (math-with-extra-prec 2
213 (let ((p (math-pow (math-add 1 rate
) num
)))
214 (math-add (math-mul amount
215 (math-div (math-mul (math-sub p
1)
218 (math-mul (or initial
0) p
)))))
219 (put 'calcFunc-fvb
'math-expandable t
)
221 (defun calcFunc-pmt (rate num amount
&optional lump
)
222 (math-check-financial rate num
)
223 (math-with-extra-prec 2
224 (let ((p (math-pow (math-add 1 rate
) num
)))
225 (math-div (math-mul (math-sub amount
226 (math-div (or lump
0) p
))
228 (math-sub 1 (math-div 1 p
))))))
229 (put 'calcFunc-pmt
'math-expandable t
)
231 (defun calcFunc-pmtb (rate num amount
&optional lump
)
232 (math-check-financial rate num
)
233 (math-with-extra-prec 2
234 (let ((p (math-pow (math-add 1 rate
) num
)))
235 (math-div (math-mul (math-sub amount
(math-div (or lump
0) p
)) rate
)
236 (math-mul (math-sub 1 (math-div 1 p
))
237 (math-add 1 rate
))))))
238 (put 'calcFunc-pmtb
'math-expandable t
)
240 (defun calcFunc-nper (rate pmt amount
&optional lump
)
241 (math-compute-nper rate pmt amount lump nil
))
242 (put 'calcFunc-nper
'math-expandable t
)
244 (defun calcFunc-nperb (rate pmt amount
&optional lump
)
245 (math-compute-nper rate pmt amount lump
'b
))
246 (put 'calcFunc-nperb
'math-expandable t
)
248 (defun calcFunc-nperl (rate pmt amount
)
249 (math-compute-nper rate pmt amount nil
'l
))
250 (put 'calcFunc-nperl
'math-expandable t
)
252 (defun math-compute-nper (rate pmt amount lump bflag
)
253 (and lump
(math-zerop lump
)
255 (and lump
(math-zerop pmt
)
259 (or (math-objectp rate
) (and math-expand-formulas
(null lump
))
260 (math-reject-arg rate
'numberp
))
261 (and (math-zerop rate
)
262 (math-reject-arg rate
'nonzerop
))
263 (or (math-objectp pmt
) (and math-expand-formulas
(null lump
))
264 (math-reject-arg pmt
'numberp
))
265 (or (math-objectp amount
) (and math-expand-formulas
(null lump
))
266 (math-reject-arg amount
'numberp
))
269 (or (math-objectp lump
)
270 (math-reject-arg lump
'numberp
))
271 (let ((root (math-find-root (list 'calcFunc-eq
276 '(var DUMMY var-DUMMY
)
280 '(var DUMMY var-DUMMY
)
283 (if (math-vectorp root
)
286 (math-with-extra-prec 2
287 (let ((temp (if (eq bflag
'l
)
288 (math-div amount pmt
)
289 (math-sub 1 (math-div (math-mul amount rate
)
291 (math-mul pmt
(math-add 1 rate
))
293 (if (or (math-posp temp
) math-expand-formulas
)
294 (math-neg (calcFunc-log temp
(math-add 1 rate
)))
295 (math-reject-arg pmt
"*Payment too small to cover interest rate"))))))
297 (defun calcFunc-rate (num pmt amount
&optional lump
)
298 (math-compute-rate num pmt amount lump
'calcFunc-pv
))
300 (defun calcFunc-rateb (num pmt amount
&optional lump
)
301 (math-compute-rate num pmt amount lump
'calcFunc-pvb
))
303 (defun math-compute-rate (num pmt amount lump func
)
304 (or (math-objectp num
)
305 (math-reject-arg num
'numberp
))
306 (or (math-objectp pmt
)
307 (math-reject-arg pmt
'numberp
))
308 (or (math-objectp amount
)
309 (math-reject-arg amount
'numberp
))
312 (math-reject-arg lump
'numberp
))
313 (let ((root (math-find-root (list 'calcFunc-eq
315 '(var DUMMY var-DUMMY
)
320 '(var DUMMY var-DUMMY
)
321 '(intv 3 (float 1 -
4) 1)
323 (if (math-vectorp root
)
327 (defun calcFunc-ratel (num pmt amount
)
328 (or (math-objectp num
) math-expand-formulas
329 (math-reject-arg num
'numberp
))
330 (or (math-objectp pmt
) math-expand-formulas
331 (math-reject-arg pmt
'numberp
))
332 (or (math-objectp amount
) math-expand-formulas
333 (math-reject-arg amount
'numberp
))
334 (math-with-extra-prec 2
335 (math-sub (math-pow (math-div pmt amount
) (math-div 1 num
)) 1)))
337 (defun calcFunc-irr (&rest vecs
)
338 (math-compute-irr vecs
'calcFunc-npv
))
340 (defun calcFunc-irrb (&rest vecs
)
341 (math-compute-irr vecs
'calcFunc-npvb
))
343 (defun math-compute-irr (vecs func
)
344 (let* ((flat (math-flatten-many-vecs vecs
))
345 (root (math-find-root (list func
346 '(var DUMMY var-DUMMY
)
348 '(var DUMMY var-DUMMY
)
349 '(intv 3 (float 1 -
4) 1)
351 (if (math-vectorp root
)
355 (defun math-check-financial (rate num
)
356 (or (math-objectp rate
) math-expand-formulas
357 (math-reject-arg rate
'numberp
))
358 (and (math-zerop rate
)
359 (math-reject-arg rate
'nonzerop
))
360 (or (math-objectp num
) math-expand-formulas
361 (math-reject-arg num
'numberp
)))
364 (defun calcFunc-sln (cost salvage life
&optional period
)
365 (or (math-realp cost
) math-expand-formulas
366 (math-reject-arg cost
'realp
))
367 (or (math-realp salvage
) math-expand-formulas
368 (math-reject-arg salvage
'realp
))
369 (or (math-realp life
) math-expand-formulas
370 (math-reject-arg life
'realp
))
371 (if (math-zerop life
) (math-reject-arg life
'nonzerop
))
373 (if (math-num-integerp period
)
374 (or (Math-lessp life period
) (not (math-posp period
)))
375 (math-reject-arg period
'integerp
)))
377 (math-div (math-sub cost salvage
) life
)))
378 (put 'calcFunc-sln
'math-expandable t
)
380 (defun calcFunc-syd (cost salvage life period
)
381 (or (math-realp cost
) math-expand-formulas
382 (math-reject-arg cost
'realp
))
383 (or (math-realp salvage
) math-expand-formulas
384 (math-reject-arg salvage
'realp
))
385 (or (math-realp life
) math-expand-formulas
386 (math-reject-arg life
'realp
))
387 (if (math-zerop life
) (math-reject-arg life
'nonzerop
))
388 (or (math-realp period
) math-expand-formulas
389 (math-reject-arg period
'realp
))
390 (if (or (Math-lessp life period
) (not (math-posp period
)))
392 (math-div (math-mul (math-sub cost salvage
)
393 (math-add (math-sub life period
) 1))
394 (math-div (math-mul life
(math-add life
1)) 2))))
395 (put 'calcFunc-syd
'math-expandable t
)
397 (defun calcFunc-ddb (cost salvage life period
)
398 (if (math-messy-integerp period
) (setq period
(math-trunc period
)))
399 (or (integerp period
) (math-reject-arg period
'fixnump
))
400 (or (math-realp cost
) (math-reject-arg cost
'realp
))
401 (or (math-realp salvage
) (math-reject-arg salvage
'realp
))
402 (or (math-realp life
) (math-reject-arg life
'realp
))
403 (if (math-zerop life
) (math-reject-arg life
'nonzerop
))
404 (if (or (Math-lessp life period
) (<= period
0))
408 (while (>= (setq period
(1- period
)) 0)
409 (setq res
(math-div (math-mul book
2) life
)
410 book
(math-sub book res
))
411 (if (Math-lessp book salvage
)
412 (setq res
(math-add res
(math-sub book salvage
))
416 ;;; arch-tag: 82f30ca8-d02f-4b33-84b4-bb6ecd84597b
417 ;;; calc-fin.el ends here