This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_renderer_sw.c
958 lines (840 loc) · 32 KB
1
2
/*
SDL - Simple DirectMedia Layer
3
Copyright (C) 1997-2010 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
SDL_VideoDisplay *display = window->display;
205
206
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
renderer->window = window;
247
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
SDL_Window *window = renderer->window;
325
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
for (i = 0; i < count; ++i) {
701
/* FIXME: We don't want to draw clipped edges */
702
703
704
705
if (!SDL_IntersectRect(rects[i], &clip, &rect)) {
/* Nothing to draw */
continue;
}
706
707
708
709
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
SDL_AddDirtyRect(&data->dirty, &rect);
}
710
711
712
713
714
715
if (data->renderer->LockTexture(data->renderer, texture, &rect, 1,
&data->surface.pixels,
&data->surface.pitch) < 0) {
return -1;
}
716
717
718
719
720
721
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) {
722
status = SDL_DrawRect(&data->surface, NULL, color);
723
724
725
726
727
728
729
730
731
} else {
status = SDL_BlendRect(&data->surface, NULL,
renderer->blendMode,
renderer->r, renderer->g, renderer->b,
renderer->a);
}
data->renderer->UnlockTexture(data->renderer, texture);
}
732
return status;
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
791
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;
}
792
static int
793
SW_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
794
const SDL_Rect * srcrect, const SDL_Rect * dstrect)
795
{
796
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
797
SDL_Window *window = renderer->window;
798
int status;
799
800
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
801
802
SDL_AddDirtyRect(&data->dirty, dstrect);
}
803
804
805
806
807
if (data->renderer->LockTexture(data->renderer,
data->texture[data->current_texture],
dstrect, 1, &data->surface.pixels,
&data->surface.pitch) < 0) {
808
809
810
return -1;
}
811
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
812
813
814
815
status =
SDL_SW_CopyYUVToRGB((SDL_SW_YUVTexture *) texture->driverdata,
srcrect, data->format, dstrect->w, dstrect->h,
data->surface.pixels, data->surface.pitch);
816
817
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
818
819
820
821
822
823
824
825
826
827
828
829
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);
830
}
831
832
833
data->renderer->UnlockTexture(data->renderer,
data->texture[data->current_texture]);
return status;
834
835
}
836
837
static int
SW_RenderReadPixels(SDL_Renderer * renderer, const SDL_Rect * rect,
838
Uint32 format, void * pixels, int pitch)
839
840
841
842
843
844
845
846
847
848
{
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;
}
849
850
851
SDL_ConvertPixels(rect->w, rect->h,
data->format, data->surface.pixels, data->surface.pitch,
format, pixels, pitch);
852
853
854
855
856
857
858
859
data->renderer->UnlockTexture(data->renderer,
data->texture[data->current_texture]);
return 0;
}
static int
SW_RenderWritePixels(SDL_Renderer * renderer, const SDL_Rect * rect,
860
Uint32 format, const void * pixels, int pitch)
861
862
863
864
865
866
867
868
869
870
871
872
873
874
{
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;
}
875
876
SDL_ConvertPixels(rect->w, rect->h, format, pixels, pitch,
data->format, data->surface.pixels, data->surface.pitch);
877
878
879
880
881
882
data->renderer->UnlockTexture(data->renderer,
data->texture[data->current_texture]);
return 0;
}
883
static void
884
SW_RenderPresent(SDL_Renderer * renderer)
885
{
886
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
887
SDL_Texture *texture = data->texture[data->current_texture];
888
889
/* Send the data to the display */
890
if (data->renderer->info.flags & SDL_RENDERER_PRESENTCOPY) {
891
892
893
SDL_DirtyRect *dirty;
for (dirty = data->dirty.list; dirty; dirty = dirty->next) {
data->renderer->RenderCopy(data->renderer, texture, &dirty->rect,
894
&dirty->rect);
895
896
897
898
899
900
901
902
}
SDL_ClearDirtyRects(&data->dirty);
} else {
SDL_Rect rect;
rect.x = 0;
rect.y = 0;
rect.w = texture->w;
rect.h = texture->h;
903
data->renderer->RenderCopy(data->renderer, texture, &rect, &rect);
904
905
906
907
}
data->renderer->RenderPresent(data->renderer);
/* Update the flipping chain, if any */
908
if (renderer->info.flags & SDL_RENDERER_PRESENTFLIP2) {
909
data->current_texture = (data->current_texture + 1) % 2;
910
} else if (renderer->info.flags & SDL_RENDERER_PRESENTFLIP3) {
911
data->current_texture = (data->current_texture + 1) % 3;
912
913
914
915
}
}
static void
916
SW_DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture)
917
918
919
920
921
922
923
924
925
926
927
{
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
928
SW_DestroyRenderer(SDL_Renderer * renderer)
929
{
930
SW_RenderData *data = (SW_RenderData *) renderer->driverdata;
931
932
SDL_Window *window = renderer->window;
SDL_VideoDisplay *display = window->display;
933
934
935
int i;
if (data) {
936
937
938
for (i = 0; i < SDL_arraysize(data->texture); ++i) {
if (data->texture[i]) {
DestroyTexture(data->renderer, data->texture[i]);
939
940
}
}
941
942
943
944
945
946
947
948
949
950
951
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);
}
952
953
954
955
956
957
958
SDL_FreeDirtyRects(&data->dirty);
SDL_free(data);
}
SDL_free(renderer);
}
/* vi: set ts=4 sw=4 expandtab: */