1 /* 2 * Benchmarks surface-to-surface blits in various formats. 3 * 4 * Written by Ryan C. Gordon. 5 */ 6 7 #include <stdio.h> 8 #include <stdlib.h> 9 #include <string.h> 10 11 #include "SDL.h" 12 13 static SDL_Surface *dest = NULL; 14 static SDL_Surface *src = NULL; 15 static int testSeconds = 10; 16 17 18 static int percent(int val, int total) 19 { 20 return((int) ((((float) val) / ((float) total)) * 100.0f)); 21 } 22 23 static int randRange(int lo, int hi) 24 { 25 return(lo + (int) (((double) hi)*rand()/(RAND_MAX+1.0))); 26 } 27 28 static void copy_trunc_str(char *str, size_t strsize, const char *flagstr) 29 { 30 if ( (strlen(str) + strlen(flagstr)) >= (strsize - 1) ) 31 strcpy(str + (strsize - 5), " ..."); 32 else 33 strcat(str, flagstr); 34 } 35 36 static void __append_sdl_surface_flag(SDL_Surface *_surface, char *str, 37 size_t strsize, Uint32 flag, 38 const char *flagstr) 39 { 40 if (_surface->flags & flag) 41 copy_trunc_str(str, strsize, flagstr); 42 } 43 44 45 #define append_sdl_surface_flag(a, b, c, fl) __append_sdl_surface_flag(a, b, c, fl, " " #fl) 46 #define print_tf_state(str, val) printf("%s: {%s}\n", str, (val) ? "true" : "false" ) 47 48 static void output_videoinfo_details(void) 49 { 50 const SDL_VideoInfo *info = SDL_GetVideoInfo(); 51 printf("SDL_GetVideoInfo():\n"); 52 if (info == NULL) 53 printf(" (null.)\n"); 54 else 55 { 56 print_tf_state(" hardware surface available", info->hw_available); 57 print_tf_state(" window manager available", info->wm_available); 58 print_tf_state(" accelerated hardware->hardware blits", info->blit_hw); 59 print_tf_state(" accelerated hardware->hardware colorkey blits", info->blit_hw_CC); 60 print_tf_state(" accelerated hardware->hardware alpha blits", info->blit_hw_A); 61 print_tf_state(" accelerated software->hardware blits", info->blit_sw); 62 print_tf_state(" accelerated software->hardware colorkey blits", info->blit_sw_CC); 63 print_tf_state(" accelerated software->hardware alpha blits", info->blit_sw_A); 64 print_tf_state(" accelerated color fills", info->blit_fill); 65 printf(" video memory: (%d)\n", info->video_mem); 66 } 67 68 printf("\n"); 69 } 70 71 static void output_surface_details(const char *name, SDL_Surface *surface) 72 { 73 printf("Details for %s:\n", name); 74 75 if (surface == NULL) 76 { 77 printf("-WARNING- You've got a NULL surface!"); 78 } 79 else 80 { 81 char f[256]; 82 printf(" width : %d\n", surface->w); 83 printf(" height : %d\n", surface->h); 84 printf(" depth : %d bits per pixel\n", surface->format->BitsPerPixel); 85 printf(" pitch : %d\n", (int) surface->pitch); 86 printf(" alpha : %d\n", (int) surface->format->alpha); 87 printf(" colorkey : 0x%X\n", (unsigned int) surface->format->colorkey); 88 89 printf(" red bits : 0x%08X mask, %d shift, %d loss\n", 90 (int) surface->format->Rmask, 91 (int) surface->format->Rshift, 92 (int) surface->format->Rloss); 93 printf(" green bits : 0x%08X mask, %d shift, %d loss\n", 94 (int) surface->format->Gmask, 95 (int) surface->format->Gshift, 96 (int) surface->format->Gloss); 97 printf(" blue bits : 0x%08X mask, %d shift, %d loss\n", 98 (int) surface->format->Bmask, 99 (int) surface->format->Bshift, 100 (int) surface->format->Bloss); 101 printf(" alpha bits : 0x%08X mask, %d shift, %d loss\n", 102 (int) surface->format->Amask, 103 (int) surface->format->Ashift, 104 (int) surface->format->Aloss); 105 106 f[0] = '\0'; 107 108 /*append_sdl_surface_flag(surface, f, sizeof (f), SDL_SWSURFACE);*/ 109 if ((surface->flags & SDL_HWSURFACE) == 0) 110 copy_trunc_str(f, sizeof (f), " SDL_SWSURFACE"); 111 112 append_sdl_surface_flag(surface, f, sizeof (f), SDL_HWSURFACE); 113 append_sdl_surface_flag(surface, f, sizeof (f), SDL_ASYNCBLIT); 114 append_sdl_surface_flag(surface, f, sizeof (f), SDL_ANYFORMAT); 115 append_sdl_surface_flag(surface, f, sizeof (f), SDL_HWPALETTE); 116 append_sdl_surface_flag(surface, f, sizeof (f), SDL_DOUBLEBUF); 117 append_sdl_surface_flag(surface, f, sizeof (f), SDL_FULLSCREEN); 118 append_sdl_surface_flag(surface, f, sizeof (f), SDL_OPENGL); 119 append_sdl_surface_flag(surface, f, sizeof (f), SDL_OPENGLBLIT); 120 append_sdl_surface_flag(surface, f, sizeof (f), SDL_RESIZABLE); 121 append_sdl_surface_flag(surface, f, sizeof (f), SDL_NOFRAME); 122 append_sdl_surface_flag(surface, f, sizeof (f), SDL_HWACCEL); 123 append_sdl_surface_flag(surface, f, sizeof (f), SDL_SRCCOLORKEY); 124 append_sdl_surface_flag(surface, f, sizeof (f), SDL_RLEACCELOK); 125 append_sdl_surface_flag(surface, f, sizeof (f), SDL_RLEACCEL); 126 append_sdl_surface_flag(surface, f, sizeof (f), SDL_SRCALPHA); 127 append_sdl_surface_flag(surface, f, sizeof (f), SDL_PREALLOC); 128 129 if (f[0] == '\0') 130 strcpy(f, " (none)"); 131 132 printf(" flags :%s\n", f); 133 } 134 135 printf("\n"); 136 } 137 138 static void output_details(void) 139 { 140 output_videoinfo_details(); 141 output_surface_details("Source Surface", src); 142 output_surface_details("Destination Surface", dest); 143 } 144 145 static Uint32 blit(SDL_Surface *dst, SDL_Surface *src, int x, int y) 146 { 147 Uint32 start = 0; 148 SDL_Rect srcRect; 149 SDL_Rect dstRect; 150 151 srcRect.x = 0; 152 srcRect.y = 0; 153 dstRect.x = x; 154 dstRect.y = y; 155 dstRect.w = srcRect.w = src->w; /* SDL will clip as appropriate. */ 156 dstRect.h = srcRect.h = src->h; 157 158 start = SDL_GetTicks(); 159 SDL_BlitSurface(src, &srcRect, dst, &dstRect); 160 return(SDL_GetTicks() - start); 161 } 162 163 static void blitCentered(SDL_Surface *dst, SDL_Surface *src) 164 { 165 int x = (dst->w - src->w) / 2; 166 int y = (dst->h - src->h) / 2; 167 blit(dst, src, x, y); 168 } 169 170 static int atoi_hex(const char *str) 171 { 172 if (str == NULL) 173 return 0; 174 175 if (strlen(str) > 2) 176 { 177 int retval = 0; 178 if ((str[0] == '0') && (str[1] == 'x')) 179 sscanf(str + 2, "%X", &retval); 180 return(retval); 181 } 182 183 return(atoi(str)); 184 } 185 186 187 static int setup_test(int argc, char **argv) 188 { 189 const char *dumpfile = NULL; 190 SDL_Surface *bmp = NULL; 191 Uint32 dstbpp = 32; 192 Uint32 dstrmask = 0x00FF0000; 193 Uint32 dstgmask = 0x0000FF00; 194 Uint32 dstbmask = 0x000000FF; 195 Uint32 dstamask = 0x00000000; 196 Uint32 dstflags = 0; 197 int dstw = 640; 198 int dsth = 480; 199 Uint32 srcbpp = 32; 200 Uint32 srcrmask = 0x00FF0000; 201 Uint32 srcgmask = 0x0000FF00; 202 Uint32 srcbmask = 0x000000FF; 203 Uint32 srcamask = 0x00000000; 204 Uint32 srcflags = 0; 205 int srcw = 640; 206 int srch = 480; 207 Uint32 origsrcalphaflags = 0; 208 Uint32 origdstalphaflags = 0; 209 Uint32 srcalphaflags = 0; 210 Uint32 dstalphaflags = 0; 211 int srcalpha = 255; 212 int dstalpha = 255; 213 int screenSurface = 0; 214 int i = 0; 215 216 for (i = 1; i < argc; i++) 217 { 218 const char *arg = argv[i]; 219 220 if (strcmp(arg, "--dstbpp") == 0) 221 dstbpp = atoi(argv[++i]); 222 else if (strcmp(arg, "--dstrmask") == 0) 223 dstrmask = atoi_hex(argv[++i]); 224 else if (strcmp(arg, "--dstgmask") == 0) 225 dstgmask = atoi_hex(argv[++i]); 226 else if (strcmp(arg, "--dstbmask") == 0) 227 dstbmask = atoi_hex(argv[++i]); 228 else if (strcmp(arg, "--dstamask") == 0) 229 dstamask = atoi_hex(argv[++i]); 230 else if (strcmp(arg, "--dstwidth") == 0) 231 dstw = atoi(argv[++i]); 232 else if (strcmp(arg, "--dstheight") == 0) 233 dsth = atoi(argv[++i]); 234 else if (strcmp(arg, "--dsthwsurface") == 0) 235 dstflags |= SDL_HWSURFACE; 236 else if (strcmp(arg, "--srcbpp") == 0) 237 srcbpp = atoi(argv[++i]); 238 else if (strcmp(arg, "--srcrmask") == 0) 239 srcrmask = atoi_hex(argv[++i]); 240 else if (strcmp(arg, "--srcgmask") == 0) 241 srcgmask = atoi_hex(argv[++i]); 242 else if (strcmp(arg, "--srcbmask") == 0) 243 srcbmask = atoi_hex(argv[++i]); 244 else if (strcmp(arg, "--srcamask") == 0) 245 srcamask = atoi_hex(argv[++i]); 246 else if (strcmp(arg, "--srcwidth") == 0) 247 srcw = atoi(argv[++i]); 248 else if (strcmp(arg, "--srcheight") == 0) 249 srch = atoi(argv[++i]); 250 else if (strcmp(arg, "--srchwsurface") == 0) 251 srcflags |= SDL_HWSURFACE; 252 else if (strcmp(arg, "--seconds") == 0) 253 testSeconds = atoi(argv[++i]); 254 else if (strcmp(arg, "--screen") == 0) 255 screenSurface = 1; 256 else if (strcmp(arg, "--dumpfile") == 0) 257 dumpfile = argv[++i]; 258 /* !!! FIXME: set colorkey. */ 259 else if (0) /* !!! FIXME: we handle some commandlines elsewhere now */ 260 { 261 fprintf(stderr, "Unknown commandline option: %s\n", arg); 262 return(0); 263 } 264 } 265 266 if (SDL_Init(SDL_INIT_VIDEO) == -1) 267 { 268 fprintf(stderr, "SDL_Init failed: %s\n", SDL_GetError()); 269 return(0); 270 } 271 272 bmp = SDL_LoadBMP("sample.bmp"); 273 if (bmp == NULL) 274 { 275 fprintf(stderr, "SDL_LoadBMP failed: %s\n", SDL_GetError()); 276 SDL_Quit(); 277 return(0); 278 } 279 280 if ((dstflags & SDL_HWSURFACE) == 0) dstflags |= SDL_SWSURFACE; 281 if ((srcflags & SDL_HWSURFACE) == 0) srcflags |= SDL_SWSURFACE; 282 283 if (screenSurface) 284 dest = SDL_SetVideoMode(dstw, dsth, dstbpp, dstflags); 285 else 286 { 287 dest = SDL_CreateRGBSurface(dstflags, dstw, dsth, dstbpp, 288 dstrmask, dstgmask, dstbmask, dstamask); 289 } 290 291 if (dest == NULL) 292 { 293 fprintf(stderr, "dest surface creation failed: %s\n", SDL_GetError()); 294 SDL_Quit(); 295 return(0); 296 } 297 298 src = SDL_CreateRGBSurface(srcflags, srcw, srch, srcbpp, 299 srcrmask, srcgmask, srcbmask, srcamask); 300 if (src == NULL) 301 { 302 fprintf(stderr, "src surface creation failed: %s\n", SDL_GetError()); 303 SDL_Quit(); 304 return(0); 305 } 306 307 /* handle alpha settings... */ 308 srcalphaflags = (src->flags&SDL_SRCALPHA) | (src->flags&SDL_RLEACCEL); 309 dstalphaflags = (dest->flags&SDL_SRCALPHA) | (dest->flags&SDL_RLEACCEL); 310 origsrcalphaflags = srcalphaflags; 311 origdstalphaflags = dstalphaflags; 312 srcalpha = src->format->alpha; 313 dstalpha = dest->format->alpha; 314 for (i = 1; i < argc; i++) 315 { 316 const char *arg = argv[i]; 317 318 if (strcmp(arg, "--srcalpha") == 0) 319 srcalpha = atoi(argv[++i]); 320 else if (strcmp(arg, "--dstalpha") == 0) 321 dstalpha = atoi(argv[++i]); 322 else if (strcmp(arg, "--srcsrcalpha") == 0) 323 srcalphaflags |= SDL_SRCALPHA; 324 else if (strcmp(arg, "--srcnosrcalpha") == 0) 325 srcalphaflags &= ~SDL_SRCALPHA; 326 else if (strcmp(arg, "--srcrleaccel") == 0) 327 srcalphaflags |= SDL_RLEACCEL; 328 else if (strcmp(arg, "--srcnorleaccel") == 0) 329 srcalphaflags &= ~SDL_RLEACCEL; 330 else if (strcmp(arg, "--dstsrcalpha") == 0) 331 dstalphaflags |= SDL_SRCALPHA; 332 else if (strcmp(arg, "--dstnosrcalpha") == 0) 333 dstalphaflags &= ~SDL_SRCALPHA; 334 else if (strcmp(arg, "--dstrleaccel") == 0) 335 dstalphaflags |= SDL_RLEACCEL; 336 else if (strcmp(arg, "--dstnorleaccel") == 0) 337 dstalphaflags &= ~SDL_RLEACCEL; 338 } 339 if ((dstalphaflags != origdstalphaflags) || (dstalpha != dest->format->alpha)) 340 SDL_SetAlpha(dest, dstalphaflags, (Uint8) dstalpha); 341 if ((srcalphaflags != origsrcalphaflags) || (srcalpha != src->format->alpha)) 342 SDL_SetAlpha(src, srcalphaflags, (Uint8) srcalpha); 343 344 /* set some sane defaults so we can see if the blit code is broken... */ 345 SDL_FillRect(dest, NULL, SDL_MapRGB(dest->format, 0, 0, 0)); 346 SDL_FillRect(src, NULL, SDL_MapRGB(src->format, 0, 0, 0)); 347 348 blitCentered(src, bmp); 349 SDL_FreeSurface(bmp); 350 351 if (dumpfile) 352 SDL_SaveBMP(src, dumpfile); /* make sure initial convert is sane. */ 353 354 output_details(); 355 356 return(1); 357 } 358 359 360 static void test_blit_speed(void) 361 { 362 Uint32 clearColor = SDL_MapRGB(dest->format, 0, 0, 0); 363 Uint32 iterations = 0; 364 Uint32 elasped = 0; 365 Uint32 end = 0; 366 Uint32 now = 0; 367 Uint32 last = 0; 368 int testms = testSeconds * 1000; 369 int wmax = (dest->w - src->w); 370 int hmax = (dest->h - src->h); 371 int isScreen = (SDL_GetVideoSurface() == dest); 372 SDL_Event event; 373 374 printf("Testing blit speed for %d seconds...\n", testSeconds); 375 376 now = SDL_GetTicks(); 377 end = now + testms; 378 379 do 380 { 381 /* pump the event queue occasionally to keep OS happy... */ 382 if (now - last > 1000) 383 { 384 last = now; 385 while (SDL_PollEvent(&event)) { /* no-op. */ } 386 } 387 388 iterations++; 389 elasped += blit(dest, src, randRange(0, wmax), randRange(0, hmax)); 390 if (isScreen) 391 { 392 SDL_Flip(dest); /* show it! */ 393 SDL_FillRect(dest, NULL, clearColor); /* blank it for next time! */ 394 } 395 396 now = SDL_GetTicks(); 397 } while (now < end); 398 399 printf("Non-blitting crap accounted for %d percent of this run.\n", 400 percent(testms - elasped, testms)); 401 402 printf("%d blits took %d ms (%d fps).\n", 403 (int) iterations, 404 (int) elasped, 405 (int) (((float)iterations) / (((float)elasped) / 1000.0f))); 406 } 407 408 int main(int argc, char **argv) 409 { 410 int initialized = setup_test(argc, argv); 411 if (initialized) 412 { 413 test_blit_speed(); 414 SDL_Quit(); 415 } 416 return(!initialized); 417 } 418 419 /* end of testblitspeed.c ... */ 420 421