/external/cffi.darcs/src/cffi-clisp.lisp

http://github.com/blindglobe/common-lisp-stat · Lisp · 407 lines · 273 code · 55 blank · 79 comment · 11 complexity · ee4acec20abd0eb3583fa2093070657e MD5 · raw file

  1. ;;;; -*- Mode: lisp; indent-tabs-mode: nil -*-
  2. ;;;
  3. ;;; cffi-clisp.lisp --- CFFI-SYS implementation for CLISP.
  4. ;;;
  5. ;;; Copyright (C) 2005-2006, James Bielman <jamesjb@jamesjb.com>
  6. ;;; Copyright (C) 2005-2006, Joerg Hoehle <hoehle@users.sourceforge.net>
  7. ;;;
  8. ;;; Permission is hereby granted, free of charge, to any person
  9. ;;; obtaining a copy of this software and associated documentation
  10. ;;; files (the "Software"), to deal in the Software without
  11. ;;; restriction, including without limitation the rights to use, copy,
  12. ;;; modify, merge, publish, distribute, sublicense, and/or sell copies
  13. ;;; of the Software, and to permit persons to whom the Software is
  14. ;;; furnished to do so, subject to the following conditions:
  15. ;;;
  16. ;;; The above copyright notice and this permission notice shall be
  17. ;;; included in all copies or substantial portions of the Software.
  18. ;;;
  19. ;;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  20. ;;; EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  21. ;;; MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  22. ;;; NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
  23. ;;; HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  24. ;;; WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  25. ;;; OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
  26. ;;; DEALINGS IN THE SOFTWARE.
  27. ;;;
  28. ;;;# Administrivia
  29. (defpackage #:cffi-sys
  30. (:use #:common-lisp #:cffi-utils #:alexandria)
  31. (:export
  32. #:canonicalize-symbol-name-case
  33. #:foreign-pointer
  34. #:pointerp
  35. #:pointer-eq
  36. #:null-pointer
  37. #:null-pointer-p
  38. #:inc-pointer
  39. #:make-pointer
  40. #:pointer-address
  41. #:%foreign-alloc
  42. #:foreign-free
  43. #:with-foreign-pointer
  44. #:%foreign-funcall
  45. #:%foreign-funcall-pointer
  46. #:%foreign-type-alignment
  47. #:%foreign-type-size
  48. #:%load-foreign-library
  49. #:%close-foreign-library
  50. #:native-namestring
  51. #:%mem-ref
  52. #:%mem-set
  53. #:make-shareable-byte-vector
  54. #:with-pointer-to-vector-data
  55. #:%foreign-symbol-pointer
  56. #:%defcallback
  57. #:%callback))
  58. (in-package #:cffi-sys)
  59. ;;;# Symbol Case
  60. (defun canonicalize-symbol-name-case (name)
  61. (declare (string name))
  62. (string-upcase name))
  63. ;;;# Built-In Foreign Types
  64. (defun convert-foreign-type (type)
  65. "Convert a CFFI built-in type keyword to a CLisp FFI type."
  66. (ecase type
  67. (:char 'ffi:char)
  68. (:unsigned-char 'ffi:uchar)
  69. (:short 'ffi:short)
  70. (:unsigned-short 'ffi:ushort)
  71. (:int 'ffi:int)
  72. (:unsigned-int 'ffi:uint)
  73. (:long 'ffi:long)
  74. (:unsigned-long 'ffi:ulong)
  75. (:long-long 'ffi:sint64)
  76. (:unsigned-long-long 'ffi:uint64)
  77. (:float 'ffi:single-float)
  78. (:double 'ffi:double-float)
  79. ;; Clisp's FFI:C-POINTER converts NULL to NIL. For now
  80. ;; we have a workaround in the pointer operations...
  81. (:pointer 'ffi:c-pointer)
  82. (:void nil)))
  83. (defun %foreign-type-size (type)
  84. "Return the size in bytes of objects having foreign type TYPE."
  85. (nth-value 0 (ffi:sizeof (convert-foreign-type type))))
  86. ;; Remind me to buy a beer for whoever made getting the alignment
  87. ;; of foreign types part of the public interface in CLisp. :-)
  88. (defun %foreign-type-alignment (type)
  89. "Return the structure alignment in bytes of foreign TYPE."
  90. #+(and cffi-features:darwin cffi-features:ppc32)
  91. (case type
  92. ((:double :long-long :unsigned-long-long)
  93. (return-from %foreign-type-alignment 8)))
  94. ;; Override not necessary for the remaining types...
  95. (nth-value 1 (ffi:sizeof (convert-foreign-type type))))
  96. ;;;# Basic Pointer Operations
  97. (deftype foreign-pointer ()
  98. '(or null ffi:foreign-address))
  99. (defun pointerp (ptr)
  100. "Return true if PTR is a foreign pointer."
  101. (or (null ptr) (typep ptr 'ffi:foreign-address)))
  102. (defun pointer-eq (ptr1 ptr2)
  103. "Return true if PTR1 and PTR2 point to the same address."
  104. (eql (ffi:foreign-address-unsigned ptr1)
  105. (ffi:foreign-address-unsigned ptr2)))
  106. (defun null-pointer ()
  107. "Return a null foreign pointer."
  108. (ffi:unsigned-foreign-address 0))
  109. (defun null-pointer-p (ptr)
  110. "Return true if PTR is a null foreign pointer."
  111. (or (null ptr) (zerop (ffi:foreign-address-unsigned ptr))))
  112. (defun inc-pointer (ptr offset)
  113. "Return a pointer pointing OFFSET bytes past PTR."
  114. (ffi:unsigned-foreign-address
  115. (+ offset (if (null ptr) 0 (ffi:foreign-address-unsigned ptr)))))
  116. (defun make-pointer (address)
  117. "Return a pointer pointing to ADDRESS."
  118. (ffi:unsigned-foreign-address address))
  119. (defun pointer-address (ptr)
  120. "Return the address pointed to by PTR."
  121. (ffi:foreign-address-unsigned ptr))
  122. ;;;# Foreign Memory Allocation
  123. (defun %foreign-alloc (size)
  124. "Allocate SIZE bytes of foreign-addressable memory and return a
  125. pointer to the allocated block. An implementation-specific error
  126. is signalled if the memory cannot be allocated."
  127. (ffi:foreign-address (ffi:allocate-shallow 'ffi:uint8 :count size)))
  128. (defun foreign-free (ptr)
  129. "Free a pointer PTR allocated by FOREIGN-ALLOC. The results
  130. are undefined if PTR is used after being freed."
  131. (ffi:foreign-free ptr))
  132. (defmacro with-foreign-pointer ((var size &optional size-var) &body body)
  133. "Bind VAR to a pointer to SIZE bytes of foreign-addressable
  134. memory during BODY. Both PTR and the memory block pointed to
  135. have dynamic extent and may be stack allocated if supported by
  136. the implementation. If SIZE-VAR is supplied, it will be bound to
  137. SIZE during BODY."
  138. (unless size-var
  139. (setf size-var (gensym "SIZE")))
  140. (let ((obj-var (gensym)))
  141. `(let ((,size-var ,size))
  142. (ffi:with-foreign-object
  143. (,obj-var `(ffi:c-array ffi:uint8 ,,size-var))
  144. (let ((,var (ffi:foreign-address ,obj-var)))
  145. ,@body)))))
  146. ;;;# Memory Access
  147. (defun %mem-ref (ptr type &optional (offset 0))
  148. "Dereference a pointer OFFSET bytes from PTR to an object of
  149. built-in foreign TYPE. Returns the object as a foreign pointer
  150. or Lisp number."
  151. (ffi:memory-as ptr (convert-foreign-type type) offset))
  152. (define-compiler-macro %mem-ref (&whole form ptr type &optional (offset 0))
  153. "Compiler macro to open-code when TYPE is constant."
  154. (if (constantp type)
  155. `(ffi:memory-as ,ptr ',(convert-foreign-type (eval type)) ,offset)
  156. form))
  157. (defun %mem-set (value ptr type &optional (offset 0))
  158. "Set a pointer OFFSET bytes from PTR to an object of built-in
  159. foreign TYPE to VALUE."
  160. (setf (ffi:memory-as ptr (convert-foreign-type type) offset) value))
  161. (define-compiler-macro %mem-set
  162. (&whole form value ptr type &optional (offset 0))
  163. (if (constantp type)
  164. ;; (setf (ffi:memory-as) value) is exported, but not so nice
  165. ;; w.r.t. the left to right evaluation rule
  166. `(ffi::write-memory-as
  167. ,value ,ptr ',(convert-foreign-type (eval type)) ,offset)
  168. form))
  169. ;;;# Shareable Vectors
  170. ;;;
  171. ;;; This interface is very experimental. WITH-POINTER-TO-VECTOR-DATA
  172. ;;; should be defined to perform a copy-in/copy-out if the Lisp
  173. ;;; implementation can't do this.
  174. (declaim (inline make-shareable-byte-vector))
  175. (defun make-shareable-byte-vector (size)
  176. "Create a Lisp vector of SIZE bytes can passed to
  177. WITH-POINTER-TO-VECTOR-DATA."
  178. (make-array size :element-type '(unsigned-byte 8)))
  179. (deftype shareable-byte-vector ()
  180. `(vector (unsigned-byte 8)))
  181. (defmacro with-pointer-to-vector-data ((ptr-var vector) &body body)
  182. "Bind PTR-VAR to a foreign pointer to the data in VECTOR."
  183. (with-unique-names (vector-var size-var)
  184. `(let ((,vector-var ,vector))
  185. (check-type ,vector-var shareable-byte-vector)
  186. (with-foreign-pointer (,ptr-var (length ,vector-var) ,size-var)
  187. ;; copy-in
  188. (loop for i below ,size-var do
  189. (%mem-set (aref ,vector-var i) ,ptr-var :unsigned-char i))
  190. (unwind-protect (progn ,@body)
  191. ;; copy-out
  192. (loop for i below ,size-var do
  193. (setf (aref ,vector-var i)
  194. (%mem-ref ,ptr-var :unsigned-char i))))))))
  195. ;;;# Foreign Function Calling
  196. (defun parse-foreign-funcall-args (args)
  197. "Return three values, a list of CLISP FFI types, a list of
  198. values to pass to the function, and the CLISP FFI return type."
  199. (let ((return-type nil))
  200. (loop for (type arg) on args by #'cddr
  201. if arg collect (list (gensym) (convert-foreign-type type)) into types
  202. and collect arg into fargs
  203. else do (setf return-type (convert-foreign-type type))
  204. finally (return (values types fargs return-type)))))
  205. (defun convert-cconv (calling-convention)
  206. (ecase calling-convention
  207. (:stdcall :stdc-stdcall)
  208. (:cdecl :stdc)))
  209. (defun c-function-type (arg-types rettype calling-convention)
  210. "Generate the apropriate CLISP foreign type specification. Also
  211. takes care of converting the calling convention names."
  212. `(ffi:c-function (:arguments ,@arg-types)
  213. (:return-type ,rettype)
  214. (:language ,(convert-cconv calling-convention))))
  215. ;;; Quick hack around the fact that the CFFI package is not yet
  216. ;;; defined when this file is loaded. I suppose we could arrange for
  217. ;;; the CFFI package to be defined a bit earlier, though.
  218. (defun library-handle-form (name)
  219. (flet ((find-cffi-symbol (symbol)
  220. (find-symbol (symbol-name symbol) '#:cffi)))
  221. `(,(find-cffi-symbol '#:foreign-library-handle)
  222. (,(find-cffi-symbol '#:get-foreign-library) ',name))))
  223. (eval-when (:compile-toplevel :load-toplevel :execute)
  224. ;; version 2.40 (CVS 2006-09-03, to be more precise) added a
  225. ;; PROPERTIES argument to FFI::FOREIGN-LIBRARY-FUNCTION.
  226. (defun post-2.40-ffi-interface-p ()
  227. (let ((f-l-f (find-symbol (string '#:foreign-library-function) '#:ffi)))
  228. (if (and f-l-f (= (length (ext:arglist f-l-f)) 5))
  229. '(:and)
  230. '(:or))))
  231. ;; FFI::FOREIGN-LIBRARY-FUNCTION and FFI::FOREIGN-LIBRARY-VARIABLE
  232. ;; were deprecated in 2.41 and removed in 2.45.
  233. (defun post-2.45-ffi-interface-p ()
  234. (if (find-symbol (string '#:foreign-library-function) '#:ffi)
  235. '(:or)
  236. '(:and))))
  237. #+#.(cffi-sys::post-2.45-ffi-interface-p)
  238. (defun %foreign-funcall-aux (name type library)
  239. `(ffi::find-foreign-function ,name ,type nil ,library nil nil))
  240. #-#.(cffi-sys::post-2.45-ffi-interface-p)
  241. (defun %foreign-funcall-aux (name type library)
  242. `(ffi::foreign-library-function
  243. ,name ,library nil
  244. #+#.(cffi-sys::post-2.40-ffi-interface-p)
  245. nil
  246. ,type))
  247. (defmacro %foreign-funcall (name args &key library calling-convention)
  248. "Invoke a foreign function called NAME, taking pairs of
  249. foreign-type/value pairs from ARGS. If a single element is left
  250. over at the end of ARGS, it specifies the foreign return type of
  251. the function call."
  252. (multiple-value-bind (types fargs rettype)
  253. (parse-foreign-funcall-args args)
  254. `(funcall
  255. (load-time-value
  256. (handler-case
  257. ,(%foreign-funcall-aux
  258. name
  259. `(ffi:parse-c-type
  260. ',(c-function-type types rettype calling-convention))
  261. (if (eq library :default)
  262. :default
  263. (library-handle-form library)))
  264. (error (err)
  265. (warn "~A" err))))
  266. ,@fargs)))
  267. (defmacro %foreign-funcall-pointer (ptr args &key calling-convention)
  268. "Similar to %foreign-funcall but takes a pointer instead of a string."
  269. (multiple-value-bind (types fargs rettype)
  270. (parse-foreign-funcall-args args)
  271. `(funcall (ffi:foreign-function
  272. ,ptr (load-time-value
  273. (ffi:parse-c-type ',(c-function-type
  274. types rettype calling-convention))))
  275. ,@fargs)))
  276. ;;;# Callbacks
  277. ;;; *CALLBACKS* contains the callbacks defined by the CFFI DEFCALLBACK
  278. ;;; macro. The symbol naming the callback is the key, and the value
  279. ;;; is a list containing a Lisp function, the parsed CLISP FFI type of
  280. ;;; the callback, and a saved pointer that should not persist across
  281. ;;; saved images.
  282. (defvar *callbacks* (make-hash-table))
  283. ;;; Return a CLISP FFI function type for a CFFI callback function
  284. ;;; given a return type and list of argument names and types.
  285. (eval-when (:compile-toplevel :load-toplevel :execute)
  286. (defun callback-type (rettype arg-names arg-types calling-convention)
  287. (ffi:parse-c-type
  288. `(ffi:c-function
  289. (:arguments ,@(mapcar (lambda (sym type)
  290. (list sym (convert-foreign-type type)))
  291. arg-names arg-types))
  292. (:return-type ,(convert-foreign-type rettype))
  293. (:language ,(convert-cconv calling-convention))))))
  294. ;;; Register and create a callback function.
  295. (defun register-callback (name function parsed-type)
  296. (setf (gethash name *callbacks*)
  297. (list function parsed-type
  298. (ffi:with-foreign-object (ptr 'ffi:c-pointer)
  299. ;; Create callback by converting Lisp function to foreign
  300. (setf (ffi:memory-as ptr parsed-type) function)
  301. (ffi:foreign-value ptr)))))
  302. ;;; Restore all saved callback pointers when restarting the Lisp
  303. ;;; image. This is pushed onto CUSTOM:*INIT-HOOKS*.
  304. ;;; Needs clisp > 2.35, bugfix 2005-09-29
  305. (defun restore-callback-pointers ()
  306. (maphash
  307. (lambda (name list)
  308. (register-callback name (first list) (second list)))
  309. *callbacks*))
  310. ;;; Add RESTORE-CALLBACK-POINTERS to the lists of functions to run
  311. ;;; when an image is restarted.
  312. (eval-when (:load-toplevel :execute)
  313. (pushnew 'restore-callback-pointers custom:*init-hooks*))
  314. ;;; Define a callback function NAME to run BODY with arguments
  315. ;;; ARG-NAMES translated according to ARG-TYPES and the return type
  316. ;;; translated according to RETTYPE. Obtain a pointer that can be
  317. ;;; passed to C code for this callback by calling %CALLBACK.
  318. (defmacro %defcallback (name rettype arg-names arg-types body
  319. &key calling-convention)
  320. `(register-callback ',name (lambda ,arg-names ,body)
  321. ,(callback-type rettype arg-names arg-types
  322. calling-convention)))
  323. ;;; Look up the name of a callback and return a pointer that can be
  324. ;;; passed to a C function. Signals an error if no callback is
  325. ;;; defined called NAME.
  326. (defun %callback (name)
  327. (multiple-value-bind (list winp) (gethash name *callbacks*)
  328. (unless winp
  329. (error "Undefined callback: ~S" name))
  330. (third list)))
  331. ;;;# Loading and Closing Foreign Libraries
  332. (defun %load-foreign-library (name path)
  333. "Load a foreign library from PATH."
  334. (declare (ignore name))
  335. #+#.(cffi-sys::post-2.45-ffi-interface-p)
  336. (ffi:open-foreign-library path)
  337. #-#.(cffi-sys::post-2.45-ffi-interface-p)
  338. (ffi::foreign-library path))
  339. (defun %close-foreign-library (handle)
  340. "Close a foreign library."
  341. (ffi:close-foreign-library handle))
  342. (defun native-namestring (pathname)
  343. (namestring pathname))
  344. ;;;# Foreign Globals
  345. (defun %foreign-symbol-pointer (name library)
  346. "Returns a pointer to a foreign symbol NAME."
  347. (prog1 (ignore-errors
  348. (ffi:foreign-address
  349. #+#.(cffi-sys::post-2.45-ffi-interface-p)
  350. (ffi::find-foreign-variable name nil library nil nil)
  351. #-#.(cffi-sys::post-2.45-ffi-interface-p)
  352. (ffi::foreign-library-variable name library nil nil)))))