This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_renderer_sw.c
957 lines (839 loc) · 32 KB
1
2
/*
SDL - Simple DirectMedia Layer
3
Copyright (C) 1997-2009 Sam Lantinga
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
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"
#include "SDL_video.h"
#include "SDL_sysvideo.h"
26
#include "SDL_pixels_c.h"
27
28
29
30
31
32
#include "SDL_rect_c.h"
#include "SDL_yuv_sw_c.h"
/* SDL surface based renderer implementation */
33
static SDL_Renderer *SW_CreateRenderer(SDL_Window * window, Uint32 flags);
34
35
static int SW_ActivateRenderer(SDL_Renderer * renderer);
static int SW_DisplayModeChanged(SDL_Renderer * renderer);
36
37
38
39
40
41
42
43
44
45
46
static int SW_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture);
static int SW_QueryTexturePixels(SDL_Renderer * renderer,
SDL_Texture * texture, void **pixels,
int *pitch);
static int SW_SetTexturePalette(SDL_Renderer * renderer,
SDL_Texture * texture,
const SDL_Color * colors, int firstcolor,
int ncolors);
static int SW_GetTexturePalette(SDL_Renderer * renderer,
SDL_Texture * texture, SDL_Color * colors,
int firstcolor, int ncolors);
47
48
49
50
51
52
53
54
55
56
57
static int SW_SetTextureColorMod(SDL_Renderer * renderer,
SDL_Texture * texture);
static int SW_SetTextureAlphaMod(SDL_Renderer * renderer,
SDL_Texture * texture);
static int SW_SetTextureBlendMode(SDL_Renderer * renderer,
SDL_Texture * texture);
static int SW_SetTextureScaleMode(SDL_Renderer * renderer,
SDL_Texture * texture);
static int SW_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, const void *pixels,
int pitch);
58
static int SW_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
59
60
const SDL_Rect * rect, int markDirty, void **pixels,
int *pitch);
61
static void SW_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture);
62
63
64
65
66
67
68
69
static int SW_RenderDrawPoints(SDL_Renderer * renderer,
const SDL_Point * points, int count);
static int SW_RenderDrawLines(SDL_Renderer * renderer,
const SDL_Point * points, int count);
static int SW_RenderDrawRects(SDL_Renderer * renderer,
const SDL_Rect ** rects, int count);
static int SW_RenderFillRects(SDL_Renderer * renderer,
const SDL_Rect ** rects, int count);
70
static int SW_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
71
const SDL_Rect * srcrect, const SDL_Rect * dstrect);
72
static int SW_RenderReadPixels(SDL_Renderer * renderer, const SDL_Rect * rect,
73
Uint32 format, void * pixels, int pitch);
74
static int SW_RenderWritePixels(SDL_Renderer * renderer, const SDL_Rect * rect,
75
Uint32 format, const void * pixels, int pitch);
76
77
78
79
80
81
82
static void SW_RenderPresent(SDL_Renderer * renderer);
static void SW_DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture);
static void SW_DestroyRenderer(SDL_Renderer * renderer);
SDL_RenderDriver SW_RenderDriver = {
SW_CreateRenderer,
83
84
{
"software",
85
86
87
(SDL_RENDERER_SINGLEBUFFER | SDL_RENDERER_PRESENTCOPY |
SDL_RENDERER_PRESENTFLIP2 | SDL_RENDERER_PRESENTFLIP3 |
SDL_RENDERER_PRESENTDISCARD | SDL_RENDERER_PRESENTVSYNC),
88
89
(SDL_TEXTUREMODULATE_NONE | SDL_TEXTUREMODULATE_COLOR |
SDL_TEXTUREMODULATE_ALPHA),
90
91
(SDL_BLENDMODE_NONE | SDL_BLENDMODE_MASK |
SDL_BLENDMODE_BLEND | SDL_BLENDMODE_ADD | SDL_BLENDMODE_MOD),
92
(SDL_TEXTURESCALEMODE_NONE | SDL_TEXTURESCALEMODE_FAST),
93
14,
94
{
95
96
97
98
99
100
101
102
103
SDL_PIXELFORMAT_INDEX8,
SDL_PIXELFORMAT_RGB555,
SDL_PIXELFORMAT_RGB565,
SDL_PIXELFORMAT_RGB888,
SDL_PIXELFORMAT_BGR888,
SDL_PIXELFORMAT_ARGB8888,
SDL_PIXELFORMAT_RGBA8888,
SDL_PIXELFORMAT_ABGR8888,
SDL_PIXELFORMAT_BGRA8888,
104
105
SDL_PIXELFORMAT_YV12,
SDL_PIXELFORMAT_IYUV,
106
SDL_PIXELFORMAT_YUY2,
107
108
SDL_PIXELFORMAT_UYVY,
SDL_PIXELFORMAT_YVYU},
109
110
111
112
113
114
0,
0}
};
typedef struct
{
115
Uint32 format;
116
SDL_bool updateSize;
117
118
119
int current_texture;
SDL_Texture *texture[3];
SDL_Surface surface;
120
121
SDL_Renderer *renderer;
SDL_DirtyRectList dirty;
122
} SW_RenderData;
123
124
125
126
127
128
static SDL_Texture *
CreateTexture(SDL_Renderer * renderer, Uint32 format, int w, int h)
{
SDL_Texture *texture;
129
texture = (SDL_Texture *) SDL_calloc(1, sizeof(*texture));
130
131
132
133
134
135
if (!texture) {
SDL_OutOfMemory();
return NULL;
}
texture->format = format;
136
texture->access = SDL_TEXTUREACCESS_STREAMING;
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
texture->w = w;
texture->h = h;
texture->renderer = renderer;
if (renderer->CreateTexture(renderer, texture) < 0) {
SDL_free(texture);
return NULL;
}
return texture;
}
static void
DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture)
{
renderer->DestroyTexture(renderer, texture);
SDL_free(texture);
}
static int
DisplayPaletteChanged(void *userdata, SDL_Palette * palette)
{
158
SW_RenderData *data = (SW_RenderData *) userdata;
159
160
161
162
163
164
165
166
167
168
169
170
171
int i;
for (i = 0; i < SDL_arraysize(data->texture); ++i) {
if (data->texture[i] && data->renderer->SetTexturePalette) {
data->renderer->SetTexturePalette(data->renderer,
data->texture[i],
palette->colors, 0,
palette->ncolors);
}
}
return 0;
}
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
void
Setup_SoftwareRenderer(SDL_Renderer * renderer)
{
renderer->CreateTexture = SW_CreateTexture;
renderer->QueryTexturePixels = SW_QueryTexturePixels;
renderer->SetTexturePalette = SW_SetTexturePalette;
renderer->GetTexturePalette = SW_GetTexturePalette;
renderer->SetTextureColorMod = SW_SetTextureColorMod;
renderer->SetTextureAlphaMod = SW_SetTextureAlphaMod;
renderer->SetTextureBlendMode = SW_SetTextureBlendMode;
renderer->SetTextureScaleMode = SW_SetTextureScaleMode;
renderer->UpdateTexture = SW_UpdateTexture;
renderer->LockTexture = SW_LockTexture;
renderer->UnlockTexture = SW_UnlockTexture;
renderer->DestroyTexture = SW_DestroyTexture;
renderer->info.mod_modes = SW_RenderDriver.info.mod_modes;
renderer->info.blend_modes = SW_RenderDriver.info.blend_modes;
renderer->info.scale_modes = SW_RenderDriver.info.scale_modes;
renderer->info.num_texture_formats =
SW_RenderDriver.info.num_texture_formats;
SDL_memcpy(renderer->info.texture_formats,
SW_RenderDriver.info.texture_formats,
sizeof(renderer->info.texture_formats));;
renderer->info.max_texture_width = SW_RenderDriver.info.max_texture_width;
renderer->info.max_texture_height =
SW_RenderDriver.info.max_texture_height;
}
201
SDL_Renderer *
202
SW_CreateRenderer(SDL_Window * window, Uint32 flags)
203
204
205
206
{
SDL_VideoDisplay *display = SDL_GetDisplayFromWindow(window);
SDL_DisplayMode *displayMode = &display->current_mode;
SDL_Renderer *renderer;
207
SW_RenderData *data;
208
209
210
int i, n;
int bpp;
Uint32 Rmask, Gmask, Bmask, Amask;
211
Uint32 renderer_flags;
212
const char *desired_driver;
213
214
215
216
217
218
219
220
221
222
223
224
225
if (!SDL_PixelFormatEnumToMasks
(displayMode->format, &bpp, &Rmask, &Gmask, &Bmask, &Amask)) {
SDL_SetError("Unknown display format");
return NULL;
}
renderer = (SDL_Renderer *) SDL_calloc(1, sizeof(*renderer));
if (!renderer) {
SDL_OutOfMemory();
return NULL;
}
226
data = (SW_RenderData *) SDL_calloc(1, sizeof(*data));
227
if (!data) {
228
SW_DestroyRenderer(renderer);
229
230
231
SDL_OutOfMemory();
return NULL;
}
232
233
renderer->ActivateRenderer = SW_ActivateRenderer;
renderer->DisplayModeChanged = SW_DisplayModeChanged;
234
235
236
237
238
renderer->RenderDrawPoints = SW_RenderDrawPoints;
renderer->RenderDrawLines = SW_RenderDrawLines;
renderer->RenderDrawRects = SW_RenderDrawRects;
renderer->RenderFillRects = SW_RenderFillRects;
239
renderer->RenderCopy = SW_RenderCopy;
240
241
renderer->RenderReadPixels = SW_RenderReadPixels;
renderer->RenderWritePixels = SW_RenderWritePixels;
242
243
renderer->RenderPresent = SW_RenderPresent;
renderer->DestroyRenderer = SW_DestroyRenderer;
244
245
renderer->info.name = SW_RenderDriver.info.name;
renderer->info.flags = 0;
246
247
renderer->window = window->id;
renderer->driverdata = data;
248
Setup_SoftwareRenderer(renderer);
249
250
251
if (flags & SDL_RENDERER_PRESENTFLIP2) {
renderer->info.flags |= SDL_RENDERER_PRESENTFLIP2;
252
n = 2;
253
254
} else if (flags & SDL_RENDERER_PRESENTFLIP3) {
renderer->info.flags |= SDL_RENDERER_PRESENTFLIP3;
255
256
n = 3;
} else {
257
renderer->info.flags |= SDL_RENDERER_PRESENTCOPY;
258
259
n = 1;
}
260
data->format = displayMode->format;
261
262
/* Find a render driver that we can use to display data */
263
264
265
266
renderer_flags = (SDL_RENDERER_SINGLEBUFFER |
SDL_RENDERER_PRESENTDISCARD);
if (flags & SDL_RENDERER_PRESENTVSYNC) {
renderer_flags |= SDL_RENDERER_PRESENTVSYNC;
267
}
268
desired_driver = SDL_getenv("SDL_VIDEO_RENDERER_SWDRIVER");
269
270
for (i = 0; i < display->num_render_drivers; ++i) {
SDL_RenderDriver *driver = &display->render_drivers[i];
271
272
273
274
275
276
277
278
279
280
if (driver->info.name == SW_RenderDriver.info.name) {
continue;
}
if (desired_driver
&& SDL_strcasecmp(desired_driver, driver->info.name) != 0) {
continue;
}
data->renderer = driver->CreateRenderer(window, renderer_flags);
if (data->renderer) {
break;
281
282
283
}
}
if (i == display->num_render_drivers) {
284
SW_DestroyRenderer(renderer);
285
286
287
SDL_SetError("Couldn't find display render driver");
return NULL;
}
288
289
if (data->renderer->info.flags & SDL_RENDERER_PRESENTVSYNC) {
renderer->info.flags |= SDL_RENDERER_PRESENTVSYNC;
290
291
292
293
294
295
296
}
/* Create the textures we'll use for display */
for (i = 0; i < n; ++i) {
data->texture[i] =
CreateTexture(data->renderer, data->format, window->w, window->h);
if (!data->texture[i]) {
297
SW_DestroyRenderer(renderer);
298
299
300
301
302
303
304
305
306
return NULL;
}
}
data->current_texture = 0;
/* Create a surface we'll use for rendering */
data->surface.flags = SDL_PREALLOC;
data->surface.format = SDL_AllocFormat(bpp, Rmask, Gmask, Bmask, Amask);
if (!data->surface.format) {
307
SW_DestroyRenderer(renderer);
308
309
310
311
312
313
314
315
316
return NULL;
}
SDL_SetSurfacePalette(&data->surface, display->palette);
/* Set up a palette watch on the display palette */
if (display->palette) {
SDL_AddPaletteWatch(display->palette, DisplayPaletteChanged, data);
}
317
318
319
return renderer;
}
320
321
322
323
static int
SW_ActivateRenderer(SDL_Renderer * renderer)
{
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
324
325
SDL_Window *window = SDL_GetWindowFromID(renderer->window);
int i, n;
326
327
328
329
330
331
if (data->renderer && data->renderer->ActivateRenderer) {
if (data->renderer->ActivateRenderer(data->renderer) < 0) {
return -1;
}
}
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
if (data->updateSize) {
/* Recreate the textures for the new window size */
if (renderer->info.flags & SDL_RENDERER_PRESENTFLIP2) {
n = 2;
} else if (renderer->info.flags & SDL_RENDERER_PRESENTFLIP3) {
n = 3;
} else {
n = 1;
}
for (i = 0; i < n; ++i) {
if (data->texture[i]) {
DestroyTexture(data->renderer, data->texture[i]);
data->texture[i] = 0;
}
}
for (i = 0; i < n; ++i) {
data->texture[i] =
CreateTexture(data->renderer, data->format, window->w,
window->h);
if (!data->texture[i]) {
return -1;
}
}
data->updateSize = SDL_FALSE;
}
357
358
359
360
361
362
363
364
365
366
367
368
369
return 0;
}
static int
SW_DisplayModeChanged(SDL_Renderer * renderer)
{
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
if (data->renderer && data->renderer->DisplayModeChanged) {
if (data->renderer->DisplayModeChanged(data->renderer) < 0) {
return -1;
}
}
370
/* Rebind the context to the window area */
371
data->updateSize = SDL_TRUE;
372
return SW_ActivateRenderer(renderer);
373
374
}
375
static int
376
SW_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture)
377
378
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
379
380
texture->driverdata =
SDL_SW_CreateYUVTexture(texture->format, texture->w, texture->h);
381
382
383
384
385
386
387
388
389
390
391
392
393
} else {
int bpp;
Uint32 Rmask, Gmask, Bmask, Amask;
if (!SDL_PixelFormatEnumToMasks
(texture->format, &bpp, &Rmask, &Gmask, &Bmask, &Amask)) {
SDL_SetError("Unknown texture format");
return -1;
}
texture->driverdata =
SDL_CreateRGBSurface(0, texture->w, texture->h, bpp, Rmask, Gmask,
Bmask, Amask);
394
395
396
397
398
399
SDL_SetSurfaceColorMod(texture->driverdata, texture->r, texture->g,
texture->b);
SDL_SetSurfaceAlphaMod(texture->driverdata, texture->a);
SDL_SetSurfaceBlendMode(texture->driverdata, texture->blendMode);
SDL_SetSurfaceScaleMode(texture->driverdata, texture->scaleMode);
400
401
402
if (texture->access == SDL_TEXTUREACCESS_STATIC) {
SDL_SetSurfaceRLE(texture->driverdata, 1);
}
403
404
405
406
407
408
409
410
411
}
if (!texture->driverdata) {
return -1;
}
return 0;
}
static int
412
413
SW_QueryTexturePixels(SDL_Renderer * renderer, SDL_Texture * texture,
void **pixels, int *pitch)
414
415
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
416
417
418
return SDL_SW_QueryYUVTexturePixels((SDL_SW_YUVTexture *)
texture->driverdata, pixels,
pitch);
419
420
421
422
423
424
425
426
427
428
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
*pixels = surface->pixels;
*pitch = surface->pitch;
return 0;
}
}
static int
429
430
SW_SetTexturePalette(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Color * colors, int firstcolor, int ncolors)
431
432
433
434
435
436
437
438
439
440
441
442
443
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("YUV textures don't have a palette");
return -1;
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
return SDL_SetPaletteColors(surface->format->palette, colors,
firstcolor, ncolors);
}
}
static int
444
445
SW_GetTexturePalette(SDL_Renderer * renderer, SDL_Texture * texture,
SDL_Color * colors, int firstcolor, int ncolors)
446
447
448
449
450
451
452
453
454
455
456
457
458
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("YUV textures don't have a palette");
return -1;
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
SDL_memcpy(colors, &surface->format->palette->colors[firstcolor],
ncolors * sizeof(*colors));
return 0;
}
}
459
460
461
static int
SW_SetTextureColorMod(SDL_Renderer * renderer, SDL_Texture * texture)
{
462
463
464
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
return SDL_SetSurfaceColorMod(surface, texture->r, texture->g,
texture->b);
465
466
467
468
469
}
static int
SW_SetTextureAlphaMod(SDL_Renderer * renderer, SDL_Texture * texture)
{
470
471
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
return SDL_SetSurfaceAlphaMod(surface, texture->a);
472
473
474
475
476
}
static int
SW_SetTextureBlendMode(SDL_Renderer * renderer, SDL_Texture * texture)
{
477
478
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
return SDL_SetSurfaceBlendMode(surface, texture->blendMode);
479
480
481
482
483
}
static int
SW_SetTextureScaleMode(SDL_Renderer * renderer, SDL_Texture * texture)
{
484
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
485
return SDL_SetSurfaceScaleMode(surface, texture->scaleMode);
486
487
}
488
static int
489
490
SW_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, const void *pixels, int pitch)
491
492
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
493
494
495
return SDL_SW_UpdateYUVTexture((SDL_SW_YUVTexture *)
texture->driverdata, rect, pixels,
pitch);
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
Uint8 *src, *dst;
int row;
size_t length;
src = (Uint8 *) pixels;
dst =
(Uint8 *) surface->pixels + rect->y * surface->pitch +
rect->x * surface->format->BytesPerPixel;
length = rect->w * surface->format->BytesPerPixel;
for (row = 0; row < rect->h; ++row) {
SDL_memcpy(dst, src, length);
src += pitch;
dst += surface->pitch;
}
return 0;
}
}
static int
517
518
519
SW_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, int markDirty, void **pixels,
int *pitch)
520
521
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
522
523
524
return SDL_SW_LockYUVTexture((SDL_SW_YUVTexture *)
texture->driverdata, rect, markDirty,
pixels, pitch);
525
526
527
528
529
530
531
532
533
534
535
536
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
*pixels =
(void *) ((Uint8 *) surface->pixels + rect->y * surface->pitch +
rect->x * surface->format->BytesPerPixel);
*pitch = surface->pitch;
return 0;
}
}
static void
537
SW_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture)
538
539
540
541
542
543
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SW_UnlockYUVTexture((SDL_SW_YUVTexture *) texture->driverdata);
}
}
544
static int
545
546
SW_RenderDrawPoints(SDL_Renderer * renderer, const SDL_Point * points,
int count)
547
548
{
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
549
SDL_Texture *texture = data->texture[data->current_texture];
550
SDL_Rect rect;
551
552
553
554
555
556
557
558
559
560
561
562
563
int i;
int x, y;
int status = 0;
/* Get the smallest rectangle that contains everything */
rect.x = 0;
rect.y = 0;
rect.w = texture->w;
rect.h = texture->h;
if (!SDL_EnclosePoints(points, count, &rect, &rect)) {
/* Nothing to draw */
return 0;
}
564
565
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
566
567
568
SDL_AddDirtyRect(&data->dirty, &rect);
}
569
if (data->renderer->LockTexture(data->renderer, texture, &rect, 1,
570
571
572
573
574
&data->surface.pixels,
&data->surface.pitch) < 0) {
return -1;
}
575
576
data->surface.clip_rect.w = data->surface.w = rect.w;
data->surface.clip_rect.h = data->surface.h = rect.h;
577
578
/* Draw the points! */
579
580
if (renderer->blendMode == SDL_BLENDMODE_NONE ||
renderer->blendMode == SDL_BLENDMODE_MASK) {
581
582
583
584
585
586
587
Uint32 color = SDL_MapRGBA(data->surface.format,
renderer->r, renderer->g, renderer->b,
renderer->a);
for (i = 0; i < count; ++i) {
x = points[i].x - rect.x;
y = points[i].y - rect.y;
588
589
590
status = SDL_DrawPoint(&data->surface, x, y, color);
}
591
} else {
592
593
594
595
596
597
598
599
600
for (i = 0; i < count; ++i) {
x = points[i].x - rect.x;
y = points[i].y - rect.y;
status = SDL_BlendPoint(&data->surface, x, y,
renderer->blendMode,
renderer->r, renderer->g, renderer->b,
renderer->a);
}
601
602
}
603
604
data->renderer->UnlockTexture(data->renderer, texture);
605
606
607
return status;
}
608
static int
609
610
SW_RenderDrawLines(SDL_Renderer * renderer, const SDL_Point * points,
int count)
611
612
{
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
SDL_Texture *texture = data->texture[data->current_texture];
SDL_Rect clip, rect;
int i;
int x1, y1, x2, y2;
int status = 0;
/* Get the smallest rectangle that contains everything */
clip.x = 0;
clip.y = 0;
clip.w = texture->w;
clip.h = texture->h;
SDL_EnclosePoints(points, count, NULL, &rect);
if (!SDL_IntersectRect(&rect, &clip, &rect)) {
/* Nothing to draw */
return 0;
628
}
629
630
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
631
632
633
SDL_AddDirtyRect(&data->dirty, &rect);
}
634
if (data->renderer->LockTexture(data->renderer, texture, &rect, 1,
635
&data->surface.pixels,
636
637
638
639
&data->surface.pitch) < 0) {
return -1;
}
640
641
data->surface.clip_rect.w = data->surface.w = rect.w;
data->surface.clip_rect.h = data->surface.h = rect.h;
642
643
/* Draw the points! */
644
645
if (renderer->blendMode == SDL_BLENDMODE_NONE ||
renderer->blendMode == SDL_BLENDMODE_MASK) {
646
647
648
649
650
651
652
653
654
Uint32 color = SDL_MapRGBA(data->surface.format,
renderer->r, renderer->g, renderer->b,
renderer->a);
for (i = 1; i < count; ++i) {
x1 = points[i-1].x - rect.x;
y1 = points[i-1].y - rect.y;
x2 = points[i].x - rect.x;
y2 = points[i].y - rect.y;
655
656
657
status = SDL_DrawLine(&data->surface, x1, y1, x2, y2, color);
}
658
} else {
659
660
661
662
663
664
665
666
667
668
669
for (i = 1; i < count; ++i) {
x1 = points[i-1].x - rect.x;
y1 = points[i-1].y - rect.y;
x2 = points[i].x - rect.x;
y2 = points[i].y - rect.y;
status = SDL_BlendLine(&data->surface, x1, y1, x2, y2,
renderer->blendMode,
renderer->r, renderer->g, renderer->b,
renderer->a);
}
670
671
}
672
673
data->renderer->UnlockTexture(data->renderer, texture);
674
675
676
return status;
}
677
static int
678
679
SW_RenderDrawRects(SDL_Renderer * renderer, const SDL_Rect ** rects,
int count)
680
{
681
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
682
683
684
685
686
SDL_Texture *texture = data->texture[data->current_texture];
SDL_Rect clip, rect;
Uint32 color = 0;
int i;
int status = 0;
687
688
689
690
691
clip.x = 0;
clip.y = 0;
clip.w = texture->w;
clip.h = texture->h;
692
693
694
695
696
697
if (renderer->blendMode == SDL_BLENDMODE_NONE ||
renderer->blendMode == SDL_BLENDMODE_MASK) {
color = SDL_MapRGBA(data->surface.format,
renderer->r, renderer->g, renderer->b,
renderer->a);
698
}
699
700
701
702
703
704
for (i = 0; i < count; ++i) {
if (!SDL_IntersectRect(rects[i], &clip, &rect)) {
/* Nothing to draw */
continue;
}
705
706
707
708
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
SDL_AddDirtyRect(&data->dirty, &rect);
}
709
710
711
712
713
714
if (data->renderer->LockTexture(data->renderer, texture, &rect, 1,
&data->surface.pixels,
&data->surface.pitch) < 0) {
return -1;
}
715
716
717
718
719
720
data->surface.clip_rect.w = data->surface.w = rect.w;
data->surface.clip_rect.h = data->surface.h = rect.h;
if (renderer->blendMode == SDL_BLENDMODE_NONE ||
renderer->blendMode == SDL_BLENDMODE_MASK) {
721
status = SDL_DrawRect(&data->surface, NULL, color);
722
723
724
725
726
727
728
729
730
} else {
status = SDL_BlendRect(&data->surface, NULL,
renderer->blendMode,
renderer->r, renderer->g, renderer->b,
renderer->a);
}
data->renderer->UnlockTexture(data->renderer, texture);
}
731
return status;
732
733
}
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
static int
SW_RenderFillRects(SDL_Renderer * renderer, const SDL_Rect ** rects,
int count)
{
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
SDL_Texture *texture = data->texture[data->current_texture];
SDL_Rect clip, rect;
Uint32 color = 0;
int i;
int status = 0;
clip.x = 0;
clip.y = 0;
clip.w = texture->w;
clip.h = texture->h;
if (renderer->blendMode == SDL_BLENDMODE_NONE ||
renderer->blendMode == SDL_BLENDMODE_MASK) {
color = SDL_MapRGBA(data->surface.format,
renderer->r, renderer->g, renderer->b,
renderer->a);
}
for (i = 0; i < count; ++i) {
if (!SDL_IntersectRect(rects[i], &clip, &rect)) {
/* Nothing to draw */
continue;
}
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
SDL_AddDirtyRect(&data->dirty, &rect);
}
if (data->renderer->LockTexture(data->renderer, texture, &rect, 1,
&data->surface.pixels,
&data->surface.pitch) < 0) {
return -1;
}
data->surface.clip_rect.w = data->surface.w = rect.w;
data->surface.clip_rect.h = data->surface.h = rect.h;
if (renderer->blendMode == SDL_BLENDMODE_NONE ||
renderer->blendMode == SDL_BLENDMODE_MASK) {
status = SDL_FillRect(&data->surface, NULL, color);
} else {
status = SDL_BlendFillRect(&data->surface, NULL,
renderer->blendMode,
renderer->r, renderer->g, renderer->b,
renderer->a);
}
data->renderer->UnlockTexture(data->renderer, texture);
}
return status;
}
791
static int
792
SW_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
793
const SDL_Rect * srcrect, const SDL_Rect * dstrect)
794
{
795
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
796
SDL_Window *window = SDL_GetWindowFromID(renderer->window);
797
int status;
798
799
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
800
801
SDL_AddDirtyRect(&data->dirty, dstrect);
}
802
803
804
805
806
if (data->renderer->LockTexture(data->renderer,
data->texture[data->current_texture],
dstrect, 1, &data->surface.pixels,
&data->surface.pitch) < 0) {
807
808
809
return -1;
}
810
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
811
812
813
814
status =
SDL_SW_CopyYUVToRGB((SDL_SW_YUVTexture *) texture->driverdata,
srcrect, data->format, dstrect->w, dstrect->h,
data->surface.pixels, data->surface.pitch);
815
816
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
817
818
819
820
821
822
823
824
825
826
827
828
SDL_Rect real_srcrect = *srcrect;
SDL_Rect real_dstrect;
data->surface.w = dstrect->w;
data->surface.h = dstrect->h;
data->surface.clip_rect.w = dstrect->w;
data->surface.clip_rect.h = dstrect->h;
real_dstrect = data->surface.clip_rect;
status =
SDL_LowerBlit(surface, &real_srcrect, &data->surface,
&real_dstrect);
829
}
830
831
832
data->renderer->UnlockTexture(data->renderer,
data->texture[data->current_texture]);
return status;
833
834
}
835
836
static int
SW_RenderReadPixels(SDL_Renderer * renderer, const SDL_Rect * rect,
837
Uint32 format, void * pixels, int pitch)
838
839
840
841
842
843
844
845
846
847
{
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
if (data->renderer->LockTexture(data->renderer,
data->texture[data->current_texture],
rect, 0, &data->surface.pixels,
&data->surface.pitch) < 0) {
return -1;
}
848
849
850
SDL_ConvertPixels(rect->w, rect->h,
data->format, data->surface.pixels, data->surface.pitch,
format, pixels, pitch);
851
852
853
854
855
856
857
858
data->renderer->UnlockTexture(data->renderer,
data->texture[data->current_texture]);
return 0;
}
static int
SW_RenderWritePixels(SDL_Renderer * renderer, const SDL_Rect * rect,
859
Uint32 format, const void * pixels, int pitch)
860
861
862
863
864
865
866
867
868
869
870
871
872
873
{
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
SDL_AddDirtyRect(&data->dirty, rect);
}
if (data->renderer->LockTexture(data->renderer,
data->texture[data->current_texture],
rect, 1, &data->surface.pixels,
&data->surface.pitch) < 0) {
return -1;
}
874
875
SDL_ConvertPixels(rect->w, rect->h, format, pixels, pitch,
data->format, data->surface.pixels, data->surface.pitch);
876
877
878
879
880
881
data->renderer->UnlockTexture(data->renderer,
data->texture[data->current_texture]);
return 0;
}
882
static void
883
SW_RenderPresent(SDL_Renderer * renderer)
884
{
885
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
886
SDL_Texture *texture = data->texture[data->current_texture];
887
888
/* Send the data to the display */
889
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
890
891
892
SDL_DirtyRect *dirty;
for (dirty = data->dirty.list; dirty; dirty = dirty->next) {
data->renderer->RenderCopy(data->renderer, texture, &dirty->rect,
893
&dirty->rect);
894
895
896
897
898
899
900
901
}
SDL_ClearDirtyRects(&data->dirty);
} else {
SDL_Rect rect;
rect.x = 0;
rect.y = 0;
rect.w = texture->w;
rect.h = texture->h;
902
data->renderer->RenderCopy(data->renderer, texture, &rect, &rect);
903
904
905
906
}
data->renderer->RenderPresent(data->renderer);
/* Update the flipping chain, if any */
907
if (renderer->info.flags & SDL_RENDERER_PRESENTFLIP2) {
908
data->current_texture = (data->current_texture + 1) % 2;
909
} else if (renderer->info.flags & SDL_RENDERER_PRESENTFLIP3) {
910
data->current_texture = (data->current_texture + 1) % 3;
911
912
913
914
}
}
static void
915
SW_DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture)
916
917
918
919
920
921
922
923
924
925
926
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SW_DestroyYUVTexture((SDL_SW_YUVTexture *) texture->driverdata);
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
SDL_FreeSurface(surface);
}
}
static void
927
SW_DestroyRenderer(SDL_Renderer * renderer)
928
{
929
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
930
931
SDL_Window *window = SDL_GetWindowFromID(renderer->window);
SDL_VideoDisplay *display = SDL_GetDisplayFromWindow(window);
932
933
934
int i;
if (data) {
935
936
937
for (i = 0; i < SDL_arraysize(data->texture); ++i) {
if (data->texture[i]) {
DestroyTexture(data->renderer, data->texture[i]);
938
939
}
}
940
941
942
943
944
945
946
947
948
949
950
if (data->surface.format) {
SDL_SetSurfacePalette(&data->surface, NULL);
SDL_FreeFormat(data->surface.format);
}
if (display->palette) {
SDL_DelPaletteWatch(display->palette, DisplayPaletteChanged,
data);
}
if (data->renderer) {
data->renderer->DestroyRenderer(data->renderer);
}
951
952
953
954
955
956
957
SDL_FreeDirtyRects(&data->dirty);
SDL_free(data);
}
SDL_free(renderer);
}
/* vi: set ts=4 sw=4 expandtab: */