5 ;; to do -- get these from alexandria or kmcrl or somesuch
6 (defmacro with-gensyms
((&rest names
) &body body
)
7 `(let ,(loop for n in names collect
`(,n
(gensym)))
10 (defmacro once-only
((&rest names
) &body body
)
11 (let ((gensyms (loop for n in names collect
(gensym))))
12 `(let (,@(loop for g in gensyms collect
`(,g
(gensym))))
13 `(let (,,@(loop for g in gensyms for n in names collect
``(,,g
,,n
)))
14 ,(let (,@(loop for n in names for g in gensyms collect
`(,n
,g
)))
17 (defmacro with-opengl
(&body forms
)
18 (with-gensyms (error-sym)
20 (let ((,error-sym
(gl:get-error
)))
22 (error "OpenGL Error ~A~%"
24 (gl:+INVALID-ENUM
+ "Invalid Enum")
25 (gl:+INVALID-VALUE
+ "Invalid value")
26 (gl:+INVALID-OPERATION
+ "Invalid Operation")
27 (gl:+OUT-OF-MEMORY
+ "Out of memory")
28 (gl:+STACK-OVERFLOW
+ "Stack overflow")
29 (gl:+STACK-UNDERFLOW
+ "Stack underflow"))))))))
31 (defclass rgba-image
()
32 ((name :accessor name-of
)
33 (width :accessor width-of
:initform
0)
34 (height :accessor height-of
:initform
0)
35 (format :reader format-of
:initform gl
:+rgba
+)
36 (bpp :reader bpp-of
:initform
4)
37 (data :accessor data-of
)
38 (size :accessor size-of
))
39 (:documentation
"Data for an opengl RGBA texture"))
41 (defmethod make-image ((self rgba-image
) &key width height
)
42 "Create a sized rgba texture"
43 (setf (width-of self
) width
)
44 (setf (height-of self
) height
)
45 (setf (slot-value self
'name
) (cffi:foreign-alloc
:uint32
))
47 (gl:gen-textures
1 (name-of self
))
48 (gl:bind-texture gl
:+texture-2d
+ (cffi::mem-ref
(name-of self
) :uint32
))
49 (gl:tex-parameter-i gl
:+texture-2d
+ gl
:+texture-wrap-s
+ gl
:+repeat
+)
50 (gl:tex-parameter-i gl
:+texture-2d
+ gl
:+texture-wrap-t
+ gl
:+repeat
+)
51 (gl:tex-parameter-i gl
:+texture-2d
+ gl
:+texture-mag-filter
+ gl
:+linear
+)
52 (gl:tex-parameter-i gl
:+texture-2d
+ gl
:+texture-min-filter
+ gl
:+linear
+)
53 (gl:pixel-store-i gl
:+unpack-alignment
+ 1)
54 (gl:tex-env-f gl
:+texture-env
+ gl
:+texture-env-mode
+ gl
:+decal
+))
55 (setf (slot-value self
'data
)
56 (cffi:foreign-alloc
:uint32
57 :count
(* (width-of self
) (height-of self
))
60 (defmethod update-image ((self rgba-image
))
61 "Upload an RGBA texture"
63 (gl:bind-texture gl
:+texture-2d
+ (cffi::mem-ref
(name-of self
) :uint32
))
64 (gl:tex-image-2d gl
:+texture-2d
+ 0 gl
:+rgba
+
65 (width-of self
) (height-of self
)
66 0 gl
:+rgba
+ gl
:+unsigned-byte
+ (data-of self
))))
69 (defgeneric render
(self &key target
))
71 (defmethod render ((self rgba-image
) &key target
)
72 "Render an RGBA texture"
73 (declare (ignore target
))
75 (gl:bind-texture gl
:+texture-2d
+ (cffi::mem-ref
(name-of self
) :uint32
))
76 (gl:tex-env-f gl
:+texture-env
+ gl
:+texture-env-mode
+ gl
:+decal
+)
77 (gl:with-begin gl
:+quads
+
79 (gl:vertex-2f -
0.5 -
0.5) ;; top lhs
81 (gl:vertex-2f
0.5 -
0.5) ;; top rhs
83 (gl:vertex-2f
0.5 0.5) ;; bot rhs
85 (gl:vertex-2f -
0.5 0.5)))) ;; bot lhs
87 (defmethod destroy-image ((self rgba-image
))
88 "Release the memory used by an RGBA texture"
89 (setf (width-of self
) 0)
90 (setf (height-of self
) 0)
92 (gl:delete-textures
1 (name-of self
))
93 (cffi:foreign-free
(name-of self
))
94 (cffi:foreign-free
(data-of self
))))
97 (defmethod image-size ((image rgba-image
))
98 "Overall RGBA texture size in bytes"
99 (* (width-of image
) (height-of image
)))
101 (defmethod pixel ((image rgba-image
) i
)
102 "Access a pixel in an RGBA texture"
103 (cffi:mem-aref
(data-of image
) :uint32 i
))
105 (defmethod (setf pixel
) (pixval (image rgba-image
) i
)
106 "Set a pixel in an RGBA texture"
107 (setf (cffi:mem-aref
(data-of image
) :uint32 i
) pixval
))
109 (defmethod indexxy ((image rgba-image
) index
)
110 "Map an i index to an x,y index of a RGBA texure"
111 (values (mod index
(width-of image
))
112 (rem index
(width-of image
))))
114 (defmethod xyindex ((image rgba-image
) x y
)
115 "Map an x,y index to an i index of a RGBA texure"
116 (the (unsigned-byte 32) (+ x
(* (width-of image
) y
))))
118 (defmethod pixel-xy ((image rgba-image
) x y
)
119 "Get a pixel in an image using x y oords"
120 (pixel image
(xyindex image x y
)))
122 (defmethod (setf pixelxy
) (pixval (image rgba-image
) x y
)
123 "Set a pixel in an image using x y oords"
124 (setf (cffi:mem-aref
(data-of image
) :uint32
(xyindex image x y
)) pixval
))
126 (defun pixval (r g b
&optional
(a 0))
127 "Convert rgb values to a pixel uint32"
128 (declare ((unsigned-byte 8) r g b a
))
129 (the (unsigned-byte 32)