-
Notifications
You must be signed in to change notification settings - Fork 38
/
Copy pathfloat.lisp
72 lines (68 loc) · 2.88 KB
/
float.lisp
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
;;; Parsing floats
;;; taken from ARNESI
(in-package #:redis)
(defun radix-values (radix)
(assert (<= 2 radix 35)
(radix)
"RADIX must be between 2 and 35 (inclusive), not ~D." radix)
(make-array radix
:displaced-to "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"
:displaced-index-offset 0
:element-type
#+lispworks 'base-char
#-lispworks 'character))
(defun parse-float (float-string &key (start 0) (end nil) (radix 10)
(junk-allowed t)
(type 'single-float)
(decimal-character #\.))
(let ((radix-array (radix-values radix))
(integer-part 0)
(mantissa 0)
(mantissa-size 1)
(sign 1))
(with-input-from-string (float-stream
(string-upcase (string-trim '(#\Space #\Tab) float-string))
:start start :end end)
(labels ((peek () (peek-char nil float-stream nil nil nil))
(next () (read-char float-stream nil nil nil))
(sign () ;; reads the (optional) sign of the number
(cond
((char= (peek) #\+) (next) (setf sign 1))
((char= (peek) #\-) (next) (setf sign -1)))
(integer-part))
(integer-part ()
(cond
((position (peek) radix-array)
;; the next char is a valid char
(setf integer-part (+ (* integer-part radix)
(position (next) radix-array)))
;; again
(return-from integer-part (integer-part)))
((null (peek))
;; end of string
(done))
((char= decimal-character (peek))
;; the decimal seperator
(next)
(return-from integer-part (mantissa)))
;; junk
(junk-allowed (done))
(t (bad-string))))
(mantissa ()
(cond
((position (peek) radix-array)
(setf mantissa (+ (* mantissa radix)
(position (next) radix-array))
mantissa-size (* mantissa-size radix))
(return-from mantissa
(mantissa)))
((or (null (peek)) junk-allowed)
;; end of string
(done))
(t (bad-string))))
(bad-string ()
(error "Unable to parse ~S." float-string))
(done ()
(return-from parse-float
(coerce (* sign (+ integer-part (/ mantissa mantissa-size))) type))))
(sign)))))