Skip to content
This repository has been archived by the owner on Feb 11, 2021. It is now read-only.

Latest commit

 

History

History
826 lines (708 loc) · 25.1 KB

SDL_render_gl.c

File metadata and controls

826 lines (708 loc) · 25.1 KB
 
Jul 19, 2006
Jul 19, 2006
1
2
/*
SDL - Simple DirectMedia Layer
Jan 24, 2010
Jan 24, 2010
3
Copyright (C) 1997-2010 Sam Lantinga
Jul 19, 2006
Jul 19, 2006
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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.1 of the License, or (at your option) any later version.
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.
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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
Sam Lantinga
slouken@libsdl.org
*/
#include "SDL_config.h"
Jul 28, 2006
Jul 28, 2006
24
#if SDL_VIDEO_RENDER_OGL
Jul 22, 2006
Jul 22, 2006
25
26
#include "SDL_opengl.h"
Feb 2, 2011
Feb 2, 2011
27
#include "../SDL_sysrender.h"
Jul 19, 2006
Jul 19, 2006
28
Aug 15, 2007
Aug 15, 2007
29
30
31
32
#ifdef __MACOSX__
#include <OpenGL/OpenGL.h>
#endif
Nov 22, 2008
Nov 22, 2008
33
Jul 19, 2006
Jul 19, 2006
34
35
/* OpenGL renderer implementation */
Aug 12, 2007
Aug 12, 2007
36
37
38
39
/* Details on optimizing the texture path on Mac OS X:
http://developer.apple.com/documentation/GraphicsImaging/Conceptual/OpenGL-MacProgGuide/opengl_texturedata/chapter_10_section_2.html
*/
Feb 2, 2011
Feb 2, 2011
40
41
/* Used to re-create the window with OpenGL capability */
extern int SDL_RecreateWindow(SDL_Window * window, Uint32 flags);
Dec 6, 2008
Dec 6, 2008
42
Aug 28, 2006
Aug 28, 2006
43
44
static const float inv255f = 1.0f / 255.0f;
Jul 19, 2006
Jul 19, 2006
45
static SDL_Renderer *GL_CreateRenderer(SDL_Window * window, Uint32 flags);
Feb 2, 2011
Feb 2, 2011
46
47
static void GL_WindowEvent(SDL_Renderer * renderer,
const SDL_WindowEvent *event);
Jul 19, 2006
Jul 19, 2006
48
49
50
51
52
static int GL_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture);
static int GL_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, const void *pixels,
int pitch);
static int GL_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
Feb 3, 2011
Feb 3, 2011
53
const SDL_Rect * rect, void **pixels, int *pitch);
Jul 19, 2006
Jul 19, 2006
54
static void GL_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture);
Dec 23, 2009
Dec 23, 2009
55
56
57
58
59
60
61
static int GL_RenderClear(SDL_Renderer * renderer);
static int GL_RenderDrawPoints(SDL_Renderer * renderer,
const SDL_Point * points, int count);
static int GL_RenderDrawLines(SDL_Renderer * renderer,
const SDL_Point * points, int count);
static int GL_RenderFillRects(SDL_Renderer * renderer,
const SDL_Rect ** rects, int count);
Jul 19, 2006
Jul 19, 2006
62
static int GL_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
Aug 28, 2006
Aug 28, 2006
63
const SDL_Rect * srcrect, const SDL_Rect * dstrect);
Nov 15, 2009
Nov 15, 2009
64
static int GL_RenderReadPixels(SDL_Renderer * renderer, const SDL_Rect * rect,
Nov 16, 2009
Nov 16, 2009
65
Uint32 pixel_format, void * pixels, int pitch);
Jul 19, 2006
Jul 19, 2006
66
67
68
69
70
71
72
73
74
static void GL_RenderPresent(SDL_Renderer * renderer);
static void GL_DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture);
static void GL_DestroyRenderer(SDL_Renderer * renderer);
SDL_RenderDriver GL_RenderDriver = {
GL_CreateRenderer,
{
"opengl",
Feb 1, 2011
Feb 1, 2011
75
(SDL_RENDERER_PRESENTVSYNC | SDL_RENDERER_ACCELERATED),
Feb 3, 2011
Feb 3, 2011
76
77
1,
{SDL_PIXELFORMAT_ARGB8888},
Jul 19, 2006
Jul 19, 2006
78
79
80
81
82
83
84
0,
0}
};
typedef struct
{
SDL_GLContext context;
Dec 5, 2008
Dec 5, 2008
85
SDL_bool updateSize;
Jul 22, 2006
Jul 22, 2006
86
SDL_bool GL_ARB_texture_rectangle_supported;
Jul 22, 2006
Jul 22, 2006
87
88
89
90
int blendMode;
/* OpenGL functions */
#define SDL_PROC(ret,func,params) ret (APIENTRY *func) params;
Feb 2, 2011
Feb 2, 2011
91
#include "../../video/SDL_glfuncs.h"
Jul 22, 2006
Jul 22, 2006
92
#undef SDL_PROC
Aug 6, 2006
Aug 6, 2006
93
Aug 12, 2007
Aug 12, 2007
94
95
void (*glTextureRangeAPPLE) (GLenum target, GLsizei length,
const GLvoid * pointer);
Jul 19, 2006
Jul 19, 2006
96
97
98
99
100
} GL_RenderData;
typedef struct
{
GLuint texture;
Jul 22, 2006
Jul 22, 2006
101
GLenum type;
Jul 19, 2006
Jul 19, 2006
102
103
GLfloat texw;
GLfloat texh;
Jul 22, 2006
Jul 22, 2006
104
105
GLenum format;
GLenum formattype;
Jul 19, 2006
Jul 19, 2006
106
107
108
109
110
void *pixels;
int pitch;
} GL_TextureData;
Jul 22, 2006
Jul 22, 2006
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
static void
GL_SetError(const char *prefix, GLenum result)
{
const char *error;
switch (result) {
case GL_NO_ERROR:
error = "GL_NO_ERROR";
break;
case GL_INVALID_ENUM:
error = "GL_INVALID_ENUM";
break;
case GL_INVALID_VALUE:
error = "GL_INVALID_VALUE";
break;
case GL_INVALID_OPERATION:
error = "GL_INVALID_OPERATION";
break;
case GL_STACK_OVERFLOW:
error = "GL_STACK_OVERFLOW";
break;
case GL_STACK_UNDERFLOW:
error = "GL_STACK_UNDERFLOW";
break;
case GL_OUT_OF_MEMORY:
error = "GL_OUT_OF_MEMORY";
break;
case GL_TABLE_TOO_LARGE:
error = "GL_TABLE_TOO_LARGE";
break;
default:
error = "UNKNOWN";
break;
}
SDL_SetError("%s: %s", prefix, error);
}
Jul 22, 2006
Jul 22, 2006
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
static int
GL_LoadFunctions(GL_RenderData * data)
{
#ifdef __SDL_NOGETPROCADDR__
#define SDL_PROC(ret,func,params) data->func=func;
#else
#define SDL_PROC(ret,func,params) \
do { \
data->func = SDL_GL_GetProcAddress(#func); \
if ( ! data->func ) { \
SDL_SetError("Couldn't load GL function %s: %s\n", #func, SDL_GetError()); \
return -1; \
} \
} while ( 0 );
#endif /* __SDL_NOGETPROCADDR__ */
Feb 2, 2011
Feb 2, 2011
164
#include "../../video/SDL_glfuncs.h"
Jul 22, 2006
Jul 22, 2006
165
166
167
168
#undef SDL_PROC
return 0;
}
Jul 19, 2006
Jul 19, 2006
169
170
171
172
173
SDL_Renderer *
GL_CreateRenderer(SDL_Window * window, Uint32 flags)
{
SDL_Renderer *renderer;
GL_RenderData *data;
Jul 28, 2006
Jul 28, 2006
174
GLint value;
Feb 2, 2011
Feb 2, 2011
175
Uint32 window_flags;
Jul 19, 2006
Jul 19, 2006
176
Feb 2, 2011
Feb 2, 2011
177
178
179
window_flags = SDL_GetWindowFlags(window);
if (!(window_flags & SDL_WINDOW_OPENGL)) {
if (SDL_RecreateWindow(window, window_flags | SDL_WINDOW_OPENGL) < 0) {
Jul 22, 2006
Jul 22, 2006
180
181
return NULL;
}
Jul 19, 2006
Jul 19, 2006
182
183
}
Jul 22, 2006
Jul 22, 2006
184
renderer = (SDL_Renderer *) SDL_calloc(1, sizeof(*renderer));
Jul 19, 2006
Jul 19, 2006
185
186
187
188
189
if (!renderer) {
SDL_OutOfMemory();
return NULL;
}
Jul 22, 2006
Jul 22, 2006
190
data = (GL_RenderData *) SDL_calloc(1, sizeof(*data));
Jul 19, 2006
Jul 19, 2006
191
192
193
194
195
196
if (!data) {
GL_DestroyRenderer(renderer);
SDL_OutOfMemory();
return NULL;
}
Feb 2, 2011
Feb 2, 2011
197
renderer->WindowEvent = GL_WindowEvent;
Jul 19, 2006
Jul 19, 2006
198
199
200
201
renderer->CreateTexture = GL_CreateTexture;
renderer->UpdateTexture = GL_UpdateTexture;
renderer->LockTexture = GL_LockTexture;
renderer->UnlockTexture = GL_UnlockTexture;
Dec 23, 2009
Dec 23, 2009
202
203
204
205
renderer->RenderClear = GL_RenderClear;
renderer->RenderDrawPoints = GL_RenderDrawPoints;
renderer->RenderDrawLines = GL_RenderDrawLines;
renderer->RenderFillRects = GL_RenderFillRects;
Jul 19, 2006
Jul 19, 2006
206
renderer->RenderCopy = GL_RenderCopy;
Nov 15, 2009
Nov 15, 2009
207
renderer->RenderReadPixels = GL_RenderReadPixels;
Jul 19, 2006
Jul 19, 2006
208
209
210
211
212
213
renderer->RenderPresent = GL_RenderPresent;
renderer->DestroyTexture = GL_DestroyTexture;
renderer->DestroyRenderer = GL_DestroyRenderer;
renderer->info = GL_RenderDriver.info;
renderer->driverdata = data;
Feb 1, 2011
Feb 1, 2011
214
renderer->info.flags = SDL_RENDERER_ACCELERATED;
Jul 19, 2006
Jul 19, 2006
215
Jul 22, 2006
Jul 22, 2006
216
217
218
219
220
if (GL_LoadFunctions(data) < 0) {
GL_DestroyRenderer(renderer);
return NULL;
}
Jan 21, 2010
Jan 21, 2010
221
data->context = SDL_GL_CreateContext(window);
Jul 19, 2006
Jul 19, 2006
222
223
224
225
if (!data->context) {
GL_DestroyRenderer(renderer);
return NULL;
}
Jan 21, 2010
Jan 21, 2010
226
if (SDL_GL_MakeCurrent(window, data->context) < 0) {
Jul 19, 2006
Jul 19, 2006
227
228
229
GL_DestroyRenderer(renderer);
return NULL;
}
Aug 15, 2007
Aug 15, 2007
230
231
232
#ifdef __MACOSX__
/* Enable multi-threaded rendering */
/* Disabled until Ryan finishes his VBO/PBO code...
Jan 8, 2008
Jan 8, 2008
233
234
CGLEnable(CGLGetCurrentContext(), kCGLCEMPEngine);
*/
Aug 15, 2007
Aug 15, 2007
235
236
#endif
Aug 5, 2006
Aug 5, 2006
237
if (flags & SDL_RENDERER_PRESENTVSYNC) {
Jul 19, 2006
Jul 19, 2006
238
239
240
241
242
SDL_GL_SetSwapInterval(1);
} else {
SDL_GL_SetSwapInterval(0);
}
if (SDL_GL_GetSwapInterval() > 0) {
Aug 5, 2006
Aug 5, 2006
243
renderer->info.flags |= SDL_RENDERER_PRESENTVSYNC;
Jul 19, 2006
Jul 19, 2006
244
245
}
Jul 28, 2006
Jul 28, 2006
246
247
248
249
data->glGetIntegerv(GL_MAX_TEXTURE_SIZE, &value);
renderer->info.max_texture_width = value;
data->glGetIntegerv(GL_MAX_TEXTURE_SIZE, &value);
renderer->info.max_texture_height = value;
Jul 22, 2006
Jul 22, 2006
250
Jul 22, 2006
Jul 22, 2006
251
252
253
254
if (SDL_GL_ExtensionSupported("GL_ARB_texture_rectangle")
|| SDL_GL_ExtensionSupported("GL_EXT_texture_rectangle")) {
data->GL_ARB_texture_rectangle_supported = SDL_TRUE;
}
Aug 12, 2007
Aug 12, 2007
255
256
257
258
259
if (SDL_GL_ExtensionSupported("GL_APPLE_texture_range")) {
data->glTextureRangeAPPLE =
(void (*)(GLenum, GLsizei, const GLvoid *))
SDL_GL_GetProcAddress("glTextureRangeAPPLE");
}
Jul 22, 2006
Jul 22, 2006
260
Jul 19, 2006
Jul 19, 2006
261
/* Set up parameters for rendering */
Jul 22, 2006
Jul 22, 2006
262
263
264
data->blendMode = -1;
data->glDisable(GL_DEPTH_TEST);
data->glDisable(GL_CULL_FACE);
Sep 19, 2009
Sep 19, 2009
265
266
/* This ended up causing video discrepancies between OpenGL and Direct3D */
/*data->glEnable(GL_LINE_SMOOTH);*/
Jul 22, 2006
Jul 22, 2006
267
if (data->GL_ARB_texture_rectangle_supported) {
Jul 22, 2006
Jul 22, 2006
268
data->glEnable(GL_TEXTURE_RECTANGLE_ARB);
Jul 22, 2006
Jul 22, 2006
269
} else {
Jul 22, 2006
Jul 22, 2006
270
data->glEnable(GL_TEXTURE_2D);
Jul 22, 2006
Jul 22, 2006
271
}
Dec 5, 2008
Dec 5, 2008
272
data->updateSize = SDL_TRUE;
Jul 19, 2006
Jul 19, 2006
273
274
275
276
return renderer;
}
Feb 2, 2011
Feb 2, 2011
277
278
static SDL_GLContext SDL_CurrentContext = NULL;
Jul 22, 2006
Jul 22, 2006
279
280
281
282
static int
GL_ActivateRenderer(SDL_Renderer * renderer)
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
Jan 21, 2010
Jan 21, 2010
283
SDL_Window *window = renderer->window;
Jul 22, 2006
Jul 22, 2006
284
Feb 2, 2011
Feb 2, 2011
285
286
287
288
289
if (SDL_CurrentContext != data->context) {
if (SDL_GL_MakeCurrent(window, data->context) < 0) {
return -1;
}
SDL_CurrentContext = data->context;
Aug 6, 2006
Aug 6, 2006
290
}
Dec 5, 2008
Dec 5, 2008
291
if (data->updateSize) {
Feb 2, 2011
Feb 2, 2011
292
293
294
int w, h;
SDL_GetWindowSize(window, &w, &h);
Dec 6, 2008
Dec 6, 2008
295
296
297
298
data->glMatrixMode(GL_PROJECTION);
data->glLoadIdentity();
data->glMatrixMode(GL_MODELVIEW);
data->glLoadIdentity();
Feb 2, 2011
Feb 2, 2011
299
300
data->glViewport(0, 0, w, h);
data->glOrtho(0.0, (GLdouble) w, (GLdouble) h, 0.0, 0.0, 1.0);
Dec 5, 2008
Dec 5, 2008
301
302
data->updateSize = SDL_FALSE;
}
Aug 6, 2006
Aug 6, 2006
303
304
305
return 0;
}
Feb 2, 2011
Feb 2, 2011
306
307
static void
GL_WindowEvent(SDL_Renderer * renderer, const SDL_WindowEvent *event)
Aug 6, 2006
Aug 6, 2006
308
309
310
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
Feb 2, 2011
Feb 2, 2011
311
312
313
314
315
if (event->event == SDL_WINDOWEVENT_RESIZED) {
/* Rebind the context to the window area and update matrices */
SDL_CurrentContext = NULL;
data->updateSize = SDL_TRUE;
}
Jul 22, 2006
Jul 22, 2006
316
317
}
Jul 22, 2006
Jul 22, 2006
318
319
320
321
322
323
324
325
326
327
328
static __inline__ int
power_of_2(int input)
{
int value = 1;
while (value < input) {
value <<= 1;
}
return value;
}
Nov 15, 2009
Nov 15, 2009
329
330
331
static __inline__ SDL_bool
convert_format(GL_RenderData *renderdata, Uint32 pixel_format,
GLint* internalFormat, GLenum* format, GLenum* type)
Jul 19, 2006
Jul 19, 2006
332
{
Nov 15, 2009
Nov 15, 2009
333
switch (pixel_format) {
Aug 5, 2006
Aug 5, 2006
334
335
case SDL_PIXELFORMAT_RGB888:
case SDL_PIXELFORMAT_ARGB8888:
Nov 15, 2009
Nov 15, 2009
336
337
*internalFormat = GL_RGBA8;
*format = GL_BGRA;
Feb 3, 2011
Feb 3, 2011
338
*type = GL_UNSIGNED_INT_8_8_8_8_REV;
Dec 7, 2008
Dec 7, 2008
339
break;
Jul 22, 2006
Jul 22, 2006
340
default:
Nov 15, 2009
Nov 15, 2009
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
return SDL_FALSE;
}
return SDL_TRUE;
}
static int
GL_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture)
{
GL_RenderData *renderdata = (GL_RenderData *) renderer->driverdata;
GL_TextureData *data;
GLint internalFormat;
GLenum format, type;
int texture_w, texture_h;
GLenum result;
Feb 2, 2011
Feb 2, 2011
356
357
GL_ActivateRenderer(renderer);
Nov 15, 2009
Nov 15, 2009
358
359
if (!convert_format(renderdata, texture->format, &internalFormat,
&format, &type)) {
Jan 13, 2011
Jan 13, 2011
360
361
SDL_SetError("Texture format %s not supported by OpenGL",
SDL_GetPixelFormatName(texture->format));
Jul 22, 2006
Jul 22, 2006
362
363
return -1;
}
Jul 19, 2006
Jul 19, 2006
364
Jul 22, 2006
Jul 22, 2006
365
data = (GL_TextureData *) SDL_calloc(1, sizeof(*data));
Jul 19, 2006
Jul 19, 2006
366
367
368
369
370
if (!data) {
SDL_OutOfMemory();
return -1;
}
Aug 11, 2007
Aug 11, 2007
371
if (texture->access == SDL_TEXTUREACCESS_STREAMING) {
Feb 3, 2011
Feb 3, 2011
372
data->pitch = texture->w * SDL_BYTESPERPIXEL(texture->format);
Aug 11, 2007
Aug 11, 2007
373
374
375
376
377
378
379
380
data->pixels = SDL_malloc(texture->h * data->pitch);
if (!data->pixels) {
SDL_OutOfMemory();
SDL_free(data);
return -1;
}
}
Jul 19, 2006
Jul 19, 2006
381
382
texture->driverdata = data;
Jul 22, 2006
Jul 22, 2006
383
384
renderdata->glGetError();
renderdata->glGenTextures(1, &data->texture);
Jul 22, 2006
Jul 22, 2006
385
386
387
388
if (renderdata->GL_ARB_texture_rectangle_supported) {
data->type = GL_TEXTURE_RECTANGLE_ARB;
texture_w = texture->w;
texture_h = texture->h;
Dec 6, 2008
Dec 6, 2008
389
390
data->texw = (GLfloat) texture_w;
data->texh = (GLfloat) texture_h;
Jul 22, 2006
Jul 22, 2006
391
392
393
394
} else {
data->type = GL_TEXTURE_2D;
texture_w = power_of_2(texture->w);
texture_h = power_of_2(texture->h);
Dec 6, 2008
Dec 6, 2008
395
data->texw = (GLfloat) (texture->w) / texture_w;
Jul 22, 2006
Jul 22, 2006
396
397
data->texh = (GLfloat) texture->h / texture_h;
}
Dec 6, 2008
Dec 6, 2008
398
Jul 22, 2006
Jul 22, 2006
399
400
data->format = format;
data->formattype = type;
Dec 20, 2008
Dec 20, 2008
401
renderdata->glEnable(data->type);
Jul 22, 2006
Jul 22, 2006
402
renderdata->glBindTexture(data->type, data->texture);
Aug 12, 2007
Aug 12, 2007
403
renderdata->glTexParameteri(data->type, GL_TEXTURE_MIN_FILTER,
Feb 1, 2011
Feb 1, 2011
404
GL_LINEAR);
Aug 12, 2007
Aug 12, 2007
405
renderdata->glTexParameteri(data->type, GL_TEXTURE_MAG_FILTER,
Feb 1, 2011
Feb 1, 2011
406
GL_LINEAR);
Aug 12, 2007
Aug 12, 2007
407
408
409
410
renderdata->glTexParameteri(data->type, GL_TEXTURE_WRAP_S,
GL_CLAMP_TO_EDGE);
renderdata->glTexParameteri(data->type, GL_TEXTURE_WRAP_T,
GL_CLAMP_TO_EDGE);
Dec 6, 2008
Dec 6, 2008
411
#ifdef __MACOSX__
Aug 12, 2007
Aug 12, 2007
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
#ifndef GL_TEXTURE_STORAGE_HINT_APPLE
#define GL_TEXTURE_STORAGE_HINT_APPLE 0x85BC
#endif
#ifndef STORAGE_CACHED_APPLE
#define STORAGE_CACHED_APPLE 0x85BE
#endif
#ifndef STORAGE_SHARED_APPLE
#define STORAGE_SHARED_APPLE 0x85BF
#endif
if (texture->access == SDL_TEXTUREACCESS_STREAMING) {
renderdata->glTexParameteri(data->type, GL_TEXTURE_STORAGE_HINT_APPLE,
GL_STORAGE_SHARED_APPLE);
} else {
renderdata->glTexParameteri(data->type, GL_TEXTURE_STORAGE_HINT_APPLE,
GL_STORAGE_CACHED_APPLE);
}
Jan 8, 2008
Jan 8, 2008
428
429
if (texture->access == SDL_TEXTUREACCESS_STREAMING
&& texture->format == SDL_PIXELFORMAT_ARGB8888) {
Aug 12, 2007
Aug 12, 2007
430
431
432
renderdata->glPixelStorei(GL_UNPACK_CLIENT_STORAGE_APPLE, GL_TRUE);
renderdata->glTexImage2D(data->type, 0, internalFormat, texture_w,
texture_h, 0, format, type, data->pixels);
Feb 3, 2011
Feb 3, 2011
433
434
}
else
Aug 12, 2007
Aug 12, 2007
435
436
437
438
439
#endif
{
renderdata->glTexImage2D(data->type, 0, internalFormat, texture_w,
texture_h, 0, format, type, NULL);
}
Jan 14, 2009
Jan 14, 2009
440
renderdata->glDisable(data->type);
Jul 22, 2006
Jul 22, 2006
441
result = renderdata->glGetError();
Jul 22, 2006
Jul 22, 2006
442
443
444
445
if (result != GL_NO_ERROR) {
GL_SetError("glTexImage2D()", result);
return -1;
}
Jul 19, 2006
Jul 19, 2006
446
447
448
return 0;
}
Jul 22, 2006
Jul 22, 2006
449
static void
Jul 22, 2006
Jul 22, 2006
450
451
SetupTextureUpdate(GL_RenderData * renderdata, SDL_Texture * texture,
int pitch)
Jul 22, 2006
Jul 22, 2006
452
{
Jul 22, 2006
Jul 22, 2006
453
renderdata->glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
Nov 29, 2008
Nov 29, 2008
454
renderdata->glPixelStorei(GL_UNPACK_ROW_LENGTH,
Feb 3, 2011
Feb 3, 2011
455
(pitch / SDL_BYTESPERPIXEL(texture->format)));
Jul 22, 2006
Jul 22, 2006
456
457
}
Jul 19, 2006
Jul 19, 2006
458
459
460
461
static int
GL_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, const void *pixels, int pitch)
{
Jul 22, 2006
Jul 22, 2006
462
GL_RenderData *renderdata = (GL_RenderData *) renderer->driverdata;
Jul 19, 2006
Jul 19, 2006
463
GL_TextureData *data = (GL_TextureData *) texture->driverdata;
Jul 22, 2006
Jul 22, 2006
464
GLenum result;
Jul 19, 2006
Jul 19, 2006
465
Feb 2, 2011
Feb 2, 2011
466
467
GL_ActivateRenderer(renderer);
Jul 22, 2006
Jul 22, 2006
468
469
renderdata->glGetError();
SetupTextureUpdate(renderdata, texture, pitch);
Dec 20, 2008
Dec 20, 2008
470
renderdata->glEnable(data->type);
Jul 22, 2006
Jul 22, 2006
471
472
473
474
renderdata->glBindTexture(data->type, data->texture);
renderdata->glTexSubImage2D(data->type, 0, rect->x, rect->y, rect->w,
rect->h, data->format, data->formattype,
pixels);
Jan 14, 2009
Jan 14, 2009
475
renderdata->glDisable(data->type);
Jul 22, 2006
Jul 22, 2006
476
result = renderdata->glGetError();
Jul 22, 2006
Jul 22, 2006
477
478
479
480
if (result != GL_NO_ERROR) {
GL_SetError("glTexSubImage2D()", result);
return -1;
}
Jul 19, 2006
Jul 19, 2006
481
482
483
484
485
return 0;
}
static int
GL_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
Feb 3, 2011
Feb 3, 2011
486
const SDL_Rect * rect, void **pixels, int *pitch)
Jul 19, 2006
Jul 19, 2006
487
488
489
{
GL_TextureData *data = (GL_TextureData *) texture->driverdata;
Jul 22, 2006
Jul 22, 2006
490
491
*pixels =
(void *) ((Uint8 *) data->pixels + rect->y * data->pitch +
Feb 3, 2011
Feb 3, 2011
492
rect->x * SDL_BYTESPERPIXEL(texture->format));
Jul 22, 2006
Jul 22, 2006
493
*pitch = data->pitch;
Jul 19, 2006
Jul 19, 2006
494
495
496
497
498
499
return 0;
}
static void
GL_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture)
{
Feb 3, 2011
Feb 3, 2011
500
GL_RenderData *renderdata = (GL_RenderData *) renderer->driverdata;
Jul 19, 2006
Jul 19, 2006
501
502
GL_TextureData *data = (GL_TextureData *) texture->driverdata;
Feb 3, 2011
Feb 3, 2011
503
504
505
506
507
508
509
510
GL_ActivateRenderer(renderer);
SetupTextureUpdate(renderdata, texture, data->pitch);
renderdata->glEnable(data->type);
renderdata->glBindTexture(data->type, data->texture);
renderdata->glTexSubImage2D(data->type, 0, 0, 0, texture->w, texture->h,
data->format, data->formattype, data->pixels);
renderdata->glDisable(data->type);
Jul 19, 2006
Jul 19, 2006
511
512
}
Dec 31, 2008
Dec 31, 2008
513
static void
Feb 1, 2011
Feb 1, 2011
514
GL_SetBlendMode(GL_RenderData * data, int blendMode)
Dec 31, 2008
Dec 31, 2008
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
{
if (blendMode != data->blendMode) {
switch (blendMode) {
case SDL_BLENDMODE_NONE:
data->glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
data->glDisable(GL_BLEND);
break;
case SDL_BLENDMODE_BLEND:
data->glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
data->glEnable(GL_BLEND);
data->glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
break;
case SDL_BLENDMODE_ADD:
data->glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
data->glEnable(GL_BLEND);
data->glBlendFunc(GL_SRC_ALPHA, GL_ONE);
break;
Feb 5, 2011
Feb 5, 2011
532
533
534
535
536
case SDL_BLENDMODE_MOD:
data->glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
data->glEnable(GL_BLEND);
data->glBlendFunc(GL_ZERO, GL_SRC_COLOR);
break;
Dec 31, 2008
Dec 31, 2008
537
538
539
540
541
}
data->blendMode = blendMode;
}
}
Dec 20, 2008
Dec 20, 2008
542
static int
Dec 23, 2009
Dec 23, 2009
543
544
545
546
GL_RenderClear(SDL_Renderer * renderer)
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
Feb 2, 2011
Feb 2, 2011
547
548
GL_ActivateRenderer(renderer);
Dec 23, 2009
Dec 23, 2009
549
550
551
552
553
554
555
556
557
558
559
560
561
data->glClearColor((GLfloat) renderer->r * inv255f,
(GLfloat) renderer->g * inv255f,
(GLfloat) renderer->b * inv255f,
(GLfloat) renderer->a * inv255f);
data->glClear(GL_COLOR_BUFFER_BIT);
return 0;
}
static int
GL_RenderDrawPoints(SDL_Renderer * renderer, const SDL_Point * points,
int count)
Jul 19, 2006
Jul 19, 2006
562
563
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
Dec 9, 2009
Dec 9, 2009
564
int i;
Dec 30, 2008
Dec 30, 2008
565
Feb 2, 2011
Feb 2, 2011
566
567
GL_ActivateRenderer(renderer);
Feb 1, 2011
Feb 1, 2011
568
GL_SetBlendMode(data, renderer->blendMode);
Jul 19, 2006
Jul 19, 2006
569
Dec 20, 2008
Dec 20, 2008
570
571
572
573
data->glColor4f((GLfloat) renderer->r * inv255f,
(GLfloat) renderer->g * inv255f,
(GLfloat) renderer->b * inv255f,
(GLfloat) renderer->a * inv255f);
Dec 30, 2008
Dec 30, 2008
574
575
data->glBegin(GL_POINTS);
Dec 9, 2009
Dec 9, 2009
576
577
578
for (i = 0; i < count; ++i) {
data->glVertex2f(0.5f + points[i].x, 0.5f + points[i].y);
}
Dec 30, 2008
Dec 30, 2008
579
data->glEnd();
Dec 20, 2008
Dec 20, 2008
580
581
582
583
return 0;
}
Dec 21, 2008
Dec 21, 2008
584
static int
Dec 23, 2009
Dec 23, 2009
585
586
GL_RenderDrawLines(SDL_Renderer * renderer, const SDL_Point * points,
int count)
Dec 21, 2008
Dec 21, 2008
587
588
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
Dec 9, 2009
Dec 9, 2009
589
int i;
Dec 21, 2008
Dec 21, 2008
590
Feb 2, 2011
Feb 2, 2011
591
592
GL_ActivateRenderer(renderer);
Feb 1, 2011
Feb 1, 2011
593
GL_SetBlendMode(data, renderer->blendMode);
Dec 21, 2008
Dec 21, 2008
594
595
596
597
598
599
data->glColor4f((GLfloat) renderer->r * inv255f,
(GLfloat) renderer->g * inv255f,
(GLfloat) renderer->b * inv255f,
(GLfloat) renderer->a * inv255f);
Dec 9, 2009
Dec 9, 2009
600
601
602
603
604
605
606
607
608
609
if (count > 2 &&
points[0].x == points[count-1].x && points[0].y == points[count-1].y) {
data->glBegin(GL_LINE_LOOP);
/* GL_LINE_LOOP takes care of the final segment */
--count;
for (i = 0; i < count; ++i) {
data->glVertex2f(0.5f + points[i].x, 0.5f + points[i].y);
}
data->glEnd();
} else {
Jan 24, 2011
Jan 24, 2011
610
#if defined(__APPLE__) || defined(__WIN32__)
Nov 20, 2010
Nov 20, 2010
611
#else
Oct 17, 2010
Oct 17, 2010
612
int x1, y1, x2, y2;
Nov 20, 2010
Nov 20, 2010
613
#endif
Oct 17, 2010
Oct 17, 2010
614
Dec 9, 2009
Dec 9, 2009
615
616
617
618
619
620
621
622
623
624
625
626
627
628
data->glBegin(GL_LINE_STRIP);
for (i = 0; i < count; ++i) {
data->glVertex2f(0.5f + points[i].x, 0.5f + points[i].y);
}
data->glEnd();
/* The line is half open, so we need one more point to complete it.
* http://www.opengl.org/documentation/specs/version1.1/glspec1.1/node47.html
* If we have to, we can use vertical line and horizontal line textures
* for vertical and horizontal lines, and then create custom textures
* for diagonal lines and software render those. It's terrible, but at
* least it would be pixel perfect.
*/
data->glBegin(GL_POINTS);
Jan 24, 2011
Jan 24, 2011
629
#if defined(__APPLE__) || defined(__WIN32__)
Dec 9, 2009
Dec 9, 2009
630
631
/* Mac OS X and Windows seem to always leave the second point open */
data->glVertex2f(0.5f + points[count-1].x, 0.5f + points[count-1].y);
Nov 21, 2009
Nov 21, 2009
632
#else
Dec 9, 2009
Dec 9, 2009
633
/* Linux seems to leave the right-most or bottom-most point open */
Oct 17, 2010
Oct 17, 2010
634
635
636
637
x1 = points[0].x;
y1 = points[0].y;
x2 = points[count-1].x;
y2 = points[count-1].y;
Dec 9, 2009
Dec 9, 2009
638
639
640
641
642
643
644
645
646
647
if (x1 > x2) {
data->glVertex2f(0.5f + x1, 0.5f + y1);
} else if (x2 > x1) {
data->glVertex2f(0.5f + x2, 0.5f + y2);
} else if (y1 > y2) {
data->glVertex2f(0.5f + x1, 0.5f + y1);
} else if (y2 > y1) {
data->glVertex2f(0.5f + x2, 0.5f + y2);
}
Nov 21, 2009
Nov 21, 2009
648
#endif
Dec 9, 2009
Dec 9, 2009
649
650
data->glEnd();
}
Nov 19, 2009
Nov 19, 2009
651
Dec 21, 2008
Dec 21, 2008
652
653
654
return 0;
}
Dec 23, 2009
Dec 23, 2009
655
656
static int
GL_RenderFillRects(SDL_Renderer * renderer, const SDL_Rect ** rects, int count)
Dec 20, 2008
Dec 20, 2008
657
658
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
Dec 9, 2009
Dec 9, 2009
659
int i;
Dec 20, 2008
Dec 20, 2008
660
Feb 2, 2011
Feb 2, 2011
661
662
GL_ActivateRenderer(renderer);
Feb 1, 2011
Feb 1, 2011
663
GL_SetBlendMode(data, renderer->blendMode);
Dec 31, 2008
Dec 31, 2008
664
Dec 20, 2008
Dec 20, 2008
665
666
667
668
data->glColor4f((GLfloat) renderer->r * inv255f,
(GLfloat) renderer->g * inv255f,
(GLfloat) renderer->b * inv255f,
(GLfloat) renderer->a * inv255f);
Dec 31, 2008
Dec 31, 2008
669
Dec 9, 2009
Dec 9, 2009
670
671
672
673
674
for (i = 0; i < count; ++i) {
const SDL_Rect *rect = rects[i];
data->glRecti(rect->x, rect->y, rect->x + rect->w, rect->y + rect->h);
}
Dec 21, 2008
Dec 21, 2008
675
Jul 19, 2006
Jul 19, 2006
676
677
678
679
680
return 0;
}
static int
GL_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
Aug 28, 2006
Aug 28, 2006
681
const SDL_Rect * srcrect, const SDL_Rect * dstrect)
Jul 19, 2006
Jul 19, 2006
682
683
684
685
686
687
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
GL_TextureData *texturedata = (GL_TextureData *) texture->driverdata;
int minx, miny, maxx, maxy;
GLfloat minu, maxu, minv, maxv;
Feb 2, 2011
Feb 2, 2011
688
689
GL_ActivateRenderer(renderer);
Jul 19, 2006
Jul 19, 2006
690
691
692
693
694
695
minx = dstrect->x;
miny = dstrect->y;
maxx = dstrect->x + dstrect->w;
maxy = dstrect->y + dstrect->h;
minu = (GLfloat) srcrect->x / texture->w;
Jul 22, 2006
Jul 22, 2006
696
minu *= texturedata->texw;
Jul 19, 2006
Jul 19, 2006
697
maxu = (GLfloat) (srcrect->x + srcrect->w) / texture->w;
Jul 22, 2006
Jul 22, 2006
698
maxu *= texturedata->texw;
Jul 19, 2006
Jul 19, 2006
699
minv = (GLfloat) srcrect->y / texture->h;
Jul 22, 2006
Jul 22, 2006
700
minv *= texturedata->texh;
Jul 19, 2006
Jul 19, 2006
701
maxv = (GLfloat) (srcrect->y + srcrect->h) / texture->h;
Jul 22, 2006
Jul 22, 2006
702
maxv *= texturedata->texh;
Jul 19, 2006
Jul 19, 2006
703
Dec 20, 2008
Dec 20, 2008
704
data->glEnable(texturedata->type);
Jul 22, 2006
Jul 22, 2006
705
706
data->glBindTexture(texturedata->type, texturedata->texture);
Aug 28, 2006
Aug 28, 2006
707
708
709
710
711
712
713
714
715
if (texture->modMode) {
data->glColor4f((GLfloat) texture->r * inv255f,
(GLfloat) texture->g * inv255f,
(GLfloat) texture->b * inv255f,
(GLfloat) texture->a * inv255f);
} else {
data->glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
}
Feb 1, 2011
Feb 1, 2011
716
GL_SetBlendMode(data, texture->blendMode);
Nov 21, 2009
Nov 21, 2009
717
Jul 22, 2006
Jul 22, 2006
718
719
data->glBegin(GL_TRIANGLE_STRIP);
data->glTexCoord2f(minu, minv);
Nov 21, 2009
Nov 21, 2009
720
data->glVertex2f((GLfloat) minx, (GLfloat) miny);
Jul 22, 2006
Jul 22, 2006
721
data->glTexCoord2f(maxu, minv);
Nov 21, 2009
Nov 21, 2009
722
data->glVertex2f((GLfloat) maxx, (GLfloat) miny);
Jul 22, 2006
Jul 22, 2006
723
data->glTexCoord2f(minu, maxv);
Nov 21, 2009
Nov 21, 2009
724
data->glVertex2f((GLfloat) minx, (GLfloat) maxy);
Jul 22, 2006
Jul 22, 2006
725
data->glTexCoord2f(maxu, maxv);
Nov 21, 2009
Nov 21, 2009
726
data->glVertex2f((GLfloat) maxx, (GLfloat) maxy);
Jul 22, 2006
Jul 22, 2006
727
data->glEnd();
Jul 19, 2006
Jul 19, 2006
728
Dec 20, 2008
Dec 20, 2008
729
730
data->glDisable(texturedata->type);
Jul 19, 2006
Jul 19, 2006
731
732
733
return 0;
}
Nov 15, 2009
Nov 15, 2009
734
735
static int
GL_RenderReadPixels(SDL_Renderer * renderer, const SDL_Rect * rect,
Nov 16, 2009
Nov 16, 2009
736
Uint32 pixel_format, void * pixels, int pitch)
Nov 15, 2009
Nov 15, 2009
737
{
Nov 15, 2009
Nov 15, 2009
738
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
Jan 21, 2010
Jan 21, 2010
739
SDL_Window *window = renderer->window;
Nov 15, 2009
Nov 15, 2009
740
741
GLint internalFormat;
GLenum format, type;
Nov 16, 2009
Nov 16, 2009
742
Uint8 *src, *dst, *tmp;
Feb 2, 2011
Feb 2, 2011
743
int w, h, length, rows;
Nov 15, 2009
Nov 15, 2009
744
Feb 2, 2011
Feb 2, 2011
745
746
GL_ActivateRenderer(renderer);
Nov 15, 2009
Nov 15, 2009
747
if (!convert_format(data, pixel_format, &internalFormat, &format, &type)) {
Nov 16, 2009
Nov 16, 2009
748
/* FIXME: Do a temp copy to a format that is supported */
Nov 15, 2009
Nov 15, 2009
749
750
751
752
SDL_SetError("Unsupported pixel format");
return -1;
}
Feb 2, 2011
Feb 2, 2011
753
754
SDL_GetWindowSize(window, &w, &h);
Nov 18, 2009
Nov 18, 2009
755
756
data->glPixelStorei(GL_PACK_ALIGNMENT, 1);
data->glPixelStorei(GL_PACK_ROW_LENGTH,
Feb 3, 2011
Feb 3, 2011
757
(pitch / SDL_BYTESPERPIXEL(pixel_format)));
Nov 15, 2009
Nov 15, 2009
758
Feb 2, 2011
Feb 2, 2011
759
data->glReadPixels(rect->x, (h-rect->y)-rect->h, rect->w, rect->h,
Nov 16, 2009
Nov 16, 2009
760
761
762
format, type, pixels);
/* Flip the rows to be top-down */
Feb 3, 2011
Feb 3, 2011
763
length = rect->w * SDL_BYTESPERPIXEL(pixel_format);
Nov 16, 2009
Nov 16, 2009
764
765
766
767
768
769
770
771
src = (Uint8*)pixels + (rect->h-1)*pitch;
dst = (Uint8*)pixels;
tmp = SDL_stack_alloc(Uint8, length);
rows = rect->h / 2;
while (rows--) {
SDL_memcpy(tmp, dst, length);
SDL_memcpy(dst, src, length);
SDL_memcpy(src, tmp, length);
Nov 18, 2009
Nov 18, 2009
772
773
dst += pitch;
src -= pitch;
Nov 16, 2009
Nov 16, 2009
774
775
}
SDL_stack_free(tmp);
Nov 17, 2009
Nov 17, 2009
776
777
return 0;
Nov 15, 2009
Nov 15, 2009
778
779
}
Jul 19, 2006
Jul 19, 2006
780
781
782
static void
GL_RenderPresent(SDL_Renderer * renderer)
{
Feb 2, 2011
Feb 2, 2011
783
784
GL_ActivateRenderer(renderer);
Jul 19, 2006
Jul 19, 2006
785
786
787
788
789
790
SDL_GL_SwapWindow(renderer->window);
}
static void
GL_DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture)
{
Jul 22, 2006
Jul 22, 2006
791
GL_RenderData *renderdata = (GL_RenderData *) renderer->driverdata;
Jul 19, 2006
Jul 19, 2006
792
793
GL_TextureData *data = (GL_TextureData *) texture->driverdata;
Feb 2, 2011
Feb 2, 2011
794
795
GL_ActivateRenderer(renderer);
Jul 19, 2006
Jul 19, 2006
796
797
798
799
if (!data) {
return;
}
if (data->texture) {
Jul 22, 2006
Jul 22, 2006
800
renderdata->glDeleteTextures(1, &data->texture);
Jul 22, 2006
Jul 22, 2006
801
802
803
}
if (data->pixels) {
SDL_free(data->pixels);
Jul 19, 2006
Jul 19, 2006
804
805
806
807
808
}
SDL_free(data);
texture->driverdata = NULL;
}
Aug 7, 2006
Aug 7, 2006
809
static void
Jul 19, 2006
Jul 19, 2006
810
811
812
813
814
GL_DestroyRenderer(SDL_Renderer * renderer)
{
GL_RenderData *data = (GL_RenderData *) renderer->driverdata;
if (data) {
Jul 22, 2006
Jul 22, 2006
815
if (data->context) {
Mar 14, 2008
Mar 14, 2008
816
/* SDL_GL_MakeCurrent(0, NULL); *//* doesn't do anything */
Jul 22, 2006
Jul 22, 2006
817
SDL_GL_DeleteContext(data->context);
Jul 19, 2006
Jul 19, 2006
818
819
820
821
822
823
}
SDL_free(data);
}
SDL_free(renderer);
}
Jul 28, 2006
Jul 28, 2006
824
#endif /* SDL_VIDEO_RENDER_OGL */
Jul 19, 2006
Jul 19, 2006
825
826
/* vi: set ts=4 sw=4 expandtab: */