SDL_QuartzVideo.m
上传用户:sun1608
上传日期:2007-02-02
资源大小:6116k
文件大小:40k
源码类别:

流媒体/Mpeg4/MP4

开发平台:

Visual C++

  1. /*
  2.     SDL - Simple DirectMedia Layer
  3.     Copyright (C) 1997, 1998, 1999, 2000, 2001, 2002  Sam Lantinga
  4.     This library is free software; you can redistribute it and/or
  5.     modify it under the terms of the GNU Library General Public
  6.     License as published by the Free Software Foundation; either
  7.     version 2 of the License, or (at your option) any later version.
  8.     This library is distributed in the hope that it will be useful,
  9.     but WITHOUT ANY WARRANTY; without even the implied warranty of
  10.     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  11.     Library General Public License for more details.
  12.     You should have received a copy of the GNU Library General Public
  13.     License along with this library; if not, write to the Free
  14.     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
  15.     Sam Lantinga
  16.     slouken@libsdl.org
  17. */
  18. #include "SDL_QuartzVideo.h"
  19. /* Some variables to share among files, put in device structure eventually */
  20. static SDL_GrabMode currentGrabMode = SDL_GRAB_OFF;
  21. static BOOL   inForeground = YES;
  22. static char QZ_Error[255]; /* Global error buffer to temporarily store more informative error messages */
  23. /* Include files into one compile unit...break apart eventually */
  24. #include "SDL_QuartzWM.m"
  25. #include "SDL_QuartzEvents.m"
  26. #include "SDL_QuartzWindow.m"
  27. /* Bootstrap binding, enables entry point into the driver */
  28. VideoBootStrap QZ_bootstrap = {
  29.     "Quartz", "Mac OS X CoreGraphics", QZ_Available, QZ_CreateDevice
  30. };
  31. /* Bootstrap functions */
  32. static int QZ_Available () {
  33.     return 1;
  34. }
  35. static SDL_VideoDevice* QZ_CreateDevice (int device_index) {
  36. #pragma unused (device_index)
  37.     SDL_VideoDevice *device;
  38.     SDL_PrivateVideoData *hidden;
  39.     device = (SDL_VideoDevice*) malloc (sizeof (*device) );
  40.     hidden = (SDL_PrivateVideoData*) malloc (sizeof (*hidden) );
  41.     if (device == NULL || hidden == NULL)
  42.         SDL_OutOfMemory ();
  43.     memset (device, 0, sizeof (*device) );
  44.     memset (hidden, 0, sizeof (*hidden) );
  45.     device->hidden = hidden;
  46.     device->VideoInit        = QZ_VideoInit;
  47.     device->ListModes        = QZ_ListModes;
  48.     device->SetVideoMode     = QZ_SetVideoMode;
  49.     device->ToggleFullScreen = QZ_ToggleFullScreen;
  50.     device->SetColors        = QZ_SetColors;
  51.     /* device->UpdateRects      = QZ_UpdateRects; this is determined by SetVideoMode() */
  52.     device->VideoQuit        = QZ_VideoQuit;
  53.     device->LockHWSurface   = QZ_LockHWSurface;
  54.     device->UnlockHWSurface = QZ_UnlockHWSurface;
  55.     device->FreeHWSurface   = QZ_FreeHWSurface;
  56.     /* device->FlipHWSurface   = QZ_FlipHWSurface */;
  57.     device->SetGamma     = QZ_SetGamma;
  58.     device->GetGamma     = QZ_GetGamma;
  59.     device->SetGammaRamp = QZ_SetGammaRamp;
  60.     device->GetGammaRamp = QZ_GetGammaRamp;
  61.     device->GL_GetProcAddress = QZ_GL_GetProcAddress;
  62.     device->GL_GetAttribute   = QZ_GL_GetAttribute;
  63.     device->GL_MakeCurrent    = QZ_GL_MakeCurrent;
  64.     device->GL_SwapBuffers    = QZ_GL_SwapBuffers;
  65.     device->GL_LoadLibrary    = QZ_GL_LoadLibrary;
  66.     device->FreeWMCursor   = QZ_FreeWMCursor;
  67.     device->CreateWMCursor = QZ_CreateWMCursor;
  68.     device->ShowWMCursor   = QZ_ShowWMCursor;
  69.     device->WarpWMCursor   = QZ_WarpWMCursor;
  70.     device->MoveWMCursor   = QZ_MoveWMCursor;
  71.     device->CheckMouseMode = QZ_CheckMouseMode;
  72.     device->InitOSKeymap   = QZ_InitOSKeymap;
  73.     device->PumpEvents     = QZ_PumpEvents;
  74.     device->SetCaption    = QZ_SetCaption;
  75.     device->SetIcon       = QZ_SetIcon;
  76.     device->IconifyWindow = QZ_IconifyWindow;
  77.     /*device->GetWMInfo     = QZ_GetWMInfo;*/
  78.     device->GrabInput     = QZ_GrabInput;
  79.     device->free = QZ_DeleteDevice;
  80.     return device;
  81. }
  82. static void QZ_DeleteDevice (SDL_VideoDevice *device) {
  83.     free (device->hidden);
  84.     free (device);
  85. }
  86. static int QZ_VideoInit (_THIS, SDL_PixelFormat *video_format) {
  87.     /* Initialize the video settings; this data persists between mode switches */
  88.     display_id = kCGDirectMainDisplay;
  89.     save_mode  = CGDisplayCurrentMode    (display_id);
  90.     mode_list  = CGDisplayAvailableModes (display_id);
  91.     palette    = CGPaletteCreateDefaultColorPalette ();
  92.     /* Gather some information that is useful to know about the display */
  93.     CFNumberGetValue (CFDictionaryGetValue (save_mode, kCGDisplayBitsPerPixel),
  94.                       kCFNumberSInt32Type, &device_bpp);
  95.     CFNumberGetValue (CFDictionaryGetValue (save_mode, kCGDisplayWidth),
  96.                       kCFNumberSInt32Type, &device_width);
  97.     CFNumberGetValue (CFDictionaryGetValue (save_mode, kCGDisplayHeight),
  98.                       kCFNumberSInt32Type, &device_height);
  99.     video_format->BitsPerPixel = device_bpp;
  100.     return 0;
  101. }
  102. static SDL_Rect** QZ_ListModes (_THIS, SDL_PixelFormat *format, Uint32 flags) {
  103.     
  104.     CFIndex num_modes;
  105.     CFIndex i;
  106.     static SDL_Rect **list = NULL;
  107.     int list_size = 0;
  108.     
  109.     /* Any windowed mode is acceptable */
  110.     if ( (flags & SDL_FULLSCREEN) == 0 )
  111.         return (SDL_Rect**)-1;
  112.         
  113.     /* Free memory from previous call, if any */
  114.     if ( list != NULL ) {
  115.       int i;
  116.       for (i = 0; list[i] != NULL; i++)
  117. free (list[i]);
  118.       free (list);
  119.       list = NULL;
  120.     }
  121.     
  122.     num_modes = CFArrayGetCount (mode_list);
  123.     /* Build list of modes with the requested bpp */
  124.     for (i = 0; i < num_modes; i++) {
  125.    
  126.         CFDictionaryRef onemode;
  127.         CFNumberRef     number;
  128. int bpp;
  129.         onemode = CFArrayGetValueAtIndex (mode_list, i); 
  130. number = CFDictionaryGetValue (onemode, kCGDisplayBitsPerPixel);
  131. CFNumberGetValue (number, kCFNumberSInt32Type, &bpp);
  132. if (bpp == format->BitsPerPixel) {
  133.   
  134.   int intvalue;
  135.           int hasMode;
  136.           int width, height;
  137.           
  138.   number = CFDictionaryGetValue (onemode, kCGDisplayWidth);
  139.   CFNumberGetValue (number, kCFNumberSInt32Type, &intvalue);
  140.   width = (Uint16) intvalue;
  141.   
  142.   number = CFDictionaryGetValue (onemode, kCGDisplayHeight);
  143.   CFNumberGetValue (number, kCFNumberSInt32Type, &intvalue);
  144.   height = (Uint16) intvalue;
  145.           
  146.           /* Check if mode is already in the list */
  147.           {
  148.             int i;
  149.             hasMode = SDL_FALSE;
  150.             for (i = 0; i < list_size; i++) {
  151.                 if (list[i]->w == width && list[i]->h == height) {
  152.                     hasMode = SDL_TRUE;
  153.                     break;
  154.                 }
  155.             }
  156.           }
  157.           
  158.           /* Grow the list and add mode to the list */
  159.           if ( ! hasMode ) {
  160.   
  161.             SDL_Rect *rect;
  162.             
  163.             list_size++;
  164.             if (list == NULL)
  165.                 list = (SDL_Rect**) malloc (sizeof(*list) * (list_size+1) );
  166.             else
  167.                 list = (SDL_Rect**) realloc (list, sizeof(*list) * (list_size+1));
  168.             
  169.             rect = (SDL_Rect*) malloc (sizeof(**list));
  170.             
  171.             if (list == NULL || rect == NULL) {
  172.                 SDL_OutOfMemory ();
  173.                 return NULL;
  174.             }
  175.             
  176.             rect->w = width;
  177.             rect->h = height;
  178.     
  179.             list[list_size-1] = rect;
  180.             list[list_size]   = NULL;
  181.         }
  182.       }
  183.     }
  184.     
  185.     /* Sort list largest to smallest (by area) */
  186.     {
  187.         int i, j;
  188.         for (i = 0; i < list_size; i++) {
  189.             for (j = 0; j < list_size-1; j++) {
  190.             
  191.                 int area1, area2;
  192.                 area1 = list[j]->w * list[j]->h;
  193.                 area2 = list[j+1]->w * list[j+1]->h;
  194.                 
  195.                 if (area1 < area2) {
  196.                     SDL_Rect *tmp = list[j];
  197.                     list[j] = list[j+1];
  198.                     list[j+1] = tmp;
  199.                 }
  200.             }
  201.         }
  202.     }
  203.     return list;
  204. }
  205. /* Gamma functions to try to hide the flash from a rez switch */
  206. /* Fade the display from normal to black */
  207. /* Save gamma tables for fade back to normal */
  208. static UInt32 QZ_FadeGammaOut (_THIS, SDL_QuartzGammaTable *table) {
  209.     CGGammaValue redTable[QZ_GAMMA_TABLE_SIZE],
  210.     greenTable[QZ_GAMMA_TABLE_SIZE],
  211.     blueTable[QZ_GAMMA_TABLE_SIZE];
  212.     float percent;
  213.     int j;
  214.     int actual;
  215.     if ( (CGDisplayNoErr != CGGetDisplayTransferByTable
  216.           (display_id, QZ_GAMMA_TABLE_SIZE,
  217.            table->red, table->green, table->blue, &actual)) ||
  218.          actual != QZ_GAMMA_TABLE_SIZE) {
  219.         return 1;
  220.     }
  221.     memcpy (redTable, table->red, sizeof(redTable));
  222.     memcpy (greenTable, table->green, sizeof(greenTable));
  223.     memcpy (blueTable, table->blue, sizeof(greenTable));
  224.     for (percent = 1.0; percent >= 0.0; percent -= 0.01) {
  225.         for (j = 0; j < QZ_GAMMA_TABLE_SIZE; j++) {
  226.             redTable[j]   = redTable[j]   * percent;
  227.             greenTable[j] = greenTable[j] * percent;
  228.             blueTable[j]  = blueTable[j]  * percent;
  229.         }
  230.         if (CGDisplayNoErr != CGSetDisplayTransferByTable
  231.             (display_id, QZ_GAMMA_TABLE_SIZE,
  232.              redTable, greenTable, blueTable)) {
  233.             CGDisplayRestoreColorSyncSettings();
  234.             return 1;
  235.         }
  236.         SDL_Delay (10);
  237.     }
  238.     return 0;
  239. }
  240. /* Fade the display from black to normal */
  241. /* Restore previously saved gamma values */
  242. static UInt32 QZ_FadeGammaIn (_THIS, SDL_QuartzGammaTable *table) {
  243.     CGGammaValue redTable[QZ_GAMMA_TABLE_SIZE],
  244.         greenTable[QZ_GAMMA_TABLE_SIZE],
  245.         blueTable[QZ_GAMMA_TABLE_SIZE];
  246.     float percent;
  247.     int j;
  248.     memset (redTable, 0, sizeof(redTable));
  249.     memset (greenTable, 0, sizeof(greenTable));
  250.     memset (blueTable, 0, sizeof(greenTable));
  251.     
  252.     for (percent = 0.0; percent <= 1.0; percent += 0.01) {
  253.         for (j = 0; j < QZ_GAMMA_TABLE_SIZE; j++) {
  254.             redTable[j]   = table->red[j]   * percent;
  255.             greenTable[j] = table->green[j] * percent;
  256.             blueTable[j]  = table->blue[j]  * percent;
  257.         }
  258.         if (CGDisplayNoErr != CGSetDisplayTransferByTable
  259.             (display_id, QZ_GAMMA_TABLE_SIZE,
  260.              redTable, greenTable, blueTable)) {
  261.             CGDisplayRestoreColorSyncSettings();
  262.             return 1;
  263.         }
  264.         SDL_Delay (10);
  265.     }
  266.     return 0;
  267. }
  268. static void QZ_UnsetVideoMode (_THIS) {
  269.     /* Reset values that may change between switches */
  270.     this->info.blit_fill = 0;
  271.     this->FillHWRect     = NULL;
  272.     this->UpdateRects    = NULL;
  273.    
  274.     /* Release fullscreen resources */
  275.     if ( mode_flags & SDL_FULLSCREEN ) {
  276.         SDL_QuartzGammaTable gamma_table;
  277.         int gamma_error;
  278.         gamma_error = QZ_FadeGammaOut (this, &gamma_table);
  279.         /* Release the OpenGL context */
  280.         /* Do this first to avoid trash on the display before fade */
  281.         if ( mode_flags & SDL_OPENGL )
  282.             QZ_TearDownOpenGL (this);
  283.         
  284.         if (mode_flags & SDL_OPENGL)
  285.             CGLSetFullScreen(NULL);
  286.         /* Restore original screen resolution/bpp */
  287.         CGDisplaySwitchToMode (display_id, save_mode);
  288.         CGDisplayRelease (display_id);
  289.         ShowMenuBar ();
  290.         
  291.         if (! gamma_error)
  292.             QZ_FadeGammaIn (this, &gamma_table);
  293.     }
  294.     /* Release window mode resources */
  295.     else { 
  296.         if ( (mode_flags & SDL_OPENGL) == 0 ) {
  297.             UnlockPortBits ( [ window_view qdPort ] );
  298.             [ window_view release  ];
  299.         }
  300.         [ qz_window setContentView:nil ];
  301.         [ qz_window setDelegate:nil ];
  302.         [ qz_window close ];
  303.         [ qz_window release ];
  304.         /* Release the OpenGL context */
  305.         if ( mode_flags & SDL_OPENGL )
  306.             QZ_TearDownOpenGL (this);
  307.     }
  308.     /* Restore gamma settings */
  309.     CGDisplayRestoreColorSyncSettings ();
  310.     
  311.     /* Set pixels to null (so other code doesn't try to free it) */
  312.     if (this->screen != NULL)
  313.         this->screen->pixels = NULL;
  314.         
  315.     /* Ensure the cursor will be visible and working when we quit */
  316.     CGDisplayShowCursor (display_id);
  317.     CGAssociateMouseAndMouseCursorPosition (1);
  318.     
  319.     /* Signal successful teardown */
  320.     video_set = SDL_FALSE;
  321. }
  322. static SDL_Surface* QZ_SetVideoFullScreen (_THIS, SDL_Surface *current, int width,
  323.                                            int height, int bpp, Uint32 flags) {
  324.     int exact_match;
  325.     int gamma_error;
  326.     SDL_QuartzGammaTable gamma_table;
  327.     
  328.     /* See if requested mode exists */
  329.     mode = CGDisplayBestModeForParameters (display_id, bpp, width, 
  330.    height, &exact_match);
  331.     
  332.     /* Require an exact match to the requested mode */
  333.     if ( ! exact_match ) {
  334.         sprintf (QZ_Error, "Failed to find display resolution: %dx%dx%d", width, height, bpp);
  335.         SDL_SetError (QZ_Error);
  336.         goto ERR_NO_MATCH;
  337.     }
  338.     /* Fade display to zero gamma */
  339.     gamma_error = QZ_FadeGammaOut (this, &gamma_table);
  340.     /* Put up the blanking window (a window above all other windows) */
  341.     if ( CGDisplayNoErr != CGDisplayCapture (display_id) ) {
  342.         SDL_SetError ("Failed capturing display");
  343.         goto ERR_NO_CAPTURE;
  344.     }
  345.     
  346.     /* Do the physical switch */
  347.     if ( CGDisplayNoErr != CGDisplaySwitchToMode (display_id, mode) ) {
  348.         SDL_SetError ("Failed switching display resolution");
  349.         goto ERR_NO_SWITCH;
  350.     }
  351.     current->pixels = (Uint32*) CGDisplayBaseAddress (display_id);
  352.     current->pitch  = CGDisplayBytesPerRow (display_id);
  353.     current->flags = 0;
  354.     current->w = width;
  355.     current->h = height;
  356.     current->flags |= SDL_FULLSCREEN;  
  357.     current->flags |= SDL_HWSURFACE;
  358.    
  359.     this->UpdateRects = QZ_DirectUpdate;
  360.     
  361.     /* Setup some mode-dependant info */
  362.     if ( CGSDisplayCanHWFill (display_id) ) {
  363.          this->info.blit_fill = 1;
  364.          this->FillHWRect = QZ_FillHWRect;
  365.     }
  366.         
  367.     if ( CGDisplayCanSetPalette (display_id) )
  368.         current->flags |= SDL_HWPALETTE;
  369.     
  370.     /* Setup OpenGL for a fullscreen context */
  371.     if (flags & SDL_OPENGL) {
  372.         CGLError err;
  373.         CGLContextObj ctx;
  374.         
  375.         if ( ! QZ_SetupOpenGL (this, bpp, flags) ) {
  376.             goto ERR_NO_GL;
  377.         }
  378.        
  379.         ctx = [ gl_context cglContext ];
  380.         err = CGLSetFullScreen (ctx);
  381.         
  382.         if (err) {
  383.             sprintf (QZ_Error, "Error setting OpenGL fullscreen: %s", CGLErrorString(err));
  384.             SDL_SetError (QZ_Error);
  385.             goto ERR_NO_GL;
  386.         }
  387.          
  388.         [ gl_context makeCurrentContext];
  389.         glClear (GL_COLOR_BUFFER_BIT);
  390.         [ gl_context flushBuffer ];
  391.         
  392.         current->flags |= SDL_OPENGL;
  393.     }
  394.     /* If we don't hide menu bar, it will get events and interrupt the program */
  395.     HideMenuBar ();
  396.     /* Fade the display to original gamma */
  397.     if (! gamma_error )
  398.         QZ_FadeGammaIn (this, &gamma_table);
  399.     
  400.     /* Save the flags to ensure correct tear-down */
  401.     mode_flags = current->flags;
  402.     
  403.     return current;
  404.  /* Since the blanking window covers *all* windows (even force quit) correct recovery is crucial */
  405.  ERR_NO_GL:      CGDisplaySwitchToMode (display_id, save_mode);
  406.  ERR_NO_SWITCH:  CGDisplayRelease (display_id);
  407.  ERR_NO_CAPTURE: if (!gamma_error) { QZ_FadeGammaIn (this, &gamma_table); }
  408.  ERR_NO_MATCH: return NULL;
  409. }
  410. static SDL_Surface* QZ_SetVideoWindowed (_THIS, SDL_Surface *current, int width,
  411.                                            int height, int bpp, Uint32 flags) {    
  412.     unsigned int style;
  413.     NSRect rect;    
  414.     rect = NSMakeRect (0, 0, width, height);
  415. #if 1 // FIXME - the resize button doesn't show?  Also need resize events...
  416.     flags &= ~SDL_RESIZABLE;
  417. #endif
  418.     /* Set the window style based on input flags */
  419.     if ( flags & SDL_NOFRAME ) {
  420.         style = NSBorderlessWindowMask;
  421.     } else {
  422.         style = NSTitledWindowMask;
  423.         style |= (NSMiniaturizableWindowMask | NSClosableWindowMask);
  424.         if ( flags & SDL_RESIZABLE )
  425.             style |= NSResizableWindowMask;
  426.     }
  427.     /* Manually create a window, avoids having a nib file resource */
  428.     qz_window = [ [ SDL_QuartzWindow alloc ] initWithContentRect:rect 
  429.         styleMask:style backing:NSBackingStoreBuffered defer:NO ];
  430.     if (qz_window == nil) {
  431.         SDL_SetError ("Could not create the Cocoa window");
  432.         return NULL;
  433.     }
  434.     
  435.     current->flags = 0;
  436.     current->w = width;
  437.     current->h = height;
  438.     
  439.     [ qz_window setReleasedWhenClosed:YES ];
  440.     QZ_SetCaption(this, this->wm_title, this->wm_icon);
  441.     [ qz_window setAcceptsMouseMovedEvents:YES ];
  442.     [ qz_window setViewsNeedDisplay:NO ];
  443.     [ qz_window center ];
  444.     [ qz_window setDelegate:
  445.         [ [ [ SDL_QuartzWindowDelegate alloc ] init ] autorelease ] ];
  446.     
  447.     /* For OpenGL, we set the content view to a NSOpenGLView */
  448.     if ( flags & SDL_OPENGL ) {
  449.     
  450.         if ( ! QZ_SetupOpenGL (this, bpp, flags) ) {
  451.             return NULL;
  452.         }
  453.         
  454.         [ gl_context setView: [ qz_window contentView ] ];
  455.         [ gl_context makeCurrentContext];
  456.         [ qz_window makeKeyAndOrderFront:nil ];
  457.         current->flags |= SDL_OPENGL;
  458.     }
  459.     /* For 2D, we set the content view to a NSQuickDrawView */
  460.     else {
  461.     
  462.         window_view = [ [ SDL_QuartzWindowView alloc ] init ];
  463.         [ qz_window setContentView:window_view ];
  464.         [ qz_window makeKeyAndOrderFront:nil ];    
  465.         
  466.         LockPortBits ( [ window_view qdPort ] );
  467.         current->pixels = GetPixBaseAddr ( GetPortPixMap ( [ window_view qdPort ] ) );
  468.         current->pitch  = GetPixRowBytes ( GetPortPixMap ( [ window_view qdPort ] ) );
  469.         
  470.         current->flags |= SDL_SWSURFACE;
  471.         current->flags |= SDL_PREALLOC;
  472.         
  473. if ( flags & SDL_NOFRAME )
  474.          current->flags |= SDL_NOFRAME;
  475. if ( flags & SDL_RESIZABLE )
  476.          current->flags |= SDL_RESIZABLE;
  477.         /* Offset 22 pixels down to fill the full content region */
  478. if ( ! (current->flags & SDL_NOFRAME) ) {
  479.          current->pixels += 22 * current->pitch;
  480. }
  481.         this->UpdateRects = QZ_UpdateRects;
  482.     }
  483.     
  484.     /* Save flags to ensure correct teardown */
  485.     mode_flags = current->flags;
  486.       
  487.     return current;
  488. }
  489. static SDL_Surface* QZ_SetVideoMode (_THIS, SDL_Surface *current, int width, 
  490.      int height, int bpp, Uint32 flags) {
  491.     if (video_set == SDL_TRUE)
  492.         QZ_UnsetVideoMode (this);
  493.        
  494.     current->flags = 0;
  495.     
  496.     /* Setup full screen video */
  497.     if ( flags & SDL_FULLSCREEN ) {
  498.         current = QZ_SetVideoFullScreen (this, current, width, height, bpp, flags );
  499.         if (current == NULL)
  500.             return NULL;
  501.     }
  502.     /* Setup windowed video */
  503.     else {
  504.         /* Force bpp to the device's bpp */
  505.         bpp = device_bpp;
  506.         current = QZ_SetVideoWindowed (this, current, width, height, bpp, flags);
  507.         if (current == NULL)
  508.             return NULL;
  509.     }
  510.     
  511.     /* Setup the new pixel format */
  512.     {
  513.         int amask = 0, 
  514.             rmask = 0, 
  515.             gmask = 0,
  516.             bmask = 0;
  517.             
  518.         switch (bpp) {
  519.             case 16:   /* (1)-5-5-5 RGB */
  520.                 amask = 0; 
  521.                 rmask = 0x7C00;
  522.                 gmask = 0x03E0;
  523.                 bmask = 0x001F;
  524.                 break;
  525.             case 24:
  526.                 SDL_SetError ("24bpp is not available");
  527.                 return NULL;
  528.             case 32:   /* (8)-8-8-8 ARGB */
  529.                 amask = 0x00000000;
  530.                 rmask = 0x00FF0000;
  531.                 gmask = 0x0000FF00;
  532.                 bmask = 0x000000FF;
  533.                 break;
  534.         }
  535.         
  536.         if ( ! SDL_ReallocFormat (current, bpp,
  537.                                   rmask, gmask, bmask, amask ) ) {
  538.            SDL_SetError ("Couldn't reallocate pixel format");
  539.            return NULL;
  540.         }
  541.     }
  542.     
  543.     /* Signal successful completion (used internally) */
  544.     video_set = SDL_TRUE;
  545.     
  546.     return current;
  547. }
  548. static int QZ_ToggleFullScreen (_THIS, int on) { 
  549.     return -1;
  550. }
  551. static int QZ_SetColors (_THIS, int first_color, int num_colors, 
  552.  SDL_Color *colors) {
  553.     CGTableCount  index;
  554.     CGDeviceColor color;
  555.     
  556.     for (index = first_color; index < first_color+num_colors; index++) {
  557.     
  558.         /* Clamp colors between 0.0 and 1.0 */
  559.         color.red   = colors->r / 255.0;
  560.         color.blue  = colors->b / 255.0;
  561.         color.green = colors->g / 255.0;
  562.         
  563.         colors++;
  564.         
  565.         CGPaletteSetColorAtIndex (palette, color, index);
  566.     }
  567.      
  568.     if ( CGDisplayNoErr != CGDisplaySetPalette (display_id, palette) )
  569.         return 0;
  570.         
  571.     return 1;
  572. }
  573. static void QZ_DirectUpdate (_THIS, int num_rects, SDL_Rect *rects) {
  574.     #pragma unused(this,num_rects,rects)
  575. }
  576. /** 
  577.  *  The obscured code is based on work by Matt Slot fprefect@ambrosiasw.com,
  578.  *  who supplied sample code for Carbon.
  579.  **/
  580. static int QZ_IsWindowObscured (NSWindow *window) {
  581. //#define TEST_OBSCURED 1
  582. #if TEST_OBSCURED
  583.     
  584.     /*  In order to determine if a direct copy to the screen is possible,
  585.        we must figure out if there are any windows covering ours (including shadows).
  586.        This can be done by querying the window server about the on screen            
  587.        windows for their screen rectangle and window level.                          
  588.        The procedure used below is puts accuracy before speed; however, it aims to call 
  589.        the window server the fewest number of times possible to keep things reasonable.
  590.        In my testing on a 300mhz G3, this routine typically takes < 2 ms. -DW
  591.     
  592.         Notes:
  593.             -Calls into the Window Server involve IPC which is slow.
  594.             -Getting a rectangle seems slower than getting the window level
  595.             -The window list we get back is in sorted order, top to bottom
  596.             -On average, I suspect, most windows above ours are dock icon windows (hence optimization)
  597.             -Some windows above ours are always there, and cannot move or obscure us (menu bar)
  598.             
  599.         Bugs:
  600.             -no way (yet) to deactivate direct drawing when a window is dragged, 
  601.                or suddenly obscured, so drawing continues and can produce garbage
  602.                We need some kind of locking mechanism on window movement to prevent this
  603.                
  604.             -deactivated normal windows use activated normal 
  605.                window shadows (slight inaccuraccy)
  606.     */
  607.     
  608.     /* Cache the connection to the window server */
  609.     static CGSConnectionID cgsConnection = (CGSConnectionID) -1;
  610.     
  611.     /* Cache the dock icon windows */
  612.     static CGSWindowID          dockIcons[kMaxWindows];
  613.     static int                  numCachedDockIcons = 0;
  614.     
  615.     CGSWindowID         windows[kMaxWindows];
  616.     CGSWindowCount         i, count;
  617.     CGSWindowLevel         winLevel;
  618.     CGSRect winRect;
  619.     CGSRect contentRect;
  620.     int     windowNumber;
  621.     //int     isMainWindow;
  622.     int     firstDockIcon;    
  623.     int     dockIconCacheMiss;
  624.     int     windowContentOffset;
  625.     
  626.     int     obscured = SDL_TRUE;
  627.         
  628.     if ( [ window isVisible ] ) {
  629.         
  630.         /*  walk the window list looking for windows over top of 
  631.                 (or casting a shadow on) ours */
  632.        
  633.         /* Get a connection to the window server */
  634.         /* Should probably be moved out into SetVideoMode() or InitVideo() */
  635.         if (cgsConnection == (CGSConnectionID) -1) {
  636.             cgsConnection = (CGSConnectionID) 0;
  637.             cgsConnection = _CGSDefaultConnection ();
  638.         }
  639.         
  640.         if (cgsConnection) { 
  641.         
  642.             if ( ! [ window styleMask ] & NSBorderlessWindowMask )
  643.                 windowContentOffset = 22;
  644.             else
  645.                 windowContentOffset = 0;
  646.                 
  647.             windowNumber = [ window windowNumber ];
  648.             //isMainWindow = [ window isMainWindow ];
  649.             
  650.             /* The window list is sorted according to order on the screen */
  651.             count = 0;
  652.             CGSGetOnScreenWindowList (cgsConnection, 0, kMaxWindows, windows, &count);
  653.             CGSGetScreenRectForWindow (cgsConnection, windowNumber, &contentRect);
  654.     
  655.             /* adjust rect for window title bar (if present) */
  656.             contentRect.origin.y    += windowContentOffset;
  657.             contentRect.size.height -= windowContentOffset;
  658.             
  659.             firstDockIcon = -1;
  660.             dockIconCacheMiss = SDL_FALSE;
  661.             
  662.             /* The first window is always an empty window with level kCGSWindowLevelTop 
  663.                so start at index 1 */
  664.             for (i = 1; i < count; i++) {
  665.                 
  666.                 /* If we reach our window in the list, it cannot be obscured */
  667.                 if (windows[i] == windowNumber) {
  668.                     
  669.                     obscured = SDL_FALSE;
  670.                     break;
  671.                 }
  672.                 else {
  673.                     
  674.                     float shadowSide;
  675.                     float shadowTop;
  676.                     float shadowBottom;
  677.                     CGSGetWindowLevel (cgsConnection, windows[i], &winLevel);
  678.                     
  679.                     if (winLevel == kCGSWindowLevelDockIcon) {
  680.                     
  681.                         int j;
  682.                         
  683.                         if (firstDockIcon < 0) {
  684.                             
  685.                             firstDockIcon = i;
  686.                         
  687.                             if (numCachedDockIcons > 0) {
  688.                             
  689.                                 for (j = 0; j < numCachedDockIcons; j++) {
  690.                             
  691.                                     if (windows[i] == dockIcons[j])
  692.                                         i++;
  693.                                     else
  694.                                         break;
  695.                                 }
  696.                             
  697.                                 if (j != 0) {
  698.                                  
  699.                                     i--;
  700.                                                                     
  701.                                     if (j < numCachedDockIcons) {
  702.                                         
  703.                                         dockIconCacheMiss = SDL_TRUE;
  704.                                     }
  705.                                 }
  706.                             }
  707.                         }
  708.                                                
  709.                         continue;
  710.                     }
  711.                     else if (winLevel == kCGSWindowLevelMenuIgnore
  712.                              /* winLevel == kCGSWindowLevelTop */) {
  713.                      
  714.                         continue; /* cannot obscure window */
  715.                     }
  716.                     else if (winLevel == kCGSWindowLevelDockMenu ||
  717.                              winLevel == kCGSWindowLevelMenu) {
  718.                      
  719.                         shadowSide = 18;
  720.                         shadowTop = 4;
  721.                         shadowBottom = 22;   
  722.                     }
  723.                     else if (winLevel == kCGSWindowLevelUtility) {
  724.                     
  725.                         shadowSide = 8;
  726.                         shadowTop = 4;
  727.                         shadowBottom = 12;
  728.                     }
  729.                     else if (winLevel == kCGSWindowLevelNormal) {
  730.                     
  731.                         /* These numbers are for foreground windows,
  732.                            they are too big (but will work) for background windows */
  733.                         shadowSide = 20;
  734.                         shadowTop = 10;
  735.                         shadowBottom = 24;
  736.                     }
  737.                     else if (winLevel == kCGSWindowLevelDock) {
  738.                     
  739.                         /* Create dock icon cache */
  740.                         if (numCachedDockIcons != (i-firstDockIcon) ||
  741.                             dockIconCacheMiss) {
  742.                         
  743.                             numCachedDockIcons = i - firstDockIcon;
  744.                             memcpy (dockIcons, &(windows[firstDockIcon]), 
  745.                                     numCachedDockIcons * sizeof(*windows));
  746.                         }
  747.                         
  748.                         /* no shadow */
  749.                         shadowSide = 0;
  750.                         shadowTop = 0;
  751.                         shadowBottom = 0;
  752.                     }
  753.                     else {
  754.                     
  755.                         /*   kCGSWindowLevelDockLabel,
  756.                              kCGSWindowLevelDock,
  757.                              kOther??? */
  758.                         
  759.                         /* no shadow */
  760.                         shadowSide = 0;
  761.                         shadowTop = 0;
  762.                         shadowBottom = 0;
  763.                     }
  764.                     
  765.                     CGSGetScreenRectForWindow (cgsConnection, windows[i], &winRect);
  766.                     
  767.                     winRect.origin.x -= shadowSide;
  768.                     winRect.origin.y -= shadowTop;
  769.                     winRect.size.width += shadowSide;
  770.                     winRect.size.height += shadowBottom;
  771.                     
  772.                     if (NSIntersectsRect (contentRect, winRect)) {
  773.                     
  774.                         obscured = SDL_TRUE;
  775.                         break;
  776.                     }
  777.                
  778.                } /* window was not our window */
  779.             
  780.             } /* iterate over windows */
  781.             
  782.         } /* get cgsConnection */
  783.     
  784.     } /* window is visible */
  785.     
  786.     return obscured;
  787. #else
  788.     return SDL_TRUE;
  789. #endif
  790. }
  791. static void QZ_UpdateRects (_THIS, int numRects, SDL_Rect *rects) { 
  792.     if (SDL_VideoSurface->flags & SDL_OPENGLBLIT) {
  793.         QZ_GL_SwapBuffers (this);
  794.     }
  795.     else if ( [ qz_window isMiniaturized ] && 
  796.               ! (SDL_VideoSurface->flags & SDL_OPENGL)) {
  797.     
  798.         /** 
  799.          * Set port alpha opaque so deminiaturize looks right
  800.          * This isn't so nice, but there is no 
  801.          * initial deminatureize notification (before demini starts)
  802.          **/
  803.         QZ_SetPortAlphaOpaque ([ [ qz_window contentView ] qdPort],
  804.                                 [ qz_window styleMask ] & NSBorderlessWindowMask);
  805.     }
  806.     else if ( ! QZ_IsWindowObscured (qz_window) ) {
  807.         
  808.         /* Use direct copy to flush contents to the display */
  809.         CGrafPtr savePort;
  810.         CGrafPtr dstPort, srcPort;
  811.         const BitMap  *dstBits, *srcBits;
  812.         Rect     dstRect, srcRect;      
  813.         Point    offset;
  814.         int i;
  815.         
  816.         GetPort (&savePort);
  817.         
  818.         dstPort = CreateNewPortForCGDisplayID ((UInt32)display_id);
  819.         srcPort = [ window_view qdPort ];
  820.         
  821.         offset.h = 0;
  822.         offset.v = 0;
  823.         SetPort (srcPort);
  824.         LocalToGlobal (&offset);
  825.         
  826.         SetPort (dstPort);
  827.         
  828.         LockPortBits (dstPort);
  829.         LockPortBits (srcPort);
  830.         
  831.         dstBits = GetPortBitMapForCopyBits (dstPort);
  832.         srcBits = GetPortBitMapForCopyBits (srcPort);
  833.         
  834.         for (i = 0; i < numRects; i++) {
  835.             
  836.             SetRect (&srcRect, rects[i].x, rects[i].y,
  837.                      rects[i].x + rects[i].w,
  838.                      rects[i].y + rects[i].h);
  839.             
  840.             SetRect (&dstRect,
  841.                      rects[i].x + offset.h, 
  842.                      rects[i].y + offset.v,
  843.                      rects[i].x + rects[i].w + offset.h,
  844.                      rects[i].y + rects[i].h + offset.v);
  845.                         
  846.             CopyBits (srcBits, dstBits,
  847.                       &srcRect, &dstRect, srcCopy, NULL);
  848.                         
  849.         }
  850.         
  851.         SetPort (savePort);
  852.     }
  853.     else {
  854.     
  855.         /* Use QDFlushPortBuffer() to flush content to display */
  856.         int i;
  857.         RgnHandle dirty = NewRgn ();
  858.         RgnHandle temp  = NewRgn ();
  859.         
  860.         SetEmptyRgn (dirty);
  861.         
  862.         /* Build the region of dirty rectangles */
  863.         for (i = 0; i < numRects; i++) {
  864.         
  865.             MacSetRectRgn (temp, rects[i].x, rects[i].y, 
  866.                 rects[i].x + rects[i].w, rects[i].y + rects[i].h);
  867.             MacUnionRgn (dirty, temp, dirty);
  868.         }
  869.         
  870.         /* Flush the dirty region */
  871.         QDFlushPortBuffer ( [ window_view qdPort ], dirty );
  872.         DisposeRgn (dirty);
  873.         DisposeRgn (temp);
  874.     }
  875. }
  876. static void QZ_VideoQuit (_THIS) {
  877.     QZ_UnsetVideoMode (this);
  878.     CGPaletteRelease (palette);
  879. }
  880. static int  QZ_FillHWRect (_THIS, SDL_Surface *dst, SDL_Rect *rect, Uint32 color) {
  881.     CGSDisplayHWFill (display_id, rect->x, rect->y, rect->w, rect->h, color);
  882.     return 0;
  883. }
  884. static int  QZ_LockHWSurface(_THIS, SDL_Surface *surface) { 
  885.     return 1;
  886. }
  887. static void QZ_UnlockHWSurface(_THIS, SDL_Surface *surface) {
  888. }
  889. static void QZ_FreeHWSurface (_THIS, SDL_Surface *surface) {
  890. }
  891. /*
  892. int QZ_FlipHWSurface (_THIS, SDL_Surface *surface) {
  893.     return 0;
  894. }
  895. */
  896. /* Gamma functions */
  897. static int QZ_SetGamma (_THIS, float red, float green, float blue) {
  898.     const CGGammaValue min = 0.0, max = 1.0;
  899.     if (red == 0.0)
  900.         red = FLT_MAX;
  901.     else
  902.         red = 1.0 / red;
  903.     if (green == 0.0)
  904.         green = FLT_MAX;
  905.     else
  906.         green = 1.0 / green;
  907.     if (blue == 0.0)
  908.         blue = FLT_MAX;
  909.     else
  910.         blue  = 1.0 / blue;
  911.     
  912.     if ( CGDisplayNoErr == CGSetDisplayTransferByFormula 
  913.         (display_id, min, max, red, min, max, green, min, max, blue) ) {
  914.             
  915.         return 0;
  916.     }
  917.     else {
  918.     
  919.         return -1;
  920.     }
  921. }
  922. static int QZ_GetGamma (_THIS, float *red, float *green, float *blue) {
  923.     CGGammaValue dummy;
  924.     if ( CGDisplayNoErr == CGGetDisplayTransferByFormula
  925.         (display_id, &dummy, &dummy, red, 
  926.  &dummy, &dummy, green, &dummy, &dummy, blue) )
  927.         
  928.         return 0;
  929.     else
  930.         return -1;
  931. }
  932. static int QZ_SetGammaRamp (_THIS, Uint16 *ramp) {
  933.  
  934.    const CGTableCount tableSize = 255;
  935.    CGGammaValue redTable[tableSize];
  936.    CGGammaValue greenTable[tableSize];
  937.    CGGammaValue blueTable[tableSize];
  938.    
  939.    int i;
  940.    
  941.    /* Extract gamma values into separate tables, convert to floats between 0.0 and 1.0 */
  942.    for (i = 0; i < 256; i++)
  943.     redTable[i % 256] = ramp[i] / 65535.0;
  944.    
  945.    for (i=256; i < 512; i++)
  946.     greenTable[i % 256] = ramp[i] / 65535.0;
  947.    
  948.    for (i=512; i < 768; i++)
  949.      blueTable[i % 256] = ramp[i] / 65535.0;
  950.      
  951.     if ( CGDisplayNoErr == CGSetDisplayTransferByTable 
  952.             (display_id, tableSize, redTable, greenTable, blueTable) )        
  953.         return 0;
  954.     else
  955.         return -1;
  956. }
  957. static int QZ_GetGammaRamp (_THIS, Uint16 *ramp) {
  958.     
  959.     const CGTableCount tableSize = 255;
  960.     CGGammaValue redTable[tableSize];
  961.     CGGammaValue greenTable[tableSize];
  962.     CGGammaValue blueTable[tableSize];
  963.     CGTableCount actual;
  964.     int i;
  965.     
  966.     if ( CGDisplayNoErr != CGGetDisplayTransferByTable 
  967.             (display_id, tableSize, redTable, greenTable, blueTable, &actual) ||
  968.           actual != tableSize)
  969.         
  970.         return -1;
  971.     
  972.     /* Pack tables into one array, with values from 0 to 65535 */
  973.     for (i = 0; i < 256; i++)
  974.         ramp[i] = redTable[i % 256] * 65535.0;
  975.    
  976.     for (i=256; i < 512; i++)
  977.         ramp[i] = greenTable[i % 256] * 65535.0;
  978.    
  979.     for (i=512; i < 768; i++)
  980.         ramp[i] = blueTable[i % 256] * 65535.0;
  981.       
  982.     return 0;    
  983. }
  984. /* OpenGL helper functions (used internally) */
  985. static int QZ_SetupOpenGL (_THIS, int bpp, Uint32 flags) {
  986.     NSOpenGLPixelFormatAttribute attr[32];
  987.     NSOpenGLPixelFormat *fmt;
  988.     int i = 0;
  989.     int colorBits = bpp;
  990.     if ( flags & SDL_FULLSCREEN ) {
  991.     
  992.         attr[i++] = NSOpenGLPFAFullScreen;
  993.     }
  994.     /* In windowed mode, the OpenGL pixel depth must match device pixel depth */
  995.     else if ( colorBits != device_bpp ) { 
  996.         colorBits = device_bpp;
  997.     }
  998.     
  999.     attr[i++] = NSOpenGLPFAColorSize;
  1000.     attr[i++] = colorBits;
  1001.     
  1002.     attr[i++] = NSOpenGLPFADepthSize;
  1003.     attr[i++] = this->gl_config.depth_size;
  1004.     if ( this->gl_config.double_buffer ) {
  1005.         attr[i++] = NSOpenGLPFADoubleBuffer;
  1006.     }
  1007.     
  1008.     if ( this->gl_config.stencil_size != 0 ) {
  1009.         attr[i++] = NSOpenGLPFAStencilSize;
  1010.         attr[i++] = this->gl_config.stencil_size;
  1011.     }
  1012.     
  1013.     attr[i++] = NSOpenGLPFAScreenMask;
  1014.     attr[i++] = CGDisplayIDToOpenGLDisplayMask (display_id);
  1015.     attr[i] = 0;
  1016.     
  1017.     fmt = [ [ NSOpenGLPixelFormat alloc ] initWithAttributes:attr ];
  1018.     if (fmt == nil) {
  1019.         SDL_SetError ("Failed creating OpenGL pixel format");
  1020.         return 0;
  1021.     }
  1022.     
  1023.     gl_context = [ [ NSOpenGLContext alloc ] initWithFormat:fmt 
  1024.         shareContext:nil];
  1025.     
  1026.     if (gl_context == nil) {
  1027.         SDL_SetError ("Failed creating OpenGL context");
  1028.         return 0;
  1029.     }
  1030.     
  1031.     /* Convince SDL that the GL "driver" is loaded */
  1032.     this->gl_config.driver_loaded = 1;
  1033.     
  1034.     [ fmt release ];
  1035.     
  1036.     return 1;
  1037. }
  1038. static void QZ_TearDownOpenGL (_THIS) {
  1039.     [ NSOpenGLContext clearCurrentContext ];
  1040.     [ gl_context clearDrawable ];
  1041.     [ gl_context release ];
  1042. }
  1043. /* SDL OpenGL functions */
  1044. static int    QZ_GL_LoadLibrary    (_THIS, const char *location) {
  1045.     this->gl_config.driver_loaded = 1;
  1046.     return 1;
  1047. }
  1048. static void*  QZ_GL_GetProcAddress (_THIS, const char *proc) {
  1049.     /* We may want to cache the bundleRef at some point */
  1050.     CFBundleRef bundle;
  1051.     CFURLRef bundleURL = CFURLCreateWithFileSystemPath (kCFAllocatorDefault, 
  1052.         CFSTR("/System/Library/Frameworks/OpenGL.framework"), kCFURLPOSIXPathStyle, true);
  1053.     
  1054.     CFStringRef functionName = CFStringCreateWithCString 
  1055.         (kCFAllocatorDefault, proc, kCFStringEncodingASCII);
  1056.     
  1057.     void *function;
  1058.     
  1059.     bundle = CFBundleCreate (kCFAllocatorDefault, bundleURL);
  1060.     assert (bundle != NULL);
  1061.     
  1062.     function = CFBundleGetFunctionPointerForName (bundle, functionName);
  1063.     CFRelease ( bundleURL );
  1064.     CFRelease ( functionName );
  1065.     CFRelease ( bundle );
  1066.     
  1067.     return function;
  1068. }
  1069. static int    QZ_GL_GetAttribute   (_THIS, SDL_GLattr attrib, int* value) {
  1070.     GLenum attr = 0;
  1071.     QZ_GL_MakeCurrent (this);
  1072.     switch (attrib) {
  1073.     case SDL_GL_RED_SIZE: attr = GL_RED_BITS;   break;
  1074.     case SDL_GL_BLUE_SIZE: attr = GL_BLUE_BITS;  break;
  1075.     case SDL_GL_GREEN_SIZE: attr = GL_GREEN_BITS; break;
  1076.     case SDL_GL_ALPHA_SIZE: attr = GL_ALPHA_BITS; break;
  1077.     case SDL_GL_DOUBLEBUFFER: attr = GL_DOUBLEBUFFER; break;
  1078.     case SDL_GL_DEPTH_SIZE: attr = GL_DEPTH_BITS;  break;
  1079.     case SDL_GL_STENCIL_SIZE: attr = GL_STENCIL_BITS; break;
  1080.     case SDL_GL_ACCUM_RED_SIZE: attr = GL_ACCUM_RED_BITS; break;
  1081.     case SDL_GL_ACCUM_GREEN_SIZE: attr = GL_ACCUM_GREEN_BITS; break;
  1082.     case SDL_GL_ACCUM_BLUE_SIZE: attr = GL_ACCUM_BLUE_BITS; break;
  1083.     case SDL_GL_ACCUM_ALPHA_SIZE: attr = GL_ACCUM_ALPHA_BITS; break;
  1084.     case SDL_GL_BUFFER_SIZE:
  1085.         {
  1086.             GLint bits = 0;
  1087.             GLint component;
  1088.             /* there doesn't seem to be a single flag in OpenGL for this! */
  1089.             glGetIntegerv (GL_RED_BITS, &component);   bits += component;
  1090.             glGetIntegerv (GL_GREEN_BITS,&component);  bits += component;
  1091.             glGetIntegerv (GL_BLUE_BITS, &component);  bits += component;
  1092.             glGetIntegerv (GL_ALPHA_BITS, &component); bits += component;
  1093.             *value = bits;
  1094.         }
  1095.         return 0;
  1096.     }
  1097.     glGetIntegerv (attr, (GLint *)value);
  1098.     return 0;
  1099. }
  1100. static int    QZ_GL_MakeCurrent    (_THIS) {
  1101.     [ gl_context makeCurrentContext ];
  1102.     return 0;
  1103. }
  1104. static void   QZ_GL_SwapBuffers    (_THIS) {
  1105.     [ gl_context flushBuffer ];
  1106. }