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

Latest commit

 

History

History
1004 lines (878 loc) · 25.9 KB

SDL_surface.c

File metadata and controls

1004 lines (878 loc) · 25.9 KB
 
Apr 26, 2001
Apr 26, 2001
1
2
/*
SDL - Simple DirectMedia Layer
Jan 24, 2010
Jan 24, 2010
3
Copyright (C) 1997-2010 Sam Lantinga
Apr 26, 2001
Apr 26, 2001
4
5
This library is free software; you can redistribute it and/or
Feb 1, 2006
Feb 1, 2006
6
modify it under the terms of the GNU Lesser General Public
Apr 26, 2001
Apr 26, 2001
7
License as published by the Free Software Foundation; either
Feb 1, 2006
Feb 1, 2006
8
version 2.1 of the License, or (at your option) any later version.
Apr 26, 2001
Apr 26, 2001
9
10
11
12
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
Feb 1, 2006
Feb 1, 2006
13
Lesser General Public License for more details.
Apr 26, 2001
Apr 26, 2001
14
Feb 1, 2006
Feb 1, 2006
15
16
17
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
Apr 26, 2001
Apr 26, 2001
18
19
Sam Lantinga
Dec 14, 2001
Dec 14, 2001
20
slouken@libsdl.org
Apr 26, 2001
Apr 26, 2001
21
*/
Feb 21, 2006
Feb 21, 2006
22
#include "SDL_config.h"
Apr 26, 2001
Apr 26, 2001
23
Feb 7, 2006
Feb 7, 2006
24
#include "SDL_video.h"
Jul 10, 2006
Jul 10, 2006
25
#include "SDL_compat.h"
Apr 26, 2001
Apr 26, 2001
26
27
28
29
30
31
#include "SDL_sysvideo.h"
#include "SDL_blit.h"
#include "SDL_RLEaccel_c.h"
#include "SDL_pixels_c.h"
#include "SDL_leaks.h"
Jan 19, 2006
Jan 19, 2006
32
Apr 26, 2001
Apr 26, 2001
33
34
35
36
/* Public routines */
/*
* Create an empty RGB surface of the appropriate depth
*/
Jul 10, 2006
Jul 10, 2006
37
38
39
40
SDL_Surface *
SDL_CreateRGBSurface(Uint32 flags,
int width, int height, int depth,
Uint32 Rmask, Uint32 Gmask, Uint32 Bmask, Uint32 Amask)
Apr 26, 2001
Apr 26, 2001
41
{
Jul 10, 2006
Jul 10, 2006
42
43
SDL_Surface *surface;
Nov 29, 2008
Nov 29, 2008
44
/* The flags are no longer used, make the compiler happy */
May 9, 2010
May 9, 2010
45
(void)flags;
Nov 29, 2008
Nov 29, 2008
46
Jul 10, 2006
Jul 10, 2006
47
/* Allocate the surface */
Jul 22, 2006
Jul 22, 2006
48
surface = (SDL_Surface *) SDL_calloc(1, sizeof(*surface));
Jul 10, 2006
Jul 10, 2006
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
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
if (surface == NULL) {
SDL_OutOfMemory();
return NULL;
}
surface->format = SDL_AllocFormat(depth, Rmask, Gmask, Bmask, Amask);
if (!surface->format) {
SDL_FreeSurface(surface);
return NULL;
}
surface->w = width;
surface->h = height;
surface->pitch = SDL_CalculatePitch(surface);
SDL_SetClipRect(surface, NULL);
if (surface->format->BitsPerPixel <= 8) {
SDL_Palette *palette =
SDL_AllocPalette((1 << surface->format->BitsPerPixel));
if (!palette) {
SDL_FreeSurface(surface);
return NULL;
}
if (Rmask || Bmask || Gmask) {
const SDL_PixelFormat *format = surface->format;
/* create palette according to masks */
int i;
int Rm = 0, Gm = 0, Bm = 0;
int Rw = 0, Gw = 0, Bw = 0;
if (Rmask) {
Rw = 8 - format->Rloss;
for (i = format->Rloss; i > 0; i -= Rw)
Rm |= 1 << i;
}
if (Gmask) {
Gw = 8 - format->Gloss;
for (i = format->Gloss; i > 0; i -= Gw)
Gm |= 1 << i;
}
if (Bmask) {
Bw = 8 - format->Bloss;
for (i = format->Bloss; i > 0; i -= Bw)
Bm |= 1 << i;
}
for (i = 0; i < palette->ncolors; ++i) {
int r, g, b;
r = (i & Rmask) >> format->Rshift;
r = (r << format->Rloss) | ((r * Rm) >> Rw);
palette->colors[i].r = r;
g = (i & Gmask) >> format->Gshift;
g = (g << format->Gloss) | ((g * Gm) >> Gw);
palette->colors[i].g = g;
b = (i & Bmask) >> format->Bshift;
b = (b << format->Bloss) | ((b * Bm) >> Bw);
palette->colors[i].b = b;
}
} else if (palette->ncolors == 2) {
/* Create a black and white bitmap palette */
palette->colors[0].r = 0xFF;
palette->colors[0].g = 0xFF;
palette->colors[0].b = 0xFF;
palette->colors[1].r = 0x00;
palette->colors[1].g = 0x00;
palette->colors[1].b = 0x00;
}
SDL_SetSurfacePalette(surface, palette);
SDL_FreePalette(palette);
}
/* Get the pixels */
if (surface->w && surface->h) {
surface->pixels = SDL_malloc(surface->h * surface->pitch);
if (!surface->pixels) {
SDL_FreeSurface(surface);
SDL_OutOfMemory();
return NULL;
}
/* This is important for bitmaps */
SDL_memset(surface->pixels, 0, surface->h * surface->pitch);
}
/* Allocate an empty mapping */
surface->map = SDL_AllocBlitMap();
if (!surface->map) {
SDL_FreeSurface(surface);
return NULL;
}
SDL_FormatChanged(surface);
Aug 18, 2007
Aug 18, 2007
141
142
/* By default surface with an alpha mask are set up for blending */
if (Amask) {
Dec 20, 2008
Dec 20, 2008
143
SDL_SetSurfaceBlendMode(surface, SDL_BLENDMODE_BLEND);
Aug 18, 2007
Aug 18, 2007
144
145
}
Jul 10, 2006
Jul 10, 2006
146
147
/* The surface is ready to go */
surface->refcount = 1;
Apr 26, 2001
Apr 26, 2001
148
#ifdef CHECK_LEAKS
Jul 10, 2006
Jul 10, 2006
149
++surfaces_allocated;
Apr 26, 2001
Apr 26, 2001
150
#endif
Jul 10, 2006
Jul 10, 2006
151
return surface;
Apr 26, 2001
Apr 26, 2001
152
}
Jul 10, 2006
Jul 10, 2006
153
Apr 26, 2001
Apr 26, 2001
154
155
156
/*
* Create an RGB surface from an existing memory buffer
*/
Jul 10, 2006
Jul 10, 2006
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
SDL_Surface *
SDL_CreateRGBSurfaceFrom(void *pixels,
int width, int height, int depth, int pitch,
Uint32 Rmask, Uint32 Gmask, Uint32 Bmask,
Uint32 Amask)
{
SDL_Surface *surface;
surface =
SDL_CreateRGBSurface(0, 0, 0, depth, Rmask, Gmask, Bmask, Amask);
if (surface != NULL) {
surface->flags |= SDL_PREALLOC;
surface->pixels = pixels;
surface->w = width;
surface->h = height;
surface->pitch = pitch;
SDL_SetClipRect(surface, NULL);
}
return surface;
}
static int
SDL_SurfacePaletteChanged(void *userdata, SDL_Palette * palette)
{
SDL_Surface *surface = (SDL_Surface *) userdata;
SDL_FormatChanged(surface);
return 0;
}
int
SDL_SetSurfacePalette(SDL_Surface * surface, SDL_Palette * palette)
Apr 26, 2001
Apr 26, 2001
190
{
Jul 10, 2006
Jul 10, 2006
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
if (!surface || !surface->format) {
SDL_SetError("SDL_SetSurfacePalette() passed a NULL surface");
return -1;
}
if (palette && palette->ncolors != (1 << surface->format->BitsPerPixel)) {
SDL_SetError
("SDL_SetSurfacePalette() passed a palette that doesn't match the surface format");
return -1;
}
if (surface->format->palette == palette) {
return 0;
}
if (surface->format->palette) {
SDL_DelPaletteWatch(surface->format->palette,
SDL_SurfacePaletteChanged, surface);
}
surface->format->palette = palette;
if (surface->format->palette) {
SDL_AddPaletteWatch(surface->format->palette,
SDL_SurfacePaletteChanged, surface);
}
return 0;
Apr 26, 2001
Apr 26, 2001
218
}
Jul 10, 2006
Jul 10, 2006
219
Aug 18, 2007
Aug 18, 2007
220
221
int
SDL_SetSurfaceRLE(SDL_Surface * surface, int flag)
Apr 26, 2001
Apr 26, 2001
222
{
Aug 18, 2007
Aug 18, 2007
223
int flags;
Aug 18, 2007
Aug 18, 2007
224
225
226
227
228
if (!surface) {
return -1;
}
Aug 18, 2007
Aug 18, 2007
229
flags = surface->map->info.flags;
Aug 18, 2007
Aug 18, 2007
230
if (flag) {
Aug 18, 2007
Aug 18, 2007
231
surface->map->info.flags |= SDL_COPY_RLE_DESIRED;
Jul 10, 2006
Jul 10, 2006
232
} else {
Aug 18, 2007
Aug 18, 2007
233
surface->map->info.flags &= ~SDL_COPY_RLE_DESIRED;
Aug 18, 2007
Aug 18, 2007
234
}
Aug 18, 2007
Aug 18, 2007
235
if (surface->map->info.flags != flags) {
Aug 18, 2007
Aug 18, 2007
236
SDL_InvalidateMap(surface->map);
Jul 10, 2006
Jul 10, 2006
237
}
Aug 18, 2007
Aug 18, 2007
238
239
return 0;
}
Jul 10, 2006
Jul 10, 2006
240
Aug 18, 2007
Aug 18, 2007
241
int
Dec 14, 2009
Dec 14, 2009
242
SDL_SetColorKey(SDL_Surface * surface, int flag, Uint32 key)
Aug 18, 2007
Aug 18, 2007
243
244
245
246
247
{
int flags;
if (!surface) {
return -1;
Jul 10, 2006
Jul 10, 2006
248
249
}
Aug 18, 2007
Aug 18, 2007
250
251
if (flag & SDL_RLEACCEL) {
SDL_SetSurfaceRLE(surface, 1);
Jul 10, 2006
Jul 10, 2006
252
253
}
Aug 18, 2007
Aug 18, 2007
254
flags = surface->map->info.flags;
Jul 10, 2006
Jul 10, 2006
255
if (flag) {
Aug 18, 2007
Aug 18, 2007
256
257
surface->map->info.flags |= SDL_COPY_COLORKEY;
surface->map->info.colorkey = key;
Jul 10, 2006
Jul 10, 2006
258
} else {
Aug 18, 2007
Aug 18, 2007
259
surface->map->info.flags &= ~SDL_COPY_COLORKEY;
Jul 10, 2006
Jul 10, 2006
260
}
Aug 18, 2007
Aug 18, 2007
261
262
263
if (surface->map->info.flags != flags) {
SDL_InvalidateMap(surface->map);
}
Aug 18, 2007
Aug 18, 2007
264
265
266
267
268
269
270
271
/* Compatibility mode */
if (surface->map->info.flags & SDL_COPY_COLORKEY) {
surface->flags |= SDL_SRCCOLORKEY;
} else {
surface->flags &= ~SDL_SRCCOLORKEY;
}
Aug 18, 2007
Aug 18, 2007
272
return 0;
Apr 26, 2001
Apr 26, 2001
273
}
Jul 10, 2006
Jul 10, 2006
274
Apr 3, 2009
Apr 3, 2009
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
int
SDL_GetColorKey(SDL_Surface * surface, Uint32 * key)
{
if (!surface) {
return -1;
}
if (!(surface->map->info.flags & SDL_COPY_COLORKEY)) {
return -1;
}
if (key) {
*key = surface->map->info.colorkey;
}
return 0;
}
Nov 25, 2008
Nov 25, 2008
292
/* This is a fairly slow function to switch from colorkey to alpha */
Nov 25, 2008
Nov 25, 2008
293
static void
Nov 25, 2008
Nov 25, 2008
294
SDL_ConvertColorkeyToAlpha(SDL_Surface * surface)
Nov 25, 2008
Nov 25, 2008
295
{
Nov 25, 2008
Nov 25, 2008
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
int x, y;
if (!surface) {
return;
}
if (!(surface->map->info.flags & SDL_COPY_COLORKEY) ||
!surface->format->Amask) {
return;
}
SDL_LockSurface(surface);
switch (surface->format->BytesPerPixel) {
case 2:
{
Uint16 *row, *spot;
Uint16 ckey = (Uint16) surface->map->info.colorkey;
Uint16 mask = (Uint16) (~surface->format->Amask);
row = (Uint16 *) surface->pixels;
for (y = surface->h; y--;) {
spot = row;
for (x = surface->w; x--;) {
if (*spot == ckey) {
*spot &= mask;
}
++spot;
}
row += surface->pitch / 2;
}
}
break;
case 3:
/* FIXME */
break;
case 4:
{
Uint32 *row, *spot;
Uint32 ckey = surface->map->info.colorkey;
Uint32 mask = ~surface->format->Amask;
row = (Uint32 *) surface->pixels;
for (y = surface->h; y--;) {
spot = row;
for (x = surface->w; x--;) {
if (*spot == ckey) {
*spot &= mask;
}
++spot;
}
row += surface->pitch / 4;
}
}
break;
}
SDL_UnlockSurface(surface);
SDL_SetColorKey(surface, 0, 0);
Dec 20, 2008
Dec 20, 2008
356
SDL_SetSurfaceBlendMode(surface, SDL_BLENDMODE_BLEND);
Nov 25, 2008
Nov 25, 2008
357
358
}
Aug 18, 2007
Aug 18, 2007
359
360
int
SDL_SetSurfaceColorMod(SDL_Surface * surface, Uint8 r, Uint8 g, Uint8 b)
Apr 26, 2001
Apr 26, 2001
361
{
Aug 18, 2007
Aug 18, 2007
362
363
364
365
366
367
368
369
370
371
372
373
374
int flags;
if (!surface) {
return -1;
}
surface->map->info.r = r;
surface->map->info.g = g;
surface->map->info.b = b;
flags = surface->map->info.flags;
if (r != 0xFF || g != 0xFF || b != 0xFF) {
surface->map->info.flags |= SDL_COPY_MODULATE_COLOR;
Jul 10, 2006
Jul 10, 2006
375
} else {
Aug 18, 2007
Aug 18, 2007
376
377
378
379
surface->map->info.flags &= ~SDL_COPY_MODULATE_COLOR;
}
if (surface->map->info.flags != flags) {
SDL_InvalidateMap(surface->map);
Jul 10, 2006
Jul 10, 2006
380
}
Aug 18, 2007
Aug 18, 2007
381
382
383
return 0;
}
Jul 10, 2006
Jul 10, 2006
384
Aug 18, 2007
Aug 18, 2007
385
386
int
SDL_GetSurfaceColorMod(SDL_Surface * surface, Uint8 * r, Uint8 * g, Uint8 * b)
Aug 18, 2007
Aug 18, 2007
387
388
389
{
if (!surface) {
return -1;
Jul 10, 2006
Jul 10, 2006
390
391
}
Aug 18, 2007
Aug 18, 2007
392
393
394
395
396
397
398
399
400
401
402
if (r) {
*r = surface->map->info.r;
}
if (g) {
*g = surface->map->info.g;
}
if (b) {
*b = surface->map->info.b;
}
return 0;
}
Jul 10, 2006
Jul 10, 2006
403
Aug 18, 2007
Aug 18, 2007
404
405
int
SDL_SetSurfaceAlphaMod(SDL_Surface * surface, Uint8 alpha)
Aug 18, 2007
Aug 18, 2007
406
407
408
409
410
411
412
413
414
415
416
417
{
int flags;
if (!surface) {
return -1;
}
surface->map->info.a = alpha;
flags = surface->map->info.flags;
if (alpha != 0xFF) {
surface->map->info.flags |= SDL_COPY_MODULATE_ALPHA;
Jul 10, 2006
Jul 10, 2006
418
} else {
Aug 18, 2007
Aug 18, 2007
419
420
421
surface->map->info.flags &= ~SDL_COPY_MODULATE_ALPHA;
}
if (surface->map->info.flags != flags) {
Jul 10, 2006
Jul 10, 2006
422
423
SDL_InvalidateMap(surface->map);
}
Aug 18, 2007
Aug 18, 2007
424
return 0;
Apr 26, 2001
Apr 26, 2001
425
}
Aug 1, 2002
Aug 1, 2002
426
Aug 18, 2007
Aug 18, 2007
427
428
int
SDL_GetSurfaceAlphaMod(SDL_Surface * surface, Uint8 * alpha)
Jul 10, 2006
Jul 10, 2006
429
{
Aug 18, 2007
Aug 18, 2007
430
431
432
433
434
435
436
437
438
if (!surface) {
return -1;
}
if (alpha) {
*alpha = surface->map->info.a;
}
return 0;
}
Jul 10, 2006
Jul 10, 2006
439
Aug 18, 2007
Aug 18, 2007
440
441
int
SDL_SetSurfaceBlendMode(SDL_Surface * surface, int blendMode)
Aug 18, 2007
Aug 18, 2007
442
443
444
445
{
int flags, status;
if (!surface) {
Jul 10, 2006
Jul 10, 2006
446
447
return -1;
}
Aug 18, 2007
Aug 18, 2007
448
449
450
status = 0;
flags = surface->map->info.flags;
Aug 18, 2007
Aug 18, 2007
451
452
surface->map->info.flags &=
~(SDL_COPY_MASK | SDL_COPY_BLEND | SDL_COPY_ADD | SDL_COPY_MOD);
Aug 18, 2007
Aug 18, 2007
453
switch (blendMode) {
Dec 20, 2008
Dec 20, 2008
454
case SDL_BLENDMODE_NONE:
Aug 18, 2007
Aug 18, 2007
455
break;
Dec 20, 2008
Dec 20, 2008
456
case SDL_BLENDMODE_MASK:
Aug 18, 2007
Aug 18, 2007
457
458
surface->map->info.flags |= SDL_COPY_MASK;
break;
Dec 20, 2008
Dec 20, 2008
459
case SDL_BLENDMODE_BLEND:
Aug 18, 2007
Aug 18, 2007
460
461
surface->map->info.flags |= SDL_COPY_BLEND;
break;
Dec 20, 2008
Dec 20, 2008
462
case SDL_BLENDMODE_ADD:
Aug 18, 2007
Aug 18, 2007
463
464
surface->map->info.flags |= SDL_COPY_ADD;
break;
Dec 20, 2008
Dec 20, 2008
465
case SDL_BLENDMODE_MOD:
Aug 18, 2007
Aug 18, 2007
466
467
468
469
470
471
472
473
474
475
surface->map->info.flags |= SDL_COPY_MOD;
break;
default:
SDL_Unsupported();
status = -1;
break;
}
if (surface->map->info.flags != flags) {
SDL_InvalidateMap(surface->map);
Jul 10, 2006
Jul 10, 2006
476
}
Aug 18, 2007
Aug 18, 2007
477
478
479
480
481
482
483
484
/* Compatibility mode */
if (surface->map->info.flags & SDL_COPY_BLEND) {
surface->flags |= SDL_SRCALPHA;
} else {
surface->flags &= ~SDL_SRCALPHA;
}
Aug 18, 2007
Aug 18, 2007
485
486
487
return status;
}
Aug 18, 2007
Aug 18, 2007
488
489
int
SDL_GetSurfaceBlendMode(SDL_Surface * surface, int *blendMode)
Aug 18, 2007
Aug 18, 2007
490
491
492
{
if (!surface) {
return -1;
Jul 10, 2006
Jul 10, 2006
493
}
Aug 1, 2002
Aug 1, 2002
494
Aug 18, 2007
Aug 18, 2007
495
496
if (!blendMode) {
return 0;
Jul 10, 2006
Jul 10, 2006
497
}
Aug 18, 2007
Aug 18, 2007
498
Jan 10, 2009
Jan 10, 2009
499
500
501
switch (surface->map->
info.flags & (SDL_COPY_MASK | SDL_COPY_BLEND | SDL_COPY_ADD |
SDL_COPY_MOD)) {
Aug 18, 2007
Aug 18, 2007
502
case SDL_COPY_MASK:
Dec 20, 2008
Dec 20, 2008
503
*blendMode = SDL_BLENDMODE_MASK;
Aug 18, 2007
Aug 18, 2007
504
505
break;
case SDL_COPY_BLEND:
Dec 20, 2008
Dec 20, 2008
506
*blendMode = SDL_BLENDMODE_BLEND;
Aug 18, 2007
Aug 18, 2007
507
508
break;
case SDL_COPY_ADD:
Dec 20, 2008
Dec 20, 2008
509
*blendMode = SDL_BLENDMODE_ADD;
Aug 18, 2007
Aug 18, 2007
510
511
break;
case SDL_COPY_MOD:
Dec 20, 2008
Dec 20, 2008
512
*blendMode = SDL_BLENDMODE_MOD;
Aug 18, 2007
Aug 18, 2007
513
514
break;
default:
Dec 20, 2008
Dec 20, 2008
515
*blendMode = SDL_BLENDMODE_NONE;
Aug 18, 2007
Aug 18, 2007
516
break;
Jul 10, 2006
Jul 10, 2006
517
}
Aug 18, 2007
Aug 18, 2007
518
519
520
return 0;
}
Aug 18, 2007
Aug 18, 2007
521
522
int
SDL_SetSurfaceScaleMode(SDL_Surface * surface, int scaleMode)
Aug 18, 2007
Aug 18, 2007
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
{
int flags, status;
if (!surface) {
return -1;
}
status = 0;
flags = surface->map->info.flags;
surface->map->info.flags &= ~(SDL_COPY_NEAREST);
switch (scaleMode) {
case SDL_TEXTURESCALEMODE_NONE:
break;
case SDL_TEXTURESCALEMODE_FAST:
surface->map->info.flags |= SDL_COPY_NEAREST;
break;
case SDL_TEXTURESCALEMODE_SLOW:
case SDL_TEXTURESCALEMODE_BEST:
SDL_Unsupported();
surface->map->info.flags |= SDL_COPY_NEAREST;
status = -1;
break;
default:
SDL_Unsupported();
status = -1;
break;
}
if (surface->map->info.flags != flags) {
SDL_InvalidateMap(surface->map);
}
return status;
}
Aug 18, 2007
Aug 18, 2007
557
558
int
SDL_GetSurfaceScaleMode(SDL_Surface * surface, int *scaleMode)
Aug 18, 2007
Aug 18, 2007
559
560
561
562
563
564
565
566
567
{
if (!surface) {
return -1;
}
if (!scaleMode) {
return 0;
}
Dec 7, 2008
Dec 7, 2008
568
switch (surface->map->info.flags & SDL_COPY_NEAREST) {
Aug 18, 2007
Aug 18, 2007
569
570
case SDL_COPY_NEAREST:
*scaleMode = SDL_TEXTURESCALEMODE_FAST;
Aug 18, 2007
Aug 18, 2007
571
572
break;
default:
Aug 18, 2007
Aug 18, 2007
573
*scaleMode = SDL_TEXTURESCALEMODE_NONE;
Aug 18, 2007
Aug 18, 2007
574
break;
Jul 10, 2006
Jul 10, 2006
575
576
}
return 0;
Aug 1, 2002
Aug 1, 2002
577
}
Apr 26, 2001
Apr 26, 2001
578
Jul 10, 2006
Jul 10, 2006
579
580
SDL_bool
SDL_SetClipRect(SDL_Surface * surface, const SDL_Rect * rect)
Apr 26, 2001
Apr 26, 2001
581
{
Jul 10, 2006
Jul 10, 2006
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
SDL_Rect full_rect;
/* Don't do anything if there's no surface to act on */
if (!surface) {
return SDL_FALSE;
}
/* Set up the full surface rectangle */
full_rect.x = 0;
full_rect.y = 0;
full_rect.w = surface->w;
full_rect.h = surface->h;
/* Set the clipping rectangle */
if (!rect) {
surface->clip_rect = full_rect;
return 1;
}
return SDL_IntersectRect(rect, &full_rect, &surface->clip_rect);
Apr 26, 2001
Apr 26, 2001
601
}
Jul 10, 2006
Jul 10, 2006
602
603
604
void
SDL_GetClipRect(SDL_Surface * surface, SDL_Rect * rect)
Apr 26, 2001
Apr 26, 2001
605
{
Jul 10, 2006
Jul 10, 2006
606
607
608
if (surface && rect) {
*rect = surface->clip_rect;
}
Apr 26, 2001
Apr 26, 2001
609
}
Jul 10, 2006
Jul 10, 2006
610
Apr 26, 2001
Apr 26, 2001
611
612
613
614
615
616
617
618
619
620
621
/*
* Set up a blit between two surfaces -- split into three parts:
* The upper part, SDL_UpperBlit(), performs clipping and rectangle
* verification. The lower part is a pointer to a low level
* accelerated blitting function.
*
* These parts are separated out and each used internally by this
* library in the optimimum places. They are exported so that if
* you know exactly what you are doing, you can optimize your code
* by calling the one(s) you need.
*/
Jul 10, 2006
Jul 10, 2006
622
623
624
int
SDL_LowerBlit(SDL_Surface * src, SDL_Rect * srcrect,
SDL_Surface * dst, SDL_Rect * dstrect)
Apr 26, 2001
Apr 26, 2001
625
{
Jul 10, 2006
Jul 10, 2006
626
627
628
629
630
631
/* Check to make sure the blit mapping is valid */
if ((src->map->dst != dst) ||
(src->map->dst->format_version != src->map->format_version)) {
if (SDL_MapSurface(src, dst) < 0) {
return (-1);
}
Mar 14, 2008
Mar 14, 2008
632
/* just here for debugging */
Mar 14, 2008
Mar 14, 2008
633
634
635
636
/* printf */
/* ("src = 0x%08X src->flags = %08X src->map->info.flags = %08x\ndst = 0x%08X dst->flags = %08X dst->map->info.flags = %08X\nsrc->map->blit = 0x%08x\n", */
/* src, dst->flags, src->map->info.flags, dst, dst->flags, */
/* dst->map->info.flags, src->map->blit); */
Jul 10, 2006
Jul 10, 2006
637
}
Aug 17, 2007
Aug 17, 2007
638
return (src->map->blit(src, srcrect, dst, dstrect));
Apr 26, 2001
Apr 26, 2001
639
640
641
}
Jul 10, 2006
Jul 10, 2006
642
643
644
int
SDL_UpperBlit(SDL_Surface * src, SDL_Rect * srcrect,
SDL_Surface * dst, SDL_Rect * dstrect)
Apr 26, 2001
Apr 26, 2001
645
{
Jul 10, 2006
Jul 10, 2006
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
SDL_Rect fulldst;
int srcx, srcy, w, h;
/* Make sure the surfaces aren't locked */
if (!src || !dst) {
SDL_SetError("SDL_UpperBlit: passed a NULL surface");
return (-1);
}
if (src->locked || dst->locked) {
SDL_SetError("Surfaces must not be locked during blit");
return (-1);
}
/* If the destination rectangle is NULL, use the entire dest surface */
if (dstrect == NULL) {
fulldst.x = fulldst.y = 0;
dstrect = &fulldst;
}
/* clip the source rectangle to the source surface */
if (srcrect) {
int maxw, maxh;
srcx = srcrect->x;
w = srcrect->w;
if (srcx < 0) {
w += srcx;
dstrect->x -= srcx;
srcx = 0;
}
maxw = src->w - srcx;
if (maxw < w)
w = maxw;
srcy = srcrect->y;
h = srcrect->h;
if (srcy < 0) {
h += srcy;
dstrect->y -= srcy;
srcy = 0;
}
maxh = src->h - srcy;
if (maxh < h)
h = maxh;
} else {
srcx = srcy = 0;
w = src->w;
h = src->h;
}
/* clip the destination rectangle against the clip rectangle */
{
SDL_Rect *clip = &dst->clip_rect;
int dx, dy;
dx = clip->x - dstrect->x;
if (dx > 0) {
w -= dx;
dstrect->x += dx;
srcx += dx;
}
dx = dstrect->x + w - clip->x - clip->w;
if (dx > 0)
w -= dx;
dy = clip->y - dstrect->y;
if (dy > 0) {
h -= dy;
dstrect->y += dy;
srcy += dy;
}
dy = dstrect->y + h - clip->y - clip->h;
if (dy > 0)
h -= dy;
}
if (w > 0 && h > 0) {
SDL_Rect sr;
sr.x = srcx;
sr.y = srcy;
sr.w = dstrect->w = w;
sr.h = dstrect->h = h;
return SDL_LowerBlit(src, &sr, dst, dstrect);
}
dstrect->w = dstrect->h = 0;
return 0;
Apr 26, 2001
Apr 26, 2001
733
734
735
736
737
}
/*
* Lock a surface to directly access the pixels
*/
Jul 10, 2006
Jul 10, 2006
738
739
int
SDL_LockSurface(SDL_Surface * surface)
Apr 26, 2001
Apr 26, 2001
740
{
Jul 10, 2006
Jul 10, 2006
741
742
743
744
745
746
747
748
749
750
751
752
753
if (!surface->locked) {
/* Perform the lock */
if (surface->flags & SDL_RLEACCEL) {
SDL_UnRLESurface(surface, 1);
surface->flags |= SDL_RLEACCEL; /* save accel'd state */
}
}
/* Increment the surface lock count, for recursive locks */
++surface->locked;
/* Ready to go.. */
return (0);
Apr 26, 2001
Apr 26, 2001
754
}
Jul 10, 2006
Jul 10, 2006
755
Apr 26, 2001
Apr 26, 2001
756
757
758
/*
* Unlock a previously locked surface
*/
Jul 10, 2006
Jul 10, 2006
759
760
void
SDL_UnlockSurface(SDL_Surface * surface)
Apr 26, 2001
Apr 26, 2001
761
{
Jul 10, 2006
Jul 10, 2006
762
763
764
765
766
767
768
769
770
771
/* Only perform an unlock if we are locked */
if (!surface->locked || (--surface->locked > 0)) {
return;
}
/* Update RLE encoded surface with new data */
if ((surface->flags & SDL_RLEACCEL) == SDL_RLEACCEL) {
surface->flags &= ~SDL_RLEACCEL; /* stop lying */
SDL_RLESurface(surface);
}
Apr 26, 2001
Apr 26, 2001
772
773
774
775
776
}
/*
* Convert a surface into the specified pixel format.
*/
Jul 10, 2006
Jul 10, 2006
777
SDL_Surface *
Nov 29, 2008
Nov 29, 2008
778
779
SDL_ConvertSurface(SDL_Surface * surface, SDL_PixelFormat * format,
Uint32 flags)
Apr 26, 2001
Apr 26, 2001
780
{
Jul 10, 2006
Jul 10, 2006
781
SDL_Surface *convert;
Aug 18, 2007
Aug 18, 2007
782
Uint32 copy_flags;
Jul 10, 2006
Jul 10, 2006
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
SDL_Rect bounds;
/* Check for empty destination palette! (results in empty image) */
if (format->palette != NULL) {
int i;
for (i = 0; i < format->palette->ncolors; ++i) {
if ((format->palette->colors[i].r != 0xFF) ||
(format->palette->colors[i].g != 0xFF) ||
(format->palette->colors[i].b != 0xFF))
break;
}
if (i == format->palette->ncolors) {
SDL_SetError("Empty destination palette");
return (NULL);
}
}
/* Create a new surface with the desired format */
Nov 29, 2008
Nov 29, 2008
801
convert = SDL_CreateRGBSurface(flags, surface->w, surface->h,
Jul 10, 2006
Jul 10, 2006
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
format->BitsPerPixel, format->Rmask,
format->Gmask, format->Bmask,
format->Amask);
if (convert == NULL) {
return (NULL);
}
/* Copy the palette if any */
if (format->palette && convert->format->palette) {
SDL_memcpy(convert->format->palette->colors,
format->palette->colors,
format->palette->ncolors * sizeof(SDL_Color));
convert->format->palette->ncolors = format->palette->ncolors;
}
Aug 18, 2007
Aug 18, 2007
817
818
819
/* Save the original copy flags */
copy_flags = surface->map->info.flags;
surface->map->info.flags = 0;
Jul 10, 2006
Jul 10, 2006
820
821
822
823
824
825
826
827
828
/* Copy over the image data */
bounds.x = 0;
bounds.y = 0;
bounds.w = surface->w;
bounds.h = surface->h;
SDL_LowerBlit(surface, &bounds, convert, &bounds);
/* Clean up the original surface, and update converted surface */
Dec 2, 2008
Dec 2, 2008
829
830
831
832
833
834
835
836
837
838
convert->map->info.r = surface->map->info.r;
convert->map->info.g = surface->map->info.g;
convert->map->info.b = surface->map->info.b;
convert->map->info.a = surface->map->info.a;
convert->map->info.flags =
(copy_flags &
~(SDL_COPY_COLORKEY | SDL_COPY_BLEND
| SDL_COPY_RLE_DESIRED | SDL_COPY_RLE_COLORKEY |
SDL_COPY_RLE_ALPHAKEY));
surface->map->info.flags = copy_flags;
Aug 18, 2007
Aug 18, 2007
839
840
841
if (copy_flags & SDL_COPY_COLORKEY) {
Uint8 keyR, keyG, keyB, keyA;
Aug 18, 2007
Aug 18, 2007
842
843
SDL_GetRGBA(surface->map->info.colorkey, surface->format, &keyR,
&keyG, &keyB, &keyA);
Aug 18, 2007
Aug 18, 2007
844
845
SDL_SetColorKey(convert, 1,
SDL_MapRGBA(convert->format, keyR, keyG, keyB, keyA));
Dec 2, 2008
Dec 2, 2008
846
/* This is needed when converting for 3D texture upload */
Nov 25, 2008
Nov 25, 2008
847
SDL_ConvertColorkeyToAlpha(convert);
Aug 18, 2007
Aug 18, 2007
848
}
Dec 2, 2008
Dec 2, 2008
849
SDL_SetClipRect(convert, &surface->clip_rect);
Aug 18, 2007
Aug 18, 2007
850
851
852
/* Enable alpha blending by default if the new surface has an
* alpha channel or alpha modulation */
Dec 2, 2008
Dec 2, 2008
853
if ((surface->format->Amask && format->Amask) ||
Jul 8, 2010
Jul 8, 2010
854
(copy_flags & (SDL_COPY_COLORKEY|SDL_COPY_MODULATE_ALPHA))) {
Dec 20, 2008
Dec 20, 2008
855
SDL_SetSurfaceBlendMode(convert, SDL_BLENDMODE_BLEND);
Jul 10, 2006
Jul 10, 2006
856
}
Dec 2, 2008
Dec 2, 2008
857
858
859
if ((copy_flags & SDL_COPY_RLE_DESIRED) || (flags & SDL_RLEACCEL)) {
SDL_SetSurfaceRLE(convert, SDL_RLEACCEL);
}
Jul 10, 2006
Jul 10, 2006
860
861
862
/* We're ready to go! */
return (convert);
Apr 26, 2001
Apr 26, 2001
863
864
}
Nov 15, 2009
Nov 15, 2009
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
/*
* Create a surface on the stack for quick blit operations
*/
static __inline__ SDL_bool
SDL_CreateSurfaceOnStack(int width, int height, Uint32 pixel_format,
void * pixels, int pitch, SDL_Surface * surface,
SDL_PixelFormat * format, SDL_BlitMap * blitmap)
{
int bpp;
Uint32 Rmask, Gmask, Bmask, Amask;
if (!SDL_PixelFormatEnumToMasks(pixel_format,
&bpp, &Rmask, &Gmask, &Bmask, &Amask)) {
return SDL_FALSE;
}
if (bpp <= 8) {
SDL_SetError("Indexed pixel formats not supported");
return SDL_FALSE;
}
SDL_zerop(surface);
surface->flags = SDL_PREALLOC;
surface->format = SDL_InitFormat(format, bpp, Rmask, Gmask, Bmask, Amask);
surface->pixels = pixels;
surface->w = width;
surface->h = height;
surface->pitch = pitch;
/* We don't actually need to set up the clip rect for our purposes */
/*SDL_SetClipRect(surface, NULL);*/
/* Allocate an empty mapping */
SDL_zerop(blitmap);
blitmap->info.r = 0xFF;
blitmap->info.g = 0xFF;
blitmap->info.b = 0xFF;
blitmap->info.a = 0xFF;
surface->map = blitmap;
SDL_FormatChanged(surface);
/* The surface is ready to go */
surface->refcount = 1;
return SDL_TRUE;
}
/*
* Copy a block of pixels of one format to another format
*/
int SDL_ConvertPixels(int width, int height,
Uint32 src_format, const void * src, int src_pitch,
Uint32 dst_format, void * dst, int dst_pitch)
{
SDL_Surface src_surface, dst_surface;
SDL_PixelFormat src_fmt, dst_fmt;
SDL_BlitMap src_blitmap, dst_blitmap;
SDL_Rect rect;
/* Fast path for same format copy */
if (src_format == dst_format) {
int bpp;
if (SDL_ISPIXELFORMAT_FOURCC(src_format)) {
switch (src_format) {
case SDL_PIXELFORMAT_YV12:
case SDL_PIXELFORMAT_IYUV:
case SDL_PIXELFORMAT_YUY2:
case SDL_PIXELFORMAT_UYVY:
case SDL_PIXELFORMAT_YVYU:
bpp = 2;
default:
SDL_SetError("Unknown FOURCC pixel format");
return -1;
}
} else {
bpp = SDL_BYTESPERPIXEL(src_format);
}
width *= bpp;
while (height-- > 0) {
SDL_memcpy(dst, src, width);
src = (Uint8*)src + src_pitch;
dst = (Uint8*)dst + dst_pitch;
}
return SDL_TRUE;
}
if (!SDL_CreateSurfaceOnStack(width, height, src_format, (void*)src,
src_pitch,
&src_surface, &src_fmt, &src_blitmap)) {
return -1;
}
if (!SDL_CreateSurfaceOnStack(width, height, dst_format, dst, dst_pitch,
&dst_surface, &dst_fmt, &dst_blitmap)) {
return -1;
}
/* Set up the rect and go! */
rect.x = 0;
rect.y = 0;
rect.w = width;
Nov 16, 2009
Nov 16, 2009
964
rect.h = height;
Nov 15, 2009
Nov 15, 2009
965
966
967
return SDL_LowerBlit(&src_surface, &rect, &dst_surface, &rect);
}
Apr 26, 2001
Apr 26, 2001
968
969
970
/*
* Free a surface created by the above function.
*/
Jul 10, 2006
Jul 10, 2006
971
972
void
SDL_FreeSurface(SDL_Surface * surface)
Apr 26, 2001
Apr 26, 2001
973
{
Jul 10, 2006
Jul 10, 2006
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
if (surface == NULL) {
return;
}
if (--surface->refcount > 0) {
return;
}
while (surface->locked > 0) {
SDL_UnlockSurface(surface);
}
if (surface->flags & SDL_RLEACCEL) {
SDL_UnRLESurface(surface, 0);
}
if (surface->format) {
SDL_SetSurfacePalette(surface, NULL);
SDL_FreeFormat(surface->format);
surface->format = NULL;
}
if (surface->map != NULL) {
SDL_FreeBlitMap(surface->map);
surface->map = NULL;
}
if (surface->pixels && ((surface->flags & SDL_PREALLOC) != SDL_PREALLOC)) {
SDL_free(surface->pixels);
}
SDL_free(surface);
Apr 26, 2001
Apr 26, 2001
999
#ifdef CHECK_LEAKS
Jul 10, 2006
Jul 10, 2006
1000
--surfaces_allocated;