This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_ndsrender.c
576 lines (519 loc) · 19.3 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
/*
SDL - Simple DirectMedia Layer
Copyright (C) 1997-2006 Sam Lantinga
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 <stdio.h>
#include <stdlib.h>
#include <nds.h>
#include "SDL_config.h"
#include "SDL_video.h"
#include "../SDL_sysvideo.h"
#include "../SDL_yuv_sw_c.h"
#include "../SDL_renderer_sw.h"
/* SDL surface based renderer implementation */
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
static SDL_Renderer *NDS_CreateRenderer(SDL_Window * window, Uint32 flags);
static int NDS_ActivateRenderer(SDL_Renderer * renderer);
static int NDS_DisplayModeChanged(SDL_Renderer * renderer);
static int NDS_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture);
static int NDS_QueryTexturePixels(SDL_Renderer * renderer,
SDL_Texture * texture, void **pixels,
int *pitch);
static int NDS_SetTexturePalette(SDL_Renderer * renderer,
SDL_Texture * texture,
const SDL_Color * colors, int firstcolor,
int ncolors);
static int NDS_GetTexturePalette(SDL_Renderer * renderer,
SDL_Texture * texture, SDL_Color * colors,
int firstcolor, int ncolors);
static int NDS_SetTextureColorMod(SDL_Renderer * renderer,
SDL_Texture * texture);
static int NDS_SetTextureAlphaMod(SDL_Renderer * renderer,
SDL_Texture * texture);
static int NDS_SetTextureBlendMode(SDL_Renderer * renderer,
SDL_Texture * texture);
static int NDS_SetTextureScaleMode(SDL_Renderer * renderer,
SDL_Texture * texture);
static int NDS_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, const void *pixels,
int pitch);
static int NDS_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, int markDirty, void **pixels,
int *pitch);
static void NDS_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture);
static void NDS_DirtyTexture(SDL_Renderer * renderer,
SDL_Texture * texture, int numrects,
const SDL_Rect * rects);
static int NDS_RenderFill(SDL_Renderer * renderer, Uint8 r, Uint8 g,
70
Uint8 b, Uint8 a, const SDL_Rect * rect);
71
static int NDS_RenderCopy(SDL_Renderer * renderer,
72
73
74
SDL_Texture * texture,
const SDL_Rect * srcrect,
const SDL_Rect * dstrect);
75
76
77
static void NDS_RenderPresent(SDL_Renderer * renderer);
static void NDS_DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture);
static void NDS_DestroyRenderer(SDL_Renderer * renderer);
78
79
80
81
SDL_RenderDriver NDS_RenderDriver = {
NDS_CreateRenderer,
82
83
84
85
86
87
88
89
90
91
92
93
94
95
{ "nds", /* char* name */
(SDL_RENDERER_SINGLEBUFFER|SDL_RENDERER_ACCELERATED), /* u32 flags */
(SDL_TEXTUREMODULATE_NONE), /* u32 mod_modes */
(SDL_TEXTUREBLENDMODE_NONE), /* u32 blend_modes */
(SDL_TEXTURESCALEMODE_NONE), /* u32 scale_modes */
3, /* u32 num_texture_formats */
{
SDL_PIXELFORMAT_INDEX8,
SDL_PIXELFORMAT_RGB555,
SDL_PIXELFORMAT_RGB565
}, /* u32 texture_formats[20] */
(256), /* int max_texture_width */
(256), /* int max_texture_height */
}
96
97
98
99
};
typedef struct
{
100
101
102
bg_attribute *bg;
u8 bg_taken[4];
int sub;
103
} NDS_RenderData;
104
105
106
107
typedef struct
{
enum { NDSTX_BG, NDSTX_SPR } type;
108
int hw_index;
109
110
111
struct { int w, h, pitch, bpp; } dim;
u16 *vram;
} NDS_TextureData;
112
113
114
115
116
117
118
/* this is mainly hackish testing/debugging stuff to get cleaned up soon
anything named sdlds_blah shouldn't make it into the stable version
*/
u16
119
120
121
sdlds_rgb2bgr(u16 c)
{
/* hack to get the proper colors until I actually get BGR555 to work right */
122
123
124
125
126
127
128
u16 Rmask = 0x7C00, Bmask = 0x001F, GAmask = 0x83E0, r, b;
r = (c & Rmask) >> 10;
b = (c & Bmask) << 10;
return (c & GAmask) | r | b;
}
void
129
130
131
132
133
134
sdlds_print_pixfmt_info(SDL_PixelFormat * f)
{
if (!f)
return;
printf("bpp: %d\nRGBA: %x %x %x %x\n",
f->BitsPerPixel, f->Rmask, f->Gmask, f->Bmask, f->Amask);
135
136
137
}
void
138
139
140
141
142
143
144
sdlds_print_surface_info(SDL_Surface * s)
{
if (!s)
return;
printf("flags: %x\nsize: %dx%d, pitch: %d\nlocked: %d, refcount: %d\n",
s->flags, s->w, s->h, s->pitch, s->locked, s->refcount);
sdlds_print_pixfmt_info(s->format);
145
146
}
147
/* again the above shouldn't make it into the stable version */
148
149
SDL_Renderer *
150
NDS_CreateRenderer(SDL_Window * window, Uint32 flags)
151
152
153
154
{
SDL_VideoDisplay *display = SDL_GetDisplayFromWindow(window);
SDL_DisplayMode *displayMode = &display->current_mode;
SDL_Renderer *renderer;
155
NDS_RenderData *data;
156
int i, n;
157
int bpp = 15;
158
159
Uint32 Rmask, Gmask, Bmask, Amask;
/* Uint32 Rmask = 0x7C00, Gmask = 0x03E0, Bmask = 0x001F, Amask = 0x8000;
160
Uint32 Rmask = 0x001F, Gmask = 0x03E0, Bmask = 0x7C00, Amask = 0x8000;
161
*/
162
printf("+NDS_CreateRenderer\n");
163
164
165
166
167
168
169
/* hard coded this to BGR555 for now */
if (!SDL_PixelFormatEnumToMasks(SDL_PIXELFORMAT_BGR555, &bpp,
&Rmask, &Gmask, &Bmask, &Amask)) {
SDL_SetError("Unknown display format");
return NULL;
}
170
171
172
173
174
175
176
renderer = (SDL_Renderer *) SDL_calloc(1, sizeof(*renderer));
if (!renderer) {
SDL_OutOfMemory();
return NULL;
}
177
data = (NDS_RenderData *) SDL_malloc(sizeof(*data));
178
if (!data) {
179
NDS_DestroyRenderer(renderer);
180
181
182
183
184
SDL_OutOfMemory();
return NULL;
}
SDL_zerop(data);
185
186
187
188
189
renderer->RenderFill = NDS_RenderFill;
renderer->RenderCopy = NDS_RenderCopy;
renderer->RenderPresent = NDS_RenderPresent;
renderer->DestroyRenderer = NDS_DestroyRenderer;
renderer->info.name = NDS_RenderDriver.info.name;
190
191
192
renderer->info.flags = 0;
renderer->window = window->id;
renderer->driverdata = data;
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
218
219
220
221
222
renderer->CreateTexture = NDS_CreateTexture;
renderer->QueryTexturePixels = NDS_QueryTexturePixels;
renderer->SetTexturePalette = NDS_SetTexturePalette;
renderer->GetTexturePalette = NDS_GetTexturePalette;
renderer->SetTextureColorMod = NDS_SetTextureColorMod;
renderer->SetTextureAlphaMod = NDS_SetTextureAlphaMod;
renderer->SetTextureBlendMode = NDS_SetTextureBlendMode;
renderer->SetTextureScaleMode = NDS_SetTextureScaleMode;
renderer->UpdateTexture = NDS_UpdateTexture;
renderer->LockTexture = NDS_LockTexture;
renderer->UnlockTexture = NDS_UnlockTexture;
renderer->DirtyTexture = NDS_DirtyTexture;
renderer->DestroyTexture = NDS_DestroyTexture;
renderer->info.mod_modes = NDS_RenderDriver.info.mod_modes;
renderer->info.blend_modes = NDS_RenderDriver.info.blend_modes;
renderer->info.scale_modes = NDS_RenderDriver.info.scale_modes;
renderer->info.num_texture_formats =
NDS_RenderDriver.info.num_texture_formats;
SDL_memcpy(renderer->info.texture_formats,
NDS_RenderDriver.info.texture_formats,
sizeof(renderer->info.texture_formats));;
renderer->info.max_texture_width = NDS_RenderDriver.info.max_texture_width;
renderer->info.max_texture_height =
NDS_RenderDriver.info.max_texture_height;
/*data->fb = (u16*)0x06020000;*/
data->bg = &BACKGROUND;
data->bg_taken[2] = data->bg_taken[3] = 0;
data->sub = 0;
223
224
printf("-NDS_CreateRenderer\n");
225
226
227
return renderer;
}
228
229
230
231
232
static int
NDS_ActivateRenderer(SDL_Renderer * renderer)
{
NDS_RenderData *data = (NDS_RenderData *) renderer->driverdata;
/* stub. TODO: figure out what needs to be done, if anything. */
233
printf("!NDS_ActivateRenderer\n");
234
235
236
237
238
239
240
241
return 0;
}
static int
NDS_DisplayModeChanged(SDL_Renderer * renderer)
{
NDS_RenderData *data = (NDS_RenderData *) renderer->driverdata;
/* stub. TODO: figure out what needs to be done */
242
printf("!NDS_DisplayModeChanged\n");
243
244
245
246
247
248
249
return 0;
}
static int
NDS_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture)
{
NDS_RenderData *data = (NDS_RenderData *) renderer->driverdata;
250
NDS_TextureData *txdat = NULL;
251
printf("+NDS_CreateTexture\n");
252
253
254
255
256
257
258
259
260
261
262
263
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("Unsupported texture format");
return -1;
} 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;
}
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
/* conditional statements on w/h to place it as bg/sprite */
/*if(texture->w <= 64 && texture->h <= 64) {
sprites not implemented yet. elegant, I know.
} else*/ if(texture->w <= 256 && texture->h <= 256) {
int whichbg = -1;
if(!data->bg_taken[2]) {
whichbg = 2;
data->bg->bg2_rotation.xdx = 0x100;
data->bg->bg2_rotation.xdy = 0;
data->bg->bg2_rotation.ydx = 0;
data->bg->bg2_rotation.ydy = 0x100;
data->bg->bg2_rotation.centerX = 0;
data->bg->bg2_rotation.centerY = 0;
} else if(!data->bg_taken[3]) {
whichbg = 3;
data->bg->bg3_rotation.xdx = 0x100;
data->bg->bg3_rotation.xdy = 0;
data->bg->bg3_rotation.ydx = 0;
data->bg->bg3_rotation.ydy = 0x100;
data->bg->bg3_rotation.centerX = 0;
data->bg->bg3_rotation.centerY = 0;
}
if(whichbg >= 0) {
data->bg->control[whichbg] = (bpp == 8) ?
BG_BMP8_256x256 : BG_BMP16_256x256;
data->bg->scroll[whichbg].x = 0;
data->bg->scroll[whichbg].y = 0;
texture->driverdata = SDL_calloc(1, sizeof(NDS_TextureData));
txdat = (NDS_TextureData*)texture->driverdata;
txdat->type = NDSTX_BG;
txdat->hw_index = whichbg;
txdat->dim.w = texture->w;
txdat->dim.h = texture->h;
txdat->dim.pitch = 256 * (bpp/8);
txdat->dim.bpp = bpp;
txdat->vram = (u16*)(data->sub ?
BG_BMP_RAM_SUB(whichbg) : BG_BMP_RAM(whichbg));
} else {
SDL_SetError("Out of NDS backgrounds.");
}
} else {
SDL_SetError("Texture too big for NDS hardware.");
}
307
308
}
309
printf("-NDS_CreateTexture\n");
310
311
312
313
314
315
316
317
318
319
if (!texture->driverdata) {
return -1;
}
return 0;
}
static int
NDS_QueryTexturePixels(SDL_Renderer * renderer, SDL_Texture * texture,
void **pixels, int *pitch)
{
320
printf("+NDS_QueryTexturePixels\n");
321
322
323
324
325
326
327
328
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("Unsupported texture format");
return -1;
} else {
NDS_TextureData *txdat = (NDS_TextureData *) texture->driverdata;
*pixels = txdat->vram;
*pitch = txdat->dim.pitch;
329
printf("-NDS_QueryTexturePixels\n");
330
331
332
333
334
335
336
337
return 0;
}
}
static int
NDS_SetTexturePalette(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Color * colors, int firstcolor, int ncolors)
{
338
printf("+NDS_SetTexturePalette\n");
339
340
341
342
343
344
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("YUV textures don't have a palette");
return -1;
} else {
NDS_TextureData *txdat = (NDS_TextureData *) texture->driverdata;
/* TODO: mess with 8-bit modes and/or 16-color palette modes */
345
printf("-NDS_SetTexturePalette\n");
346
347
348
349
350
351
352
353
return 0;
}
}
static int
NDS_GetTexturePalette(SDL_Renderer * renderer, SDL_Texture * texture,
SDL_Color * colors, int firstcolor, int ncolors)
{
354
printf("+NDS_GetTexturePalette\n");
355
356
357
358
359
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("YUV textures don't have a palette");
return -1;
} else {
NDS_TextureData *txdat = (NDS_TextureData *) texture->driverdata;
360
printf("-NDS_GetTexturePalette\n");
361
362
363
364
365
366
367
368
/* TODO: mess with 8-bit modes and/or 16-color palette modes */
return 0;
}
}
static int
NDS_SetTextureColorMod(SDL_Renderer * renderer, SDL_Texture * texture)
{
369
printf("!NDS_SetTextureColorMod\n");
370
371
372
373
374
375
376
/* stub. TODO: figure out what needs to be done, if anything */
return 0;
}
static int
NDS_SetTextureAlphaMod(SDL_Renderer * renderer, SDL_Texture * texture)
{
377
printf("!NDS_SetTextureAlphaMod\n");
378
379
380
381
382
383
384
/* stub. TODO: figure out what needs to be done, if anything */
return 0;
}
static int
NDS_SetTextureBlendMode(SDL_Renderer * renderer, SDL_Texture * texture)
{
385
printf("!NDS_SetTextureBlendMode\n");
386
387
388
389
390
391
392
/* stub. TODO: figure out what needs to be done, if anything */
return 0;
}
static int
NDS_SetTextureScaleMode(SDL_Renderer * renderer, SDL_Texture * texture)
{
393
printf("!NDS_SetTextureScaleMode\n");
394
395
396
397
398
399
400
401
402
/* stub. TODO: figure out what needs to be done.
(NDS hardware scaling is nearest neighbor.) */
return 0;
}
static int
NDS_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, const void *pixels, int pitch)
{
403
printf("+NDS_UpdateTexture\n");
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("Unsupported texture format");
return -1;
} else {
NDS_TextureData *txdat = (NDS_TextureData *) texture->driverdata;
Uint8 *src, *dst;
int row;
size_t length;
/* IMPORTANT! copy the new pixels into the sprite or bg. */
src = (Uint8 *) pixels;
dst =
(Uint8 *) txdat->vram + rect->y * txdat->dim.pitch +
rect->x * (txdat->dim.bpp/8);
length = rect->w * (txdat->dim.bpp/8);
for (row = 0; row < rect->h; ++row) {
SDL_memcpy(dst, src, length);
src += pitch;
421
dst += txdat->dim.pitch;
422
}
423
printf("-NDS_UpdateTexture\n");
424
425
426
427
428
429
430
431
432
return 0;
}
}
static int
NDS_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
const SDL_Rect * rect, int markDirty, void **pixels,
int *pitch)
{
433
printf("+NDS_LockTexture\n");
434
435
436
437
438
439
440
441
442
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("Unsupported texture format");
return -1;
} else {
NDS_TextureData *txdat = (NDS_TextureData *) texture->driverdata;
*pixels = (void *) ((u8 *)txdat->vram + rect->y * txdat->dim.pitch
+ rect->x * (txdat->dim.bpp/8));
*pitch = txdat->dim.pitch;
443
printf("-NDS_LockTexture\n");
444
445
446
447
448
449
450
451
452
453
return 0;
}
}
static void
NDS_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture)
{
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("Unsupported texture format");
}
454
printf("!NDS_UnlockTexture\n");
455
456
457
458
459
460
}
static void
NDS_DirtyTexture(SDL_Renderer * renderer, SDL_Texture * texture,
int numrects, const SDL_Rect * rects)
{ /* stub */
461
printf("!NDS_DirtyTexture\n");
462
463
}
464
static int
465
NDS_RenderFill(SDL_Renderer * renderer, Uint8 r, Uint8 g, Uint8 b,
466
Uint8 a, const SDL_Rect * rect)
467
{
468
NDS_RenderData *data = (NDS_RenderData *) renderer->driverdata;
469
SDL_Rect real_rect = *rect;
470
471
472
u16 color;
int i, j;
473
printf("+NDS_RenderFill\n");
474
/* TODO: make a single-color sprite and stretch it.
475
476
477
478
479
480
color = RGB15(r>>3,g>>3,b>>3);
for (i = real_rect.x; i < real_rect.x+real_rect.w; ++i) {
for (j = real_rect.y; j < real_rect.y+real_rect.h; ++j) {
data->fb[(j + real_rect.y) * 256 + i + real_rect.x] =
0x8000 | color;
}
481
}*/
482
printf("-NDS_RenderFill\n");
483
return 0;
484
}
485
486
static int
487
NDS_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
488
const SDL_Rect * srcrect, const SDL_Rect * dstrect)
489
{
490
NDS_RenderData *data = (NDS_RenderData *) renderer->driverdata;
491
492
SDL_Window *window = SDL_GetWindowFromID(renderer->window);
SDL_VideoDisplay *display = SDL_GetDisplayFromWindow(window);
493
494
printf("+NDS_RenderCopy\n");
495
#if 0
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_Surface *target = data->screens[data->current_screen];
void *pixels =
(Uint8 *) target->pixels + dstrect->y * target->pitch +
dstrect->x * target->format->BytesPerPixel;
return SDL_SW_CopyYUVToRGB((SDL_SW_YUVTexture *) texture->driverdata,
srcrect, display->current_mode.format,
dstrect->w, dstrect->h, pixels,
target->pitch);
} else {
SDL_Surface *surface = (SDL_Surface *) texture->driverdata;
SDL_Surface *target = data->screens[data->current_screen];
SDL_Rect real_srcrect = *srcrect;
SDL_Rect real_dstrect = *dstrect;
return SDL_LowerBlit(surface, &real_srcrect, target, &real_dstrect);
}
512
#endif
513
printf("-NDS_RenderCopy\n");
514
return 0;
515
516
}
517
518
static void
519
NDS_RenderPresent(SDL_Renderer * renderer)
520
{
521
522
NDS_RenderData *data = (NDS_RenderData *) renderer->driverdata;
/* Send the data to the display TODO */
523
524
printf("+NDS_RenderPresent\n");
525
/* Update the flipping chain, if any */
526
527
if (renderer->info.flags & SDL_RENDERER_PRESENTVSYNC) {
swiWaitForVBlank();
528
}
529
printf("-NDS_RenderPresent\n");
530
531
}
532
533
534
static void
NDS_DestroyTexture(SDL_Renderer * renderer, SDL_Texture * texture)
{
535
printf("+NDS_DestroyTexture\n");
536
537
538
if (SDL_ISPIXELFORMAT_FOURCC(texture->format)) {
SDL_SetError("Unsupported texture format");
} else {
539
540
/* free anything else allocated for texture */
SDL_free(texture->driverdata);
541
}
542
printf("-NDS_DestroyTexture\n");
543
544
}
545
static void
546
NDS_DestroyRenderer(SDL_Renderer * renderer)
547
{
548
NDS_RenderData *data = (NDS_RenderData *) renderer->driverdata;
549
550
/*SDL_Window *window = SDL_GetWindowFromID(renderer->window);
SDL_VideoDisplay *display = SDL_GetDisplayFromWindow(window);*/
551
552
int i;
553
printf("+NDS_DestroyRenderer\n");
554
if (data) {
555
556
/* TODO: free anything relevant. */
/*for (i = 0; i < SDL_arraysize(data->texture); ++i) {
557
558
if (data->texture[i]) {
DestroyTexture(data->renderer, data->texture[i]);
559
}
560
561
562
563
564
565
566
567
568
}
if (data->surface.format) {
SDL_SetSurfacePalette(&data->surface, NULL);
SDL_FreeFormat(data->surface.format);
}
if (display->palette) {
SDL_DelPaletteWatch(display->palette, DisplayPaletteChanged,
data);
}
569
SDL_FreeDirtyRects(&data->dirty);*/
570
571
572
SDL_free(data);
}
SDL_free(renderer);
573
printf("-NDS_DestroyRenderer\n");
574
575
576
}
/* vi: set ts=4 sw=4 expandtab: */