-
Notifications
You must be signed in to change notification settings - Fork 29
/
Copy pathtypes.lisp
323 lines (283 loc) · 12.6 KB
/
types.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
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
;;;; +----------------------------------------------------------------+
;;;; | DBUS |
;;;; +----------------------------------------------------------------+
(defpackage #:dbus/types
(:use #:cl #:dbus/utils)
(:import-from #:alexandria #:with-gensyms #:circular-list)
(:import-from #:babel #:string-to-octets #:octets-to-string)
(:export
#:sigexp
#:signature
#:pack
#:unpack
#:valid-body-p
#:define-dbus-type
#:pack-string
#:unpack-string
#:pack-array
#:unpack-array
#:pack-seq
#:unpack-seq
#:pack-variant
#:unpack-variant
#:valid-signature-p
#:valid-array-p
#:valid-struct-p
#:valid-variant-p
#:valid-dict-entry-p
#:stream
#:endianness
#:value
#:element-types))
(in-package #:dbus/types)
;;;; Defining DBUS types
(defclass dbus-type ()
((name :initarg :name :reader dbus-type-name)
(signature :initarg :signature :reader dbus-type-signature)
(sigexp-formatter :initarg :sigexp-formatter :reader dbus-type-sigexp-formatter)
(signature-parser :initarg :signature-parser :reader dbus-type-signature-parser)
(alignment :initarg :alignment :reader dbus-type-alignment)
(packer :initarg :packer :reader dbus-type-packer)
(unpacker :initarg :unpacker :reader dbus-type-unpacker)
(checker :initarg :checker :reader dbus-type-checker)))
(defmethod print-object ((type dbus-type) stream)
(print-unreadable-object (type stream :type t)
(format stream "~S" (dbus-type-name type)))
type)
(defclass dbus-type-table ()
((by-name :initform (make-hash-table) :reader dbus-type-table-by-name)
(by-signature :initform (make-hash-table) :reader dbus-type-table-by-signature)))
(defvar *dbus-type-table*
(make-instance 'dbus-type-table))
(defun find-dbus-type (designator &optional (table *dbus-type-table*))
(etypecase designator
(dbus-type (values designator '()))
(symbol
(values
(or (gethash designator (dbus-type-table-by-name table))
(error "Can't find DBUS type with name ~S." designator))
'()))
(character
(values
(or (gethash designator (dbus-type-table-by-signature table))
(error "Can't find DBUS type with signature ~S." designator))
'()))
((cons symbol)
(values (find-dbus-type (first designator) table) (rest designator)))))
(defun register-dbus-type (type &optional (table *dbus-type-table*))
(setf (gethash (dbus-type-name type) (dbus-type-table-by-name table)) type)
(setf (gethash (dbus-type-signature type) (dbus-type-table-by-signature table)) type)
table)
(defun make-dbus-type-formatter/parser (name signature composite)
(etypecase composite
((eql nil)
(values (lambda (stream element-types)
(declare (ignore element-types))
(write-char signature stream))
(lambda (stream)
(declare (ignore stream))
name)))
((eql t)
(values (lambda (stream element-types)
(write-char signature stream)
(format-sigexp-to-stream element-types stream))
(lambda (stream)
(cons name (parse-signature-from-stream stream nil 1)))))
(character
(values (lambda (stream element-types)
(write-char signature stream)
(format-sigexp-to-stream element-types stream)
(write-char composite stream))
(lambda (stream)
(prog1 (cons name (parse-signature-from-stream stream composite))
(read-char stream)))))))
(defmacro define-dbus-type (name &key signature composite alignment pack unpack (checker t))
(with-gensyms (formatter parser)
`(progn
(register-dbus-type
(multiple-value-bind (,formatter ,parser)
(make-dbus-type-formatter/parser ',name ',signature ',composite)
(make-instance 'dbus-type
:name ',name
:signature ',signature
:sigexp-formatter ,formatter
:signature-parser ,parser
:alignment ',alignment
:packer (lambda (stream endianness element-types value)
(declare (ignorable element-types value))
(with-binary-writers (stream endianness)
(align ',alignment)
,pack))
:unpacker (lambda (stream endianness element-types)
(declare (ignorable element-types))
(with-binary-readers (stream endianness)
(align ',alignment)
,unpack))
:checker ,(if (and (consp checker) (eq (car checker) 'function))
checker
`(lambda (value element-types)
(declare (ignore element-types))
(typep value ',checker))))))
',name)))
(defun pack-1 (stream endianness type value)
"Pack a single DBUS value into stream."
(multiple-value-bind (type element-types) (find-dbus-type type)
(funcall (dbus-type-packer type) stream endianness element-types value)))
(defun unpack-1 (stream endianness type)
"Unpack a single DBUS value from stream."
(multiple-value-bind (type element-types) (find-dbus-type type)
(funcall (dbus-type-unpacker type) stream endianness element-types)))
(defun pack-seq (stream endianness types values)
"Pack a sequence of values into stream."
(map nil (lambda (type value) (pack-1 stream endianness type value)) types values))
(defun unpack-seq (stream endianness types)
"Unpack a sequence of DBUS values from stream."
(map 'list (lambda (type) (unpack-1 stream endianness type)) types))
(defun pack-string (stream endianness value length-size)
"Pack DBUS string into stream."
(with-binary-writers (stream endianness)
(let ((octets (string-to-octets value :encoding :utf-8)))
(ecase length-size
(8 (u8 (length octets)))
(32 (u32 (length octets))))
(map nil #'u8 octets)
(u8 0))))
(defun unpack-string (stream endianness length)
"Unpack DBUS string from stream."
(with-binary-readers (stream endianness)
(prog1 (octets-to-string
(map-into (make-octet-vector length) #'u8)
:encoding :utf-8)
(u8))))
(defun pack-array (stream endianness element-type value)
"Pack DBUS array into stream."
(with-binary-writers (stream endianness)
(let ((length-position (file-position stream)))
(u32 0)
(align (alignment element-type))
(let ((start-position (file-position stream)))
(pack-seq stream endianness (circular-list element-type) value)
(let ((end-position (file-position stream)))
(file-position stream length-position)
(u32 (- end-position start-position))
(file-position stream end-position))))))
(defun unpack-array (stream endianness element-type length)
"Unpack DBUS array from stream."
(with-binary-readers (stream endianness)
(align (alignment element-type))
(loop with start = (stream-read-position stream)
with end = (+ start length)
until (= end (stream-read-position stream))
collect (unpack-1 stream endianness element-type))))
(defun pack-variant (stream endianness element-types value)
"Pack DBUS variant into stream."
(pack-1 stream endianness :signature element-types)
(pack-1 stream endianness (first element-types) value))
(defun unpack-variant (stream endianness)
"Unpack DBUS variant from stream."
(with-binary-readers (stream endianness)
(unpack-1 stream endianness
(first (sigexp (unpack-string stream endianness (u8)))))))
(defun alignment (type)
"Return the number of octets to which elements of the supplied type
should be aligned."
(dbus-type-alignment (find-dbus-type type)))
(defun parse-signature-from-stream (stream &optional terminator-char num-elements)
"Parse a signature string from a character stream and return the
corresponding signature expression.
The value of TERMINATOR-CHAR determines when to stop parsing. If it
is NIL (the default), parsing is stopped when there are no more
characters left to read from the stream. If it is a character,
parsing is stopped when the same character is read from the stream.
The value of NUM-ELEMENTS determines how many elements (types) should
be read before parsing is stopped. If it is NIL (the default), there
is no bound on the number of elements to be read."
(loop for num-read from 0
for char = (peek-char nil stream nil nil)
until (or (null char) (eql char terminator-char) (eql num-read num-elements))
collect (let ((type (find-dbus-type char)))
(read-char stream)
(funcall (dbus-type-signature-parser type) stream))))
(defun format-sigexp-to-stream (sigexp stream)
"Format a signature expression as a signature string into a
character stream."
(dolist (subexp sigexp)
(multiple-value-bind (type element-types) (find-dbus-type subexp)
(funcall (dbus-type-sigexp-formatter type) stream element-types))))
(defun valid-signature-p (value element-types)
"Return true if the value is a valid signature string or signature
expression, and false otherwise."
(declare (ignore element-types))
(handler-case
(progn (signature (sigexp value)) t)
(error () nil)))
(defun valid-array-p (value element-types)
"Return true if the value is a sequence with elements of the first
type supplied in ELEMENT-TYPES, and false otherwise."
(when element-types
(let ((element-type (first element-types)))
(and (typep value 'sequence)
(every (lambda (element) (valid-value-p element element-type))
value)))))
(defun valid-struct-p (value element-types)
"Return true if the value is a sequence with elements matching the
types supplied in ELEMENT-TYPES, and false otherwise."
(and (typep value 'sequence)
(= (length value) (length element-types))
(every (lambda (element element-type) (valid-value-p element element-type))
value element-types)))
(defun valid-variant-p (value element-types)
"Return true if the value is a variant value specification, and
false otherwise."
(declare (ignore element-types))
(and (listp value)
(= (length value) 2)
(valid-value-p (first value) :signature)
(let ((actual-value (second value))
(sigexp (sigexp (first value))))
(valid-value-p actual-value (first sigexp)))))
(defun valid-dict-entry-p (value element-types)
"Return true if the value is a sequence with two elements, both
matching the types supplied in ELEMENT-TYPES, and false otherwise."
(and (typep value 'sequence)
(= (length value) (length element-types) 2)
(every (lambda (element element-type) (valid-value-p element element-type))
value element-types)))
(defun valid-value-p (value type)
"Return true if the value is of the supplied DBUS type, and false
otherwise."
(multiple-value-bind (type element-types) (find-dbus-type type)
(funcall (dbus-type-checker type) value element-types)))
;;;; Operators related to DBUS types
(defun sigexp (object)
"Return the signature expression corresponding to the object passed.
If the object is a string, it is assumed to be a signature string,
otherwise it is assumed to be a signature expression and is returned
as-is."
(if (stringp object)
(with-input-from-string (in object)
(parse-signature-from-stream in))
object))
(defun signature (object)
"Return the signature string corresponding to the object passed.
If the object is a string, it is assumed to be a signature string and
is returned as-is, otherwise it is assumed to be a signature
expression."
(if (stringp object)
object
(with-output-to-string (out)
(format-sigexp-to-stream object out))))
(defun pack (stream endianness sigexp &rest values)
"Pack values according to the signature expression and endianness
into stream."
(pack-seq stream endianness (sigexp sigexp) values))
(defun unpack (stream endianness sigexp)
"Unpack values from stream according to endianness and the signature
expression and return them as a list."
(unpack-seq stream endianness (sigexp sigexp)))
(defun valid-body-p (body sigexp)
"Return true if the message body (which is a list of values) is
valid according to the signature expression, and false otherwise."
(setf sigexp (sigexp sigexp))
(and (= (length body) (length sigexp))
(every #'valid-value-p body sigexp)))