-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathcl-wiringpi2.lisp
195 lines (164 loc) · 7.19 KB
/
cl-wiringpi2.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
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
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
;;;; cl-wiringpi2.lisp
(in-package #:cl-wiringpi2)
(defvar *setup-mode* nil "Mode in which Wiring Pi was initialized.")
(defun initializedp ()
"Check if the WiringPi library was initialized."
*setup-mode*)
(defun in-sys-mode ()
"Check if initialization was done in :SYS mode."
(eql *setup-mode* :SYS))
(defun setup (&key (mode :wiring-pi) (terminate-on-error nil))
"Initialize the Wiring Pi library.
`MODE' determines the type of initialization:
- :WIRING-PI - use Wiring Pi pin numeration
- :PHYS - use physical pin numeration
- :GPIO, :BROADCOM - use Broadcom pin numeration
- :SYS - use the /sys/class/gpio interface instead of accessing the hardware directly
Note that all modes except :SYS require the program to be run with superuser privileges.
Pin numeration under :SYS mode is the same as in :GPIO/:BROADCOM mode.
You should call this function only once per execution of the program. For convenience,
calling it again with the same `MODE' will be ignored; otherwise an error will be signalled.
Set `TERMINATE-ON-ERROR' to `NIL' to restore the default behaviour of Wiring Pi 2 causing
the process to terminate if setup function fails."
(unless terminate-on-error
(setenv "WIRINGPI_CODES" "TRUE"))
(if (initializedp)
(unless (eql *setup-mode* mode)
(error "Wiring Pi 2 setup may be done only once per execution."))
(prog1
(ccase mode
(:wiring-pi
(wpi2-ffi:wiring-pi-setup))
(:phys
(wpi2-ffi:wiring-pi-setup-phys))
((:gpio :broadcom)
(wpi2-ffi:wiring-pi-setup-gpio))
(:sys
(wpi2-ffi:wiring-pi-setup-sys)))
(setf *setup-mode* mode))))
(defun pin-mode (pin)
"Try to get the mode `PIN' is in.
Returns either :INPUT or :OUTPUT for valid I/O modes,
or :ALT-0 - :ALT-5 for 6 different 'alternate functions'.
See also `(SETF PIN-MODE)', `PIN-MODE*'."
(let ((mode (wpi2-ffi:get-alt pin)))
(ecase mode
(#.wpi2-ffi:+input+ :INPUT)
(#.wpi2-ffi:+output+ :OUTPUT)
(#.wpi2-ffi:+alt-0+ :ALT-0)
(#.wpi2-ffi:+alt-1+ :ALT-1)
(#.wpi2-ffi:+alt-2+ :ALT-2)
(#.wpi2-ffi:+alt-3+ :ALT-3)
(#.wpi2-ffi:+alt-4+ :ALT-4)
(#.wpi2-ffi:+alt-5+ :ALT-5))))
(defun (setf pin-mode) (mode pin)
"Try to set `PIN' to `MODE'.
Available modes are:
- :INPUT
- :OUTPUT
- :PWM-OUTPUT
- :GPIO-CLOCK
This function has no effect in :SYS mode."
(let ((mode-id (ccase mode
(:INPUT wpi2-ffi:+input+)
(:OUTPUT wpi2-ffi:+output+)
(:PWM-OUTPUT wpi2-ffi:+pwm-output+)
(:GPIO-CLOCK wpi2-ffi:+gpio-clock+))))
(wpi2-ffi:pin-mode pin mode-id)))
(defun pin-mode* (pin)
"Returns the raw bits of `PIN' mode as an integer."
(wpi2-ffi:get-alt pin))
(defun set-pull-resistor-mode (pin mode)
"Set the mode of pull-up / pull-down resistors on the `PIN'.
Available modes are:
- :OFF - no pull up / down
- :PULL-UP - pull up (to 3.3V)
- :PULL-DOWN - pull down (pull to ground)
`PIN' should be set to input."
(let ((pud (ccase mode
(:OFF wpi2-ffi:+pud-off+)
(:PULL-UP wpi2-ffi:+pud-up+)
(:PULL-DOWN wpi2-ffi:+pud-down+))))
(wpi2-ffi:pull-up-dn-control pin pud)))
(defun digital-write (pin value)
"Write `VALUE' to `PIN'.
Use constants `+LOW+' and `+HIGH+' for `VALUE'."
(wpi2-ffi:digital-write pin value))
(defun pwm-write (pin value)
"Write `VALUE' to PWM register for the given `PIN'.
The range for `VALUE' is 0-1024 for Raspberry Pi's on-board PWM pin (pin 1, BMC_GPIO 18, physical: 12)."
(wpi2-ffi:pwm-write pin value))
(defun digital-read (pin)
"Read the state of `PIN'.
Returns numeric value of the pin state; compare with constants `+LOW+' and `+HIGH+'."
(wpi2-ffi:digital-read pin))
(defun analog-read (pin)
"Returns the value on the analog input `PIN'.
Note, apparently there are no analog pins on the Raspberry Pi itself; read will return a dummy value there."
(wpi2-ffi:analog-read pin))
(defun analog-write (pin value)
"Writes the given `VALUE' to the supplied analog `PIN'.
Note, apparently there are no analog pins on the Raspberry Pi itself - the write will be non-effective there."
(wpi2-ffi:analog-write pin value))
(defun pi-board-revision ()
"Returns the board version of the Raspberry Pi. It will be either 1 or 2.
Revision 1 means early Model A and B's.
Revision 2 is everything else - B, B+, CM, Pi 2, Pi Zero, Pi3.
Some of the pins in Broadcom numeration change their numbers depending on the board revision."
(wpi2-ffi:pi-board-rev))
(defun pi-board-info ()
"Returns the hardware information of Raspberry Pi board.
Result is an plist with following keys:
- :MODEL - keyword describing the Raspberry Pi model
- :REVISION - number describing the Raspberry Pi revision (see `PI-BOARD-REVISION')
- :MEMORY - number of megabytes of RAM the board has
- :MAKER - keyword identifying the maker of the board
- :OVERVOLTED - `T' if the board was overclocked; `NIL' otherwise.
Models: :A, :B, :B+, :CM, :ALPHA, :2, :3, :ZERO, :07, :UNKNOWN
Makers: :SONY :EGOMAN :MBEST :UNKNOWN"
(destructuring-bind (model revision memory maker overvolted)
(pi-board-info*)
(let ((r-model (case model
(#.wpi2-ffi:+pi-model-a+ :A)
(#.wpi2-ffi:+pi-model-b+ :B)
(#.wpi2-ffi:+pi-model-ap+ :A+)
(#.wpi2-ffi:+pi-model-bp+ :B+)
(#.wpi2-ffi:+pi-model-2+ :2)
(#.wpi2-ffi:+pi-alpha+ :ALPHA)
(#.wpi2-ffi:+pi-model-cm+ :CM)
(#.wpi2-ffi:+pi-model-07+ :07)
(#.wpi2-ffi:+pi-model-3+ :3)
(#.wpi2-ffi:+pi-model-zero+ :ZERO)
(otherwise :UNKNOWN)))
(r-memory (case memory
(0 256)
(1 512)
(2 1024)
(otherwise 0)))
(r-maker (case maker
(#.wpi2-ffi:+pi-maker-sony+ :SONY)
(#.wpi2-ffi:+pi-maker-egoman+ :EGOMAN)
(#.wpi2-ffi:+pi-maker-mbest+ :MBEST)
(otherwise :UNKNOWN)))
(r-overvolted (not (zerop overvolted))))
(list :model r-model :revision revision :memory r-memory :maker r-maker :overvolted r-overvolted))))
(defun pi-board-info* ()
"Like `PI-BOARD-INFO', but gives the raw values returned from the Wiring Pi library.
They are, in sequence: model id, revision id, memory, maker id, overvolted state."
(wpi2-ffi:wrapped-pi-board-id))
(defun milliseconds ()
"Returns the number of milliseconds since the call to setup function.
Note that the number will wrap around after around 49 days."
(wpi2-ffi:millis))
(defun microseconds ()
"Returns the number of microseconds since the call to setup function.
Note that the number will wrap around after around 71 minutes."
(wpi2-ffi:micros))
(defun delay (milliseconds)
"Pause the execution for at least `MILLISECONDS' milliseconds.
Maximum delay is approximately 49 days."
(wpi2-ffi:delay milliseconds))
(defun delay-microseconds (microseconds)
"Pause the execution for at least `MICROSECONDS' microseconds.
Maximum delay is approximately 71 minutes."
(wpi2-ffi:delay-microseconds microseconds))