blob: f19d1efab2d73105f40c18f019415c862058c2d6 [file] [log] [blame]
J. Duke319a3b92007-12-01 00:00:00 +00001/*
2 * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation. Sun designates this
8 * particular file as subject to the "Classpath" exception as provided
9 * by Sun in the LICENSE file that accompanied this code.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
22 * CA 95054 USA or visit www.sun.com if you need additional information or
23 * have any questions.
24 */
25
26package sun.java2d.opengl;
27
28import java.awt.Graphics;
29import java.awt.GraphicsConfiguration;
30import java.awt.Rectangle;
31import sun.java2d.SunGraphics2D;
32import sun.java2d.SurfaceData;
33import sun.java2d.pipe.Region;
34
35/**
36 * This class contains a number of static utility methods that may be
37 * called (via reflection) by a third-party library, such as JOGL, in order
38 * to interoperate with the OGL-based Java 2D pipeline.
39 *
40 * WARNING: These methods are being made available as a temporary measure
41 * until we offer a more complete, public solution. Like any sun.* class,
42 * this class is not an officially supported public API; it may be modified
43 * at will or removed completely in a future release.
44 */
45class OGLUtilities {
46
47 /**
48 * These OGL-specific surface type constants are the same as those
49 * defined in the OGLSurfaceData class and are duplicated here so that
50 * clients of this API can access them more easily via reflection.
51 */
52 public static final int UNDEFINED = OGLSurfaceData.UNDEFINED;
53 public static final int WINDOW = OGLSurfaceData.WINDOW;
54 public static final int PBUFFER = OGLSurfaceData.PBUFFER;
55 public static final int TEXTURE = OGLSurfaceData.TEXTURE;
56 public static final int FLIP_BACKBUFFER = OGLSurfaceData.FLIP_BACKBUFFER;
57 public static final int FBOBJECT = OGLSurfaceData.FBOBJECT;
58
59 private OGLUtilities() {
60 }
61
62 /**
63 * Returns true if the current thread is the OGL QueueFlusher thread.
64 */
65 public static boolean isQueueFlusherThread() {
66 return OGLRenderQueue.isQueueFlusherThread();
67 }
68
69 /**
70 * Invokes the given Runnable on the OGL QueueFlusher thread with the
71 * OpenGL context corresponding to the given Graphics object made
72 * current. It is legal for OpenGL code executed in the given
73 * Runnable to change the current OpenGL context; it will be reset
74 * once the Runnable completes. No guarantees are made as to the
75 * state of the OpenGL context of the Graphics object; for
76 * example, calling code must set the scissor box using the return
77 * value from {@link #getOGLScissorBox} to avoid drawing
78 * over other Swing components, and must typically set the OpenGL
79 * viewport using the return value from {@link #getOGLViewport} to
80 * make the client's OpenGL rendering appear in the correct place
81 * relative to the scissor region.
82 *
83 * In order to avoid deadlock, it is important that the given Runnable
84 * does not attempt to acquire the AWT lock, as that will be handled
85 * automatically as part of the <code>rq.flushAndInvokeNow()</code> step.
86 *
87 * @param g the Graphics object for the corresponding destination surface;
88 * if null, the step making a context current to the destination surface
89 * will be skipped
90 * @param r the action to be performed on the QFT; cannot be null
91 * @return true if the operation completed successfully, or false if
92 * there was any problem making a context current to the surface
93 * associated with the given Graphics object
94 */
95 public static boolean invokeWithOGLContextCurrent(Graphics g, Runnable r) {
96 OGLRenderQueue rq = OGLRenderQueue.getInstance();
97 rq.lock();
98 try {
99 if (g != null) {
100 if (!(g instanceof SunGraphics2D)) {
101 return false;
102 }
103 SurfaceData sData = ((SunGraphics2D)g).surfaceData;
104 if (!(sData instanceof OGLSurfaceData)) {
105 return false;
106 }
107
108 // make a context current to the destination surface
109 OGLContext.validateContext((OGLSurfaceData)sData);
110 }
111
112 // invoke the given runnable on the QFT
113 rq.flushAndInvokeNow(r);
114
115 // invalidate the current context so that the next time we render
116 // with Java 2D, the context state will be completely revalidated
117 OGLContext.invalidateCurrentContext();
118 } finally {
119 rq.unlock();
120 }
121
122 return true;
123 }
124
125 /**
126 * Invokes the given Runnable on the OGL QueueFlusher thread with the
127 * "shared" OpenGL context (corresponding to the given
128 * GraphicsConfiguration object) made current. This method is typically
129 * used when the Runnable needs a current context to complete its
130 * operation, but does not require that the context be made current to
131 * a particular surface. For example, an application may call this
132 * method so that the given Runnable can query the OpenGL capabilities
133 * of the given GraphicsConfiguration, without making a context current
134 * to a dummy surface (or similar hacky techniques).
135 *
136 * In order to avoid deadlock, it is important that the given Runnable
137 * does not attempt to acquire the AWT lock, as that will be handled
138 * automatically as part of the <code>rq.flushAndInvokeNow()</code> step.
139 *
140 * @param config the GraphicsConfiguration object whose "shared"
141 * context will be made current during this operation; if this value is
142 * null or if OpenGL is not enabled for the GraphicsConfiguration, this
143 * method will return false
144 * @param r the action to be performed on the QFT; cannot be null
145 * @return true if the operation completed successfully, or false if
146 * there was any problem making the shared context current
147 */
148 public static boolean
149 invokeWithOGLSharedContextCurrent(GraphicsConfiguration config,
150 Runnable r)
151 {
152 if (!(config instanceof OGLGraphicsConfig)) {
153 return false;
154 }
155
156 OGLRenderQueue rq = OGLRenderQueue.getInstance();
157 rq.lock();
158 try {
159 // make the "shared" context current for the given GraphicsConfig
160 OGLContext.setScratchSurface((OGLGraphicsConfig)config);
161
162 // invoke the given runnable on the QFT
163 rq.flushAndInvokeNow(r);
164
165 // invalidate the current context so that the next time we render
166 // with Java 2D, the context state will be completely revalidated
167 OGLContext.invalidateCurrentContext();
168 } finally {
169 rq.unlock();
170 }
171
172 return true;
173 }
174
175 /**
176 * Returns the Rectangle describing the OpenGL viewport on the
177 * Java 2D surface associated with the given Graphics object and
178 * component width and height. When a third-party library is
179 * performing OpenGL rendering directly into the visible region of
180 * the associated surface, this viewport helps the application
181 * position the OpenGL output correctly on that surface.
182 *
183 * Note that the x/y values in the returned Rectangle object represent
184 * the lower-left corner of the viewport region, relative to the
185 * lower-left corner of the given surface.
186 *
187 * @param g the Graphics object for the corresponding destination surface;
188 * cannot be null
189 * @param componentWidth width of the component to be painted
190 * @param componentHeight height of the component to be painted
191 * @return a Rectangle describing the OpenGL viewport for the given
192 * destination surface and component dimensions, or null if the given
193 * Graphics object is invalid
194 */
195 public static Rectangle getOGLViewport(Graphics g,
196 int componentWidth,
197 int componentHeight)
198 {
199 if (!(g instanceof SunGraphics2D)) {
200 return null;
201 }
202
203 SunGraphics2D sg2d = (SunGraphics2D)g;
204 SurfaceData sData = (SurfaceData)sg2d.surfaceData;
205
206 // this is the upper-left origin of the region to be painted,
207 // relative to the upper-left origin of the surface
208 // (in Java2D coordinates)
209 int x0 = sg2d.transX;
210 int y0 = sg2d.transY;
211
212 // this is the lower-left origin of the region to be painted,
213 // relative to the lower-left origin of the surface
214 // (in OpenGL coordinates)
215 Rectangle surfaceBounds = sData.getBounds();
216 int x1 = x0;
217 int y1 = surfaceBounds.height - (y0 + componentHeight);
218
219 return new Rectangle(x1, y1, componentWidth, componentHeight);
220 }
221
222 /**
223 * Returns the Rectangle describing the OpenGL scissor box on the
224 * Java 2D surface associated with the given Graphics object. When a
225 * third-party library is performing OpenGL rendering directly
226 * into the visible region of the associated surface, this scissor box
227 * must be set to avoid drawing over existing rendering results.
228 *
229 * Note that the x/y values in the returned Rectangle object represent
230 * the lower-left corner of the scissor region, relative to the
231 * lower-left corner of the given surface.
232 *
233 * @param g the Graphics object for the corresponding destination surface;
234 * cannot be null
235 * @return a Rectangle describing the OpenGL scissor box for the given
236 * Graphics object and corresponding destination surface, or null if the
237 * given Graphics object is invalid or the clip region is non-rectangular
238 */
239 public static Rectangle getOGLScissorBox(Graphics g) {
240 if (!(g instanceof SunGraphics2D)) {
241 return null;
242 }
243
244 SunGraphics2D sg2d = (SunGraphics2D)g;
245 SurfaceData sData = (SurfaceData)sg2d.surfaceData;
246 Region r = sg2d.getCompClip();
247 if (!r.isRectangular()) {
248 // caller probably doesn't know how to handle shape clip
249 // appropriately, so just return null (Swing currently never
250 // sets a shape clip, but that could change in the future)
251 return null;
252 }
253
254 // this is the upper-left origin of the scissor box relative to the
255 // upper-left origin of the surface (in Java 2D coordinates)
256 int x0 = r.getLoX();
257 int y0 = r.getLoY();
258
259 // this is the width and height of the scissor region
260 int w = r.getWidth();
261 int h = r.getHeight();
262
263 // this is the lower-left origin of the scissor box relative to the
264 // lower-left origin of the surface (in OpenGL coordinates)
265 Rectangle surfaceBounds = sData.getBounds();
266 int x1 = x0;
267 int y1 = surfaceBounds.height - (y0 + h);
268
269 return new Rectangle(x1, y1, w, h);
270 }
271
272 /**
273 * Returns an Object identifier for the Java 2D surface associated with
274 * the given Graphics object. This identifier may be used to determine
275 * whether the surface has changed since the last invocation of this
276 * operation, and thereby whether the OpenGL state corresponding to the
277 * old surface must be destroyed and recreated.
278 *
279 * @param g the Graphics object for the corresponding destination surface;
280 * cannot be null
281 * @return an identifier for the surface associated with the given
282 * Graphics object, or null if the given Graphics object is invalid
283 */
284 public static Object getOGLSurfaceIdentifier(Graphics g) {
285 if (!(g instanceof SunGraphics2D)) {
286 return null;
287 }
288 return ((SunGraphics2D)g).surfaceData;
289 }
290
291 /**
292 * Returns one of the OGL-specific surface type constants (defined in
293 * this class), which describes the surface associated with the given
294 * Graphics object.
295 *
296 * @param g the Graphics object for the corresponding destination surface;
297 * cannot be null
298 * @return a constant that describes the surface associated with the
299 * given Graphics object; if the given Graphics object is invalid (i.e.
300 * is not associated with an OpenGL surface) this method will return
301 * <code>OGLUtilities.UNDEFINED</code>
302 */
303 public static int getOGLSurfaceType(Graphics g) {
304 if (!(g instanceof SunGraphics2D)) {
305 return UNDEFINED;
306 }
307 SurfaceData sData = ((SunGraphics2D)g).surfaceData;
308 if (!(sData instanceof OGLSurfaceData)) {
309 return UNDEFINED;
310 }
311 return ((OGLSurfaceData)sData).getType();
312 }
313
314 /**
315 * Returns the OpenGL texture target constant (either GL_TEXTURE_2D
316 * or GL_TEXTURE_RECTANGLE_ARB) for the surface associated with the
317 * given Graphics object. This method is only useful for those surface
318 * types that are backed by an OpenGL texture, namely {@code TEXTURE},
319 * {@code FBOBJECT}, and (on Windows only) {@code PBUFFER}.
320 *
321 * @param g the Graphics object for the corresponding destination surface;
322 * cannot be null
323 * @return the texture target constant for the surface associated with the
324 * given Graphics object; if the given Graphics object is invalid (i.e.
325 * is not associated with an OpenGL surface), or the associated surface
326 * is not backed by an OpenGL texture, this method will return zero.
327 */
328 public static int getOGLTextureType(Graphics g) {
329 if (!(g instanceof SunGraphics2D)) {
330 return 0;
331 }
332 SurfaceData sData = ((SunGraphics2D)g).surfaceData;
333 if (!(sData instanceof OGLSurfaceData)) {
334 return 0;
335 }
336 return ((OGLSurfaceData)sData).getTextureTarget();
337 }
338}