1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
|
;; $Header$
;; $Locker$
;; FFT -- This is an FFT benchmark written by Harry Barrow.
;; It tests a variety of floating point operations, including array references.
(eval-when (compile)
(setq *READ-DEFAULT-FLOAT-FORMAT* 'double-float)
)
(defvar **fft-re**
(make-array 1025. :element-type 'double-float
:initial-element 0.0))
(defvar **fft-im**
(make-array 1025. :element-type 'double-float
:initial-element 0.0))
(declaim (type (vector double-float) **fft-re** **fft-im**))
(defvar s-pi (float pi 0.0))
(declaim (double-float s-pi))
(defun fft (areal aimag)
(declare (type (simple-array double-float (*)) areal aimag))
(prog* ((ar areal)
(ai aimag)
(i 1)
(j 0)
(k 0)
(m 0) ;compute m = log(n)
(n (1- (array-dimension ar 0)))
(nv2 (floor n 2))
(le 0) (le1 0) (ip 0)
(ur 0.0) (ui 0.0) (wr 0.0) (wi 0.0) (tr 0.0) (ti 0.0))
(declare (type fixnum i j k n nv2 m le le1 ip))
(declare (type (simple-array double-float (*)) ar ai))
(declare (double-float ur ui wr wi tr ti))
l1 (cond ((< i n)
(setq m (the fixnum (1+ m))
i (the fixnum (+ i i)))
(go l1)))
(cond ((not (equal n (the fixnum (expt 2 m))))
(princ "error ... array size not a power of two.")
(read)
(return (terpri))))
(setq j 1 ;interchange elements
i 1) ;in bit-reversed order
l3 (cond ((< i j)
(setq tr (aref ar j)
ti (aref ai j))
(setf (aref ar j) (aref ar i))
(setf (aref ai j) (aref ai i))
(setf (aref ar i) tr)
(setf (aref ai i) ti)))
(setq k nv2)
l6 (cond ((< k j)
(setq j (the fixnum (- j k))
k (the fixnum (/ k 2)))
(go l6)))
(setq j (the fixnum (+ j k))
i (the fixnum (1+ i)))
(cond ((< i n)
(go l3)))
(do ((l 1 (the fixnum (1+ (the fixnum l)))))
((> (the fixnum l) m)) ;loop thru stages
(declare (type fixnum l))
(setq le (the fixnum (expt 2 l))
le1 (the (values fixnum fixnum) (floor le 2))
ur 1.0
ui 0.0
wr (cos (/ s-pi (float le1)))
wi (sin (/ s-pi (float le1))))
(do ((j 1 (the fixnum (1+ (the fixnum j)))))
((> (the fixnum j) le1)) ;loop thru butterflies
(declare (type fixnum j))
(do ((i j (+ (the fixnum i) le)))
((> (the fixnum i) n)) ;do a butterfly
(declare (type fixnum i))
(setq ip (the fixnum (+ i le1))
tr (- (* (aref ar ip) ur)
(* (aref ai ip) ui))
ti (+ (* (aref ar ip) ui)
(* (aref ai ip) ur)))
(setf (aref ar ip) (- (aref ar i) tr))
(setf (aref ai ip) (- (aref ai i) ti))
(setf (aref ar i) (+ (aref ar i) tr))
(setf (aref ai i) (+ (aref ai i) ti))))
(setq tr (- (* ur wr) (* ui wi))
ti (+ (* ur wi) (* ui wr))
ur tr
ui ti))
(return t)))
(defun fft-bench ()
(dotimes (i 10)
(fft **fft-re** **fft-im**)))
(defun testfft ()
(print (time (fft-bench))))
;;;
;;; the following are for verifying that the implementation gives the
;;; correct result
;;;
(defun clear-fft ()
(dotimes (i 1025)
(setf (aref **fft-re** i) 0.0
(aref **fft-im** i) 0.0))
(values))
(defun setup-fft-component (theta &optional (phase 0.0))
(let ((f (* 2 pi theta))
(c (cos (* 0.5 pi phase)))
(s (sin (* 0.5 pi phase))))
(dotimes (i 1025)
(let ((x (sin (* f (/ i 1024.0)))))
(incf (aref **fft-re** i) (float (* c x) 0.0))
(incf (aref **fft-im** i) (float (* s x) 0.0)))))
(values))
(defvar fft-delta 0.0001)
(defun print-fft ()
(dotimes (i 1025)
(let ((re (aref **fft-re** i))
(im (aref **fft-im** i)))
(unless (and (< (abs re) fft-delta) (< (abs im) fft-delta))
(format t "~4d ~10f ~10f~%" i re im))))
(values))
|