-;; Common Lisp bindings for GTK+ v2.0
-;; Copyright (C) 2000-2001 Espen S. Johnsen <esj@stud.cs.uit.no>
+;; Common Lisp bindings for GTK+ v2.x
+;; Copyright 2000-2005 Espen S. Johnsen <espen@users.sf.net>
;;
-;; This library is free software; you can redistribute it and/or
-;; modify it under the terms of the GNU Lesser General Public
-;; License as published by the Free Software Foundation; either
-;; version 2 of the License, or (at your option) any later version.
+;; Permission is hereby granted, free of charge, to any person obtaining
+;; a copy of this software and associated documentation files (the
+;; "Software"), to deal in the Software without restriction, including
+;; without limitation the rights to use, copy, modify, merge, publish,
+;; distribute, sublicense, and/or sell copies of the Software, and to
+;; permit persons to whom the Software is furnished to do so, subject to
+;; the following conditions:
;;
-;; This library is distributed in the hope that it will be useful,
-;; but WITHOUT ANY WARRANTY; without even the implied warranty of
-;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-;; Lesser General Public License for more details.
+;; The above copyright notice and this permission notice shall be
+;; included in all copies or substantial portions of the Software.
;;
-;; You should have received a copy of the GNU Lesser General Public
-;; License along with this library; if not, write to the Free Software
-;; Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+;; EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+;; MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
+;; IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
+;; CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
+;; TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
+;; SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
-;; $Id: genums.lisp,v 1.6 2005-02-01 15:24:52 espen Exp $
+;; $Id: genums.lisp,v 1.14 2005-04-24 13:24:41 espen Exp $
(in-package "GLIB")
-
-
-(defun %map-enum (args op)
- (let ((current-value 0))
- (mapcar
- #'(lambda (mapping)
- (destructuring-bind (symbol &optional (value current-value))
- (mklist mapping)
- (setf current-value (1+ value))
- (case op
- (:enum-int (list symbol value))
- (:flags-int (list symbol value))
- (:int-enum (list value symbol))
- (:int-flags (list value symbol))
- (:symbols symbol))))
- args)))
-
-(defun %query-enum-or-flags-values (query-function class type)
- (multiple-value-bind (sap length)
- (funcall query-function (type-class-ref type))
- (let ((values nil)
- (size (proxy-instance-size (find-class class)))
- (proxy (make-instance class :location sap)))
- (dotimes (i length)
- (with-slots (location nickname value) proxy
- (setf location sap)
- (setq sap (sap+ sap size))
- (push
- (list
- (intern (substitute #\- #\_ (string-upcase nickname)) "KEYWORD")
- value)
- values)))
- values)))
-
;;;; Generic enum type
+(defun %map-enum (mappings op)
+ (loop
+ as value = 0 then (1+ value)
+ for mapping in mappings
+ collect (let ((symbol (if (atom mapping) mapping (first mapping))))
+ (unless (atom mapping)
+ (setq value (second mapping)))
+ (ecase op
+ (:symbol-int `(,symbol ,value))
+ (:int-symbol `(,value ,symbol))
+ (:int-quoted-symbol `(,value ',symbol))
+ (:symbols symbol)))))
+
(deftype enum (&rest args)
`(member ,@(%map-enum args :symbols)))
(defmethod to-alien-form (form (type (eql 'enum)) &rest args)
(declare (ignore type))
- `(ecase ,form
- ,@(%map-enum args :enum-int)))
+ `(case ,form
+ ,@(%map-enum args :symbol-int)
+ (t (error 'type-error :datum ,form :expected-type '(enum ,@args)))))
+
(defmethod from-alien-form (form (type (eql 'enum)) &rest args)
(declare (ignore type))
- `(ecase ,form
- ,@(%map-enum args :int-enum)))
+ `(case ,form
+ ,@(%map-enum args :int-quoted-symbol)))
(defmethod to-alien-function ((type (eql 'enum)) &rest args)
- (let ((mappings (%map-enum args :enum-int)))
+ (declare (ignore type))
+ (let ((mappings (%map-enum args :symbol-int)))
#'(lambda (enum)
(or
(second (assoc enum mappings))
- (error "~S is not of type ~S" enum (cons type args))))))
+ (error 'type-error :datum enum :expected-type (cons 'enum args))))))
(defmethod from-alien-function ((type (eql 'enum)) &rest args)
(declare (ignore type))
- (let ((mappings (%map-enum args :int-enum)))
+ (let ((mappings (%map-enum args :int-symbol)))
#'(lambda (int)
(second (assoc int mappings)))))
#'(lambda (location &optional (offset 0))
(funcall function (funcall reader location offset)))))
-
-
-(defclass %enum-value (struct)
- ((value :allocation :alien :type int)
- (name :allocation :alien :type string)
- (nickname :allocation :alien :type string))
- (:metaclass static-struct-class))
-
-(defbinding %enum-class-values () pointer
- (class pointer)
- (n-values unsigned-int :out))
-
-(defun query-enum-values (type)
- (%query-enum-or-flags-values #'%enum-class-values '%enum-value type))
-
(defun enum-int (enum type)
(funcall (to-alien-function type) enum))
(defun enum-mapping (type)
(rest (type-expand-to 'enum type)))
+
+;;;; Named enum types
+
+(defmacro define-enum-type (name &rest args)
+ (let ((enum-int (intern (format nil "~A-TO-INT" name)))
+ (int-enum (intern (format nil "INT-TO-~A" name))))
+ `(progn
+ (deftype ,name () '(enum ,@args))
+ (defun ,enum-int (enum)
+ (case enum
+ ,@(%map-enum args :symbol-int)
+ (t (error 'type-error :datum enum :expected-type ',name))))
+ (defun ,int-enum (value)
+ (case value
+ ,@(%map-enum args :int-quoted-symbol)))
+ (defmethod to-alien-form (form (type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (list ',enum-int form))
+ (defmethod from-alien-form (form (type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (list ',int-enum form))
+ (defmethod to-alien-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ #',enum-int)
+ (defmethod from-alien-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ #',int-enum)
+ (defmethod writer-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (let ((writer (writer-function 'signed)))
+ #'(lambda (enum location &optional (offset 0))
+ (funcall writer (,enum-int enum) location offset))))
+ (defmethod reader-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (let ((reader (reader-function 'signed)))
+ #'(lambda (location &optional (offset 0))
+ (,int-enum (funcall reader location offset))))))))
+
+
;;;; Generic flags type
+(defun %map-flags (mappings op)
+ (loop
+ as value = 1 then (ash value 1)
+ for mapping in mappings
+ collect (let ((symbol (if (atom mapping) mapping (first mapping))))
+ (unless (atom mapping)
+ (setq value (second mapping)))
+ (ecase op
+ (:symbol-int `(,symbol ,value))
+ (:int-symbol `(,value ,symbol))
+ (:symbols symbol)))))
+
(deftype flags (&rest args)
- `(or null (cons (member ,@(%map-enum args :symbols)) list)))
+ `(or (member ,@(%map-flags args :symbols)) list))
(defmethod alien-type ((type (eql 'flags)) &rest args)
(declare (ignore type args))
(size-of 'unsigned))
(defmethod to-alien-form (flags (type (eql 'flags)) &rest args)
- `(loop
- with value = 0
- with flags = ,flags
- for flag in (mklist flags)
- do (let ((flagval
- (or
- (second (assoc flag ',(%map-enum args :flags-int)))
- (error "~S is not of type ~S" flags '(,type ,@args)))))
- (setq value (logior value flagval)))
- finally (return value)))
-
-(defmethod from-alien-form (int (type (eql 'flags)) &rest args)
+ `(reduce #'logior (mklist ,flags)
+ :key #'(lambda (flag)
+ (case flag
+ ,@(%map-flags args :symbol-int)
+ (t (error 'type-error :datum ,flags
+ :expected-type '(,type ,@args)))))))
+
+(defmethod from-alien-form (value (type (eql 'flags)) &rest args)
(declare (ignore type))
`(loop
- for mapping in ',(%map-enum args :int-flags)
- unless (zerop (logand int (first mapping)))
- collect (second mapping)))
+ for (int symbol) in ',(%map-flags args :int-symbol)
+ when (= (logand ,value int) int)
+ collect symbol))
(defmethod to-alien-function ((type (eql 'flags)) &rest args)
- (let ((mappings (%map-enum args :flags-int)))
- #'(lambda (flags)
- (loop
- with value = 0
- for flag in (mklist flags)
- do (let ((flagval (or
- (second (assoc flag mappings))
- (error "~S is not of type ~S" flags (cons type args)))))
- (setq value (logior value flagval)))
- finally (return value)))))
+ (declare (ignore type))
+ (let ((mappings (%map-flags args :symbol-int)))
+ #'(lambda (flags)
+ (reduce #'logior (mklist flags)
+ :key #'(lambda (flag)
+ (or
+ (second (assoc flag mappings))
+ (error 'type-error :datum flags
+ :expected-type (cons 'flags args))))))))
(defmethod from-alien-function ((type (eql 'flags)) &rest args)
(declare (ignore type))
- (let ((mappings (%map-enum args :int-flags)))
- #'(lambda (int)
+ (let ((mappings (%map-flags args :int-symbol)))
+ #'(lambda (value)
(loop
- for mapping in mappings
- unless (zerop (logand int (first mapping)))
- collect (second mapping)))))
+ for (int symbol) in mappings
+ when (= (logand value int) int)
+ collect symbol))))
(defmethod writer-function ((type (eql 'flags)) &rest args)
(declare (ignore type))
(funcall function (funcall reader location offset)))))
+;;;; Named flags types
+
+(defmacro define-flags-type (name &rest args)
+ (let ((flags-int (intern (format nil "~A-TO-INT" name)))
+ (int-flags (intern (format nil "INT-TO-~A" name)))
+ (satisfies (intern (format nil "~A-P" name))))
+ `(progn
+ (deftype ,name () '(satisfies ,satisfies))
+ (defun ,satisfies (object)
+ (flet ((valid-p (ob)
+ (find ob ',(%map-flags args :symbols))))
+ (typecase object
+ (symbol (valid-p object))
+ (list (every #'valid-p object)))))
+ (defun ,flags-int (flags)
+ (reduce #'logior (mklist flags)
+ :key #'(lambda (flag)
+ (case flag
+ ,@(%map-flags args :symbol-int)
+ (t (error 'type-error :datum flags
+ :expected-type ',name))))))
+ (defun ,int-flags (value)
+ (loop
+ for (int symbol) in ',(%map-flags args :int-symbol)
+ when(= (logand value int) int)
+ collect symbol))
+ (defmethod alien-type ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (alien-type 'flags))
+ (defmethod size-of ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (size-of 'flags))
+ (defmethod to-alien-form (form (type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (list ',flags-int form))
+ (defmethod from-alien-form (form (type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (list ',int-flags form))
+ (defmethod to-alien-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ #',flags-int)
+ (defmethod from-alien-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ #',int-flags)
+ (defmethod writer-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (let ((writer (writer-function 'signed)))
+ #'(lambda (flags location &optional (offset 0))
+ (funcall writer (,flags-int flags) location offset))))
+ (defmethod reader-function ((type (eql ',name)) &rest args)
+ (declare (ignore type args))
+ (let ((reader (reader-function 'signed)))
+ #'(lambda (location &optional (offset 0))
+ (,int-flags (funcall reader location offset))))))))
+
+
+
+;;;; Type definition by introspection
+
+(defun %query-enum-or-flags-values (query-function class type)
+ (multiple-value-bind (sap length)
+ (funcall query-function (type-class-ref type))
+ (let ((values nil)
+ (size (proxy-instance-size (find-class class)))
+ (proxy (make-instance class :location sap)))
+ (dotimes (i length)
+ (with-slots (location nickname value) proxy
+ (setf location sap)
+ (setq sap (sap+ sap size))
+ (push
+ (list
+ (intern (substitute #\- #\_ (string-upcase nickname)) "KEYWORD")
+ value)
+ values)))
+ values)))
+
+
+(defclass %enum-value (struct)
+ ((value :allocation :alien :type int)
+ (name :allocation :alien :type string)
+ (nickname :allocation :alien :type string))
+ (:metaclass static-struct-class))
+
+(defbinding %enum-class-values () pointer
+ (class pointer)
+ (n-values unsigned-int :out))
+
+(defun query-enum-values (type)
+ (%query-enum-or-flags-values #'%enum-class-values '%enum-value type))
+
(defclass %flags-value (struct)
((value :allocation :alien :type unsigned-int)
(%query-enum-or-flags-values #'%flags-class-values '%flags-value type))
-
-;;;;
-
(defun expand-enum-type (type-number forward-p options)
- (declare (ignore forward))
+ (declare (ignore forward-p))
(let* ((super (supertype type-number))
(type (type-from-number type-number))
(mappings (getf options :mappings))
(remove-if
#'(lambda (mapping) (eq (second mapping) nil)) mappings))))
`(progn
- (register-type ',type ,(find-type-name type-number))
- (deftype ,type () '(,super ,@expanded-mappings)))))
+ (register-type ',type ',(find-type-init-function type-number))
+ ,(ecase super
+ (enum `(define-enum-type ,type ,@expanded-mappings))
+ (flags `(define-flags-type ,type ,@expanded-mappings))))))
(register-derivable-type 'enum "GEnum" 'expand-enum-type)