2005-01-09 18:37:02 +01:00
/*
* IWineD3DSurface Implementation
*
2006-05-22 23:49:09 +02:00
* Copyright 1998 Lionel Ulmer
2006-05-06 18:08:33 +02:00
* Copyright 2000 - 2001 TransGaming Technologies Inc .
2005-01-09 18:37:02 +01:00
* Copyright 2002 - 2005 Jason Edmeades
* Copyright 2002 - 2003 Raphael Junqueira
* Copyright 2004 Christian Costa
2005-07-13 16:15:54 +02:00
* Copyright 2005 Oliver Stieber
2006-05-09 20:07:40 +02:00
* Copyright 2006 Stefan D <EFBFBD> singer for CodeWeavers
2005-01-09 18:37:02 +01:00
*
* 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
2006-05-18 14:49:52 +02:00
* Foundation , Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 , USA
2005-01-09 18:37:02 +01:00
*/
# include "config.h"
2005-08-03 23:17:08 +02:00
# include "wine/port.h"
2005-01-09 18:37:02 +01:00
# include "wined3d_private.h"
WINE_DEFAULT_DEBUG_CHANNEL ( d3d_surface ) ;
# define GLINFO_LOCATION ((IWineD3DImpl *)(((IWineD3DDeviceImpl *)This->resource.wineD3DDevice)->wineD3D))->gl_info
2006-07-17 17:27:19 +02:00
typedef enum {
NO_CONVERSION ,
CONVERT_PALETTED ,
CONVERT_PALETTED_CK ,
CONVERT_CK_565 ,
CONVERT_CK_5551 ,
CONVERT_CK_4444 ,
CONVERT_CK_4444_ARGB ,
CONVERT_CK_1555 ,
CONVERT_555 ,
CONVERT_CK_RGB24 ,
CONVERT_CK_8888 ,
CONVERT_CK_8888_ARGB ,
2007-02-15 03:01:44 +01:00
CONVERT_RGB32_888 ,
CONVERT_V8U8
2006-07-17 17:27:19 +02:00
} CONVERT_TYPES ;
2006-11-06 19:45:10 +01:00
HRESULT d3dfmt_convert_surface ( BYTE * src , BYTE * dst , UINT pitch , UINT width , UINT height , UINT outpitch , CONVERT_TYPES convert , IWineD3DSurfaceImpl * surf ) ;
2006-07-17 17:27:19 +02:00
2006-09-11 18:51:18 +02:00
static void surface_download_data ( IWineD3DSurfaceImpl * This ) {
if ( This - > resource . format = = WINED3DFMT_DXT1 | |
This - > resource . format = = WINED3DFMT_DXT2 | | This - > resource . format = = WINED3DFMT_DXT3 | |
This - > resource . format = = WINED3DFMT_DXT4 | | This - > resource . format = = WINED3DFMT_DXT5 ) {
if ( ! GL_SUPPORT ( EXT_TEXTURE_COMPRESSION_S3TC ) ) { /* We can assume this as the texture would not have been created otherwise */
FIXME ( " (%p) : Attempting to lock a compressed texture when texture compression isn't supported by opengl \n " , This ) ;
} else {
TRACE ( " (%p) : Calling glGetCompressedTexImageARB level %d, format %#x, type %#x, data %p \n " , This , This - > glDescription . level ,
This - > glDescription . glFormat , This - > glDescription . glType , This - > resource . allocatedMemory ) ;
ENTER_GL ( ) ;
GL_EXTCALL ( glGetCompressedTexImageARB ( This - > glDescription . target , This - > glDescription . level , This - > resource . allocatedMemory ) ) ;
checkGLcall ( " glGetCompressedTexImageARB() " ) ;
LEAVE_GL ( ) ;
}
} else {
2007-02-13 20:21:48 +01:00
void * mem ;
int src_pitch = 0 ;
int dst_pitch = 0 ;
2007-02-15 03:01:44 +01:00
if ( This - > Flags & SFLAG_CONVERTED ) {
FIXME ( " Read back converted textures unsupported \n " ) ;
return ;
}
2007-02-13 20:21:48 +01:00
if ( This - > Flags & SFLAG_NONPOW2 ) {
src_pitch = This - > bytesPerPixel * This - > pow2Width ;
dst_pitch = IWineD3DSurface_GetPitch ( ( IWineD3DSurface * ) This ) ;
src_pitch = ( src_pitch + SURFACE_ALIGNMENT - 1 ) & ~ ( SURFACE_ALIGNMENT - 1 ) ;
mem = HeapAlloc ( GetProcessHeap ( ) , 0 , src_pitch * This - > pow2Height ) ;
} else {
mem = This - > resource . allocatedMemory ;
}
2006-09-11 18:51:18 +02:00
TRACE ( " (%p) : Calling glGetTexImage level %d, format %#x, type %#x, data %p \n " , This , This - > glDescription . level ,
2007-02-13 20:21:48 +01:00
This - > glDescription . glFormat , This - > glDescription . glType , mem ) ;
2006-09-11 18:51:18 +02:00
ENTER_GL ( ) ;
glGetTexImage ( This - > glDescription . target , This - > glDescription . level , This - > glDescription . glFormat ,
2007-02-13 20:21:48 +01:00
This - > glDescription . glType , mem ) ;
2006-09-11 18:51:18 +02:00
checkGLcall ( " glGetTexImage() " ) ;
LEAVE_GL ( ) ;
2007-02-13 20:21:48 +01:00
if ( This - > Flags & SFLAG_NONPOW2 ) {
LPBYTE src_data , dst_data ;
int y ;
2006-09-11 18:51:18 +02:00
/*
* Some games ( e . g . warhammer 40 k ) don ' t work properly with the odd pitches , preventing
* the surface pitch from being used to box non - power2 textures . Instead we have to use a hack to
* repack the texture so that the bpp * width pitch can be used instead of bpp * pow2width .
*
* We ' re doing this . . .
*
* instead of boxing the texture :
* | < - texture width - > | - - > pow2width | / \
* | 111111111111111111 | | |
* | 222 Texture 222222 | boxed empty | texture height
* | 3333 Data 33333333 | | |
* | 444444444444444444 | | \ /
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - |
* | boxed empty | boxed empty | pow2height
* | | | \ /
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*
*
* we ' re repacking the data to the expected texture width
*
* | < - texture width - > | - - > pow2width | / \
* | 111111111111111111222222222222222 | |
* | 222333333333333333333444444444444 | texture height
* | 444444 | |
* | | \ /
* | | |
* | empty | pow2height
* | | \ /
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*
* = = is the same as
*
* | < - texture width - > | / \
* | 111111111111111111 |
* | 222222222222222222 | texture height
* | 333333333333333333 |
* | 444444444444444444 | \ /
* - - - - - - - - - - - - - - - - - - - -
*
* this also means that any references to allocatedMemory should work with the data as if were a
* standard texture with a non - power2 width instead of texture boxed up to be a power2 texture .
*
* internally the texture is still stored in a boxed format so any references to textureName will
* get a boxed texture with width pow2width and not a texture of width currentDesc . Width .
2007-02-13 20:21:48 +01:00
*
* Performance should not be an issue , because applications normally do not lock the surfaces when
* rendering . If an app does , the SFLAG_DYNLOCK flag will kick in and the memory copy won ' t be released ,
* and doesn ' t have to be re - read .
2006-09-11 18:51:18 +02:00
*/
2007-02-13 20:21:48 +01:00
src_data = mem ;
dst_data = This - > resource . allocatedMemory ;
TRACE ( " (%p) : Repacking the surface data from pitch %d to pitch %d \n " , This , src_pitch , dst_pitch ) ;
for ( y = 1 ; y < This - > currentDesc . Height ; y + + ) {
/* skip the first row */
src_data + = src_pitch ;
dst_data + = dst_pitch ;
memcpy ( dst_data , src_data , dst_pitch ) ;
2006-09-11 18:51:18 +02:00
}
2007-02-13 20:21:48 +01:00
HeapFree ( GetProcessHeap ( ) , 0 , mem ) ;
2006-09-11 18:51:18 +02:00
}
}
}
static void surface_upload_data ( IWineD3DSurfaceImpl * This , GLsizei width , GLsizei height , GLenum format , GLenum type , const GLvoid * data ) {
if ( This - > resource . format = = WINED3DFMT_DXT1 | |
This - > resource . format = = WINED3DFMT_DXT2 | | This - > resource . format = = WINED3DFMT_DXT3 | |
This - > resource . format = = WINED3DFMT_DXT4 | | This - > resource . format = = WINED3DFMT_DXT5 ) {
if ( ! GL_SUPPORT ( EXT_TEXTURE_COMPRESSION_S3TC ) ) {
FIXME ( " Using DXT1/3/5 without advertized support \n " ) ;
} else {
TRACE ( " (%p) : Calling glCompressedTexSubImage2D w %d, h %d, data %p \n " , This , width , height , data ) ;
ENTER_GL ( ) ;
2007-02-13 20:28:25 +01:00
/* glCompressedTexSubImage2D for uploading and glTexImage2D for allocating does not work well on some drivers(r200 dri, MacOS ATI driver)
* glCompressedTexImage2D does not accept NULL pointers . So for compressed textures surface_allocate_surface does nothing , and this
* function uses glCompressedTexImage2D instead of the SubImage call
*/
GL_EXTCALL ( glCompressedTexImage2DARB ( This - > glDescription . target , This - > glDescription . level , This - > glDescription . glFormatInternal ,
width , height , 0 /* border */ , This - > resource . size , data ) ) ;
2006-09-11 18:51:18 +02:00
checkGLcall ( " glCompressedTexSubImage2D " ) ;
LEAVE_GL ( ) ;
}
} else {
TRACE ( " (%p) : Calling glTexSubImage2D w %d, h %d, data, %p \n " , This , width , height , data ) ;
ENTER_GL ( ) ;
glTexSubImage2D ( This - > glDescription . target , This - > glDescription . level , 0 , 0 , width , height , format , type , data ) ;
checkGLcall ( " glTexSubImage2D " ) ;
LEAVE_GL ( ) ;
}
}
static void surface_allocate_surface ( IWineD3DSurfaceImpl * This , GLenum internal , GLsizei width , GLsizei height , GLenum format , GLenum type ) {
TRACE ( " (%p) : Creating surface (target %#x) level %d, d3d format %s, internal format %#x, width %d, height %d, gl format %#x, gl type=%#x \n " , This ,
This - > glDescription . target , This - > glDescription . level , debug_d3dformat ( This - > resource . format ) , internal , width , height , format , type ) ;
2007-02-13 20:28:25 +01:00
if ( This - > resource . format = = WINED3DFMT_DXT1 | |
This - > resource . format = = WINED3DFMT_DXT2 | | This - > resource . format = = WINED3DFMT_DXT3 | |
This - > resource . format = = WINED3DFMT_DXT4 | | This - > resource . format = = WINED3DFMT_DXT5 ) {
/* glCompressedTexImage2D does not accept NULL pointers, so we cannot allocate a compressed texture without uploading data */
TRACE ( " Not allocating compressed surfaces, surface_upload_data will specify them \n " ) ;
return ;
}
2006-09-11 18:51:18 +02:00
ENTER_GL ( ) ;
glTexImage2D ( This - > glDescription . target , This - > glDescription . level , internal , width , height , 0 , format , type , NULL ) ;
checkGLcall ( " glTexImage2D " ) ;
LEAVE_GL ( ) ;
}
2005-01-09 18:37:02 +01:00
/* *******************************************
IWineD3DSurface IUnknown parts follow
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
HRESULT WINAPI IWineD3DSurfaceImpl_QueryInterface ( IWineD3DSurface * iface , REFIID riid , LPVOID * ppobj )
{
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-03-14 11:12:52 +01:00
/* Warn ,but be nice about things */
2005-11-10 13:14:56 +01:00
TRACE ( " (%p)->(%s,%p) \n " , This , debugstr_guid ( riid ) , ppobj ) ;
2007-01-12 00:17:04 +01:00
2005-03-14 11:12:52 +01:00
if ( IsEqualGUID ( riid , & IID_IUnknown )
2006-02-06 11:32:41 +01:00
| | IsEqualGUID ( riid , & IID_IWineD3DBase )
2005-03-14 11:12:52 +01:00
| | IsEqualGUID ( riid , & IID_IWineD3DResource )
| | IsEqualGUID ( riid , & IID_IWineD3DSurface ) ) {
IUnknown_AddRef ( ( IUnknown * ) iface ) ;
* ppobj = This ;
2006-04-25 23:59:12 +02:00
return S_OK ;
2005-03-14 11:12:52 +01:00
}
2006-04-25 23:59:12 +02:00
* ppobj = NULL ;
2005-01-09 18:37:02 +01:00
return E_NOINTERFACE ;
}
ULONG WINAPI IWineD3DSurfaceImpl_AddRef ( IWineD3DSurface * iface ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-01-19 17:59:01 +01:00
ULONG ref = InterlockedIncrement ( & This - > resource . ref ) ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p) : AddRef increasing from %d \n " , This , ref - 1 ) ;
2005-01-19 17:59:01 +01:00
return ref ;
2005-01-09 18:37:02 +01:00
}
ULONG WINAPI IWineD3DSurfaceImpl_Release ( IWineD3DSurface * iface ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-01-19 17:59:01 +01:00
ULONG ref = InterlockedDecrement ( & This - > resource . ref ) ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p) : Releasing from %d \n " , This , ref + 1 ) ;
2005-01-09 18:37:02 +01:00
if ( ref = = 0 ) {
2006-05-15 20:49:36 +02:00
IWineD3DDeviceImpl * device = ( IWineD3DDeviceImpl * ) This - > resource . wineD3DDevice ;
2005-07-11 22:38:27 +02:00
TRACE ( " (%p) : cleaning up \n " , This ) ;
if ( This - > glDescription . textureName ! = 0 ) { /* release the openGL texture.. */
2005-03-14 11:12:52 +01:00
ENTER_GL ( ) ;
2005-07-11 22:38:27 +02:00
TRACE ( " Deleting texture %d \n " , This - > glDescription . textureName ) ;
glDeleteTextures ( 1 , & This - > glDescription . textureName ) ;
2005-03-14 11:12:52 +01:00
LEAVE_GL ( ) ;
}
2006-05-06 18:08:33 +02:00
if ( This - > Flags & SFLAG_DIBSECTION ) {
/* Release the DC */
SelectObject ( This - > hDC , This - > dib . holdbitmap ) ;
DeleteDC ( This - > hDC ) ;
/* Release the DIB section */
DeleteObject ( This - > dib . DIBsection ) ;
This - > dib . bitmap_data = NULL ;
This - > resource . allocatedMemory = NULL ;
}
2006-07-27 17:24:55 +02:00
if ( This - > Flags & SFLAG_USERPTR ) IWineD3DSurface_SetMem ( iface , NULL ) ;
2006-05-06 18:08:33 +02:00
2005-03-29 21:01:00 +02:00
IWineD3DResourceImpl_CleanUp ( ( IWineD3DResource * ) iface ) ;
2006-05-15 20:49:36 +02:00
if ( iface = = device - > ddraw_primary )
device - > ddraw_primary = NULL ;
2005-07-11 22:38:27 +02:00
2007-02-12 19:21:10 +01:00
if ( iface = = device - > lastActiveRenderTarget ) {
device - > lastActiveRenderTarget = NULL ;
}
2005-07-11 22:38:27 +02:00
TRACE ( " (%p) Released \n " , This ) ;
2005-01-09 18:37:02 +01:00
HeapFree ( GetProcessHeap ( ) , 0 , This ) ;
2005-03-14 11:12:52 +01:00
2005-01-09 18:37:02 +01:00
}
return ref ;
}
/* ****************************************************
IWineD3DSurface IWineD3DResource parts follow
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
HRESULT WINAPI IWineD3DSurfaceImpl_GetDevice ( IWineD3DSurface * iface , IWineD3DDevice * * ppDevice ) {
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_GetDevice ( ( IWineD3DResource * ) iface , ppDevice ) ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_SetPrivateData ( IWineD3DSurface * iface , REFGUID refguid , CONST void * pData , DWORD SizeOfData , DWORD Flags ) {
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_SetPrivateData ( ( IWineD3DResource * ) iface , refguid , pData , SizeOfData , Flags ) ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_GetPrivateData ( IWineD3DSurface * iface , REFGUID refguid , void * pData , DWORD * pSizeOfData ) {
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_GetPrivateData ( ( IWineD3DResource * ) iface , refguid , pData , pSizeOfData ) ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_FreePrivateData ( IWineD3DSurface * iface , REFGUID refguid ) {
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_FreePrivateData ( ( IWineD3DResource * ) iface , refguid ) ;
2005-01-09 18:37:02 +01:00
}
2005-03-29 21:01:00 +02:00
DWORD WINAPI IWineD3DSurfaceImpl_SetPriority ( IWineD3DSurface * iface , DWORD PriorityNew ) {
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_SetPriority ( ( IWineD3DResource * ) iface , PriorityNew ) ;
2005-01-09 18:37:02 +01:00
}
2005-03-29 21:01:00 +02:00
DWORD WINAPI IWineD3DSurfaceImpl_GetPriority ( IWineD3DSurface * iface ) {
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_GetPriority ( ( IWineD3DResource * ) iface ) ;
2005-01-09 18:37:02 +01:00
}
2006-06-10 13:15:32 +02:00
void WINAPI IWineD3DSurfaceImpl_PreLoad ( IWineD3DSurface * iface ) {
2005-03-14 11:12:52 +01:00
/* TODO: re-write the way textures and managed,
* use a ' opengl context manager ' to manage RenderTarget surfaces
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-07-13 16:15:54 +02:00
2005-03-14 11:12:52 +01:00
/* TODO: check for locks */
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DBaseTexture * baseTexture = NULL ;
2005-07-11 22:38:27 +02:00
TRACE ( " (%p)Checking to see if the container is a base texture \n " , This ) ;
2006-04-07 12:51:12 +02:00
if ( IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DBaseTexture , ( void * * ) & baseTexture ) = = WINED3D_OK ) {
2005-03-14 11:12:52 +01:00
TRACE ( " Passing to conatiner \n " ) ;
IWineD3DBaseTexture_PreLoad ( baseTexture ) ;
IWineD3DBaseTexture_Release ( baseTexture ) ;
2005-07-13 16:15:54 +02:00
} else {
2005-03-14 11:12:52 +01:00
TRACE ( " (%p) : About to load surface \n " , This ) ;
ENTER_GL ( ) ;
#if 0 /* TODO: context manager support */
IWineD3DContextManager_PushState ( This - > contextManager , GL_TEXTURE_2D , ENABLED , NOW /* make sure the state is applied now */ ) ;
# endif
2005-07-11 22:38:27 +02:00
glEnable ( This - > glDescription . target ) ; /* make sure texture support is enabled in this context */
2006-09-11 18:51:18 +02:00
if ( ! This - > glDescription . level ) {
if ( ! This - > glDescription . textureName ) {
glGenTextures ( 1 , & This - > glDescription . textureName ) ;
checkGLcall ( " glGenTextures " ) ;
TRACE ( " Surface %p given name %d \n " , This , This - > glDescription . textureName ) ;
2005-03-14 11:12:52 +01:00
}
2006-09-11 18:51:18 +02:00
glBindTexture ( This - > glDescription . target , This - > glDescription . textureName ) ;
checkGLcall ( " glBindTexture " ) ;
IWineD3DSurface_LoadTexture ( iface ) ;
/* This is where we should be reducing the amount of GLMemoryUsed */
} else if ( This - > glDescription . textureName ) { /* NOTE: the level 0 surface of a mpmapped texture must be loaded first! */
/* assume this is a coding error not a real error for now */
FIXME ( " Mipmap surface has a glTexture bound to it! \n " ) ;
2005-03-14 11:12:52 +01:00
}
2006-03-28 14:20:47 +02:00
if ( This - > resource . pool = = WINED3DPOOL_DEFAULT ) {
2005-03-14 11:12:52 +01:00
/* Tell opengl to try and keep this texture in video ram (well mostly) */
GLclampf tmp ;
tmp = 0.9f ;
2005-07-11 22:38:27 +02:00
glPrioritizeTextures ( 1 , & This - > glDescription . textureName , & tmp ) ;
2005-03-14 11:12:52 +01:00
}
/* TODO: disable texture support, if it wastn't enabled when we entered. */
#if 0 /* TODO: context manager support */
IWineD3DContextManager_PopState ( This - > contextManager , GL_TEXTURE_2D , DISABLED , DELAYED
/* we don't care when the state is disabled(if atall) */ ) ;
# endif
LEAVE_GL ( ) ;
}
return ;
2005-01-09 18:37:02 +01:00
}
2006-03-09 23:21:16 +01:00
WINED3DRESOURCETYPE WINAPI IWineD3DSurfaceImpl_GetType ( IWineD3DSurface * iface ) {
2005-07-13 16:15:54 +02:00
TRACE ( " (%p) : calling resourceimpl_GetType \n " , iface ) ;
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_GetType ( ( IWineD3DResource * ) iface ) ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_GetParent ( IWineD3DSurface * iface , IUnknown * * pParent ) {
2005-07-13 16:15:54 +02:00
TRACE ( " (%p) : calling resourceimpl_GetParent \n " , iface ) ;
2005-01-17 14:44:57 +01:00
return IWineD3DResourceImpl_GetParent ( ( IWineD3DResource * ) iface , pParent ) ;
2005-01-09 18:37:02 +01:00
}
/* ******************************************************
IWineD3DSurface IWineD3DSurface parts follow
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-03-14 11:12:52 +01:00
HRESULT WINAPI IWineD3DSurfaceImpl_GetContainer ( IWineD3DSurface * iface , REFIID riid , void * * ppContainer ) {
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-02-07 12:25:59 +01:00
IWineD3DBase * container = 0 ;
TRACE ( " (This %p, riid %s, ppContainer %p) \n " , This , debugstr_guid ( riid ) , ppContainer ) ;
if ( ! ppContainer ) {
ERR ( " Called without a valid ppContainer. \n " ) ;
2005-03-14 11:12:52 +01:00
}
2006-02-07 12:25:59 +01:00
2005-01-09 18:37:02 +01:00
/** From MSDN:
2005-03-14 11:12:52 +01:00
* If the surface is created using CreateImageSurface / CreateOffscreenPlainSurface , CreateRenderTarget ,
* or CreateDepthStencilSurface , the surface is considered stand alone . In this case ,
* GetContainer will return the Direct3D device used to create the surface .
2005-01-09 18:37:02 +01:00
*/
2006-02-07 12:25:59 +01:00
if ( This - > container ) {
container = This - > container ;
} else {
2006-02-24 21:40:18 +01:00
container = ( IWineD3DBase * ) This - > resource . wineD3DDevice ;
2006-02-07 12:25:59 +01:00
}
TRACE ( " Relaying to QueryInterface \n " ) ;
2006-06-06 23:42:33 +02:00
return IUnknown_QueryInterface ( container , riid , ppContainer ) ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_GetDesc ( IWineD3DSurface * iface , WINED3DSURFACE_DESC * pDesc ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p) : copying into %p \n " , This , pDesc ) ;
2005-07-12 19:02:47 +02:00
if ( pDesc - > Format ! = NULL ) * ( pDesc - > Format ) = This - > resource . format ;
if ( pDesc - > Type ! = NULL ) * ( pDesc - > Type ) = This - > resource . resourceType ;
if ( pDesc - > Usage ! = NULL ) * ( pDesc - > Usage ) = This - > resource . usage ;
if ( pDesc - > Pool ! = NULL ) * ( pDesc - > Pool ) = This - > resource . pool ;
if ( pDesc - > Size ! = NULL ) * ( pDesc - > Size ) = This - > resource . size ; /* dx8 only */
if ( pDesc - > MultiSampleType ! = NULL ) * ( pDesc - > MultiSampleType ) = This - > currentDesc . MultiSampleType ;
if ( pDesc - > MultiSampleQuality ! = NULL ) * ( pDesc - > MultiSampleQuality ) = This - > currentDesc . MultiSampleQuality ;
if ( pDesc - > Width ! = NULL ) * ( pDesc - > Width ) = This - > currentDesc . Width ;
if ( pDesc - > Height ! = NULL ) * ( pDesc - > Height ) = This - > currentDesc . Height ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
2005-07-13 16:15:54 +02:00
void WINAPI IWineD3DSurfaceImpl_SetGlTextureDesc ( IWineD3DSurface * iface , UINT textureName , int target ) {
2005-07-11 22:38:27 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p) : setting textureName %u, target %i \n " , This , textureName , target ) ;
if ( This - > glDescription . textureName = = 0 & & textureName ! = 0 ) {
2006-05-06 16:15:37 +02:00
This - > Flags | = SFLAG_DIRTY ;
2005-07-11 22:38:27 +02:00
IWineD3DSurface_AddDirtyRect ( iface , NULL ) ;
}
This - > glDescription . textureName = textureName ;
This - > glDescription . target = target ;
}
2005-07-13 16:15:54 +02:00
void WINAPI IWineD3DSurfaceImpl_GetGlDesc ( IWineD3DSurface * iface , glDescriptor * * glDescription ) {
2005-07-11 22:38:27 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p) : returning %p \n " , This , & This - > glDescription ) ;
* glDescription = & This - > glDescription ;
}
/* TODO: think about moving this down to resource? */
const void * WINAPI IWineD3DSurfaceImpl_GetData ( IWineD3DSurface * iface ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
/* This should only be called for sysmem textures, it may be a good idea to extend this to all pools at some point in the futture */
2006-03-28 14:20:47 +02:00
if ( This - > resource . pool ! = WINED3DPOOL_SYSTEMMEM ) {
2005-07-11 22:38:27 +02:00
FIXME ( " (%p)Attempting to get system memory for a non-system memory texture \n " , iface ) ;
}
return ( CONST void * ) ( This - > resource . allocatedMemory ) ;
}
2007-02-13 20:24:00 +01:00
static void read_from_framebuffer ( IWineD3DSurfaceImpl * This , CONST RECT * rect , void * dest , UINT pitch , BOOL srcUpsideDown ) {
BYTE * mem ;
2006-07-17 12:39:44 +02:00
GLint fmt ;
GLint type ;
2007-02-13 20:24:00 +01:00
BYTE * row , * top , * bottom ;
int i ;
BOOL bpp ;
2006-07-17 12:39:44 +02:00
switch ( This - > resource . format )
{
case WINED3DFMT_P8 :
{
/* GL can't return palettized data, so read ARGB pixels into a
2006-08-06 12:28:32 +02:00
* separate block of memory and convert them into palettized format
2006-07-17 12:39:44 +02:00
* in software . Slow , but if the app means to use palettized render
2006-08-06 12:28:32 +02:00
* targets and locks it . . .
2006-07-17 12:39:44 +02:00
*
* Use GL_RGB , GL_UNSIGNED_BYTE to read the surface for performance reasons
* Don ' t use GL_BGR as in the WINED3DFMT_R8G8B8 case , instead watch out
* for the color channels when palettizing the colors .
*/
fmt = GL_RGB ;
type = GL_UNSIGNED_BYTE ;
pitch * = 3 ;
2007-02-13 20:24:00 +01:00
mem = HeapAlloc ( GetProcessHeap ( ) , 0 , This - > resource . size * 3 ) ;
2006-07-17 12:39:44 +02:00
if ( ! mem ) {
ERR ( " Out of memory \n " ) ;
return ;
}
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel * 3 ;
2006-07-17 12:39:44 +02:00
}
break ;
default :
mem = dest ;
fmt = This - > glDescription . glFormat ;
type = This - > glDescription . glType ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel ;
2006-07-17 12:39:44 +02:00
}
2007-02-13 20:24:00 +01:00
glReadPixels ( rect - > left , rect - > top ,
rect - > right - rect - > left ,
rect - > bottom - rect - > top ,
fmt , type , mem ) ;
vcheckGLcall ( " glReadPixels " ) ;
/* TODO: Merge this with the palettization loop below for P8 targets */
if ( ! srcUpsideDown ) {
UINT len , off ;
/* glReadPixels returns the image upside down, and there is no way to prevent this.
Flip the lines in software */
len = ( rect - > right - rect - > left ) * bpp ;
off = rect - > left * bpp ;
row = HeapAlloc ( GetProcessHeap ( ) , 0 , len ) ;
2006-07-17 12:39:44 +02:00
if ( ! row ) {
ERR ( " Out of memory \n " ) ;
2007-02-13 20:24:00 +01:00
if ( This - > resource . format = = WINED3DFMT_P8 ) HeapFree ( GetProcessHeap ( ) , 0 , mem ) ;
2006-07-17 12:39:44 +02:00
return ;
}
2007-02-13 20:24:00 +01:00
top = mem + pitch * rect - > top ;
2006-07-17 12:39:44 +02:00
bottom = ( ( BYTE * ) mem ) + pitch * ( rect - > bottom - rect - > top - 1 ) ;
for ( i = 0 ; i < ( rect - > bottom - rect - > top ) / 2 ; i + + ) {
2007-02-13 20:24:00 +01:00
memcpy ( row , top + off , len ) ;
memcpy ( top + off , bottom + off , len ) ;
memcpy ( bottom + off , row , len ) ;
2006-07-17 12:39:44 +02:00
top + = pitch ;
bottom - = pitch ;
}
HeapFree ( GetProcessHeap ( ) , 0 , row ) ;
}
if ( This - > resource . format = = WINED3DFMT_P8 ) {
PALETTEENTRY * pal ;
DWORD width = pitch / 3 ;
int x , y , c ;
if ( This - > palette ) {
pal = This - > palette - > palents ;
} else {
pal = This - > resource . wineD3DDevice - > palettes [ This - > resource . wineD3DDevice - > currentPalette ] ;
}
for ( y = rect - > top ; y < rect - > bottom ; y + + ) {
for ( x = rect - > left ; x < rect - > right ; x + + ) {
/* start lines pixels */
BYTE * blue = ( BYTE * ) ( ( BYTE * ) mem ) + y * pitch + x * ( sizeof ( BYTE ) * 3 ) ;
BYTE * green = blue + 1 ;
BYTE * red = green + 1 ;
for ( c = 0 ; c < 256 ; c + + ) {
if ( * red = = pal [ c ] . peRed & &
* green = = pal [ c ] . peGreen & &
* blue = = pal [ c ] . peBlue )
{
* ( ( BYTE * ) dest + y * width + x ) = c ;
break ;
}
}
}
}
HeapFree ( GetProcessHeap ( ) , 0 , mem ) ;
}
}
2006-06-10 13:15:32 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_LockRect ( IWineD3DSurface * iface , WINED3DLOCKED_RECT * pLockedRect , CONST RECT * pRect , DWORD Flags ) {
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-07-07 22:35:05 +02:00
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
IWineD3DSwapChainImpl * swapchain = NULL ;
2007-02-13 20:21:44 +01:00
TRACE ( " (%p) : rect@%p flags(%08x), output lockedRect@%p, memory@%p \n " , This , pRect , Flags , pLockedRect , This - > resource . allocatedMemory ) ;
2005-01-09 18:37:02 +01:00
2006-05-06 16:15:37 +02:00
if ( ! ( This - > Flags & SFLAG_LOCKABLE ) ) {
2005-07-13 16:15:54 +02:00
/* Note: UpdateTextures calls CopyRects which calls this routine to populate the
2005-01-09 18:37:02 +01:00
texture regions , and since the destination is an unlockable region we need
to tolerate this */
2005-07-13 16:15:54 +02:00
TRACE ( " Warning: trying to lock unlockable surf@%p \n " , This ) ;
2006-04-07 12:51:12 +02:00
/*return WINED3DERR_INVALIDCALL; */
2005-01-09 18:37:02 +01:00
}
2007-02-13 20:21:44 +01:00
pLockedRect - > Pitch = IWineD3DSurface_GetPitch ( iface ) ;
2005-07-07 22:35:05 +02:00
2007-02-13 20:21:44 +01:00
/* Mark the surface locked */
This - > Flags | = SFLAG_LOCKED ;
2005-07-07 22:35:05 +02:00
2007-02-13 20:21:44 +01:00
/* Whatever surface we have, make sure that there is memory allocated for the downloaded copy */
if ( ! This - > resource . allocatedMemory ) {
This - > resource . allocatedMemory = HeapAlloc ( GetProcessHeap ( ) , 0 , This - > resource . size + 4 ) ;
This - > Flags | = SFLAG_GLDIRTY ; /* This is the marker that surface data has to be downloaded */
2005-01-09 18:37:02 +01:00
}
2007-02-13 20:21:44 +01:00
/* Calculate the correct start address to report */
2005-01-09 18:37:02 +01:00
if ( NULL = = pRect ) {
2005-03-29 21:01:00 +02:00
pLockedRect - > pBits = This - > resource . allocatedMemory ;
2005-01-09 18:37:02 +01:00
This - > lockedRect . left = 0 ;
This - > lockedRect . top = 0 ;
This - > lockedRect . right = This - > currentDesc . Width ;
This - > lockedRect . bottom = This - > currentDesc . Height ;
2006-10-01 05:20:10 +02:00
TRACE ( " Locked Rect (%p) = l %d, t %d, r %d, b %d \n " , & This - > lockedRect , This - > lockedRect . left , This - > lockedRect . top , This - > lockedRect . right , This - > lockedRect . bottom ) ;
2005-01-09 18:37:02 +01:00
} else {
2006-10-01 05:20:10 +02:00
TRACE ( " Lock Rect (%p) = l %d, t %d, r %d, b %d \n " , pRect , pRect - > left , pRect - > top , pRect - > right , pRect - > bottom ) ;
2005-01-09 18:37:02 +01:00
2006-06-29 21:13:12 +02:00
if ( ( pRect - > top < 0 ) | |
( pRect - > left < 0 ) | |
( pRect - > left > = pRect - > right ) | |
( pRect - > top > = pRect - > bottom ) | |
( pRect - > right > This - > currentDesc . Width ) | |
( pRect - > bottom > This - > currentDesc . Height ) )
{
WARN ( " Invalid values in pRect !!! \n " ) ;
2007-02-15 13:32:25 +01:00
return WINED3DERR_INVALIDCALL ;
2006-06-29 21:13:12 +02:00
}
2005-03-29 21:01:00 +02:00
if ( This - > resource . format = = WINED3DFMT_DXT1 ) { /* DXT1 is half byte per pixel */
2005-08-29 14:20:02 +02:00
pLockedRect - > pBits = This - > resource . allocatedMemory + ( pLockedRect - > Pitch * pRect - > top ) + ( ( pRect - > left * This - > bytesPerPixel / 2 ) ) ;
2005-01-09 18:37:02 +01:00
} else {
2005-03-29 21:01:00 +02:00
pLockedRect - > pBits = This - > resource . allocatedMemory + ( pLockedRect - > Pitch * pRect - > top ) + ( pRect - > left * This - > bytesPerPixel ) ;
2005-01-09 18:37:02 +01:00
}
This - > lockedRect . left = pRect - > left ;
This - > lockedRect . top = pRect - > top ;
This - > lockedRect . right = pRect - > right ;
This - > lockedRect . bottom = pRect - > bottom ;
}
2006-05-06 16:15:37 +02:00
if ( This - > Flags & SFLAG_NONPOW2 ) {
2005-07-11 16:25:54 +02:00
TRACE ( " Locking non-power 2 texture \n " ) ;
}
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
/* Performance optimization: Count how often a surface is locked, if it is locked regularly do not throw away the system memory copy.
* This avoids the need to download the surface from opengl all the time . The surface is still downloaded if the opengl texture is
* changed
*/
if ( ! ( This - > Flags & SFLAG_DYNLOCK ) ) {
This - > lockCount + + ;
/* MAXLOCKCOUNT is defined in wined3d_private.h */
if ( This - > lockCount > MAXLOCKCOUNT ) {
TRACE ( " Surface is locked regularily, not freeing the system memory copy any more \n " ) ;
This - > Flags | = SFLAG_DYNLOCK ;
}
}
2005-07-13 13:34:55 +02:00
2007-02-13 20:21:44 +01:00
if ( ( Flags & WINED3DLOCK_DISCARD ) | | ! ( This - > Flags & SFLAG_GLDIRTY ) ) {
TRACE ( " WINED3DLOCK_DISCARD flag passed, or local copy is up to date, not downloading data \n " ) ;
goto lock_end ;
}
2006-05-23 00:48:54 +02:00
2007-02-13 20:21:44 +01:00
/* Now download the surface content from opengl
* Use the render target readback if the surface is on a swapchain ( = onscreen render target ) or the current primary target
* Offscreen targets which are not active at the moment or are higher targets ( fbos ) can be locked with the texture path
*/
IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DSwapChain , ( void * * ) & swapchain ) ;
if ( swapchain | | iface = = myDevice - > render_targets [ 0 ] ) {
2007-02-13 20:24:00 +01:00
BOOL srcIsUpsideDown ;
2007-02-13 20:21:44 +01:00
if ( wined3d_settings . rendertargetlock_mode = = RTL_DISABLE ) {
static BOOL warned = FALSE ;
if ( ! warned ) {
ERR ( " The application tries to lock the render target, but render target locking is disabled \n " ) ;
warned = TRUE ;
2006-05-06 18:08:33 +02:00
}
2007-02-13 20:21:44 +01:00
if ( swapchain ) IWineD3DSwapChain_Release ( ( IWineD3DSwapChain * ) swapchain ) ;
return WINED3D_OK ;
}
2006-09-11 18:51:18 +02:00
2007-02-13 20:21:44 +01:00
/* Activate the surface. Set it up for blitting now, although not necessarily needed for LockRect.
* Certain graphics drivers seem to dislike some enabled states when reading from opengl , the blitting usage
* should help here . Furthermore unlockrect will need the context set up for blitting . The context manager will find
* context - > last_was_blit set on the unlock .
*/
ENTER_GL ( ) ;
ActivateContext ( myDevice , iface , CTXUSAGE_BLIT ) ;
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
/* Select the correct read buffer, and give some debug output.
* There is no need to keep track of the current read buffer or reset it , every part of the code
* that reads sets the read buffer as desired .
*/
if ( ! swapchain ) {
/* Locking the primary render target which is not on a swapchain(=offscreen render target).
* Read from the back buffer
*/
TRACE ( " Locking offscreen render target \n " ) ;
glReadBuffer ( GL_BACK ) ;
2007-02-13 20:24:00 +01:00
srcIsUpsideDown = TRUE ;
2007-02-13 20:21:44 +01:00
} else {
if ( iface = = swapchain - > frontBuffer ) {
TRACE ( " Locking the front buffer \n " ) ;
glReadBuffer ( GL_FRONT ) ;
} else if ( swapchain - > backBuffer & & iface = = swapchain - > backBuffer [ 0 ] ) {
TRACE ( " Locking the back buffer \n " ) ;
glReadBuffer ( GL_BACK ) ;
2005-07-13 13:34:55 +02:00
} else {
2007-02-13 20:21:44 +01:00
/* Ok, there is an issue: OpenGL does not guarant any back buffer number, so all we can do is to read GL_BACK
* and hope it gives what the app wants
*/
FIXME ( " Application is locking a 2nd or higher back buffer \n " ) ;
glReadBuffer ( GL_BACK ) ;
2005-07-13 13:34:55 +02:00
}
2007-02-13 20:21:44 +01:00
IWineD3DSwapChain_Release ( ( IWineD3DSwapChain * ) swapchain ) ;
2007-02-13 20:24:00 +01:00
srcIsUpsideDown = FALSE ;
2005-07-13 13:34:55 +02:00
}
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
switch ( wined3d_settings . rendertargetlock_mode ) {
case RTL_AUTO :
case RTL_READDRAW :
case RTL_READTEX :
2007-02-13 20:24:00 +01:00
read_from_framebuffer ( This , & This - > lockedRect , This - > resource . allocatedMemory , pLockedRect - > Pitch , srcIsUpsideDown ) ;
2007-02-13 20:21:44 +01:00
break ;
2006-06-15 12:54:19 +02:00
2007-02-13 20:21:44 +01:00
case RTL_TEXDRAW :
case RTL_TEXTEX :
2007-02-13 20:24:00 +01:00
read_from_framebuffer ( This , & This - > lockedRect , This - > resource . allocatedMemory , pLockedRect - > Pitch , srcIsUpsideDown ) ;
2007-02-13 20:21:44 +01:00
FIXME ( " Reading from render target with a texture isn't implemented yet, falling back to framebuffer reading \n " ) ;
break ;
2005-01-09 18:37:02 +01:00
}
2007-02-13 20:21:44 +01:00
LEAVE_GL ( ) ;
2007-02-13 20:24:00 +01:00
/* Mark the local copy up to date if a full download was done */
if ( This - > lockedRect . left = = 0 & &
This - > lockedRect . top = = 0 & &
This - > lockedRect . right = = This - > currentDesc . Width & &
This - > lockedRect . bottom = = This - > currentDesc . Height ) {
This - > Flags & = ~ SFLAG_GLDIRTY ;
}
2007-02-13 20:21:44 +01:00
} else if ( iface = = myDevice - > stencilBufferTarget ) {
/** the depth stencil in openGL has a format of GL_FLOAT
* which should be good for WINED3DFMT_D16_LOCKABLE
* and WINED3DFMT_D16
* it is unclear what format the stencil buffer is in except .
* ' Each index is converted to fixed point . . .
* If GL_MAP_STENCIL is GL_TRUE , indices are replaced by their
* mappings in the table GL_PIXEL_MAP_S_TO_S .
* glReadPixels ( This - > lockedRect . left ,
* This - > lockedRect . bottom - j - 1 ,
* This - > lockedRect . right - This - > lockedRect . left ,
* 1 ,
* GL_DEPTH_COMPONENT ,
* type ,
* ( char * ) pLockedRect - > pBits + ( pLockedRect - > Pitch * ( j - This - > lockedRect . top ) ) ) ;
*
* Depth Stencil surfaces which are not the current depth stencil target should have their data in a
* gl texture ( next path ) , or in local memory ( early return because of missing SFLAG_GLDIRTY above ) . If
* none of that is the case the problem is not in this function : - )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
FIXME ( " Depth stencil locking not supported yet \n " ) ;
} else {
/* This path is for normal surfaces, offscreen render targets and everything else that is in a gl texture */
TRACE ( " locking an ordinarary surface \n " ) ;
2005-07-07 22:35:05 +02:00
2007-02-13 20:21:44 +01:00
/* TODO: Make sure that *any* context is active for this thread. It is not important which context that is,
* nor that is has any special setup ( CTXUSAGE_LOADRESOURCE is fine ) , but the code below needs a context .
* A context is guaranted to be there in a single threaded environment , but not with multithreading
*/
if ( 0 ! = This - > glDescription . textureName ) {
/* Now I have to copy thing bits back */
/* Make sure that a proper texture unit is selected, bind the texture and dirtify the sampler to restore the texture on the next draw */
if ( GL_SUPPORT ( ARB_MULTITEXTURE ) ) {
ENTER_GL ( ) ;
GL_EXTCALL ( glActiveTextureARB ( GL_TEXTURE0_ARB ) ) ;
checkGLcall ( " glActiveTextureARB " ) ;
LEAVE_GL ( ) ;
}
IWineD3DDeviceImpl_MarkStateDirty ( This - > resource . wineD3DDevice , STATE_SAMPLER ( 0 ) ) ;
IWineD3DSurface_PreLoad ( iface ) ;
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
surface_download_data ( This ) ;
2005-07-07 22:35:05 +02:00
}
2007-02-13 20:24:00 +01:00
/* The local copy is now up to date to the opengl one because a full download was done */
This - > Flags & = ~ SFLAG_GLDIRTY ;
2005-01-09 18:37:02 +01:00
}
2007-02-13 20:21:44 +01:00
lock_end :
2006-03-08 00:11:28 +01:00
if ( Flags & ( WINED3DLOCK_NO_DIRTY_UPDATE | WINED3DLOCK_READONLY ) ) {
2005-01-09 18:37:02 +01:00
/* Don't dirtify */
} else {
2005-03-14 11:12:52 +01:00
IWineD3DBaseTexture * pBaseTexture ;
2005-01-09 18:37:02 +01:00
/**
* Dirtify on lock
* as seen in msdn docs
*/
IWineD3DSurface_AddDirtyRect ( iface , & This - > lockedRect ) ;
2005-07-07 22:35:05 +02:00
/** Dirtify Container if needed */
2006-04-07 12:51:12 +02:00
if ( WINED3D_OK = = IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DBaseTexture , ( void * * ) & pBaseTexture ) & & pBaseTexture ! = NULL ) {
2005-03-14 11:12:52 +01:00
TRACE ( " Making container dirty \n " ) ;
2005-07-07 22:35:05 +02:00
IWineD3DBaseTexture_SetDirty ( pBaseTexture , TRUE ) ;
2005-03-14 11:12:52 +01:00
IWineD3DBaseTexture_Release ( pBaseTexture ) ;
2005-07-13 16:15:54 +02:00
} else {
2005-03-14 11:12:52 +01:00
TRACE ( " Surface is standalone, no need to dirty the container \n " ) ;
2005-01-09 18:37:02 +01:00
}
}
2006-05-06 16:15:37 +02:00
TRACE ( " returning memory@%p, pitch(%d) dirtyfied(%d) \n " , pLockedRect - > pBits , pLockedRect - > Pitch , This - > Flags & SFLAG_DIRTY ? 0 : 1 ) ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
2006-07-17 17:27:19 +02:00
static void flush_to_framebuffer_drawpixels ( IWineD3DSurfaceImpl * This ) {
GLint prev_store ;
GLint prev_rasterpos [ 4 ] ;
2005-01-09 18:37:02 +01:00
GLint skipBytes = 0 ;
2007-02-13 20:21:44 +01:00
BOOL storechanged = FALSE , memory_allocated = FALSE ;
2006-07-17 17:27:19 +02:00
GLint fmt , type ;
2007-02-13 20:24:00 +01:00
BYTE * mem ;
UINT bpp ;
UINT pitch = IWineD3DSurface_GetPitch ( ( IWineD3DSurface * ) This ) ; /* target is argb, 4 byte */
2006-07-17 17:27:19 +02:00
glDisable ( GL_TEXTURE_2D ) ;
vcheckGLcall ( " glDisable(GL_TEXTURE_2D) " ) ;
glFlush ( ) ;
vcheckGLcall ( " glFlush " ) ;
glGetIntegerv ( GL_PACK_SWAP_BYTES , & prev_store ) ;
vcheckGLcall ( " glIntegerv " ) ;
glGetIntegerv ( GL_CURRENT_RASTER_POSITION , & prev_rasterpos [ 0 ] ) ;
vcheckGLcall ( " glIntegerv " ) ;
glPixelZoom ( 1.0 , - 1.0 ) ;
vcheckGLcall ( " glPixelZoom " ) ;
/* If not fullscreen, we need to skip a number of bytes to find the next row of data */
glGetIntegerv ( GL_UNPACK_ROW_LENGTH , & skipBytes ) ;
glPixelStorei ( GL_UNPACK_ROW_LENGTH , This - > currentDesc . Width ) ;
glRasterPos3i ( This - > lockedRect . left , This - > lockedRect . top , 1 ) ;
vcheckGLcall ( " glRasterPos2f " ) ;
/* Some drivers(radeon dri, others?) don't like exceptions during
* glDrawPixels . If the surface is a DIB section , it might be in GDIMode
* after ReleaseDC . Reading it will cause an exception , which x11drv will
* catch to put the dib section in InSync mode , which leads to a crash
* and a blocked x server on my radeon card .
*
* The following lines read the dib section so it is put in inSync mode
* before glDrawPixels is called and the crash is prevented . There won ' t
* be any interfering gdi accesses , because UnlockRect is called from
* ReleaseDC , and the app won ' t use the dc any more afterwards .
*/
if ( This - > Flags & SFLAG_DIBSECTION ) {
volatile BYTE read ;
read = This - > resource . allocatedMemory [ 0 ] ;
}
switch ( This - > resource . format ) {
/* No special care needed */
case WINED3DFMT_A4R4G4B4 :
case WINED3DFMT_R5G6B5 :
case WINED3DFMT_A1R5G5B5 :
case WINED3DFMT_R8G8B8 :
type = This - > glDescription . glType ;
fmt = This - > glDescription . glFormat ;
mem = This - > resource . allocatedMemory ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel ;
2006-07-17 17:27:19 +02:00
break ;
case WINED3DFMT_X4R4G4B4 :
{
int size ;
unsigned short * data ;
data = ( unsigned short * ) This - > resource . allocatedMemory ;
size = ( This - > lockedRect . bottom - This - > lockedRect . top ) * ( This - > lockedRect . right - This - > lockedRect . left ) ;
while ( size > 0 ) {
* data | = 0xF000 ;
data + + ;
size - - ;
}
type = This - > glDescription . glType ;
fmt = This - > glDescription . glFormat ;
mem = This - > resource . allocatedMemory ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel ;
2006-07-17 17:27:19 +02:00
}
break ;
case WINED3DFMT_X1R5G5B5 :
{
int size ;
unsigned short * data ;
data = ( unsigned short * ) This - > resource . allocatedMemory ;
size = ( This - > lockedRect . bottom - This - > lockedRect . top ) * ( This - > lockedRect . right - This - > lockedRect . left ) ;
while ( size > 0 ) {
* data | = 0x8000 ;
data + + ;
size - - ;
}
type = This - > glDescription . glType ;
fmt = This - > glDescription . glFormat ;
mem = This - > resource . allocatedMemory ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel ;
2006-07-17 17:27:19 +02:00
}
break ;
case WINED3DFMT_X8R8G8B8 :
{
/* make sure the X byte is set to alpha on, since it
2006-08-06 12:28:32 +02:00
could be any random value . This fixes the intro movie in Pirates ! */
2006-07-17 17:27:19 +02:00
int size ;
unsigned int * data ;
data = ( unsigned int * ) This - > resource . allocatedMemory ;
size = ( This - > lockedRect . bottom - This - > lockedRect . top ) * ( This - > lockedRect . right - This - > lockedRect . left ) ;
while ( size > 0 ) {
* data | = 0xFF000000 ;
data + + ;
size - - ;
}
}
2006-08-06 12:28:32 +02:00
/* Fall through */
2006-07-17 17:27:19 +02:00
case WINED3DFMT_A8R8G8B8 :
{
glPixelStorei ( GL_PACK_SWAP_BYTES , TRUE ) ;
vcheckGLcall ( " glPixelStorei " ) ;
storechanged = TRUE ;
type = This - > glDescription . glType ;
fmt = This - > glDescription . glFormat ;
mem = This - > resource . allocatedMemory ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel ;
2006-07-17 17:27:19 +02:00
}
break ;
case WINED3DFMT_A2R10G10B10 :
{
glPixelStorei ( GL_PACK_SWAP_BYTES , TRUE ) ;
vcheckGLcall ( " glPixelStorei " ) ;
storechanged = TRUE ;
type = This - > glDescription . glType ;
fmt = This - > glDescription . glFormat ;
mem = This - > resource . allocatedMemory ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel ;
2006-07-17 17:27:19 +02:00
}
break ;
case WINED3DFMT_P8 :
{
2006-07-28 13:57:49 +02:00
int height = This - > glRect . bottom - This - > glRect . top ;
2006-07-17 17:27:19 +02:00
type = GL_UNSIGNED_BYTE ;
fmt = GL_RGBA ;
mem = HeapAlloc ( GetProcessHeap ( ) , 0 , This - > resource . size * sizeof ( DWORD ) ) ;
if ( ! mem ) {
ERR ( " Out of memory \n " ) ;
return ;
}
2007-02-13 20:21:44 +01:00
memory_allocated = TRUE ;
2006-07-28 13:57:49 +02:00
d3dfmt_convert_surface ( This - > resource . allocatedMemory ,
mem ,
2006-10-05 21:35:10 +02:00
pitch ,
2006-11-06 19:45:10 +01:00
pitch ,
2006-10-05 21:35:10 +02:00
height ,
pitch * 4 ,
2006-07-28 13:57:49 +02:00
CONVERT_PALETTED ,
This ) ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel * 4 ;
pitch * = 4 ;
2006-07-17 17:27:19 +02:00
}
break ;
default :
FIXME ( " Unsupported Format %u in locking func \n " , This - > resource . format ) ;
/* Give it a try */
type = This - > glDescription . glType ;
fmt = This - > glDescription . glFormat ;
mem = This - > resource . allocatedMemory ;
2007-02-13 20:24:00 +01:00
bpp = This - > bytesPerPixel ;
2006-07-17 17:27:19 +02:00
}
glDrawPixels ( This - > lockedRect . right - This - > lockedRect . left ,
( This - > lockedRect . bottom - This - > lockedRect . top ) - 1 ,
fmt , type ,
2007-02-13 20:24:00 +01:00
mem + bpp * This - > lockedRect . left + pitch * This - > lockedRect . top ) ;
2006-07-17 17:27:19 +02:00
checkGLcall ( " glDrawPixels " ) ;
glPixelZoom ( 1.0 , 1.0 ) ;
vcheckGLcall ( " glPixelZoom " ) ;
glRasterPos3iv ( & prev_rasterpos [ 0 ] ) ;
vcheckGLcall ( " glRasterPos3iv " ) ;
/* Reset to previous pack row length */
glPixelStorei ( GL_UNPACK_ROW_LENGTH , skipBytes ) ;
vcheckGLcall ( " glPixelStorei GL_UNPACK_ROW_LENGTH " ) ;
if ( storechanged ) {
glPixelStorei ( GL_PACK_SWAP_BYTES , prev_store ) ;
vcheckGLcall ( " glPixelStorei GL_PACK_SWAP_BYTES " ) ;
}
2007-02-15 03:04:40 +01:00
/* Blitting environment requires that 2D texturing is enabled. It was turned off before,
* turn it on again
*/
glEnable ( GL_TEXTURE_2D ) ;
checkGLcall ( " glEnable(GL_TEXTURE_2D) " ) ;
2007-02-13 20:21:44 +01:00
if ( memory_allocated ) HeapFree ( GetProcessHeap ( ) , 0 , mem ) ;
2006-07-17 17:27:19 +02:00
return ;
}
2007-02-13 20:24:00 +01:00
static void flush_to_framebuffer_texture ( IWineD3DSurfaceImpl * This ) {
2006-07-28 00:43:38 +02:00
float glTexCoord [ 4 ] ;
2007-02-13 20:24:00 +01:00
glTexCoord [ 0 ] = ( float ) This - > lockedRect . left / ( float ) This - > pow2Width ; /* left */
glTexCoord [ 1 ] = ( float ) This - > lockedRect . right / ( float ) This - > pow2Width ; /* right */
glTexCoord [ 2 ] = ( float ) This - > lockedRect . top / ( float ) This - > pow2Height ; /* top */
glTexCoord [ 3 ] = ( float ) This - > lockedRect . bottom / ( float ) This - > pow2Height ; /* bottom */
2006-07-28 00:43:38 +02:00
2007-02-13 20:24:00 +01:00
IWineD3DSurface_PreLoad ( ( IWineD3DSurface * ) This ) ;
2006-07-28 00:43:38 +02:00
ENTER_GL ( ) ;
glBindTexture ( GL_TEXTURE_2D , This - > glDescription . textureName ) ;
checkGLcall ( " glEnable glBindTexture " ) ;
/* No filtering for blts */
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER , GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER , GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
/* Start drawing a quad */
glBegin ( GL_QUADS ) ;
glColor3d ( 1.0f , 1.0f , 1.0f ) ;
glTexCoord2f ( glTexCoord [ 0 ] , glTexCoord [ 2 ] ) ;
2007-02-13 20:24:00 +01:00
glVertex3f ( This - > lockedRect . left , This - > lockedRect . top , 0.0 ) ;
2006-07-28 00:43:38 +02:00
glTexCoord2f ( glTexCoord [ 0 ] , glTexCoord [ 3 ] ) ;
2007-02-13 20:24:00 +01:00
glVertex3f ( This - > lockedRect . left , This - > lockedRect . bottom , 0.0 ) ;
2006-07-28 00:43:38 +02:00
glTexCoord2f ( glTexCoord [ 1 ] , glTexCoord [ 3 ] ) ;
2007-02-13 20:24:00 +01:00
glVertex3d ( This - > lockedRect . right , This - > lockedRect . bottom , 0.0 ) ;
2006-07-28 00:43:38 +02:00
glTexCoord2f ( glTexCoord [ 1 ] , glTexCoord [ 2 ] ) ;
2007-02-13 20:24:00 +01:00
glVertex3f ( This - > lockedRect . right , This - > lockedRect . top , 0.0 ) ;
2006-07-28 00:43:38 +02:00
glEnd ( ) ;
checkGLcall ( " glEnd " ) ;
/* Unbind the texture */
glBindTexture ( GL_TEXTURE_2D , 0 ) ;
checkGLcall ( " glEnable glBindTexture " ) ;
LEAVE_GL ( ) ;
}
2006-07-17 17:27:19 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_UnlockRect ( IWineD3DSurface * iface ) {
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-07-07 22:35:05 +02:00
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
IWineD3DSwapChainImpl * swapchain = NULL ;
2005-01-09 18:37:02 +01:00
2006-05-06 16:15:37 +02:00
if ( ! ( This - > Flags & SFLAG_LOCKED ) ) {
2005-07-07 22:35:05 +02:00
WARN ( " trying to Unlock an unlocked surf@%p \n " , This ) ;
2006-04-07 12:51:12 +02:00
return WINED3DERR_INVALIDCALL ;
2005-01-09 18:37:02 +01:00
}
2007-02-13 20:21:44 +01:00
TRACE ( " (%p) : dirtyfied(%d) \n " , This , This - > Flags & SFLAG_DIRTY ? 1 : 0 ) ;
2005-07-07 22:35:05 +02:00
2006-05-06 16:15:37 +02:00
if ( ! ( This - > Flags & SFLAG_DIRTY ) ) {
2005-01-09 18:37:02 +01:00
TRACE ( " (%p) : Not Dirtified so nothing to do, return now \n " , This ) ;
goto unlock_end ;
}
2007-02-13 20:21:44 +01:00
IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DSwapChain , ( void * * ) & swapchain ) ;
if ( swapchain | | iface = = myDevice - > render_targets [ 0 ] ) {
if ( wined3d_settings . rendertargetlock_mode = = RTL_DISABLE ) {
static BOOL warned = FALSE ;
if ( ! warned ) {
ERR ( " The application tries to write to the render target, but render target locking is disabled \n " ) ;
warned = TRUE ;
2006-12-19 23:26:39 +01:00
}
2007-02-13 20:21:44 +01:00
if ( swapchain ) IWineD3DSwapChain_Release ( ( IWineD3DSwapChain * ) swapchain ) ;
goto unlock_end ;
2006-12-19 23:26:39 +01:00
}
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
/* Activate the correct context for the render target */
ENTER_GL ( ) ;
ActivateContext ( myDevice , iface , CTXUSAGE_BLIT ) ;
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
if ( ! swapchain ) {
/* Primary offscreen render target */
TRACE ( " Offscreen render target \n " ) ;
glDrawBuffer ( GL_BACK ) ;
checkGLcall ( " glDrawBuffer(GL_BACK) " ) ;
} else {
if ( iface = = swapchain - > frontBuffer ) {
TRACE ( " Onscreen front buffer \n " ) ;
2005-01-09 18:37:02 +01:00
glDrawBuffer ( GL_FRONT ) ;
2007-02-13 20:21:44 +01:00
checkGLcall ( " glDrawBuffer(GL_FRONT) " ) ;
} else if ( iface = = swapchain - > backBuffer [ 0 ] ) {
TRACE ( " Onscreen back buffer \n " ) ;
2006-06-04 16:42:57 +02:00
glDrawBuffer ( GL_BACK ) ;
2007-02-13 20:21:44 +01:00
checkGLcall ( " glDrawBuffer(GL_BACK) " ) ;
} else {
FIXME ( " Unlocking a higher back buffer \n " ) ;
2006-06-24 12:32:18 +02:00
glDrawBuffer ( GL_BACK ) ;
2007-02-13 20:21:44 +01:00
checkGLcall ( " glDrawBuffer(GL_BACK) " ) ;
2006-06-24 12:32:18 +02:00
}
2007-02-13 20:21:44 +01:00
IWineD3DSwapChain_Release ( ( IWineD3DSwapChain * ) swapchain ) ;
2005-01-09 18:37:02 +01:00
}
2007-02-13 20:21:44 +01:00
switch ( wined3d_settings . rendertargetlock_mode ) {
case RTL_AUTO :
case RTL_READDRAW :
case RTL_TEXDRAW :
flush_to_framebuffer_drawpixels ( This ) ;
break ;
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
case RTL_READTEX :
case RTL_TEXTEX :
2007-02-13 20:24:00 +01:00
flush_to_framebuffer_texture ( This ) ;
2007-02-13 20:21:44 +01:00
break ;
}
if ( ! swapchain | | swapchain - > backBuffer ) {
glDrawBuffer ( GL_BACK ) ;
checkGLcall ( " glDrawBuffer(GL_BACK) " ) ;
2005-01-09 18:37:02 +01:00
} else {
2007-02-13 20:21:44 +01:00
glDrawBuffer ( GL_FRONT ) ;
checkGLcall ( " glDrawBuffer(GL_FRONT) " ) ;
2005-01-09 18:37:02 +01:00
}
2007-02-13 20:21:44 +01:00
LEAVE_GL ( ) ;
2005-01-09 18:37:02 +01:00
2007-02-13 20:21:44 +01:00
/** restore clean dirty state */
IWineD3DSurface_CleanDirtyRect ( iface ) ;
} else if ( iface = = myDevice - > stencilBufferTarget ) {
FIXME ( " Depth Stencil buffer locking is not implemented \n " ) ;
2005-01-09 18:37:02 +01:00
} else {
2007-02-13 20:21:44 +01:00
/* The rest should be a normal texture */
IWineD3DBaseTextureImpl * impl ;
/* Check if the texture is bound, if yes dirtify the sampler to force a re-upload of the texture
* Can ' t load the texture here because PreLoad may destroy and recreate the gl texture , so sampler
* states need resetting
*/
if ( IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DBaseTexture , ( void * * ) & impl ) = = WINED3D_OK ) {
if ( impl - > baseTexture . bindCount ) {
IWineD3DDeviceImpl_MarkStateDirty ( myDevice , STATE_SAMPLER ( impl - > baseTexture . sampler ) ) ;
}
IWineD3DBaseTexture_Release ( ( IWineD3DBaseTexture * ) impl ) ;
}
2005-01-09 18:37:02 +01:00
}
unlock_end :
2006-05-06 16:15:37 +02:00
This - > Flags & = ~ SFLAG_LOCKED ;
2005-01-09 18:37:02 +01:00
memset ( & This - > lockedRect , 0 , sizeof ( RECT ) ) ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_GetDC ( IWineD3DSurface * iface , HDC * pHDC ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-05-06 18:08:33 +02:00
WINED3DLOCKED_RECT lock ;
UINT usage ;
BITMAPINFO * b_info ;
HDC ddc ;
DWORD * masks ;
HRESULT hr ;
2006-05-09 21:04:07 +02:00
RGBQUAD col [ 256 ] ;
2006-06-21 10:36:14 +02:00
const PixelFormatDesc * formatEntry = getFormatDescEntry ( This - > resource . format ) ;
2006-05-06 18:08:33 +02:00
TRACE ( " (%p)->(%p) \n " , This , pHDC ) ;
2006-07-25 16:38:02 +02:00
if ( This - > Flags & SFLAG_USERPTR ) {
ERR ( " Not supported on surfaces with an application-provided surfaces \n " ) ;
return DDERR_NODC ;
}
2006-05-06 18:08:33 +02:00
/* Give more detailed info for ddraw */
if ( This - > Flags & SFLAG_DCINUSE )
return DDERR_DCALREADYCREATED ;
/* Can't GetDC if the surface is locked */
if ( This - > Flags & SFLAG_LOCKED )
return WINED3DERR_INVALIDCALL ;
memset ( & lock , 0 , sizeof ( lock ) ) ; /* To be sure */
/* Create a DIB section if there isn't a hdc yet */
2006-05-09 21:04:07 +02:00
if ( ! This - > hDC ) {
2006-07-16 10:08:26 +02:00
int extraline = 0 ;
SYSTEM_INFO sysInfo ;
2006-05-09 21:04:07 +02:00
switch ( This - > bytesPerPixel ) {
2006-05-06 18:08:33 +02:00
case 2 :
case 4 :
/* Allocate extra space to store the RGB bit masks. */
b_info = HeapAlloc ( GetProcessHeap ( ) , HEAP_ZERO_MEMORY , sizeof ( BITMAPINFOHEADER ) + 3 * sizeof ( DWORD ) ) ;
break ;
case 3 :
b_info = HeapAlloc ( GetProcessHeap ( ) , HEAP_ZERO_MEMORY , sizeof ( BITMAPINFOHEADER ) ) ;
break ;
default :
/* Allocate extra space for a palette. */
b_info = HeapAlloc ( GetProcessHeap ( ) , HEAP_ZERO_MEMORY ,
sizeof ( BITMAPINFOHEADER )
+ sizeof ( RGBQUAD )
* ( 1 < < ( This - > bytesPerPixel * 8 ) ) ) ;
break ;
}
2006-08-16 22:35:58 +02:00
if ( ! b_info )
return E_OUTOFMEMORY ;
2006-07-16 10:08:26 +02:00
/* Some apps access the surface in via DWORDs, and do not take the necessary care at the end of the
* surface . So we need at least extra 4 bytes at the end of the surface . Check against the page size ,
* if the last page used for the surface has at least 4 spare bytes we ' re safe , otherwise
* add an extra line to the dib section
*/
GetSystemInfo ( & sysInfo ) ;
if ( ( ( This - > resource . size + 3 ) % sysInfo . dwPageSize ) < 4 ) {
extraline = 1 ;
TRACE ( " Adding an extra line to the dib section \n " ) ;
}
2006-05-06 18:08:33 +02:00
b_info - > bmiHeader . biSize = sizeof ( BITMAPINFOHEADER ) ;
2007-02-13 20:21:48 +01:00
b_info - > bmiHeader . biWidth = This - > currentDesc . Width ;
b_info - > bmiHeader . biHeight = - This - > currentDesc . Height - extraline ;
b_info - > bmiHeader . biSizeImage = ( This - > currentDesc . Height + extraline ) * IWineD3DSurface_GetPitch ( iface ) ;
2006-05-06 18:08:33 +02:00
b_info - > bmiHeader . biPlanes = 1 ;
b_info - > bmiHeader . biBitCount = This - > bytesPerPixel * 8 ;
b_info - > bmiHeader . biXPelsPerMeter = 0 ;
b_info - > bmiHeader . biYPelsPerMeter = 0 ;
b_info - > bmiHeader . biClrUsed = 0 ;
b_info - > bmiHeader . biClrImportant = 0 ;
/* Get the bit masks */
masks = ( DWORD * ) & ( b_info - > bmiColors ) ;
2006-05-09 21:04:07 +02:00
switch ( This - > resource . format ) {
2006-05-06 18:08:33 +02:00
case WINED3DFMT_R8G8B8 :
usage = DIB_RGB_COLORS ;
b_info - > bmiHeader . biCompression = BI_RGB ;
break ;
case WINED3DFMT_X1R5G5B5 :
case WINED3DFMT_A1R5G5B5 :
case WINED3DFMT_A4R4G4B4 :
case WINED3DFMT_X4R4G4B4 :
case WINED3DFMT_R3G3B2 :
case WINED3DFMT_A8R3G3B2 :
case WINED3DFMT_A2B10G10R10 :
case WINED3DFMT_A8B8G8R8 :
case WINED3DFMT_X8B8G8R8 :
case WINED3DFMT_A2R10G10B10 :
case WINED3DFMT_R5G6B5 :
case WINED3DFMT_A16B16G16R16 :
usage = 0 ;
b_info - > bmiHeader . biCompression = BI_BITFIELDS ;
2006-06-21 10:36:14 +02:00
masks [ 0 ] = formatEntry - > redMask ;
masks [ 1 ] = formatEntry - > greenMask ;
masks [ 2 ] = formatEntry - > blueMask ;
2006-05-06 18:08:33 +02:00
break ;
default :
/* Don't know palette */
b_info - > bmiHeader . biCompression = BI_RGB ;
usage = 0 ;
break ;
}
2007-01-02 17:16:36 +01:00
ddc = GetDC ( 0 ) ;
2006-05-09 21:04:07 +02:00
if ( ddc = = 0 ) {
2006-05-06 18:08:33 +02:00
HeapFree ( GetProcessHeap ( ) , 0 , b_info ) ;
return HRESULT_FROM_WIN32 ( GetLastError ( ) ) ;
}
2006-10-01 05:20:10 +02:00
TRACE ( " Creating a DIB section with size %dx%dx%d, size=%d \n " , b_info - > bmiHeader . biWidth , b_info - > bmiHeader . biHeight , b_info - > bmiHeader . biBitCount , b_info - > bmiHeader . biSizeImage ) ;
2006-05-06 18:08:33 +02:00
This - > dib . DIBsection = CreateDIBSection ( ddc , b_info , usage , & This - > dib . bitmap_data , 0 /* Handle */ , 0 /* Offset */ ) ;
2007-01-02 17:16:36 +01:00
ReleaseDC ( 0 , ddc ) ;
2006-05-06 18:08:33 +02:00
2006-05-09 21:04:07 +02:00
if ( ! This - > dib . DIBsection ) {
2006-05-06 18:08:33 +02:00
ERR ( " CreateDIBSection failed! \n " ) ;
2006-08-16 22:35:58 +02:00
HeapFree ( GetProcessHeap ( ) , 0 , b_info ) ;
2006-05-06 18:08:33 +02:00
return HRESULT_FROM_WIN32 ( GetLastError ( ) ) ;
}
TRACE ( " DIBSection at : %p \n " , This - > dib . bitmap_data ) ;
/* copy the existing surface to the dib section */
2006-05-09 21:04:07 +02:00
if ( This - > resource . allocatedMemory ) {
2006-07-13 18:35:45 +02:00
memcpy ( This - > dib . bitmap_data , This - > resource . allocatedMemory , b_info - > bmiHeader . biSizeImage ) ;
2006-05-06 18:08:33 +02:00
/* We won't need that any more */
HeapFree ( GetProcessHeap ( ) , 0 , This - > resource . allocatedMemory ) ;
2006-05-23 00:48:54 +02:00
} else {
/* This is to make LockRect read the gl Texture although memory is allocated */
This - > Flags | = SFLAG_GLDIRTY ;
2006-05-06 18:08:33 +02:00
}
2006-08-16 22:35:58 +02:00
HeapFree ( GetProcessHeap ( ) , 0 , b_info ) ;
2006-05-06 18:08:33 +02:00
/* Use the dib section from now on */
This - > resource . allocatedMemory = This - > dib . bitmap_data ;
/* Now allocate a HDC */
This - > hDC = CreateCompatibleDC ( 0 ) ;
This - > dib . holdbitmap = SelectObject ( This - > hDC , This - > dib . DIBsection ) ;
TRACE ( " using wined3d palette %p \n " , This - > palette ) ;
SelectPalette ( This - > hDC ,
This - > palette ? This - > palette - > hpal : 0 ,
FALSE ) ;
This - > Flags | = SFLAG_DIBSECTION ;
}
/* Lock the surface */
hr = IWineD3DSurface_LockRect ( iface ,
& lock ,
NULL ,
0 ) ;
2006-05-09 21:04:07 +02:00
if ( FAILED ( hr ) ) {
2006-10-01 05:20:10 +02:00
ERR ( " IWineD3DSurface_LockRect failed with hr = %08x \n " , hr ) ;
2006-05-06 18:08:33 +02:00
/* keep the dib section */
return hr ;
}
2006-05-09 21:04:07 +02:00
if ( This - > resource . format = = WINED3DFMT_P8 | |
This - > resource . format = = WINED3DFMT_A8P8 ) {
unsigned int n ;
if ( This - > palette ) {
PALETTEENTRY ent [ 256 ] ;
GetPaletteEntries ( This - > palette - > hpal , 0 , 256 , ent ) ;
for ( n = 0 ; n < 256 ; n + + ) {
col [ n ] . rgbRed = ent [ n ] . peRed ;
col [ n ] . rgbGreen = ent [ n ] . peGreen ;
col [ n ] . rgbBlue = ent [ n ] . peBlue ;
col [ n ] . rgbReserved = 0 ;
}
} else {
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
for ( n = 0 ; n < 256 ; n + + ) {
col [ n ] . rgbRed = device - > palettes [ device - > currentPalette ] [ n ] . peRed ;
col [ n ] . rgbGreen = device - > palettes [ device - > currentPalette ] [ n ] . peGreen ;
col [ n ] . rgbBlue = device - > palettes [ device - > currentPalette ] [ n ] . peBlue ;
col [ n ] . rgbReserved = 0 ;
}
}
SetDIBColorTable ( This - > hDC , 0 , 256 , col ) ;
}
2006-05-06 18:08:33 +02:00
* pHDC = This - > hDC ;
TRACE ( " returning %p \n " , * pHDC ) ;
This - > Flags | = SFLAG_DCINUSE ;
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_ReleaseDC ( IWineD3DSurface * iface , HDC hDC ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-05-06 18:08:33 +02:00
TRACE ( " (%p)->(%p) \n " , This , hDC ) ;
if ( ! ( This - > Flags & SFLAG_DCINUSE ) )
2007-02-15 13:32:25 +01:00
return WINED3DERR_INVALIDCALL ;
2006-05-06 18:08:33 +02:00
/* we locked first, so unlock now */
IWineD3DSurface_UnlockRect ( iface ) ;
This - > Flags & = ~ SFLAG_DCINUSE ;
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
/* ******************************************************
IWineD3DSurface Internal ( No mapping to directx api ) parts follow
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-05-22 23:49:09 +02:00
2006-08-03 12:39:04 +02:00
HRESULT d3dfmt_get_conv ( IWineD3DSurfaceImpl * This , BOOL need_alpha_ck , BOOL use_texturing , GLenum * format , GLenum * internal , GLenum * type , CONVERT_TYPES * convert , int * target_bpp ) {
2006-05-22 23:49:09 +02:00
BOOL colorkey_active = need_alpha_ck & & ( This - > CKeyFlags & DDSD_CKSRCBLT ) ;
2006-06-21 10:36:14 +02:00
const PixelFormatDesc * formatEntry = getFormatDescEntry ( This - > resource . format ) ;
2006-05-22 23:49:09 +02:00
/* Default values: From the surface */
2006-06-21 10:36:14 +02:00
* format = formatEntry - > glFormat ;
* internal = formatEntry - > glInternal ;
* type = formatEntry - > glType ;
2006-05-22 23:49:09 +02:00
* convert = NO_CONVERSION ;
* target_bpp = This - > bytesPerPixel ;
/* Ok, now look if we have to do any conversion */
switch ( This - > resource . format ) {
case WINED3DFMT_P8 :
/* ****************
Paletted Texture
* * * * * * * * * * * * * * * * */
2006-08-03 12:39:04 +02:00
/* Use conversion when the paletted texture extension is not available, or when it is available make sure it is used
* for texturing as it won ' t work for calls like glDraw - / glReadPixels and further also use conversion in case of color keying .
*/
if ( ! GL_SUPPORT ( EXT_PALETTED_TEXTURE ) | | colorkey_active | | ( ! use_texturing & & GL_SUPPORT ( EXT_PALETTED_TEXTURE ) ) ) {
2006-05-22 23:49:09 +02:00
* format = GL_RGBA ;
* internal = GL_RGBA ;
* type = GL_UNSIGNED_BYTE ;
* target_bpp = 4 ;
if ( colorkey_active ) {
* convert = CONVERT_PALETTED_CK ;
2006-07-25 16:41:41 +02:00
} else {
* convert = CONVERT_PALETTED ;
2006-05-22 23:49:09 +02:00
}
}
break ;
case WINED3DFMT_R3G3B2 :
/* **********************
GL_UNSIGNED_BYTE_3_3_2
* * * * * * * * * * * * * * * * * * * * * * */
if ( colorkey_active ) {
/* This texture format will never be used.. So do not care about color keying
up until the point in time it will be needed : - ) */
FIXME ( " ColorKeying not supported in the RGB 332 format ! \n " ) ;
}
break ;
case WINED3DFMT_R5G6B5 :
if ( colorkey_active ) {
* convert = CONVERT_CK_565 ;
* format = GL_RGBA ;
* internal = GL_RGBA ;
* type = GL_UNSIGNED_SHORT_5_5_5_1 ;
}
break ;
case WINED3DFMT_R8G8B8 :
if ( colorkey_active ) {
* convert = CONVERT_CK_RGB24 ;
* format = GL_RGBA ;
* internal = GL_RGBA ;
* type = GL_UNSIGNED_INT_8_8_8_8 ;
* target_bpp = 4 ;
}
break ;
case WINED3DFMT_X8R8G8B8 :
if ( colorkey_active ) {
* convert = CONVERT_RGB32_888 ;
* format = GL_RGBA ;
* internal = GL_RGBA ;
* type = GL_UNSIGNED_INT_8_8_8_8 ;
}
break ;
2007-02-15 03:01:44 +01:00
case WINED3DFMT_V8U8 :
* convert = CONVERT_V8U8 ;
* format = GL_BGR ;
* internal = GL_RGB8 ;
* type = GL_BYTE ;
* target_bpp = 3 ;
break ;
2006-05-22 23:49:09 +02:00
default :
break ;
}
return WINED3D_OK ;
}
2006-11-06 19:45:10 +01:00
HRESULT d3dfmt_convert_surface ( BYTE * src , BYTE * dst , UINT pitch , UINT width , UINT height , UINT outpitch , CONVERT_TYPES convert , IWineD3DSurfaceImpl * surf ) {
BYTE * source , * dest ;
2006-10-05 21:35:10 +02:00
TRACE ( " (%p)->(%p),(%d,%d,%d,%d,%p) \n " , src , dst , pitch , height , outpitch , convert , surf ) ;
2006-05-22 23:49:09 +02:00
switch ( convert ) {
case NO_CONVERSION :
{
2006-10-05 21:35:10 +02:00
memcpy ( dst , src , pitch * height ) ;
2006-05-22 23:49:09 +02:00
break ;
}
case CONVERT_PALETTED :
case CONVERT_PALETTED_CK :
{
IWineD3DPaletteImpl * pal = surf - > palette ;
BYTE table [ 256 ] [ 4 ] ;
unsigned int i ;
2006-10-05 21:35:10 +02:00
unsigned int x , y ;
2006-05-22 23:49:09 +02:00
if ( pal = = NULL ) {
/* TODO: If we are a sublevel, try to get the palette from level 0 */
}
if ( pal = = NULL ) {
/* Still no palette? Use the device's palette */
/* Get the surface's palette */
for ( i = 0 ; i < 256 ; i + + ) {
IWineD3DDeviceImpl * device = surf - > resource . wineD3DDevice ;
table [ i ] [ 0 ] = device - > palettes [ device - > currentPalette ] [ i ] . peRed ;
table [ i ] [ 1 ] = device - > palettes [ device - > currentPalette ] [ i ] . peGreen ;
table [ i ] [ 2 ] = device - > palettes [ device - > currentPalette ] [ i ] . peBlue ;
if ( ( convert = = CONVERT_PALETTED_CK ) & &
( i > = surf - > SrcBltCKey . dwColorSpaceLowValue ) & &
( i < = surf - > SrcBltCKey . dwColorSpaceHighValue ) ) {
/* We should maybe here put a more 'neutral' color than the standard bright purple
one often used by application to prevent the nice purple borders when bi - linear
filtering is on */
table [ i ] [ 3 ] = 0x00 ;
} else {
table [ i ] [ 3 ] = 0xFF ;
}
}
} else {
TRACE ( " Using surface palette %p \n " , pal ) ;
/* Get the surface's palette */
for ( i = 0 ; i < 256 ; i + + ) {
table [ i ] [ 0 ] = pal - > palents [ i ] . peRed ;
table [ i ] [ 1 ] = pal - > palents [ i ] . peGreen ;
table [ i ] [ 2 ] = pal - > palents [ i ] . peBlue ;
if ( ( convert = = CONVERT_PALETTED_CK ) & &
( i > = surf - > SrcBltCKey . dwColorSpaceLowValue ) & &
( i < = surf - > SrcBltCKey . dwColorSpaceHighValue ) ) {
/* We should maybe here put a more 'neutral' color than the standard bright purple
one often used by application to prevent the nice purple borders when bi - linear
filtering is on */
table [ i ] [ 3 ] = 0x00 ;
} else {
table [ i ] [ 3 ] = 0xFF ;
}
}
}
2006-10-05 21:35:10 +02:00
for ( y = 0 ; y < height ; y + + )
{
2006-11-06 19:45:10 +01:00
source = src + pitch * y ;
2006-10-05 21:35:10 +02:00
dest = dst + outpitch * y ;
2006-11-06 19:45:10 +01:00
/* This is an 1 bpp format, using the width here is fine */
for ( x = 0 ; x < width ; x + + ) {
BYTE color = * source + + ;
2006-10-05 21:35:10 +02:00
* dest + + = table [ color ] [ 0 ] ;
* dest + + = table [ color ] [ 1 ] ;
* dest + + = table [ color ] [ 2 ] ;
* dest + + = table [ color ] [ 3 ] ;
}
2006-05-22 23:49:09 +02:00
}
}
break ;
case CONVERT_CK_565 :
{
/* Converting the 565 format in 5551 packed to emulate color-keying.
Note : in all these conversion , it would be best to average the averaging
pixels to get the color of the pixel that will be color - keyed to
prevent ' color bleeding ' . This will be done later on if ever it is
too visible .
2006-10-05 21:31:50 +02:00
Note2 : Nvidia documents say that their driver does not support alpha + color keying
on the same surface and disables color keying in such a case
2006-05-22 23:49:09 +02:00
*/
2006-10-05 21:35:10 +02:00
unsigned int x , y ;
WORD * Source ;
WORD * Dest ;
2006-05-22 23:49:09 +02:00
TRACE ( " Color keyed 565 \n " ) ;
2006-10-05 21:35:10 +02:00
for ( y = 0 ; y < height ; y + + ) {
Source = ( WORD * ) ( src + y * pitch ) ;
Dest = ( WORD * ) ( dst + y * outpitch ) ;
2006-11-08 02:22:29 +01:00
for ( x = 0 ; x < width ; x + + ) {
2006-10-05 21:35:10 +02:00
WORD color = * Source + + ;
* Dest = ( ( color & 0xFFC0 ) | ( ( color & 0x1F ) < < 1 ) ) ;
if ( ( color < surf - > SrcBltCKey . dwColorSpaceLowValue ) | |
( color > surf - > SrcBltCKey . dwColorSpaceHighValue ) ) {
* Dest | = 0x0001 ;
}
Dest + + ;
2006-05-22 23:49:09 +02:00
}
}
}
break ;
2007-02-15 03:01:44 +01:00
case CONVERT_V8U8 :
{
unsigned int x , y ;
short * Source ;
char * Dest ;
for ( y = 0 ; y < height ; y + + ) {
Source = ( short * ) ( src + y * pitch ) ;
Dest = ( char * ) ( dst + y * outpitch ) ;
for ( x = 0 ; x < width ; x + + ) {
long color = ( * Source + + ) ;
Dest [ 0 ] = color > > 8 ;
Dest [ 1 ] = color ;
Dest [ 2 ] = 0xff ;
Dest + = 3 ;
}
}
break ;
}
2006-05-22 23:49:09 +02:00
default :
ERR ( " Unsupported conversation type %d \n " , convert ) ;
}
return WINED3D_OK ;
}
2006-08-03 12:39:04 +02:00
/* This function is used in case of 8bit paletted textures to upload the palette.
For now it only supports GL_EXT_paletted_texture extension but support for other
2006-11-07 00:37:42 +01:00
extensions like ARB_fragment_program and ATI_fragment_shaders will be added as well .
2006-08-03 12:39:04 +02:00
*/
void d3dfmt_p8_upload_palette ( IWineD3DSurface * iface , CONVERT_TYPES convert ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DPaletteImpl * pal = This - > palette ;
BYTE table [ 256 ] [ 4 ] ;
int i ;
if ( pal = = NULL ) {
/* Still no palette? Use the device's palette */
/* Get the surface's palette */
for ( i = 0 ; i < 256 ; i + + ) {
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
table [ i ] [ 0 ] = device - > palettes [ device - > currentPalette ] [ i ] . peRed ;
table [ i ] [ 1 ] = device - > palettes [ device - > currentPalette ] [ i ] . peGreen ;
table [ i ] [ 2 ] = device - > palettes [ device - > currentPalette ] [ i ] . peBlue ;
if ( ( convert = = CONVERT_PALETTED_CK ) & &
( i > = This - > SrcBltCKey . dwColorSpaceLowValue ) & &
( i < = This - > SrcBltCKey . dwColorSpaceHighValue ) ) {
/* We should maybe here put a more 'neutral' color than the standard bright purple
one often used by application to prevent the nice purple borders when bi - linear
filtering is on */
table [ i ] [ 3 ] = 0x00 ;
} else {
table [ i ] [ 3 ] = 0xFF ;
}
}
} else {
TRACE ( " Using surface palette %p \n " , pal ) ;
/* Get the surface's palette */
for ( i = 0 ; i < 256 ; i + + ) {
table [ i ] [ 0 ] = pal - > palents [ i ] . peRed ;
table [ i ] [ 1 ] = pal - > palents [ i ] . peGreen ;
table [ i ] [ 2 ] = pal - > palents [ i ] . peBlue ;
if ( ( convert = = CONVERT_PALETTED_CK ) & &
( i > = This - > SrcBltCKey . dwColorSpaceLowValue ) & &
( i < = This - > SrcBltCKey . dwColorSpaceHighValue ) ) {
/* We should maybe here put a more 'neutral' color than the standard bright purple
one often used by application to prevent the nice purple borders when bi - linear
filtering is on */
table [ i ] [ 3 ] = 0x00 ;
} else {
table [ i ] [ 3 ] = 0xFF ;
}
}
}
GL_EXTCALL ( glColorTableEXT ( GL_TEXTURE_2D , GL_RGBA , 256 , GL_RGBA , GL_UNSIGNED_BYTE , table ) ) ;
}
2006-06-10 13:15:32 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_LoadTexture ( IWineD3DSurface * iface ) {
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-09-11 18:51:18 +02:00
GLenum format , internal , type ;
CONVERT_TYPES convert ;
int bpp ;
2006-10-05 21:35:10 +02:00
int width , pitch , outpitch ;
2006-09-11 18:51:18 +02:00
BYTE * mem ;
2005-01-09 18:37:02 +01:00
2006-05-06 16:15:37 +02:00
if ( This - > Flags & SFLAG_INTEXTURE ) {
2005-07-11 16:25:54 +02:00
TRACE ( " Surface already in texture \n " ) ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-07-11 16:25:54 +02:00
}
2006-07-25 16:41:41 +02:00
if ( This - > Flags & SFLAG_DIRTY ) {
TRACE ( " Reloading because surface is dirty \n " ) ;
} else if ( /* Reload: gl texture has ck, now no ckey is set OR */
2006-07-26 23:19:04 +02:00
( ( This - > Flags & SFLAG_GLCKEY ) & & ( ! ( This - > CKeyFlags & DDSD_CKSRCBLT ) ) ) | |
2006-07-25 16:41:41 +02:00
/* Reload: vice versa OR */
2006-07-26 23:19:04 +02:00
( ( ! ( This - > Flags & SFLAG_GLCKEY ) ) & & ( This - > CKeyFlags & DDSD_CKSRCBLT ) ) | |
2006-07-25 16:41:41 +02:00
/* Also reload: Color key is active AND the color key has changed */
2006-07-26 23:19:04 +02:00
( ( This - > CKeyFlags & DDSD_CKSRCBLT ) & & (
( This - > glCKey . dwColorSpaceLowValue ! = This - > SrcBltCKey . dwColorSpaceLowValue ) | |
( This - > glCKey . dwColorSpaceHighValue ! = This - > SrcBltCKey . dwColorSpaceHighValue ) ) ) ) {
2006-07-25 16:41:41 +02:00
TRACE ( " Reloading because of color keying \n " ) ;
} else {
2005-07-11 16:25:54 +02:00
TRACE ( " surface isn't dirty \n " ) ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-07-11 16:25:54 +02:00
}
2006-05-06 16:15:37 +02:00
This - > Flags & = ~ SFLAG_DIRTY ;
2005-07-11 16:25:54 +02:00
2006-02-06 20:57:42 +01:00
/* Resources are placed in system RAM and do not need to be recreated when a device is lost.
* These resources are not bound by device size or format restrictions . Because of this ,
* these resources cannot be accessed by the Direct3D device nor set as textures or render targets .
* However , these resources can always be created , locked , and copied .
*/
2006-07-23 00:03:33 +02:00
if ( This - > resource . pool = = WINED3DPOOL_SCRATCH & & ! ( This - > Flags & SFLAG_FORCELOAD ) )
2005-07-11 16:25:54 +02:00
{
2006-05-18 22:59:33 +02:00
FIXME ( " (%p) Operation not supported for scratch textures \n " , This ) ;
2006-04-07 12:51:12 +02:00
return WINED3DERR_INVALIDCALL ;
2005-07-11 16:25:54 +02:00
}
2005-01-09 18:37:02 +01:00
2006-05-06 16:15:37 +02:00
if ( This - > Flags & SFLAG_INPBUFFER ) {
2005-07-11 22:38:27 +02:00
if ( This - > glDescription . level ! = 0 )
2005-01-09 18:37:02 +01:00
FIXME ( " Surface in texture is only supported for level 0 \n " ) ;
2005-03-29 21:01:00 +02:00
else if ( This - > resource . format = = WINED3DFMT_P8 | | This - > resource . format = = WINED3DFMT_A8P8 | |
2005-08-29 14:20:02 +02:00
This - > resource . format = = WINED3DFMT_DXT1 | | This - > resource . format = = WINED3DFMT_DXT2 | |
This - > resource . format = = WINED3DFMT_DXT3 | | This - > resource . format = = WINED3DFMT_DXT4 | |
2005-03-29 21:01:00 +02:00
This - > resource . format = = WINED3DFMT_DXT5 )
FIXME ( " Format %d not supported \n " , This - > resource . format ) ;
2005-01-09 18:37:02 +01:00
else {
2005-07-28 20:33:26 +02:00
GLint prevRead ;
2006-09-11 18:51:18 +02:00
ENTER_GL ( ) ;
2005-07-11 22:38:27 +02:00
glGetIntegerv ( GL_READ_BUFFER , & prevRead ) ;
vcheckGLcall ( " glGetIntegerv " ) ;
glReadBuffer ( GL_BACK ) ;
vcheckGLcall ( " glReadBuffer " ) ;
glCopyTexImage2D ( This - > glDescription . target ,
This - > glDescription . level ,
This - > glDescription . glFormatInternal ,
2005-01-09 18:37:02 +01:00
0 ,
0 ,
This - > currentDesc . Width ,
This - > currentDesc . Height ,
0 ) ;
2005-07-11 22:38:27 +02:00
checkGLcall ( " glCopyTexImage2D " ) ;
glReadBuffer ( prevRead ) ;
vcheckGLcall ( " glReadBuffer " ) ;
2006-09-11 18:51:18 +02:00
LEAVE_GL ( ) ;
2005-07-11 22:38:27 +02:00
TRACE ( " Updating target %d \n " , This - > glDescription . target ) ;
2006-05-06 16:15:37 +02:00
This - > Flags | = SFLAG_INTEXTURE ;
2005-01-09 18:37:02 +01:00
}
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
2006-09-11 18:51:18 +02:00
if ( This - > CKeyFlags & DDSD_CKSRCBLT ) {
This - > Flags | = SFLAG_GLCKEY ;
This - > glCKey = This - > SrcBltCKey ;
}
else This - > Flags & = ~ SFLAG_GLCKEY ;
2005-01-09 18:37:02 +01:00
2006-09-11 18:51:18 +02:00
d3dfmt_get_conv ( This , TRUE /* We need color keying */ , TRUE /* We will use textures */ , & format , & internal , & type , & convert , & bpp ) ;
2005-01-09 18:37:02 +01:00
2006-10-05 20:04:07 +02:00
/* The width is in 'length' not in bytes */
2007-02-13 20:21:48 +01:00
width = This - > currentDesc . Width ;
2006-10-05 21:35:10 +02:00
pitch = IWineD3DSurface_GetPitch ( iface ) ;
2006-09-11 18:51:18 +02:00
if ( ( convert ! = NO_CONVERSION ) & & This - > resource . allocatedMemory ) {
2007-02-15 13:52:21 +01:00
int height = This - > currentDesc . Height ;
2005-07-13 13:34:55 +02:00
2006-10-05 21:35:10 +02:00
/* Stick to the alignment for the converted surface too, makes it easier to load the surface */
outpitch = width * bpp ;
2006-11-06 19:43:48 +01:00
outpitch = ( outpitch + SURFACE_ALIGNMENT - 1 ) & ~ ( SURFACE_ALIGNMENT - 1 ) ;
2006-10-05 21:35:10 +02:00
mem = HeapAlloc ( GetProcessHeap ( ) , 0 , outpitch * height ) ;
2006-09-11 18:51:18 +02:00
if ( ! mem ) {
2006-10-05 21:35:10 +02:00
ERR ( " Out of memory %d, %d! \n " , outpitch , height ) ;
2006-09-11 18:51:18 +02:00
return WINED3DERR_OUTOFVIDEOMEMORY ;
2005-01-09 18:37:02 +01:00
}
2006-11-06 19:45:10 +01:00
d3dfmt_convert_surface ( This - > resource . allocatedMemory , mem , pitch , width , height , outpitch , convert , This ) ;
2006-09-11 18:51:18 +02:00
This - > Flags | = SFLAG_CONVERTED ;
} else if ( This - > resource . format = = WINED3DFMT_P8 & & GL_SUPPORT ( EXT_PALETTED_TEXTURE ) ) {
d3dfmt_p8_upload_palette ( iface , convert ) ;
This - > Flags & = ~ SFLAG_CONVERTED ;
mem = This - > resource . allocatedMemory ;
2005-01-09 18:37:02 +01:00
} else {
2006-09-11 18:51:18 +02:00
This - > Flags & = ~ SFLAG_CONVERTED ;
mem = This - > resource . allocatedMemory ;
}
2006-05-22 23:49:09 +02:00
2006-09-11 18:51:18 +02:00
/* Make sure the correct pitch is used */
2006-10-05 20:04:07 +02:00
glPixelStorei ( GL_UNPACK_ROW_LENGTH , width ) ;
2006-08-09 22:45:59 +02:00
2007-02-13 20:21:48 +01:00
if ( ( This - > Flags & SFLAG_NONPOW2 ) & & ! ( This - > Flags & SFLAG_OVERSIZE ) ) {
2006-09-11 18:51:18 +02:00
TRACE ( " non power of two support \n " ) ;
surface_allocate_surface ( This , internal , This - > pow2Width , This - > pow2Height , format , type ) ;
if ( mem ) {
2007-02-13 20:21:48 +01:00
surface_upload_data ( This , This - > currentDesc . Width , This - > currentDesc . Height , format , type , mem ) ;
2006-05-22 23:49:09 +02:00
}
2006-09-11 18:51:18 +02:00
} else {
surface_allocate_surface ( This , internal , This - > glRect . right - This - > glRect . left , This - > glRect . bottom - This - > glRect . top , format , type ) ;
if ( mem ) {
surface_upload_data ( This , This - > glRect . right - This - > glRect . left , This - > glRect . bottom - This - > glRect . top , format , type , mem ) ;
}
}
2005-01-09 18:37:02 +01:00
2006-09-11 18:51:18 +02:00
/* Restore the default pitch */
glPixelStorei ( GL_UNPACK_ROW_LENGTH , 0 ) ;
2006-05-22 23:49:09 +02:00
2006-09-11 18:51:18 +02:00
if ( mem ! = This - > resource . allocatedMemory )
HeapFree ( GetProcessHeap ( ) , 0 , mem ) ;
2005-01-09 18:37:02 +01:00
#if 0
2006-09-11 18:51:18 +02:00
{
static unsigned int gen = 0 ;
char buffer [ 4096 ] ;
+ + gen ;
if ( ( gen % 10 ) = = 0 ) {
snprintf ( buffer , sizeof ( buffer ) , " /tmp/surface%p_type%u_level%u_%u.ppm " , This , This - > glDescription . target , This - > glDescription . level , gen ) ;
IWineD3DSurfaceImpl_SaveSnapshot ( iface , buffer ) ;
2005-01-09 18:37:02 +01:00
}
2006-09-11 18:51:18 +02:00
/*
* debugging crash code
if ( gen = = 250 ) {
void * * test = NULL ;
* test = 0 ;
}
*/
}
2005-01-09 18:37:02 +01:00
# endif
2005-07-13 13:34:55 +02:00
2006-09-11 18:51:18 +02:00
if ( ! ( This - > Flags & SFLAG_DONOTFREE ) ) {
HeapFree ( GetProcessHeap ( ) , 0 , This - > resource . allocatedMemory ) ;
This - > resource . allocatedMemory = NULL ;
2005-01-09 18:37:02 +01:00
}
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
# include <errno.h>
# include <stdio.h>
HRESULT WINAPI IWineD3DSurfaceImpl_SaveSnapshot ( IWineD3DSurface * iface , const char * filename ) {
FILE * f = NULL ;
2005-09-27 12:49:59 +02:00
UINT i , y ;
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-09-27 12:49:59 +02:00
char * allocatedMemory ;
char * textureRow ;
IWineD3DSwapChain * swapChain = NULL ;
int width , height ;
2006-08-27 00:37:56 +02:00
GLuint tmpTexture = 0 ;
2005-09-27 12:49:59 +02:00
DWORD color ;
/*FIXME:
Textures my not be stored in - > allocatedgMemory and a GlTexture
2005-11-10 12:39:07 +01:00
so we should lock the surface before saving a snapshot , or at least check that
2005-09-27 12:49:59 +02:00
*/
/* TODO: Compressed texture images can be obtained from the GL in uncompressed form
by calling GetTexImage and in compressed form by calling
GetCompressedTexImageARB . Queried compressed images can be saved and
later reused by calling CompressedTexImage [ 123 ] DARB . Pre - compressed
texture images do not need to be processed by the GL and should
significantly improve texture loading performance relative to uncompressed
images . */
/* Setup the width and height to be the internal texture width and height. */
width = This - > pow2Width ;
height = This - > pow2Height ;
/* check to see if were a 'virtual' texture e.g. were not a pbuffer of texture were a back buffer*/
IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DSwapChain , ( void * * ) & swapChain ) ;
2006-05-06 16:15:37 +02:00
if ( swapChain | | ( This - > Flags & SFLAG_INPBUFFER ) ) { /* if were not a real texture then read the back buffer into a real texture*/
2005-11-10 12:39:07 +01:00
/* we don't want to interfere with the back buffer so read the data into a temporary texture and then save the data out of the temporary texture */
2005-09-27 12:49:59 +02:00
GLint prevRead ;
ENTER_GL ( ) ;
FIXME ( " (%p) This surface needs to be locked before a snapshot can be taken \n " , This ) ;
glEnable ( GL_TEXTURE_2D ) ;
glGenTextures ( 1 , & tmpTexture ) ;
glBindTexture ( GL_TEXTURE_2D , tmpTexture ) ;
glTexImage2D ( GL_TEXTURE_2D ,
0 ,
GL_RGBA ,
width ,
height ,
0 /*border*/ ,
GL_RGBA ,
GL_UNSIGNED_INT_8_8_8_8_REV ,
NULL ) ;
glGetIntegerv ( GL_READ_BUFFER , & prevRead ) ;
vcheckGLcall ( " glGetIntegerv " ) ;
glReadBuffer ( GL_BACK ) ;
vcheckGLcall ( " glReadBuffer " ) ;
glCopyTexImage2D ( GL_TEXTURE_2D ,
0 ,
GL_RGBA ,
0 ,
0 ,
width ,
height ,
0 ) ;
checkGLcall ( " glCopyTexImage2D " ) ;
glReadBuffer ( prevRead ) ;
LEAVE_GL ( ) ;
} else { /* bind the real texture */
IWineD3DSurface_PreLoad ( iface ) ;
}
allocatedMemory = HeapAlloc ( GetProcessHeap ( ) , 0 , width * height * 4 ) ;
ENTER_GL ( ) ;
FIXME ( " Saving texture level %d width %d height %d \n " , This - > glDescription . level , width , height ) ;
glGetTexImage ( GL_TEXTURE_2D ,
This - > glDescription . level ,
GL_RGBA ,
GL_UNSIGNED_INT_8_8_8_8_REV ,
allocatedMemory ) ;
checkGLcall ( " glTexImage2D " ) ;
if ( tmpTexture ) {
glBindTexture ( GL_TEXTURE_2D , 0 ) ;
glDeleteTextures ( 1 , & tmpTexture ) ;
}
LEAVE_GL ( ) ;
2005-01-09 18:37:02 +01:00
f = fopen ( filename , " w+ " ) ;
if ( NULL = = f ) {
ERR ( " opening of %s failed with: %s \n " , filename , strerror ( errno ) ) ;
2006-04-07 12:51:12 +02:00
return WINED3DERR_INVALIDCALL ;
2005-01-09 18:37:02 +01:00
}
2005-09-27 12:49:59 +02:00
/* Save the dat out to a TGA file because 1: it's an easy raw format, 2: it supports an alpha chanel*/
TRACE ( " (%p) opened %s with format %s \n " , This , filename , debug_d3dformat ( This - > resource . format ) ) ;
/* TGA header */
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 2 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
fputc ( 0 , f ) ;
/* short width*/
fwrite ( & width , 2 , 1 , f ) ;
/* short height */
fwrite ( & height , 2 , 1 , f ) ;
/* format rgba */
fputc ( 0x20 , f ) ;
fputc ( 0x28 , f ) ;
/* raw data */
/* if the data is upside down if we've fetched it from a back buffer, so it needs flipping again to make it the correct way up*/
if ( swapChain )
textureRow = allocatedMemory + ( width * ( height - 1 ) * 4 ) ;
else
textureRow = allocatedMemory ;
for ( y = 0 ; y < height ; y + + ) {
for ( i = 0 ; i < width ; i + + ) {
color = * ( ( DWORD * ) textureRow ) ;
fputc ( ( color > > 16 ) & 0xFF , f ) ; /* B */
fputc ( ( color > > 8 ) & 0xFF , f ) ; /* G */
fputc ( ( color > > 0 ) & 0xFF , f ) ; /* R */
fputc ( ( color > > 24 ) & 0xFF , f ) ; /* A */
textureRow + = 4 ;
2005-01-09 18:37:02 +01:00
}
2005-09-27 12:49:59 +02:00
/* take two rows of the pointer to the texture memory */
if ( swapChain )
( textureRow - = width < < 3 ) ;
2005-01-09 18:37:02 +01:00
}
2005-09-27 12:49:59 +02:00
TRACE ( " Closing file \n " ) ;
2005-01-09 18:37:02 +01:00
fclose ( f ) ;
2005-09-27 12:49:59 +02:00
if ( swapChain ) {
IWineD3DSwapChain_Release ( swapChain ) ;
}
HeapFree ( GetProcessHeap ( ) , 0 , allocatedMemory ) ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_CleanDirtyRect ( IWineD3DSurface * iface ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-05-06 16:15:37 +02:00
This - > Flags & = ~ SFLAG_DIRTY ;
2005-01-09 18:37:02 +01:00
This - > dirtyRect . left = This - > currentDesc . Width ;
This - > dirtyRect . top = This - > currentDesc . Height ;
This - > dirtyRect . right = 0 ;
This - > dirtyRect . bottom = 0 ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p) : Dirty?%d, Rect:(%d,%d,%d,%d) \n " , This , This - > Flags & SFLAG_DIRTY ? 1 : 0 , This - > dirtyRect . left ,
2005-01-17 14:44:57 +01:00
This - > dirtyRect . top , This - > dirtyRect . right , This - > dirtyRect . bottom ) ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
/**
* Slightly inefficient way to handle multiple dirty rects but it works : )
*/
extern HRESULT WINAPI IWineD3DSurfaceImpl_AddDirtyRect ( IWineD3DSurface * iface , CONST RECT * pDirtyRect ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-07-11 16:25:54 +02:00
IWineD3DBaseTexture * baseTexture = NULL ;
2006-05-06 16:15:37 +02:00
This - > Flags | = SFLAG_DIRTY ;
2005-01-09 18:37:02 +01:00
if ( NULL ! = pDirtyRect ) {
This - > dirtyRect . left = min ( This - > dirtyRect . left , pDirtyRect - > left ) ;
This - > dirtyRect . top = min ( This - > dirtyRect . top , pDirtyRect - > top ) ;
This - > dirtyRect . right = max ( This - > dirtyRect . right , pDirtyRect - > right ) ;
This - > dirtyRect . bottom = max ( This - > dirtyRect . bottom , pDirtyRect - > bottom ) ;
} else {
This - > dirtyRect . left = 0 ;
This - > dirtyRect . top = 0 ;
This - > dirtyRect . right = This - > currentDesc . Width ;
This - > dirtyRect . bottom = This - > currentDesc . Height ;
}
2006-10-01 05:20:10 +02:00
TRACE ( " (%p) : Dirty?%d, Rect:(%d,%d,%d,%d) \n " , This , This - > Flags & SFLAG_DIRTY , This - > dirtyRect . left ,
2005-01-17 14:44:57 +01:00
This - > dirtyRect . top , This - > dirtyRect . right , This - > dirtyRect . bottom ) ;
2005-07-11 16:25:54 +02:00
/* if the container is a basetexture then mark it dirty. */
2006-04-07 12:51:12 +02:00
if ( IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DBaseTexture , ( void * * ) & baseTexture ) = = WINED3D_OK ) {
2005-07-11 16:25:54 +02:00
TRACE ( " Passing to conatiner \n " ) ;
IWineD3DBaseTexture_SetDirty ( baseTexture , TRUE ) ;
IWineD3DBaseTexture_Release ( baseTexture ) ;
}
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-01-09 18:37:02 +01:00
}
2006-02-06 11:30:48 +01:00
HRESULT WINAPI IWineD3DSurfaceImpl_SetContainer ( IWineD3DSurface * iface , IWineD3DBase * container ) {
2005-03-14 11:12:52 +01:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-02-06 11:30:48 +01:00
TRACE ( " This %p, container %p \n " , This , container ) ;
2006-02-13 13:23:58 +01:00
/* We can't keep a reference to the container, since the container already keeps a reference to us. */
2006-02-06 11:30:48 +01:00
2005-03-14 11:12:52 +01:00
TRACE ( " Setting container to %p from %p \n " , container , This - > container ) ;
This - > container = container ;
2006-02-06 11:30:48 +01:00
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-03-14 11:12:52 +01:00
}
2005-11-02 21:01:54 +01:00
HRESULT WINAPI IWineD3DSurfaceImpl_SetFormat ( IWineD3DSurface * iface , WINED3DFORMAT format ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-06-21 10:36:14 +02:00
const PixelFormatDesc * formatEntry = getFormatDescEntry ( format ) ;
2005-11-02 21:01:54 +01:00
if ( This - > resource . format ! = WINED3DFMT_UNKNOWN ) {
FIXME ( " (%p) : The foramt of the surface must be WINED3DFORMAT_UNKNOWN \n " , This ) ;
2006-04-07 12:51:12 +02:00
return WINED3DERR_INVALIDCALL ;
2005-11-02 21:01:54 +01:00
}
TRACE ( " (%p) : Setting texture foramt to (%d,%s) \n " , This , format , debug_d3dformat ( format ) ) ;
if ( format = = WINED3DFMT_UNKNOWN ) {
This - > resource . size = 0 ;
} else if ( format = = WINED3DFMT_DXT1 ) {
/* DXT1 is half byte per pixel */
2006-06-21 10:36:14 +02:00
This - > resource . size = ( ( max ( This - > pow2Width , 4 ) * formatEntry - > bpp ) * max ( This - > pow2Height , 4 ) ) > > 1 ;
2005-11-02 21:01:54 +01:00
} else if ( format = = WINED3DFMT_DXT2 | | format = = WINED3DFMT_DXT3 | |
format = = WINED3DFMT_DXT4 | | format = = WINED3DFMT_DXT5 ) {
2006-06-21 10:36:14 +02:00
This - > resource . size = ( ( max ( This - > pow2Width , 4 ) * formatEntry - > bpp ) * max ( This - > pow2Height , 4 ) ) ;
2005-11-02 21:01:54 +01:00
} else {
2006-11-06 19:43:48 +01:00
This - > resource . size = ( ( This - > pow2Width * formatEntry - > bpp ) + SURFACE_ALIGNMENT - 1 ) & ~ ( SURFACE_ALIGNMENT - 1 ) ;
2006-09-24 09:49:05 +02:00
This - > resource . size * = This - > pow2Height ;
2005-11-02 21:01:54 +01:00
}
/* Setup some glformat defaults */
2006-06-21 10:36:14 +02:00
This - > glDescription . glFormat = formatEntry - > glFormat ;
This - > glDescription . glFormatInternal = formatEntry - > glInternal ;
This - > glDescription . glType = formatEntry - > glType ;
2005-11-02 21:01:54 +01:00
if ( format ! = WINED3DFMT_UNKNOWN ) {
2006-06-21 10:36:14 +02:00
This - > bytesPerPixel = formatEntry - > bpp ;
2005-11-02 21:01:54 +01:00
} else {
This - > bytesPerPixel = 0 ;
}
2006-05-06 16:15:37 +02:00
This - > Flags | = ( WINED3DFMT_D16_LOCKABLE = = format ) ? SFLAG_LOCKABLE : 0 ;
2005-11-02 21:01:54 +01:00
This - > resource . format = format ;
2007-02-13 20:21:41 +01:00
TRACE ( " (%p) : Size %d, bytesPerPixel %d, glFormat %d, glFotmatInternal %d, glType %d \n " , This , This - > resource . size , This - > bytesPerPixel , This - > glDescription . glFormat , This - > glDescription . glFormatInternal , This - > glDescription . glType ) ;
2005-11-02 21:01:54 +01:00
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-11-02 21:01:54 +01:00
}
2006-07-25 16:38:02 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_SetMem ( IWineD3DSurface * iface , void * Mem ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
/* Render targets depend on their hdc, and we can't create a hdc on a user pointer */
if ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) {
ERR ( " Not supported on render targets \n " ) ;
return WINED3DERR_INVALIDCALL ;
}
if ( This - > Flags & ( SFLAG_LOCKED | SFLAG_DCINUSE ) ) {
WARN ( " Surface is locked or the HDC is in use \n " ) ;
return WINED3DERR_INVALIDCALL ;
}
if ( Mem & & Mem ! = This - > resource . allocatedMemory ) {
/* Do I have to copy the old surface content? */
if ( This - > Flags & SFLAG_DIBSECTION ) {
/* Release the DC. No need to hold the critical section for the update
* Thread because this thread runs only on front buffers , but this method
* fails for render targets in the check above .
*/
SelectObject ( This - > hDC , This - > dib . holdbitmap ) ;
DeleteDC ( This - > hDC ) ;
/* Release the DIB section */
DeleteObject ( This - > dib . DIBsection ) ;
This - > dib . bitmap_data = NULL ;
This - > resource . allocatedMemory = NULL ;
This - > hDC = NULL ;
This - > Flags & = ~ SFLAG_DIBSECTION ;
} else if ( ! ( This - > Flags & SFLAG_USERPTR ) ) {
HeapFree ( GetProcessHeap ( ) , 0 , This - > resource . allocatedMemory ) ;
}
This - > resource . allocatedMemory = Mem ;
This - > Flags | = SFLAG_USERPTR ;
} else if ( This - > Flags & SFLAG_USERPTR ) {
/* Lockrect and GetDC will re-create the dib section and allocated memory */
This - > resource . allocatedMemory = NULL ;
This - > Flags & = ~ SFLAG_USERPTR ;
}
return WINED3D_OK ;
}
2005-03-14 11:12:52 +01:00
/* TODO: replace this function with context management routines */
HRESULT WINAPI IWineD3DSurfaceImpl_SetPBufferState ( IWineD3DSurface * iface , BOOL inPBuffer , BOOL inTexture ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2005-07-13 16:15:54 +02:00
2006-05-06 16:15:37 +02:00
if ( inPBuffer ) {
This - > Flags | = SFLAG_INPBUFFER ;
} else {
This - > Flags & = ~ SFLAG_INPBUFFER ;
}
if ( inTexture ) {
This - > Flags | = SFLAG_INTEXTURE ;
} else {
This - > Flags & = ~ SFLAG_INTEXTURE ;
}
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-03-14 11:12:52 +01:00
}
2006-06-10 13:15:32 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_Flip ( IWineD3DSurface * iface , IWineD3DSurface * override , DWORD Flags ) {
2006-05-09 19:49:24 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DDevice * D3D = ( IWineD3DDevice * ) This - > resource . wineD3DDevice ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p)->(%p,%x) \n " , This , override , Flags ) ;
2006-05-09 19:49:24 +02:00
/* Flipping is only supported on RenderTargets */
if ( ! ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) ) return DDERR_NOTFLIPPABLE ;
if ( override ) {
/* DDraw sets this for the X11 surfaces, so don't confuse the user
* FIXME ( " (%p) Target override is not supported by now \ n " , This) ;
* Additionally , it isn ' t really possible to support triple - buffering
* properly on opengl at all
*/
}
/* Flipping a OpenGL surface -> Use WineD3DDevice::Present */
return IWineD3DDevice_Present ( D3D , NULL , NULL , 0 , NULL ) ;
2006-04-12 22:32:44 +02:00
}
2007-02-12 19:18:46 +01:00
/* Does a direct frame buffer -> texture copy. Stretching is done
* with single pixel copy calls
*/
static inline void fb_copy_to_texture_direct ( IWineD3DSurfaceImpl * This , IWineD3DSurface * SrcSurface , IWineD3DSwapChainImpl * swapchain , WINED3DRECT * srect , WINED3DRECT * drect , BOOL upsidedown ) {
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
float xrel , yrel ;
UINT row ;
BOOL warned = FALSE ; /* deliberately not static */
IWineD3DSurfaceImpl * Src = ( IWineD3DSurfaceImpl * ) SrcSurface ;
ENTER_GL ( ) ;
ActivateContext ( myDevice , SrcSurface , CTXUSAGE_BLIT ) ;
/* Bind the target texture */
glBindTexture ( GL_TEXTURE_2D , This - > glDescription . textureName ) ;
checkGLcall ( " glBindTexture " ) ;
2007-02-15 03:02:39 +01:00
if ( ! swapchain | | ( swapchain - > backBuffer & & SrcSurface = = swapchain - > backBuffer [ 0 ] ) ) {
2007-02-12 19:18:46 +01:00
glReadBuffer ( GL_BACK ) ;
} else {
glReadBuffer ( GL_FRONT ) ;
}
checkGLcall ( " glReadBuffer " ) ;
xrel = ( float ) ( srect - > x2 - srect - > x1 ) / ( float ) ( drect - > x2 - drect - > x1 ) ;
yrel = ( float ) ( srect - > y2 - srect - > y1 ) / ( float ) ( drect - > y2 - drect - > y1 ) ;
if ( ( xrel - 1.0 < - eps ) | | ( xrel - 1.0 > eps ) ) {
FIXME ( " Doing a pixel by pixel copy from the framebuffer to a texture, expect major performance issues \n " ) ;
}
if ( upsidedown & &
! ( ( xrel - 1.0 < - eps ) | | ( xrel - 1.0 > eps ) ) & &
! ( ( yrel - 1.0 < - eps ) | | ( yrel - 1.0 > eps ) ) ) {
/* Upside down copy without stretching is nice, one glCopyTexSubImage call will do */
glCopyTexSubImage2D ( This - > glDescription . target ,
This - > glDescription . level ,
drect - > x1 , drect - > y1 , /* xoffset, yoffset */
srect - > x1 , Src - > currentDesc . Height - srect - > y2 ,
drect - > x2 - drect - > x1 , drect - > y2 - drect - > y1 ) ;
} else {
UINT yoffset = Src - > currentDesc . Height - srect - > y1 + drect - > y1 - 1 ;
/* I have to process this row by row to swap the image,
* otherwise it would be upside down , so streching in y direction
* doesn ' t cost extra time
*
* However , streching in x direction can be avoided if not necessary
*/
for ( row = drect - > y1 ; row < drect - > y2 ; row + + ) {
if ( ( xrel - 1.0 < - eps ) | | ( xrel - 1.0 > eps ) ) {
/* Well, that stuff works, but it's very slow.
* find a better way instead
*/
UINT col ;
if ( ! warned ) {
warned = TRUE ;
FIXME ( " Doing a pixel by pixel render target -> texture copy, expect performance issues \n " ) ;
}
for ( col = drect - > x1 ; col < drect - > x2 ; col + + ) {
glCopyTexSubImage2D ( This - > glDescription . target ,
This - > glDescription . level ,
drect - > x1 + col , row , /* xoffset, yoffset */
srect - > x1 + col * xrel , yoffset - ( int ) ( row * yrel ) ,
1 , 1 ) ;
}
} else {
glCopyTexSubImage2D ( This - > glDescription . target ,
This - > glDescription . level ,
drect - > x1 , row , /* xoffset, yoffset */
srect - > x1 , yoffset - ( int ) ( row * yrel ) ,
drect - > x2 - drect - > x1 , 1 ) ;
}
}
}
vcheckGLcall ( " glCopyTexSubImage2D " ) ;
LEAVE_GL ( ) ;
}
/* Uses the hardware to stretch and flip the image */
static inline void fb_copy_to_texture_hwstretch ( IWineD3DSurfaceImpl * This , IWineD3DSurface * SrcSurface , IWineD3DSwapChainImpl * swapchain , WINED3DRECT * srect , WINED3DRECT * drect , BOOL upsidedown ) {
GLuint backup , src ;
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
IWineD3DSurfaceImpl * Src = ( IWineD3DSurfaceImpl * ) SrcSurface ;
float left , right , top , bottom ; /* Texture coordinates */
UINT fbwidth = Src - > currentDesc . Width ;
UINT fbheight = Src - > currentDesc . Height ;
TRACE ( " Using hwstretch blit \n " ) ;
/* Activate the Proper context for reading from the source surface, set it up for blitting */
ENTER_GL ( ) ;
ActivateContext ( myDevice , SrcSurface , CTXUSAGE_BLIT ) ;
/* Backup the back buffer and copy the source buffer into a texture to draw an upside down stretched quad. If
* we are reading from the back buffer , the backup can be used as source texture
*/
glGenTextures ( 1 , & backup ) ;
checkGLcall ( " glGenTextures(1, &backup) " ) ;
glBindTexture ( GL_TEXTURE_2D , backup ) ;
checkGLcall ( " glBindTexture(GL_TEXTURE_2D, backup) " ) ;
glReadBuffer ( GL_BACK ) ;
checkGLcall ( " glReadBuffer(GL_BACK) " ) ;
/* TODO: Only back up the part that will be overwritten */
glTexImage2D ( GL_TEXTURE_2D , 0 , GL_RGBA , Src - > pow2Width , Src - > pow2Height , 0 ,
GL_RGBA , GL_UNSIGNED_BYTE , NULL ) ;
checkGLcall ( " glTexImage2D " ) ;
glCopyTexSubImage2D ( GL_TEXTURE_2D , 0 ,
0 , 0 /* read offsets */ ,
0 , 0 ,
fbwidth ,
fbheight ) ;
checkGLcall ( " glCopyTexSubImage2D " ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER , GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER , GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
2007-02-15 03:02:39 +01:00
if ( ! swapchain | | ( IWineD3DSurface * ) This = = swapchain - > backBuffer [ 0 ] ) {
2007-02-12 19:18:46 +01:00
src = backup ;
} else {
glReadBuffer ( GL_FRONT ) ;
checkGLcall ( " glReadBuffer(GL_FRONT) " ) ;
glGenTextures ( 1 , & src ) ;
checkGLcall ( " glGenTextures(1, &src) " ) ;
glBindTexture ( GL_TEXTURE_2D , src ) ;
checkGLcall ( " glBindTexture(GL_TEXTURE_2D, src) " ) ;
/* TODO: Only copy the part that will be read. Use srect->x1, srect->y2 as origin, but with the width watch
* out for power of 2 sizes
*/
glTexImage2D ( GL_TEXTURE_2D , 0 , GL_RGBA , Src - > pow2Width , Src - > pow2Height , 0 ,
GL_RGBA , GL_UNSIGNED_BYTE , NULL ) ;
checkGLcall ( " glTexImage2D " ) ;
glCopyTexSubImage2D ( GL_TEXTURE_2D , 0 ,
0 , 0 /* read offsets */ ,
0 , 0 ,
fbwidth ,
fbheight ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER , GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER , GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
glReadBuffer ( GL_BACK ) ;
checkGLcall ( " glReadBuffer(GL_BACK) " ) ;
}
checkGLcall ( " glEnd and previous " ) ;
left = ( float ) srect - > x1 / ( float ) Src - > pow2Width ;
right = ( float ) srect - > x2 / ( float ) Src - > pow2Width ;
if ( upsidedown ) {
top = ( float ) ( Src - > currentDesc . Height - srect - > y1 ) / ( float ) Src - > pow2Height ;
bottom = ( float ) ( Src - > currentDesc . Height - srect - > y2 ) / ( float ) Src - > pow2Height ;
} else {
top = ( float ) ( Src - > currentDesc . Height - srect - > y2 ) / ( float ) Src - > pow2Height ;
bottom = ( float ) ( Src - > currentDesc . Height - srect - > y1 ) / ( float ) Src - > pow2Height ;
}
/* draw the source texture stretched and upside down. The correct surface is bound already */
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_WRAP_S , GL_CLAMP ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_WRAP_T , GL_CLAMP ) ;
glBegin ( GL_QUADS ) ;
/* bottom left */
glTexCoord2f ( left , bottom ) ;
glVertex2i ( 0 , fbheight ) ;
/* top left */
glTexCoord2f ( left , top ) ;
glVertex2i ( 0 , fbheight - drect - > y2 - drect - > y1 ) ;
/* top right */
glTexCoord2f ( right , top ) ;
glVertex2i ( drect - > x2 - drect - > x1 , fbheight - drect - > y2 - drect - > y1 ) ;
/* bottom right */
glTexCoord2f ( right , bottom ) ;
glVertex2i ( drect - > x2 - drect - > x1 , fbheight ) ;
glEnd ( ) ;
checkGLcall ( " glEnd and previous " ) ;
/* Now read the stretched and upside down image into the destination texture */
glBindTexture ( This - > glDescription . target , This - > glDescription . textureName ) ;
checkGLcall ( " glBindTexture " ) ;
glCopyTexSubImage2D ( This - > glDescription . target ,
0 ,
drect - > x1 , drect - > y1 , /* xoffset, yoffset */
0 , 0 , /* We blitted the image to the origin */
drect - > x2 - drect - > x1 , drect - > y2 - drect - > y1 ) ;
checkGLcall ( " glCopyTexSubImage2D " ) ;
/* Write the back buffer backup back */
glBindTexture ( GL_TEXTURE_2D , backup ) ;
checkGLcall ( " glBindTexture(GL_TEXTURE_2D, backup) " ) ;
glBegin ( GL_QUADS ) ;
/* top left */
glTexCoord2f ( 0.0 , ( float ) fbheight / ( float ) Src - > pow2Height ) ;
glVertex2i ( 0 , 0 ) ;
/* bottom left */
glTexCoord2f ( 0.0 , 0.0 ) ;
glVertex2i ( 0 , fbheight ) ;
/* bottom right */
glTexCoord2f ( ( float ) fbwidth / ( float ) Src - > pow2Width , 0.0 ) ;
glVertex2i ( fbwidth , Src - > currentDesc . Height ) ;
/* top right */
glTexCoord2f ( ( float ) fbwidth / ( float ) Src - > pow2Width , ( float ) fbheight / ( float ) Src - > pow2Height ) ;
glVertex2i ( fbwidth , 0 ) ;
glEnd ( ) ;
/* Cleanup */
if ( src ! = backup ) {
glDeleteTextures ( 1 , & src ) ;
checkGLcall ( " glDeleteTextures(1, &src) " ) ;
}
glDeleteTextures ( 1 , & backup ) ;
checkGLcall ( " glDeleteTextures(1, &backup) " ) ;
LEAVE_GL ( ) ;
}
2006-05-23 00:48:54 +02:00
/* Not called from the VTable */
2006-06-10 13:15:32 +02:00
static HRESULT IWineD3DSurfaceImpl_BltOverride ( IWineD3DSurfaceImpl * This , RECT * DestRect , IWineD3DSurface * SrcSurface , RECT * SrcRect , DWORD Flags , DDBLTFX * DDBltFx ) {
2006-10-12 08:22:18 +02:00
WINED3DRECT rect ;
2006-05-23 00:48:54 +02:00
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
2007-02-15 03:02:39 +01:00
IWineD3DSwapChainImpl * srcSwapchain = NULL , * dstSwapchain = NULL ;
2006-05-23 00:48:54 +02:00
IWineD3DSurfaceImpl * Src = ( IWineD3DSurfaceImpl * ) SrcSurface ;
BOOL SrcOK = TRUE ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p)->(%p,%p,%p,%08x,%p) \n " , This , DestRect , SrcSurface , SrcRect , Flags , DDBltFx ) ;
2006-05-23 00:48:54 +02:00
/* Get the swapchain. One of the surfaces has to be a primary surface */
2007-02-15 03:02:39 +01:00
IWineD3DSurface_GetContainer ( ( IWineD3DSurface * ) This , & IID_IWineD3DSwapChain , ( void * * ) & dstSwapchain ) ;
if ( dstSwapchain ) IWineD3DSwapChain_Release ( ( IWineD3DSwapChain * ) dstSwapchain ) ;
if ( Src ) {
IWineD3DSurface_GetContainer ( ( IWineD3DSurface * ) Src , & IID_IWineD3DSwapChain , ( void * * ) & srcSwapchain ) ;
if ( srcSwapchain ) IWineD3DSwapChain_Release ( ( IWineD3DSwapChain * ) srcSwapchain ) ;
}
/* Early sort out of cases where no render target is used */
if ( ! dstSwapchain & & ! srcSwapchain & &
SrcSurface ! = myDevice - > render_targets [ 0 ] & & This ! = ( IWineD3DSurfaceImpl * ) myDevice - > render_targets [ 0 ] ) {
TRACE ( " No surface is render target, not using hardware blit. Src = %p, dst = %p \n " , Src , This ) ;
return WINED3DERR_INVALIDCALL ;
2006-05-23 00:48:54 +02:00
}
2007-02-15 03:03:05 +01:00
/* No destination color keying supported */
if ( Flags & ( DDBLT_KEYDEST | DDBLT_KEYDESTOVERRIDE ) ) {
/* Can we support that with glBlendFunc if blitting to the frame buffer? */
TRACE ( " Destination color key not supported in accelerated Blit, falling back to software \n " ) ;
return WINED3DERR_INVALIDCALL ;
}
2006-05-23 00:48:54 +02:00
if ( DestRect ) {
rect . x1 = DestRect - > left ;
rect . y1 = DestRect - > top ;
rect . x2 = DestRect - > right ;
rect . y2 = DestRect - > bottom ;
} else {
rect . x1 = 0 ;
rect . y1 = 0 ;
rect . x2 = This - > currentDesc . Width ;
rect . y2 = This - > currentDesc . Height ;
}
2007-02-15 03:02:39 +01:00
/* The only case where both surfaces on a swapchain are supported is a back buffer -> front buffer blit on the same swapchain */
if ( dstSwapchain & & dstSwapchain = = srcSwapchain ) {
/* Half-life does a Blt from the back buffer to the front buffer,
* Full surface size , no flags . . . Use present instead
*/
2006-05-23 00:48:54 +02:00
/* Check rects - IWineD3DDevice_Present doesn't handle them */
if ( SrcRect ) {
if ( ( SrcRect - > left = = 0 ) & & ( SrcRect - > top = = 0 ) & &
( SrcRect - > right = = Src - > currentDesc . Width ) & & ( SrcRect - > bottom = = Src - > currentDesc . Height ) ) {
SrcOK = TRUE ;
}
} else {
SrcOK = TRUE ;
}
/* Check the Destination rect and the surface sizes */
if ( SrcOK & &
( rect . x1 = = 0 ) & & ( rect . y1 = = 0 ) & &
( rect . x2 = = This - > currentDesc . Width ) & & ( rect . y2 = = This - > currentDesc . Height ) & &
( This - > currentDesc . Width = = Src - > currentDesc . Width ) & &
( This - > currentDesc . Height = = Src - > currentDesc . Height ) ) {
/* These flags are unimportant for the flag check, remove them */
if ( ( Flags & ~ ( DDBLT_DONOTWAIT | DDBLT_WAIT ) ) = = 0 ) {
2007-02-15 03:02:39 +01:00
if ( dstSwapchain - > backBuffer & & ( ( IWineD3DSurface * ) This = = dstSwapchain - > frontBuffer ) & &
SrcSurface = = dstSwapchain - > backBuffer [ 0 ] ) {
2006-05-23 00:48:54 +02:00
2007-02-15 13:32:25 +01:00
WINED3DSWAPEFFECT orig_swap = dstSwapchain - > presentParms . SwapEffect ;
2006-05-23 00:48:54 +02:00
/* The idea behind this is that a glReadPixels and a glDrawPixels call
* take very long , while a flip is fast .
* This applies to Half - Life , which does such Blts every time it finished
* a frame , and to Prince of Persia 3 D , which uses this to draw at least the main
* menu . This is also used by all apps when they do windowed rendering
*
* The problem is that flipping is not really the same as copying . After a
* Blt the front buffer is a copy of the back buffer , and the back buffer is
* untouched . Therefore it ' s necessary to override the swap effect
* and to set it back after the flip .
*/
2007-02-15 03:02:39 +01:00
dstSwapchain - > presentParms . SwapEffect = WINED3DSWAPEFFECT_COPY ;
2006-05-23 00:48:54 +02:00
TRACE ( " Full screen back buffer -> front buffer blt, performing a flip instead \n " ) ;
IWineD3DDevice_Present ( ( IWineD3DDevice * ) This - > resource . wineD3DDevice ,
NULL , NULL , 0 , NULL ) ;
2007-02-15 03:02:39 +01:00
dstSwapchain - > presentParms . SwapEffect = orig_swap ;
2006-05-23 00:48:54 +02:00
return WINED3D_OK ;
}
}
}
2007-02-15 03:02:39 +01:00
TRACE ( " Unsupported blit between buffers on the same swapchain \n " ) ;
return WINED3DERR_INVALIDCALL ;
} else if ( ( dstSwapchain | | This = = ( IWineD3DSurfaceImpl * ) myDevice - > render_targets [ 0 ] ) & &
( srcSwapchain | | SrcSurface = = myDevice - > render_targets [ 0 ] ) ) {
ERR ( " Can't perform hardware blit between 2 different swapchains, falling back to software \n " ) ;
return WINED3DERR_INVALIDCALL ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
if ( srcSwapchain | | SrcSurface = = myDevice - > render_targets [ 0 ] ) {
/* Blit from render target to texture */
WINED3DRECT srect ;
BOOL upsideDown , stretchx ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:03:05 +01:00
if ( Flags & ( DDBLT_KEYSRC | DDBLT_KEYSRCOVERRIDE ) ) {
TRACE ( " Color keying not supported by frame buffer to texture blit \n " ) ;
return WINED3DERR_INVALIDCALL ;
/* Destination color key is checked above */
}
2007-02-15 03:02:39 +01:00
/* Call preload for the surface to make sure it isn't dirty */
IWineD3DSurface_PreLoad ( ( IWineD3DSurface * ) This ) ;
/* Make sure that the top pixel is always above the bottom pixel, and keep a seperate upside down flag
* glCopyTexSubImage is a bit picky about the parameters we pass to it
*/
if ( SrcRect ) {
if ( SrcRect - > top < SrcRect - > bottom ) {
srect . y1 = SrcRect - > top ;
srect . y2 = SrcRect - > bottom ;
upsideDown = FALSE ;
} else {
srect . y1 = SrcRect - > bottom ;
srect . y2 = SrcRect - > top ;
upsideDown = TRUE ;
2006-05-23 00:48:54 +02:00
}
2007-02-15 03:02:39 +01:00
srect . x1 = SrcRect - > left ;
srect . x2 = SrcRect - > right ;
} else {
srect . x1 = 0 ;
srect . y1 = 0 ;
srect . x2 = Src - > currentDesc . Width ;
srect . y2 = Src - > currentDesc . Height ;
upsideDown = FALSE ;
}
if ( rect . x1 > rect . x2 ) {
UINT tmp = rect . x2 ;
rect . x2 = rect . x1 ;
rect . x1 = tmp ;
upsideDown = ! upsideDown ;
}
2007-02-15 13:51:55 +01:00
if ( ! srcSwapchain ) {
TRACE ( " Reading from an offscreen target \n " ) ;
upsideDown = ! upsideDown ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
if ( rect . x2 - rect . x1 ! = srect . x2 - srect . x1 ) {
stretchx = TRUE ;
} else {
stretchx = FALSE ;
}
2006-07-26 23:19:04 +02:00
2007-02-15 03:02:39 +01:00
/* Blt is a pretty powerful call, while glCopyTexSubImage2D is not. glCopyTexSubImage cannot
* flip the image nor scale it . If GL_EXT_framebuffer_blit is available it can be used ( hopefully ,
* not implemented by now ) . Otherwise :
*
* - > If the app asks for a unscaled , upside down copy , just perform one glCopyTexSubImage2D call
* - > If the app wants a image width an unscaled width , copy it line per line
* - > If the app wants a image that is scaled on the x axis , and the destination rectangle is smaller
* than the frame buffer , draw an upside down scaled image onto the fb , read it back and restore the
* back buffer . This is slower than reading line per line , thus not used for flipping
* - > If the app wants a scaled image with a dest rect that is bigger than the fb , it has to be copied
* pixel by pixel
*/
if ( FALSE /* GL_SUPPORT(EXT_FRAMEBUFFER_BLIT) */ ) {
TRACE ( " Using GL_EXT_framebuffer_blit for copying \n " ) ;
} else if ( ( ! stretchx ) | | rect . x2 - rect . x1 > Src - > currentDesc . Width | |
rect . y2 - rect . y1 > Src - > currentDesc . Height ) {
TRACE ( " No stretching in x direction, using direct framebuffer -> texture copy \n " ) ;
fb_copy_to_texture_direct ( This , SrcSurface , srcSwapchain , & srect , & rect , upsideDown ) ;
} else {
TRACE ( " Using hardware stretching to flip / stretch the texture \n " ) ;
fb_copy_to_texture_hwstretch ( This , SrcSurface , srcSwapchain , & srect , & rect , upsideDown ) ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
if ( ! ( This - > Flags & SFLAG_DONOTFREE ) ) {
HeapFree ( GetProcessHeap ( ) , 0 , This - > resource . allocatedMemory ) ;
This - > resource . allocatedMemory = NULL ;
} else {
This - > Flags | = SFLAG_GLDIRTY ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
return WINED3D_OK ;
} else if ( Src ) {
/* Blit from offscreen surface to render target */
float glTexCoord [ 4 ] ;
2007-02-15 03:03:05 +01:00
DWORD oldCKeyFlags = Src - > CKeyFlags ;
DDCOLORKEY oldBltCKey = This - > SrcBltCKey ;
2007-02-15 03:02:39 +01:00
RECT SourceRectangle ;
GLint oldDraw ;
TRACE ( " Blt from surface %p to rendertarget %p \n " , Src , This ) ;
if ( SrcRect ) {
SourceRectangle . left = SrcRect - > left ;
SourceRectangle . right = SrcRect - > right ;
SourceRectangle . top = SrcRect - > top ;
SourceRectangle . bottom = SrcRect - > bottom ;
} else {
SourceRectangle . left = 0 ;
SourceRectangle . right = Src - > currentDesc . Width ;
SourceRectangle . top = 0 ;
SourceRectangle . bottom = Src - > currentDesc . Height ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
if ( ! CalculateTexRect ( Src , & SourceRectangle , glTexCoord ) ) {
/* Fall back to software */
WARN ( " (%p) Source texture area (%d,%d)-(%d,%d) is too big \n " , Src ,
SourceRectangle . left , SourceRectangle . top ,
SourceRectangle . right , SourceRectangle . bottom ) ;
return WINED3DERR_INVALIDCALL ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
/* Color keying: Check if we have to do a color keyed blt,
* and if not check if a color key is activated .
2007-02-15 03:03:05 +01:00
*
* Just modify the color keying parameters in the surface and restore them afterwards
* The surface keeps track of the color key last used to load the opengl surface .
* PreLoad will catch the change to the flags and color key and reload if neccessary .
2007-02-15 03:02:39 +01:00
*/
2007-02-15 03:03:05 +01:00
if ( Flags & DDBLT_KEYSRC ) {
/* Use color key from surface */
} else if ( Flags & DDBLT_KEYSRCOVERRIDE ) {
/* Use color key from DDBltFx */
Src - > CKeyFlags | = DDSD_CKSRCBLT ;
This - > SrcBltCKey = DDBltFx - > ddckSrcColorkey ;
} else {
/* Do not use color key */
2007-02-15 03:02:39 +01:00
Src - > CKeyFlags & = ~ DDSD_CKSRCBLT ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
/* Now load the surface */
IWineD3DSurface_PreLoad ( ( IWineD3DSurface * ) Src ) ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
ENTER_GL ( ) ;
2006-07-26 23:19:04 +02:00
2007-02-15 03:02:39 +01:00
/* Activate the destination context, set it up for blitting */
ActivateContext ( myDevice , ( IWineD3DSurface * ) This , CTXUSAGE_BLIT ) ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
glGetIntegerv ( GL_DRAW_BUFFER , & oldDraw ) ;
if ( This = = ( IWineD3DSurfaceImpl * ) dstSwapchain - > frontBuffer ) {
TRACE ( " Drawing to front buffer \n " ) ;
glDrawBuffer ( GL_FRONT ) ;
checkGLcall ( " glDrawBuffer GL_FRONT " ) ;
}
2006-06-04 16:42:57 +02:00
2007-02-15 03:02:39 +01:00
/* Bind the texture */
glBindTexture ( GL_TEXTURE_2D , Src - > glDescription . textureName ) ;
checkGLcall ( " glBindTexture " ) ;
/* No filtering for blts */
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER ,
GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER ,
GL_NEAREST ) ;
checkGLcall ( " glTexParameteri " ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_WRAP_S , GL_CLAMP ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_WRAP_T , GL_CLAMP ) ;
glTexEnvi ( GL_TEXTURE_ENV , GL_TEXTURE_ENV_MODE , GL_REPLACE ) ;
checkGLcall ( " glTexEnvi " ) ;
/* This is for color keying */
2007-02-15 03:03:05 +01:00
if ( Flags & ( DDBLT_KEYSRC | DDBLT_KEYSRCOVERRIDE ) ) {
2007-02-15 03:02:39 +01:00
glEnable ( GL_ALPHA_TEST ) ;
checkGLcall ( " glEnable GL_ALPHA_TEST " ) ;
glAlphaFunc ( GL_NOTEQUAL , 0.0 ) ;
checkGLcall ( " glAlphaFunc \n " ) ;
} else {
glDisable ( GL_ALPHA_TEST ) ;
checkGLcall ( " glDisable GL_ALPHA_TEST " ) ;
2006-05-23 00:48:54 +02:00
}
2007-02-15 03:02:39 +01:00
/* Draw a textured quad
*/
glBegin ( GL_QUADS ) ;
glColor3d ( 1.0f , 1.0f , 1.0f ) ;
glTexCoord2f ( glTexCoord [ 0 ] , glTexCoord [ 2 ] ) ;
glVertex3f ( rect . x1 ,
rect . y1 ,
0.0 ) ;
glTexCoord2f ( glTexCoord [ 0 ] , glTexCoord [ 3 ] ) ;
glVertex3f ( rect . x1 , rect . y2 , 0.0 ) ;
glTexCoord2f ( glTexCoord [ 1 ] , glTexCoord [ 3 ] ) ;
glVertex3f ( rect . x2 ,
rect . y2 ,
0.0 ) ;
glTexCoord2f ( glTexCoord [ 1 ] , glTexCoord [ 2 ] ) ;
glVertex3f ( rect . x2 ,
rect . y1 ,
0.0 ) ;
glEnd ( ) ;
checkGLcall ( " glEnd " ) ;
2007-02-15 03:04:40 +01:00
if ( Flags & ( DDBLT_KEYSRC | DDBLT_KEYSRCOVERRIDE ) ) {
glDisable ( GL_ALPHA_TEST ) ;
checkGLcall ( " glDisable(GL_ALPHA_TEST) " ) ;
}
2007-02-15 03:02:39 +01:00
/* Unbind the texture */
glBindTexture ( GL_TEXTURE_2D , 0 ) ;
checkGLcall ( " glEnable glBindTexture " ) ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
if ( This = = ( IWineD3DSurfaceImpl * ) dstSwapchain - > frontBuffer & & oldDraw = = GL_BACK ) {
glDrawBuffer ( oldDraw ) ;
}
2007-02-15 03:03:05 +01:00
/* Restore the color key parameters */
Src - > CKeyFlags = oldCKeyFlags ;
This - > SrcBltCKey = oldBltCKey ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
LEAVE_GL ( ) ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
/* TODO: If the surface is locked often, perform the Blt in software on the memory instead */
This - > Flags | = SFLAG_GLDIRTY ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
return WINED3D_OK ;
} else {
/* Source-Less Blit to render target */
if ( Flags & DDBLT_COLORFILL ) {
/* This is easy to handle for the D3D Device... */
DWORD color ;
TRACE ( " Colorfill \n " ) ;
/* The color as given in the Blt function is in the format of the frame-buffer...
* ' clear ' expect it in ARGB format = > we need to do some conversion : - )
*/
if ( This - > resource . format = = WINED3DFMT_P8 ) {
if ( This - > palette ) {
color = ( ( 0xFF000000 ) |
( This - > palette - > palents [ DDBltFx - > u5 . dwFillColor ] . peRed < < 16 ) |
( This - > palette - > palents [ DDBltFx - > u5 . dwFillColor ] . peGreen < < 8 ) |
( This - > palette - > palents [ DDBltFx - > u5 . dwFillColor ] . peBlue ) ) ;
2007-02-12 19:18:46 +01:00
} else {
2007-02-15 03:02:39 +01:00
color = 0xFF000000 ;
2006-05-23 00:48:54 +02:00
}
2007-02-15 03:02:39 +01:00
}
else if ( This - > resource . format = = WINED3DFMT_R5G6B5 ) {
if ( DDBltFx - > u5 . dwFillColor = = 0xFFFF ) {
color = 0xFFFFFFFF ;
2006-05-23 00:48:54 +02:00
} else {
2007-02-15 03:02:39 +01:00
color = ( ( 0xFF000000 ) |
( ( DDBltFx - > u5 . dwFillColor & 0xF800 ) < < 8 ) |
( ( DDBltFx - > u5 . dwFillColor & 0x07E0 ) < < 5 ) |
( ( DDBltFx - > u5 . dwFillColor & 0x001F ) < < 3 ) ) ;
2006-05-23 00:48:54 +02:00
}
}
2007-02-15 03:02:39 +01:00
else if ( ( This - > resource . format = = WINED3DFMT_R8G8B8 ) | |
( This - > resource . format = = WINED3DFMT_X8R8G8B8 ) ) {
color = 0xFF000000 | DDBltFx - > u5 . dwFillColor ;
2006-05-23 00:48:54 +02:00
}
2007-02-15 03:02:39 +01:00
else if ( This - > resource . format = = WINED3DFMT_A8R8G8B8 ) {
color = DDBltFx - > u5 . dwFillColor ;
}
else {
ERR ( " Wrong surface type for BLT override(Format doesn't match) ! \n " ) ;
return WINED3DERR_INVALIDCALL ;
}
TRACE ( " Calling GetSwapChain with mydevice = %p \n " , myDevice ) ;
if ( dstSwapchain & & dstSwapchain - > backBuffer & & This = = ( IWineD3DSurfaceImpl * ) dstSwapchain - > backBuffer [ 0 ] ) {
glDrawBuffer ( GL_BACK ) ;
checkGLcall ( " glDrawBuffer(GL_BACK) " ) ;
} else if ( dstSwapchain & & This = = ( IWineD3DSurfaceImpl * ) dstSwapchain - > frontBuffer ) {
glDrawBuffer ( GL_FRONT ) ;
checkGLcall ( " glDrawBuffer(GL_FRONT) " ) ;
} else if ( This = = ( IWineD3DSurfaceImpl * ) myDevice - > render_targets [ 0 ] ) {
glDrawBuffer ( GL_BACK ) ;
checkGLcall ( " glDrawBuffer(GL_BACK) " ) ;
2006-05-23 00:48:54 +02:00
} else {
2007-02-15 03:02:39 +01:00
TRACE ( " Surface is higher back buffer, falling back to software \n " ) ;
return WINED3DERR_INVALIDCALL ;
2006-05-23 00:48:54 +02:00
}
2007-02-15 03:02:39 +01:00
TRACE ( " (%p) executing Render Target override, color = %x \n " , This , color ) ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
IWineD3DDevice_Clear ( ( IWineD3DDevice * ) myDevice ,
1 /* Number of rectangles */ ,
& rect ,
WINED3DCLEAR_TARGET ,
color ,
0.0 /* Z */ ,
0 /* Stencil */ ) ;
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
/* Restore the original draw buffer */
if ( ! dstSwapchain | | ( dstSwapchain - > backBuffer & & dstSwapchain - > backBuffer [ 0 ] ) ) {
glDrawBuffer ( GL_BACK ) ;
vcheckGLcall ( " glDrawBuffer " ) ;
}
2006-05-23 00:48:54 +02:00
2007-02-15 03:02:39 +01:00
return WINED3D_OK ;
2006-06-24 12:32:18 +02:00
}
2006-05-23 00:48:54 +02:00
}
2007-02-15 03:02:39 +01:00
/* Default: Fall back to the generic blt. Not an error, a TRACE is enought */
TRACE ( " Didn't find any usable render target setup for hw blit, falling back to software \n " ) ;
2006-04-14 19:15:46 +02:00
return WINED3DERR_INVALIDCALL ;
}
2006-06-10 13:15:32 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_Blt ( IWineD3DSurface * iface , RECT * DestRect , IWineD3DSurface * SrcSurface , RECT * SrcRect , DWORD Flags , DDBLTFX * DDBltFx ) {
2006-05-23 00:48:54 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DSurfaceImpl * Src = ( IWineD3DSurfaceImpl * ) SrcSurface ;
2007-02-09 16:35:45 +01:00
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p)->(%p,%p,%p,%x,%p) \n " , This , DestRect , SrcSurface , SrcRect , Flags , DDBltFx ) ;
2006-07-28 21:33:20 +02:00
TRACE ( " (%p): Usage is %s \n " , This , debug_d3dusage ( This - > resource . usage ) ) ;
2006-05-23 00:48:54 +02:00
2007-02-09 16:35:45 +01:00
/* Accessing the depth stencil is supposed to fail between a BeginScene and EndScene pair */
if ( myDevice - > inScene & &
( iface = = myDevice - > stencilBufferTarget | |
( SrcSurface & & SrcSurface = = myDevice - > stencilBufferTarget ) ) ) {
TRACE ( " Attempt to access the depth stencil surface in a BeginScene / EndScene pair, returning WINED3DERR_INVALIDCALL \n " ) ;
return WINED3DERR_INVALIDCALL ;
}
2006-05-23 00:48:54 +02:00
/* Special cases for RenderTargets */
if ( ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) | |
( Src & & ( Src - > resource . usage & WINED3DUSAGE_RENDERTARGET ) ) ) {
if ( IWineD3DSurfaceImpl_BltOverride ( This , DestRect , SrcSurface , SrcRect , Flags , DDBltFx ) = = WINED3D_OK ) return WINED3D_OK ;
}
/* For the rest call the X11 surface implementation.
* For RenderTargets this should be implemented OpenGL accelerated in BltOverride ,
* other Blts are rather rare
*/
return IWineGDISurfaceImpl_Blt ( iface , DestRect , SrcSurface , SrcRect , Flags , DDBltFx ) ;
}
2006-04-17 17:25:55 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_GetBltStatus ( IWineD3DSurface * iface , DWORD Flags ) {
2006-05-09 20:07:40 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p)->(%x) \n " , This , Flags ) ;
2006-05-09 20:07:40 +02:00
switch ( Flags )
{
case DDGBS_CANBLT :
case DDGBS_ISBLTDONE :
return DD_OK ;
default :
return DDERR_INVALIDPARAMS ;
}
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_GetFlipStatus ( IWineD3DSurface * iface , DWORD Flags ) {
2006-05-09 20:07:40 +02:00
/* XXX: DDERR_INVALIDSURFACETYPE */
2006-10-01 05:20:10 +02:00
TRACE ( " (%p)->(%08x) \n " , iface , Flags ) ;
2006-05-09 20:07:40 +02:00
switch ( Flags ) {
case DDGFS_CANFLIP :
case DDGFS_ISFLIPDONE :
return DD_OK ;
default :
return DDERR_INVALIDPARAMS ;
}
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_IsLost ( IWineD3DSurface * iface ) {
2006-05-09 18:47:28 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p) \n " , This ) ;
return This - > Flags & SFLAG_LOST ? DDERR_SURFACELOST : WINED3D_OK ;
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_Restore ( IWineD3DSurface * iface ) {
2006-05-09 18:47:28 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p) \n " , This ) ;
/* So far we don't lose anything :) */
This - > Flags & = ~ SFLAG_LOST ;
return WINED3D_OK ;
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_BltFast ( IWineD3DSurface * iface , DWORD dstx , DWORD dsty , IWineD3DSurface * Source , RECT * rsrc , DWORD trans ) {
2006-05-26 12:19:28 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DSurfaceImpl * srcImpl = ( IWineD3DSurfaceImpl * ) Source ;
2007-02-09 16:35:45 +01:00
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p)->(%d, %d, %p, %p, %08x \n " , iface , dstx , dsty , Source , rsrc , trans ) ;
2006-05-26 12:19:28 +02:00
2007-02-09 16:35:45 +01:00
if ( myDevice - > inScene & &
( iface = = myDevice - > stencilBufferTarget | |
( Source & & Source = = myDevice - > stencilBufferTarget ) ) ) {
TRACE ( " Attempt to access the depth stencil surface in a BeginScene / EndScene pair, returning WINED3DERR_INVALIDCALL \n " ) ;
return WINED3DERR_INVALIDCALL ;
}
2006-05-26 12:19:28 +02:00
/* Special cases for RenderTargets */
if ( ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) | |
( srcImpl & & ( srcImpl - > resource . usage & WINED3DUSAGE_RENDERTARGET ) ) ) {
RECT SrcRect , DstRect ;
2006-07-26 23:19:04 +02:00
DWORD Flags = 0 ;
2006-05-26 12:19:28 +02:00
if ( rsrc ) {
SrcRect . left = rsrc - > left ;
SrcRect . top = rsrc - > top ;
SrcRect . bottom = rsrc - > bottom ;
SrcRect . right = rsrc - > right ;
} else {
SrcRect . left = 0 ;
SrcRect . top = 0 ;
SrcRect . right = srcImpl - > currentDesc . Width ;
SrcRect . bottom = srcImpl - > currentDesc . Height ;
}
DstRect . left = dstx ;
DstRect . top = dsty ;
DstRect . right = dstx + SrcRect . right - SrcRect . left ;
DstRect . bottom = dsty + SrcRect . bottom - SrcRect . top ;
2006-09-28 21:03:22 +02:00
/* Convert BltFast flags into Btl ones because it is called from SurfaceImpl_Blt as well */
2006-07-26 23:19:04 +02:00
if ( trans & DDBLTFAST_SRCCOLORKEY )
Flags | = DDBLT_KEYSRC ;
if ( trans & DDBLTFAST_DESTCOLORKEY )
Flags | = DDBLT_KEYDEST ;
if ( trans & DDBLTFAST_WAIT )
Flags | = DDBLT_WAIT ;
if ( trans & DDBLTFAST_DONOTWAIT )
Flags | = DDBLT_DONOTWAIT ;
if ( IWineD3DSurfaceImpl_BltOverride ( This , & DstRect , Source , & SrcRect , Flags , NULL ) = = WINED3D_OK ) return WINED3D_OK ;
2006-05-26 12:19:28 +02:00
}
return IWineGDISurfaceImpl_BltFast ( iface , dstx , dsty , Source , rsrc , trans ) ;
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_GetPalette ( IWineD3DSurface * iface , IWineD3DPalette * * Pal ) {
2006-05-09 20:28:23 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p)->(%p) \n " , This , Pal ) ;
* Pal = ( IWineD3DPalette * ) This - > palette ;
return DD_OK ;
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_RealizePalette ( IWineD3DSurface * iface ) {
2006-05-09 20:28:23 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
RGBQUAD col [ 256 ] ;
IWineD3DPaletteImpl * pal = This - > palette ;
unsigned int n ;
TRACE ( " (%p) \n " , This ) ;
if ( This - > resource . format = = WINED3DFMT_P8 | |
This - > resource . format = = WINED3DFMT_A8P8 )
{
TRACE ( " Dirtifying surface \n " ) ;
This - > Flags | = SFLAG_DIRTY ;
}
2006-05-14 18:01:31 +02:00
if ( This - > Flags & SFLAG_DIBSECTION ) {
TRACE ( " (%p): Updating the hdc's palette \n " , This ) ;
for ( n = 0 ; n < 256 ; n + + ) {
if ( pal ) {
col [ n ] . rgbRed = pal - > palents [ n ] . peRed ;
col [ n ] . rgbGreen = pal - > palents [ n ] . peGreen ;
col [ n ] . rgbBlue = pal - > palents [ n ] . peBlue ;
} else {
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
/* Use the default device palette */
col [ n ] . rgbRed = device - > palettes [ device - > currentPalette ] [ n ] . peRed ;
col [ n ] . rgbGreen = device - > palettes [ device - > currentPalette ] [ n ] . peGreen ;
col [ n ] . rgbBlue = device - > palettes [ device - > currentPalette ] [ n ] . peBlue ;
}
col [ n ] . rgbReserved = 0 ;
}
SetDIBColorTable ( This - > hDC , 0 , 256 , col ) ;
2006-05-09 20:28:23 +02:00
}
return WINED3D_OK ;
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_SetPalette ( IWineD3DSurface * iface , IWineD3DPalette * Pal ) {
2006-05-09 20:28:23 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DPaletteImpl * PalImpl = ( IWineD3DPaletteImpl * ) Pal ;
TRACE ( " (%p)->(%p) \n " , This , Pal ) ;
if ( This - > palette ! = NULL )
if ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET )
This - > palette - > Flags & = ~ DDPCAPS_PRIMARYSURFACE ;
if ( PalImpl ! = NULL ) {
if ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) {
/* Set the device's main palette if the palette
* wasn ' t a primary palette before
*/
if ( ! ( PalImpl - > Flags & DDPCAPS_PRIMARYSURFACE ) ) {
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
unsigned int i ;
for ( i = 0 ; i < 256 ; i + + ) {
device - > palettes [ device - > currentPalette ] [ i ] = PalImpl - > palents [ i ] ;
}
}
( PalImpl ) - > Flags | = DDPCAPS_PRIMARYSURFACE ;
}
}
This - > palette = PalImpl ;
return IWineD3DSurface_RealizePalette ( iface ) ;
2006-04-17 17:25:55 +02:00
}
HRESULT WINAPI IWineD3DSurfaceImpl_SetColorKey ( IWineD3DSurface * iface , DWORD Flags , DDCOLORKEY * CKey ) {
2006-05-09 19:37:38 +02:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2006-10-01 05:20:10 +02:00
TRACE ( " (%p)->(%08x,%p) \n " , This , Flags , CKey ) ;
2006-05-09 19:37:38 +02:00
if ( ( Flags & DDCKEY_COLORSPACE ) ! = 0 ) {
2006-10-01 05:20:10 +02:00
FIXME ( " colorkey value not supported (%08x) ! \n " , Flags ) ;
2006-05-09 19:37:38 +02:00
return DDERR_INVALIDPARAMS ;
}
/* Dirtify the surface, but only if a key was changed */
if ( CKey ) {
switch ( Flags & ~ DDCKEY_COLORSPACE ) {
case DDCKEY_DESTBLT :
This - > DestBltCKey = * CKey ;
This - > CKeyFlags | = DDSD_CKDESTBLT ;
break ;
case DDCKEY_DESTOVERLAY :
This - > DestOverlayCKey = * CKey ;
This - > CKeyFlags | = DDSD_CKDESTOVERLAY ;
break ;
case DDCKEY_SRCOVERLAY :
This - > SrcOverlayCKey = * CKey ;
This - > CKeyFlags | = DDSD_CKSRCOVERLAY ;
break ;
case DDCKEY_SRCBLT :
This - > SrcBltCKey = * CKey ;
This - > CKeyFlags | = DDSD_CKSRCBLT ;
break ;
}
}
else {
switch ( Flags & ~ DDCKEY_COLORSPACE ) {
case DDCKEY_DESTBLT :
This - > CKeyFlags & = ~ DDSD_CKDESTBLT ;
break ;
case DDCKEY_DESTOVERLAY :
This - > CKeyFlags & = ~ DDSD_CKDESTOVERLAY ;
break ;
case DDCKEY_SRCOVERLAY :
This - > CKeyFlags & = ~ DDSD_CKSRCOVERLAY ;
break ;
case DDCKEY_SRCBLT :
This - > CKeyFlags & = ~ DDSD_CKSRCBLT ;
break ;
}
}
return WINED3D_OK ;
2006-04-17 17:25:55 +02:00
}
2006-06-10 13:15:32 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_PrivateSetup ( IWineD3DSurface * iface ) {
2006-05-19 00:13:29 +02:00
/** Check against the maximum texture sizes supported by the video card **/
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " %p \n " , This ) ;
if ( ( This - > pow2Width > GL_LIMITS ( texture_size ) | | This - > pow2Height > GL_LIMITS ( texture_size ) ) & & ! ( This - > resource . usage & ( WINED3DUSAGE_RENDERTARGET | WINED3DUSAGE_DEPTHSTENCIL ) ) ) {
/* one of three options
1 : Do the same as we do with nonpow 2 and scale the texture , ( any texture ops would require the texture to be scaled which is potentially slow )
2 : Set the texture to the maxium size ( bad idea )
3 : WARN and return WINED3DERR_NOTAVAILABLE ;
4 : Create the surface , but allow it to be used only for DirectDraw Blts . Some apps ( e . g . Swat 3 ) create textures with a Height of 16 and a Width > 3000 and blt 16 x16 letter areas from them to the render target .
*/
WARN ( " (%p) Creating an oversized surface \n " , This ) ;
This - > Flags | = SFLAG_OVERSIZE ;
/* This will be initialized on the first blt */
This - > glRect . left = 0 ;
This - > glRect . top = 0 ;
This - > glRect . right = 0 ;
This - > glRect . bottom = 0 ;
} else {
/* No oversize, gl rect is the full texture size */
This - > Flags & = ~ SFLAG_OVERSIZE ;
This - > glRect . left = 0 ;
This - > glRect . top = 0 ;
This - > glRect . right = This - > pow2Width ;
This - > glRect . bottom = This - > pow2Height ;
}
2006-04-17 17:25:55 +02:00
return WINED3D_OK ;
}
2006-05-06 17:39:45 +02:00
DWORD WINAPI IWineD3DSurfaceImpl_GetPitch ( IWineD3DSurface * iface ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
DWORD ret ;
TRACE ( " (%p) \n " , This ) ;
/* DXTn formats don't have exact pitches as they are to the new row of blocks,
where each block is 4 x4 pixels , 8 bytes ( dxt1 ) and 16 bytes ( dxt2 / 3 / 4 / 5 )
ie pitch = ( width / 4 ) * bytes per block */
if ( This - > resource . format = = WINED3DFMT_DXT1 ) /* DXT1 is 8 bytes per block */
2006-12-03 04:27:36 +01:00
ret = ( ( This - > currentDesc . Width + 3 ) > > 2 ) < < 3 ;
2006-05-06 17:39:45 +02:00
else if ( This - > resource . format = = WINED3DFMT_DXT2 | | This - > resource . format = = WINED3DFMT_DXT3 | |
This - > resource . format = = WINED3DFMT_DXT4 | | This - > resource . format = = WINED3DFMT_DXT5 ) /* DXT2/3/4/5 is 16 bytes per block */
2006-12-03 04:27:36 +01:00
ret = ( ( This - > currentDesc . Width + 3 ) > > 2 ) < < 4 ;
2006-05-06 17:39:45 +02:00
else {
2007-02-13 20:21:48 +01:00
ret = This - > bytesPerPixel * This - > currentDesc . Width ; /* Bytes / row */
2006-09-24 09:49:05 +02:00
/* Surfaces are 32 bit aligned */
2006-10-05 21:36:19 +02:00
ret = ( ret + SURFACE_ALIGNMENT - 1 ) & ~ ( SURFACE_ALIGNMENT - 1 ) ;
2006-05-06 17:39:45 +02:00
}
2006-10-01 05:20:10 +02:00
TRACE ( " (%p) Returning %d \n " , This , ret ) ;
2006-05-06 17:39:45 +02:00
return ret ;
}
2006-10-05 16:37:42 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_SetOverlayPosition ( IWineD3DSurface * iface , LONG X , LONG Y ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
FIXME ( " (%p)->(%d,%d) Stub! \n " , This , X , Y ) ;
2006-10-05 17:56:54 +02:00
if ( ! ( This - > resource . usage & WINED3DUSAGE_OVERLAY ) )
{
TRACE ( " (%p): Not an overlay surface \n " , This ) ;
return DDERR_NOTAOVERLAYSURFACE ;
}
2006-10-05 16:37:42 +02:00
return WINED3D_OK ;
}
2006-10-05 16:40:56 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_GetOverlayPosition ( IWineD3DSurface * iface , LONG * X , LONG * Y ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
FIXME ( " (%p)->(%p,%p) Stub! \n " , This , X , Y ) ;
2006-10-05 17:56:54 +02:00
if ( ! ( This - > resource . usage & WINED3DUSAGE_OVERLAY ) )
{
TRACE ( " (%p): Not an overlay surface \n " , This ) ;
return DDERR_NOTAOVERLAYSURFACE ;
}
2006-10-05 16:40:56 +02:00
return WINED3D_OK ;
}
2006-10-05 16:41:19 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_UpdateOverlayZOrder ( IWineD3DSurface * iface , DWORD Flags , IWineD3DSurface * Ref ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DSurfaceImpl * RefImpl = ( IWineD3DSurfaceImpl * ) Ref ;
FIXME ( " (%p)->(%08x,%p) Stub! \n " , This , Flags , RefImpl ) ;
2006-10-05 17:56:54 +02:00
if ( ! ( This - > resource . usage & WINED3DUSAGE_OVERLAY ) )
{
TRACE ( " (%p): Not an overlay surface \n " , This ) ;
return DDERR_NOTAOVERLAYSURFACE ;
}
2006-10-05 16:41:19 +02:00
return WINED3D_OK ;
}
2006-10-05 16:43:26 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_UpdateOverlay ( IWineD3DSurface * iface , RECT * SrcRect , IWineD3DSurface * DstSurface , RECT * DstRect , DWORD Flags , WINEDDOVERLAYFX * FX ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DSurfaceImpl * Dst = ( IWineD3DSurfaceImpl * ) DstSurface ;
FIXME ( " (%p)->(%p, %p, %p, %08x, %p) \n " , This , SrcRect , Dst , DstRect , Flags , FX ) ;
2006-10-05 17:56:54 +02:00
if ( ! ( This - > resource . usage & WINED3DUSAGE_OVERLAY ) )
{
TRACE ( " (%p): Not an overlay surface \n " , This ) ;
return DDERR_NOTAOVERLAYSURFACE ;
}
2006-10-05 16:43:26 +02:00
return WINED3D_OK ;
}
2005-06-06 21:50:35 +02:00
const IWineD3DSurfaceVtbl IWineD3DSurface_Vtbl =
2005-01-09 18:37:02 +01:00
{
2005-03-14 11:12:52 +01:00
/* IUnknown */
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl_QueryInterface ,
IWineD3DSurfaceImpl_AddRef ,
IWineD3DSurfaceImpl_Release ,
2005-03-29 21:01:00 +02:00
/* IWineD3DResource */
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl_GetParent ,
IWineD3DSurfaceImpl_GetDevice ,
IWineD3DSurfaceImpl_SetPrivateData ,
IWineD3DSurfaceImpl_GetPrivateData ,
IWineD3DSurfaceImpl_FreePrivateData ,
IWineD3DSurfaceImpl_SetPriority ,
IWineD3DSurfaceImpl_GetPriority ,
IWineD3DSurfaceImpl_PreLoad ,
IWineD3DSurfaceImpl_GetType ,
2005-07-13 16:15:54 +02:00
/* IWineD3DSurface */
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl_GetContainer ,
IWineD3DSurfaceImpl_GetDesc ,
IWineD3DSurfaceImpl_LockRect ,
IWineD3DSurfaceImpl_UnlockRect ,
IWineD3DSurfaceImpl_GetDC ,
IWineD3DSurfaceImpl_ReleaseDC ,
2006-04-12 22:32:44 +02:00
IWineD3DSurfaceImpl_Flip ,
2006-04-14 19:15:46 +02:00
IWineD3DSurfaceImpl_Blt ,
2006-04-17 17:25:55 +02:00
IWineD3DSurfaceImpl_GetBltStatus ,
IWineD3DSurfaceImpl_GetFlipStatus ,
IWineD3DSurfaceImpl_IsLost ,
IWineD3DSurfaceImpl_Restore ,
IWineD3DSurfaceImpl_BltFast ,
IWineD3DSurfaceImpl_GetPalette ,
IWineD3DSurfaceImpl_SetPalette ,
IWineD3DSurfaceImpl_RealizePalette ,
IWineD3DSurfaceImpl_SetColorKey ,
2006-05-06 17:39:45 +02:00
IWineD3DSurfaceImpl_GetPitch ,
2006-07-25 16:38:02 +02:00
IWineD3DSurfaceImpl_SetMem ,
2006-10-05 16:37:42 +02:00
IWineD3DSurfaceImpl_SetOverlayPosition ,
2006-10-05 16:40:56 +02:00
IWineD3DSurfaceImpl_GetOverlayPosition ,
2006-10-05 16:41:19 +02:00
IWineD3DSurfaceImpl_UpdateOverlayZOrder ,
2006-10-05 16:43:26 +02:00
IWineD3DSurfaceImpl_UpdateOverlay ,
2005-01-09 18:37:02 +01:00
/* Internal use: */
IWineD3DSurfaceImpl_CleanDirtyRect ,
IWineD3DSurfaceImpl_AddDirtyRect ,
IWineD3DSurfaceImpl_LoadTexture ,
2005-03-14 11:12:52 +01:00
IWineD3DSurfaceImpl_SaveSnapshot ,
IWineD3DSurfaceImpl_SetContainer ,
2005-07-11 22:38:27 +02:00
IWineD3DSurfaceImpl_SetPBufferState ,
IWineD3DSurfaceImpl_SetGlTextureDesc ,
IWineD3DSurfaceImpl_GetGlDesc ,
2005-11-02 21:01:54 +01:00
IWineD3DSurfaceImpl_GetData ,
2006-04-17 17:25:55 +02:00
IWineD3DSurfaceImpl_SetFormat ,
IWineD3DSurfaceImpl_PrivateSetup
2005-01-09 18:37:02 +01:00
} ;