SDL_cgxyuv.c
上传用户:sun1608
上传日期:2007-02-02
资源大小:6116k
文件大小:5k
源码类别:

流媒体/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. #ifdef SAVE_RCSID
  19. static char rcsid =
  20.  "@(#) $Id: SDL_cgxyuv.c,v 1.3 2002/04/22 21:38:04 wmay Exp $";
  21. #endif
  22. /* This is the XFree86 Xv extension implementation of YUV video overlays */
  23. #ifdef XFREE86_XV
  24. #include <stdlib.h>
  25. #include <string.h>
  26. #include <X11/Xlib.h>
  27. #include <sys/ipc.h>
  28. #include <sys/shm.h>
  29. #include <X11/extensions/XShm.h>
  30. #include <X11/extensions/Xvlib.h>
  31. #include "SDL_error.h"
  32. #include "SDL_video.h"
  33. #include "SDL_x11yuv_c.h"
  34. #include "SDL_yuvfuncs.h"
  35. /* The functions used to manipulate software video overlays */
  36. static struct private_yuvhwfuncs x11_yuvfuncs = {
  37. X11_LockYUVOverlay,
  38. X11_UnlockYUVOverlay,
  39. X11_DisplayYUVOverlay,
  40. X11_FreeYUVOverlay
  41. };
  42. struct private_yuvhwdata {
  43. int port;
  44. XShmSegmentInfo yuvshm;
  45. XvImage *image;
  46. };
  47. SDL_Overlay *X11_CreateYUVOverlay(_THIS, int width, int height, Uint32 format, SDL_Surface *display)
  48. {
  49. SDL_Overlay *overlay;
  50. struct private_yuvhwdata *hwdata;
  51. int xv_port;
  52. int i, j;
  53. int adaptors;
  54. XvAdaptorInfo *ainfo;
  55. XShmSegmentInfo *yuvshm;
  56. xv_port = -1;
  57. if ( (Success == XvQueryExtension(GFX_Display, &j, &j, &j, &j, &j)) &&
  58.      (Success == XvQueryAdaptors(GFX_Display,
  59.                                  RootWindow(GFX_Display, SDL_Screen),
  60.                                  &adaptors, &ainfo)) ) {
  61. for ( i=0; (i<adaptors) && (xv_port == -1); ++i ) {
  62. if ( (ainfo[i].type & XvInputMask) &&
  63.      (ainfo[i].type & XvImageMask) ) {
  64. int num_formats;
  65. XvImageFormatValues *formats;
  66. formats = XvListImageFormats(GFX_Display,
  67.               ainfo[i].base_id, &num_formats);
  68. for ( j=0; j<num_formats; ++j ) {
  69. if ( (Uint32)formats[j].id == format ) {
  70. xv_port = ainfo[i].base_id;
  71. break;
  72. }
  73. }
  74. }
  75. }
  76. }
  77. if ( xv_port == -1 ) {
  78. SDL_SetError("No available video ports for requested format");
  79. return(NULL);
  80. }
  81. /* Create the overlay structure */
  82. overlay = (SDL_Overlay *)malloc(sizeof *overlay);
  83. if ( overlay == NULL ) {
  84. SDL_OutOfMemory();
  85. return(NULL);
  86. }
  87. memset(overlay, 0, (sizeof *overlay));
  88. /* Fill in the basic members */
  89. overlay->format = format;
  90. overlay->w = width;
  91. overlay->h = height;
  92. /* Set up the YUV surface function structure */
  93. overlay->hwfuncs = &x11_yuvfuncs;
  94. /* Create the pixel data and lookup tables */
  95. hwdata = (struct private_yuvhwdata *)malloc(sizeof *hwdata);
  96. overlay->hwdata = hwdata;
  97. if ( hwdata == NULL ) {
  98. SDL_OutOfMemory();
  99. SDL_FreeYUVOverlay(overlay);
  100. return(NULL);
  101. }
  102. yuvshm = &hwdata->yuvshm;
  103. memset(yuvshm, 0, sizeof(*yuvshm));
  104. hwdata->port = xv_port;
  105. hwdata->image = XvShmCreateImage(GFX_Display, xv_port, format,
  106.                                  0, width, height, yuvshm);
  107. if ( hwdata->image == NULL ) {
  108. SDL_OutOfMemory();
  109. SDL_FreeYUVOverlay(overlay);
  110. return(NULL);
  111. }
  112. yuvshm->shmid = shmget(IPC_PRIVATE, hwdata->image->data_size,
  113.                        IPC_CREAT | 0777);
  114. if ( yuvshm->shmid < 0 ) {
  115. SDL_SetError("Unable to get %d bytes shared memory",
  116.              hwdata->image->data_size);
  117. SDL_FreeYUVOverlay(overlay);
  118. return(NULL);
  119. }
  120. yuvshm->shmaddr  = (caddr_t) shmat(yuvshm->shmid, 0, 0);
  121. yuvshm->readOnly = False;
  122. hwdata->image->data = yuvshm->shmaddr;
  123. XShmAttach(GFX_Display, yuvshm);
  124. XSync(GFX_Display, False);
  125. shmctl(yuvshm->shmid, IPC_RMID, 0);
  126. /* We're all done.. */
  127. return(overlay);
  128. }
  129. int X11_LockYUVOverlay(_THIS, SDL_Overlay *overlay)
  130. {
  131. overlay->pixels = overlay->hwdata->image->data;
  132. /* What should the pitch be set to? */
  133. return(0);
  134. }
  135. void X11_UnlockYUVOverlay(_THIS, SDL_Overlay *overlay)
  136. {
  137. overlay->pixels = NULL;
  138. }
  139. int X11_DisplayYUVOverlay(_THIS, SDL_Overlay *overlay, SDL_Rect *dstrect)
  140. {
  141. struct private_yuvhwdata *hwdata;
  142. hwdata = overlay->hwdata;
  143. XvShmPutImage(GFX_Display, hwdata->port, SDL_Window, SDL_GC,
  144.               hwdata->image, 0, 0, overlay->w, overlay->h,
  145.               dstrect->x, dstrect->y, dstrect->w, dstrect->h, False);
  146. XSync(GFX_Display, False);
  147. return(0);
  148. }
  149. void X11_FreeYUVOverlay(_THIS, SDL_Overlay *overlay)
  150. {
  151. struct private_yuvhwdata *hwdata;
  152. hwdata = overlay->hwdata;
  153. if ( hwdata ) {
  154. if ( hwdata->yuvshm.shmaddr ) {
  155. XShmDetach(GFX_Display, &hwdata->yuvshm);
  156. shmdt(hwdata->yuvshm.shmaddr);
  157. }
  158. if ( hwdata->image ) {
  159. XFree(hwdata->image);
  160. }
  161. free(hwdata);
  162. }
  163. }
  164. #endif /* XFREE86_XV */