X-Git-Url: https://www.chiark.greenend.org.uk/ucgi/~mdw/git/clg/blobdiff_plain/050b602e6c70ee9f950fc9a579377c118fcb7a61..84d58948bc9cf543ac43e17da28d1a832baeff1a:/gffi/interface.lisp diff --git a/gffi/interface.lisp b/gffi/interface.lisp index 084ed47..6b918d2 100644 --- a/gffi/interface.lisp +++ b/gffi/interface.lisp @@ -72,15 +72,19 @@ (defun default-alien-type-name (type-name) #'string-capitalize (cons prefix (split-string (symbol-name type-name) :delimiter #\-)))))) -(defun default-type-name (alien-name) - (let ((parts - (mapcar - #'string-upcase - (split-string-if alien-name #'upper-case-p)))) - (intern - (concatenate-strings (rest parts) #\-) - (find-prefix-package (first parts))))) +(defun split-alien-name (alien-name) + (let ((parts (split-string-if alien-name #'upper-case-p))) + (do ((prefix (first parts) (concatenate 'string prefix (first rest))) + (rest (rest parts) (cdr rest))) + ((null rest) + (error "Couldn't split alien name '~A' to find a registered prefix" + alien-name)) + (when (find-prefix-package prefix) + (return (values (string-upcase (concatenate-strings rest #\-)) + (find-prefix-package prefix))))))) +(defun default-type-name (alien-name) + (multiple-value-call #'intern (split-alien-name alien-name))) (defun in-arg-p (style) (find style '(:in :in/out :in/return :in-out :return))) @@ -92,6 +96,29 @@ (defun return-arg-p (style) (find style '(:in/return :return))) (defmacro defbinding (name lambda-list return-type &rest args) + "This defines a foreign function call. NAME should either be a symbol or a +list (LISP-SYM STRING). The lisp function will be given the name of the lisp +symbol and the foreign function name is either the string given or automatically +generated using DEFAULT-ALIEN-FNAME. + +If LAMBDA-LIST is nil, the lambda list for the generated lisp function is +automatically computed from the input arguments as described below. If it is +non-nil, it gives the lambda list for the function. To manually specify an empty +lambda list, pass (NIL) which gets recognised as a special value. + +RETURN-TYPE should be a valid type. + +A normal element of ARGS is a list matching + + (EXPR TYPE &OPTIONAL (STYLE :IN) (OUT-TYPE TYPE)) + +however a shorthand form for an input parameter with name the same as its type +is that you can just give the atom TYPE as an argument. The lambda-list for the +function is the list of all input arguments, although if an EXPR is repeated, it +will only appear once. To add a constant argument, define one with STYLE :IN and +EXPR the value it should take. + +To give the binding a docstring, pass a string as the first element of ARGS." (multiple-value-bind (lisp-name c-name) (if (atom name) (values name (default-alien-fname name)) @@ -420,6 +447,12 @@ (deftype callback () 'symbol)) ;;;; Type expansion +;; A hack to make the TYPE-EXPAND code for SBCL work. +#?+(pkg-config:sbcl>= 1 0 35 15) +(sb-ext:without-package-locks + (setf (symbol-function 'sb-kernel::type-expand) + (lambda (form) (typexpand form)))) + (defun type-expand-1 (form) #+(or cmu sbcl) (let ((def (cond ((symbolp form)