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
2007-04-09 01:53:32 +02:00
* Copyright 2007 Henri Verbeet
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 ) ;
2007-06-09 14:27:41 +02:00
# define GLINFO_LOCATION This->resource.wineD3DDevice->adapter->gl_info
2005-01-09 18:37:02 +01:00
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 ) {
2007-06-07 01:08:26 +02:00
if ( ! This - > resource . allocatedMemory ) This - > resource . allocatedMemory = HeapAlloc ( GetProcessHeap ( ) , 0 , This - > resource . size + 4 ) ;
2006-09-11 18:51:18 +02: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 ) {
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 ) ;
GL_EXTCALL ( glGetCompressedTexImageARB ( This - > glDescription . target , This - > glDescription . level , This - > resource . allocatedMemory ) ) ;
checkGLcall ( " glGetCompressedTexImageARB() " ) ;
}
} 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 ) {
2007-06-08 22:28:04 +02:00
unsigned char alignment = This - > resource . wineD3DDevice - > surface_alignment ;
2007-02-13 20:21:48 +01:00
src_pitch = This - > bytesPerPixel * This - > pow2Width ;
dst_pitch = IWineD3DSurface_GetPitch ( ( IWineD3DSurface * ) This ) ;
2007-06-08 22:28:04 +02:00
src_pitch = ( src_pitch + alignment - 1 ) & ~ ( alignment - 1 ) ;
2007-02-13 20:21:48 +01:00
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
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() " ) ;
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
}
}
2007-06-07 01:08:26 +02:00
/* Surface has now been downloaded */
This - > Flags | = SFLAG_INSYSMEM ;
2006-09-11 18:51:18 +02:00
}
2007-06-07 01:08:26 +02:00
static void surface_upload_data ( IWineD3DSurfaceImpl * This , GLenum internal , GLsizei width , GLsizei height , GLenum format , GLenum type , const GLvoid * data ) {
2006-09-11 18:51:18 +02: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 ) {
if ( ! GL_SUPPORT ( EXT_TEXTURE_COMPRESSION_S3TC ) ) {
FIXME ( " Using DXT1/3/5 without advertized support \n " ) ;
} else {
2007-03-31 23:02:37 +02:00
if ( GL_SUPPORT ( APPLE_CLIENT_STORAGE ) ) {
/* Neither NONPOW2, DIBSECTION nor OVERSIZE flags can be set on compressed textures */
This - > Flags | = SFLAG_CLIENT ;
}
2006-09-11 18:51:18 +02:00
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
*/
2007-06-07 01:08:26 +02:00
GL_EXTCALL ( glCompressedTexImage2DARB ( This - > glDescription . target , This - > glDescription . level , internal ,
2007-02-13 20:28:25 +01:00
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 ) {
2007-03-31 23:02:37 +02:00
BOOL enable_client_storage = FALSE ;
2006-09-11 18:51:18 +02:00
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 ( ) ;
2007-03-31 23:02:37 +02:00
if ( GL_SUPPORT ( APPLE_CLIENT_STORAGE ) ) {
if ( This - > Flags & ( SFLAG_NONPOW2 | SFLAG_DIBSECTION | SFLAG_OVERSIZE | SFLAG_CONVERTED ) | | This - > resource . allocatedMemory = = NULL ) {
/* In some cases we want to disable client storage.
* SFLAG_NONPOW2 has a bigger opengl texture than the client memory , and different pitches
* SFLAG_DIBSECTION : Dibsections may have read / write protections on the memory . Avoid issues . . .
* SFLAG_OVERSIZE : The gl texture is smaller than the allocated memory
* SFLAG_CONVERTED : The conversion destination memory is freed after loading the surface
* allocatedMemory = = NULL : Not defined in the extension . Seems to disable client storage effectively
*/
glPixelStorei ( GL_UNPACK_CLIENT_STORAGE_APPLE , GL_FALSE ) ;
checkGLcall ( " glPixelStorei(GL_UNPACK_CLIENT_STORAGE_APPLE, GL_FALSE) " ) ;
This - > Flags & = SFLAG_CLIENT ;
enable_client_storage = TRUE ;
} else {
This - > Flags | = SFLAG_CLIENT ;
/* Below point opengl to our allocated texture memory */
}
}
glTexImage2D ( This - > glDescription . target , This - > glDescription . level , internal , width , height , 0 , format , type ,
This - > Flags & SFLAG_CLIENT ? This - > resource . allocatedMemory : NULL ) ;
2006-09-11 18:51:18 +02:00
checkGLcall ( " glTexImage2D " ) ;
2007-03-31 23:02:37 +02:00
if ( enable_client_storage ) {
glPixelStorei ( GL_UNPACK_CLIENT_STORAGE_APPLE , GL_TRUE ) ;
checkGLcall ( " glPixelStorei(GL_UNPACK_CLIENT_STORAGE_APPLE, GL_TRUE) " ) ;
}
2006-09-11 18:51:18 +02:00
LEAVE_GL ( ) ;
2007-04-22 02:02:29 +02:00
This - > Flags | = SFLAG_ALLOCATED ;
2006-09-11 18:51:18 +02:00
}
2007-04-09 01:53:32 +02:00
/* In D3D the depth stencil dimensions have to be greater than or equal to the
* render target dimensions . With FBOs , the dimensions have to be an exact match . */
/* TODO: We should synchronize the renderbuffer's content with the texture's content. */
void surface_set_compatible_renderbuffer ( IWineD3DSurface * iface , unsigned int width , unsigned int height ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
renderbuffer_entry_t * entry ;
GLuint renderbuffer = 0 ;
unsigned int src_width , src_height ;
src_width = This - > pow2Width ;
src_height = This - > pow2Height ;
/* A depth stencil smaller than the render target is not valid */
if ( width > src_width | | height > src_height ) return ;
/* Remove any renderbuffer set if the sizes match */
if ( width = = src_width & & height = = src_height ) {
This - > current_renderbuffer = NULL ;
return ;
}
/* Look if we've already got a renderbuffer of the correct dimensions */
LIST_FOR_EACH_ENTRY ( entry , & This - > renderbuffers , renderbuffer_entry_t , entry ) {
if ( entry - > width = = width & & entry - > height = = height ) {
renderbuffer = entry - > id ;
This - > current_renderbuffer = entry ;
break ;
}
}
if ( ! renderbuffer ) {
2007-07-23 11:13:37 +02:00
const GlPixelFormatDesc * glDesc ;
2007-07-27 13:22:54 +02:00
getFormatDescEntry ( This - > resource . format , & GLINFO_LOCATION , & glDesc ) ;
2007-04-09 01:53:32 +02:00
GL_EXTCALL ( glGenRenderbuffersEXT ( 1 , & renderbuffer ) ) ;
GL_EXTCALL ( glBindRenderbufferEXT ( GL_RENDERBUFFER_EXT , renderbuffer ) ) ;
2007-07-23 11:13:37 +02:00
GL_EXTCALL ( glRenderbufferStorageEXT ( GL_RENDERBUFFER_EXT , glDesc - > glFormat , width , height ) ) ;
2007-04-09 01:53:32 +02:00
entry = HeapAlloc ( GetProcessHeap ( ) , 0 , sizeof ( renderbuffer_entry_t ) ) ;
entry - > width = width ;
entry - > height = height ;
entry - > id = renderbuffer ;
list_add_head ( & This - > renderbuffers , & entry - > entry ) ;
This - > current_renderbuffer = entry ;
}
checkGLcall ( " set_compatible_renderbuffer " ) ;
}
2007-04-16 21:20:25 +02:00
GLenum surface_get_gl_buffer ( IWineD3DSurface * iface , IWineD3DSwapChain * swapchain ) {
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DSwapChainImpl * swapchain_impl = ( IWineD3DSwapChainImpl * ) swapchain ;
TRACE ( " (%p) : swapchain %p \n " , This , swapchain ) ;
if ( swapchain_impl - > backBuffer & & swapchain_impl - > backBuffer [ 0 ] = = iface ) {
TRACE ( " Returning GL_BACK \n " ) ;
return GL_BACK ;
} else if ( swapchain_impl - > frontBuffer = = iface ) {
TRACE ( " Returning GL_FRONT \n " ) ;
return GL_FRONT ;
}
FIXME ( " Higher back buffer, returning GL_BACK \n " ) ;
return GL_BACK ;
}
2007-04-09 01:53:32 +02:00
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 ;
2007-04-09 01:53:32 +02:00
renderbuffer_entry_t * entry , * entry2 ;
2005-07-11 22:38:27 +02:00
TRACE ( " (%p) : cleaning up \n " , This ) ;
2007-02-17 18:06:30 +01:00
if ( iface = = device - > lastActiveRenderTarget ) {
IWineD3DSwapChainImpl * swapchain = device - > swapchains ? ( IWineD3DSwapChainImpl * ) device - > swapchains [ 0 ] : NULL ;
TRACE ( " Last active render target destroyed \n " ) ;
/* Find a replacement surface for the currently active back buffer. The context manager does not do NULL
* checks , so switch to a valid target as long as the currently set surface is still valid . Use the
* surface of the implicit swpchain . If that is the same as the destroyed surface the device is destroyed
* and the lastActiveRenderTarget member shouldn ' t matter
*/
if ( swapchain ) {
if ( swapchain - > backBuffer & & swapchain - > backBuffer [ 0 ] ! = iface ) {
TRACE ( " Activating primary back buffer \n " ) ;
ActivateContext ( device , swapchain - > backBuffer [ 0 ] , CTXUSAGE_RESOURCELOAD ) ;
} else if ( ! swapchain - > backBuffer & & swapchain - > frontBuffer ! = iface ) {
/* Single buffering environment */
TRACE ( " Activating primary front buffer \n " ) ;
ActivateContext ( device , swapchain - > frontBuffer , CTXUSAGE_RESOURCELOAD ) ;
} else {
2007-02-20 15:57:10 +01:00
TRACE ( " Device is being destroyed, setting lastActiveRenderTarget = 0xdeadbabe \n " ) ;
/* Implicit render target destroyed, that means the device is being destroyed
2007-02-17 18:06:30 +01:00
* whatever we set here , it shouldn ' t matter
*/
device - > lastActiveRenderTarget = ( IWineD3DSurface * ) 0xdeadbabe ;
}
} else {
/* May happen during ddraw uninitialization */
TRACE ( " Render target set, but swapchain does not exist! \n " ) ;
device - > lastActiveRenderTarget = ( IWineD3DSurface * ) 0xdeadcafe ;
}
}
2005-07-11 22:38:27 +02:00
if ( This - > glDescription . textureName ! = 0 ) { /* release the openGL texture.. */
2007-03-17 23:00:39 +01:00
/* Need a context to destroy the texture. Use the currently active render target, but only if
* the primary render target exists . Otherwise lastActiveRenderTarget is garbage , see above .
* When destroying the primary rt , Uninit3D will activate a context before doing anything
*/
if ( device - > render_targets [ 0 ] ) {
ActivateContext ( device , device - > lastActiveRenderTarget , CTXUSAGE_RESOURCELOAD ) ;
}
2005-07-11 22:38:27 +02:00
TRACE ( " Deleting texture %d \n " , This - > glDescription . textureName ) ;
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
2005-07-11 22:38:27 +02:00
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
2007-02-19 15:23:36 +01:00
HeapFree ( GetProcessHeap ( ) , 0 , This - > palette9 ) ;
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-04-09 01:53:32 +02:00
LIST_FOR_EACH_ENTRY_SAFE ( entry , entry2 , & This - > renderbuffers , renderbuffer_entry_t , entry ) {
GL_EXTCALL ( glDeleteRenderbuffersEXT ( 1 , & entry - > id ) ) ;
HeapFree ( GetProcessHeap ( ) , 0 , entry ) ;
}
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: check for locks */
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
IWineD3DBaseTexture * baseTexture = NULL ;
2007-03-17 23:00:39 +01:00
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
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 ) {
2007-06-06 00:21:08 +02:00
TRACE ( " Passing to container \n " ) ;
2005-03-14 11:12:52 +01:00
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 ) ;
2007-03-17 23:00:39 +01:00
if ( ! device - > isInDraw ) {
ActivateContext ( device , device - > lastActiveRenderTarget , CTXUSAGE_RESOURCELOAD ) ;
}
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
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 " ) ;
2007-06-07 01:08:26 +02:00
IWineD3DSurface_LoadTexture ( iface , FALSE ) ;
2006-09-11 18:51:18 +02:00
/* 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
}
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 ) {
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ SFLAG_INTEXTURE ;
2005-07-11 22:38:27 +02:00
IWineD3DSurface_AddDirtyRect ( iface , NULL ) ;
}
This - > glDescription . textureName = textureName ;
This - > glDescription . target = target ;
2007-04-22 02:02:29 +02:00
This - > Flags & = ~ SFLAG_ALLOCATED ;
2005-07-11 22:38:27 +02:00
}
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 ;
2007-06-06 00:21:08 +02:00
/* 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 future */
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 ) ;
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ SFLAG_INSYSMEM ; /* 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
2007-03-13 00:07:12 +01:00
/* DXTn textures are based on compressed blocks of 4x4 pixels, each
* 16 bytes large ( 8 bytes in case of DXT1 ) . Because of that Pitch has
* slightly different meaning compared to regular textures . For DXTn
* textures Pitch is the size of a row of blocks , 4 high and " width "
* long . The x offset is calculated differently as well , since moving 4
* pixels to the right actually moves an entire 4 x4 block to right , ie
* 16 bytes ( 8 in case of DXT1 ) . */
if ( This - > resource . format = = WINED3DFMT_DXT1 ) {
pLockedRect - > pBits = This - > resource . allocatedMemory + ( pLockedRect - > Pitch * pRect - > top / 4 ) + ( pRect - > left * 2 ) ;
} else if ( This - > resource . format = = WINED3DFMT_DXT2 | | This - > resource . format = = WINED3DFMT_DXT3
| | This - > resource . format = = WINED3DFMT_DXT4 | | This - > resource . format = = WINED3DFMT_DXT5 ) {
pLockedRect - > pBits = This - > resource . allocatedMemory + ( pLockedRect - > Pitch * pRect - > top / 4 ) + ( pRect - > left * 4 ) ;
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 ) {
2007-06-06 00:21:08 +02:00
TRACE ( " Surface is locked regularly, not freeing the system memory copy any more \n " ) ;
2007-02-13 20:21:44 +01:00
This - > Flags | = SFLAG_DYNLOCK ;
}
}
2005-07-13 13:34:55 +02:00
2007-08-01 22:48:42 +02:00
if ( Flags & WINED3DLOCK_DISCARD ) {
/* Set SFLAG_INSYSMEM, so we'll never try to download the data from the texture. */
TRACE ( " WINED3DLOCK_DISCARD flag passed, marking local copy as up to date \n " ) ;
This - > Flags | = SFLAG_INSYSMEM ;
}
if ( This - > Flags & SFLAG_INSYSMEM ) {
TRACE ( " Local copy is up to date, not downloading data \n " ) ;
2007-02-13 20:21:44 +01:00
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 .
*/
ActivateContext ( myDevice , iface , CTXUSAGE_BLIT ) ;
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
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 " ) ;
2007-02-27 21:32:15 +01:00
glReadBuffer ( myDevice - > offscreenBuffer ) ;
2007-02-13 20:24:00 +01:00
srcIsUpsideDown = TRUE ;
2007-02-13 20:21:44 +01:00
} else {
2007-04-16 21:20:32 +02:00
GLenum buffer = surface_get_gl_buffer ( iface , ( IWineD3DSwapChain * ) swapchain ) ;
TRACE ( " Locking %#x buffer \n " , buffer ) ;
glReadBuffer ( buffer ) ;
checkGLcall ( " glReadBuffer " ) ;
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 ) {
2007-03-06 21:47:45 +01:00
This - > Flags | = SFLAG_INSYSMEM ;
2007-02-13 20:24:00 +01:00
}
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
2007-03-06 21:47:45 +01:00
* gl texture ( next path ) , or in local memory ( early return because of set SFLAG_INSYSMEM above ) . If
2007-02-13 20:21:44 +01:00
* 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 */
2007-06-06 00:21:08 +02:00
TRACE ( " locking an ordinary surface \n " ) ;
2005-07-07 22:35:05 +02:00
2007-02-13 20:21:44 +01:00
if ( 0 ! = This - > glDescription . textureName ) {
/* Now I have to copy thing bits back */
2007-03-17 23:00:39 +01:00
if ( myDevice - > createParms . BehaviorFlags & WINED3DCREATE_MULTITHREADED ) {
ActivateContext ( myDevice , myDevice - > lastActiveRenderTarget , CTXUSAGE_RESOURCELOAD ) ;
}
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
2007-02-13 20:21:44 +01:00
/* 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 ) ) {
GL_EXTCALL ( glActiveTextureARB ( GL_TEXTURE0_ARB ) ) ;
checkGLcall ( " glActiveTextureARB " ) ;
}
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 ) ;
2007-03-17 23:00:39 +01:00
LEAVE_GL ( ) ;
2005-07-07 22:35:05 +02:00
}
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
}
}
2007-03-06 21:47:45 +01:00
TRACE ( " returning memory@%p, pitch(%d) dirtyfied(%d) \n " , pLockedRect - > pBits , pLockedRect - > Pitch ,
This - > Flags & ( SFLAG_INDRAWABLE | SFLAG_INTEXTURE ) ? 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-03-06 21:47:45 +01:00
TRACE ( " (%p) : dirtyfied(%d) \n " , This , This - > Flags & ( SFLAG_INDRAWABLE | SFLAG_INTEXTURE ) ? 0 : 1 ) ;
2005-07-07 22:35:05 +02:00
2007-03-06 21:47:45 +01:00
if ( This - > Flags & ( SFLAG_INDRAWABLE | SFLAG_INTEXTURE ) ) {
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 ) ;
2007-08-21 16:52:01 +02:00
if ( swapchain | | ( myDevice - > render_targets & & iface = = myDevice - > render_targets [ 0 ] ) ) {
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 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 */
ActivateContext ( myDevice , iface , CTXUSAGE_BLIT ) ;
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
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 " ) ;
2007-02-27 21:32:15 +01:00
glDrawBuffer ( myDevice - > offscreenBuffer ) ;
checkGLcall ( " glDrawBuffer(myDevice->offscreenBuffer) " ) ;
2007-02-13 20:21:44 +01:00
} else {
2007-04-16 21:20:32 +02:00
GLenum buffer = surface_get_gl_buffer ( iface , ( IWineD3DSwapChain * ) swapchain ) ;
TRACE ( " Unlocking %#x buffer \n " , buffer ) ;
glDrawBuffer ( buffer ) ;
checkGLcall ( " glDrawBuffer " ) ;
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 ;
}
2007-02-27 21:32:15 +01:00
if ( ! swapchain ) {
glDrawBuffer ( myDevice - > offscreenBuffer ) ;
checkGLcall ( " glDrawBuffer(myDevice->offscreenBuffer) " ) ;
} else if ( swapchain - > backBuffer ) {
2007-02-13 20:21:44 +01:00
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-03-06 21:47:45 +01:00
This - > dirtyRect . left = This - > currentDesc . Width ;
This - > dirtyRect . top = This - > currentDesc . Height ;
This - > dirtyRect . right = 0 ;
This - > dirtyRect . bottom = 0 ;
This - > Flags | = SFLAG_INDRAWABLE ;
2007-02-13 20:21:44 +01:00
} 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 ] ;
2007-07-27 13:22:54 +02:00
const StaticPixelFormatDesc * formatEntry = getFormatDescEntry ( This - > resource . format , NULL , NULL ) ;
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 " ) ;
2007-04-14 22:44:55 +02:00
return WINEDDERR_NODC ;
2006-07-25 16:38:02 +02:00
}
2006-05-06 18:08:33 +02:00
/* Give more detailed info for ddraw */
if ( This - > Flags & SFLAG_DCINUSE )
2007-04-14 22:44:55 +02:00
return WINEDDERR_DCALREADYCREATED ;
2006-05-06 18:08:33 +02:00
/* 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 ;
2007-03-31 23:02:37 +02:00
void * oldmem = This - > resource . allocatedMemory ;
2006-07-16 10:08:26 +02:00
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-06-08 22:28:04 +02:00
/* TODO: Is there a nicer way to force a specific alignment? (8 byte for ddraw) */
b_info - > bmiHeader . biWidth = IWineD3DSurface_GetPitch ( iface ) / This - > bytesPerPixel ;
2007-02-13 20:21:48 +01:00
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 */
2006-05-23 00:48:54 +02:00
} else {
/* This is to make LockRect read the gl Texture although memory is allocated */
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ SFLAG_INSYSMEM ;
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 ;
2007-03-31 23:02:37 +02:00
if ( This - > Flags & SFLAG_CLIENT ) {
IWineD3DSurface_PreLoad ( iface ) ;
}
HeapFree ( GetProcessHeap ( ) , 0 , oldmem ) ;
2006-05-06 18:08:33 +02:00
}
/* 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
2007-08-11 20:02:01 +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 , BOOL srgb_mode ) {
2007-04-14 22:44:55 +02:00
BOOL colorkey_active = need_alpha_ck & & ( This - > CKeyFlags & WINEDDSD_CKSRCBLT ) ;
2007-07-23 11:13:37 +02:00
const GlPixelFormatDesc * glDesc ;
2007-07-27 13:22:54 +02:00
getFormatDescEntry ( This - > resource . format , & GLINFO_LOCATION , & glDesc ) ;
2006-05-22 23:49:09 +02:00
/* Default values: From the surface */
2007-07-23 11:13:37 +02:00
* format = glDesc - > glFormat ;
* internal = srgb_mode ? glDesc - > glGammaInternal : glDesc - > glInternal ;
* type = glDesc - > 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 ;
2007-05-22 03:29:44 +02:00
case WINED3DFMT_X1R5G5B5 :
if ( colorkey_active ) {
* convert = CONVERT_CK_5551 ;
* format = GL_BGRA ;
* internal = GL_RGBA ;
* type = GL_UNSIGNED_SHORT_1_5_5_5_REV ;
}
break ;
2006-05-22 23:49:09 +02:00
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 :
2007-03-12 13:11:04 +01:00
if ( GL_SUPPORT ( NV_TEXTURE_SHADER3 ) ) break ;
2007-03-20 23:42:37 +01:00
else if ( GL_SUPPORT ( ATI_ENVMAP_BUMPMAP ) ) {
* format = GL_DUDV_ATI ;
* internal = GL_DU8DV8_ATI ;
* type = GL_BYTE ;
/* No conversion - Just change the gl type */
break ;
}
2007-02-15 03:01:44 +01:00
* convert = CONVERT_V8U8 ;
* format = GL_BGR ;
* internal = GL_RGB8 ;
2007-03-21 00:06:04 +01:00
* type = GL_UNSIGNED_BYTE ;
2007-02-15 03:01:44 +01:00
* target_bpp = 3 ;
break ;
2007-03-12 13:11:04 +01:00
case WINED3DFMT_X8L8V8U8 :
if ( GL_SUPPORT ( NV_TEXTURE_SHADER3 ) ) break ;
2007-03-21 00:06:04 +01:00
* convert = CONVERT_X8L8V8U8 ;
2007-03-12 13:11:04 +01:00
* format = GL_BGRA ;
* internal = GL_RGBA8 ;
2007-03-21 00:06:04 +01:00
* type = GL_UNSIGNED_BYTE ;
2007-03-12 13:11:04 +01:00
* target_bpp = 4 ;
2007-03-20 23:42:37 +01:00
/* Not supported by GL_ATI_envmap_bumpmap */
2007-03-12 13:11:04 +01:00
break ;
case WINED3DFMT_Q8W8V8U8 :
if ( GL_SUPPORT ( NV_TEXTURE_SHADER3 ) ) break ;
2007-03-21 00:06:04 +01:00
* convert = CONVERT_Q8W8V8U8 ;
2007-03-12 13:11:04 +01:00
* format = GL_BGRA ;
* internal = GL_RGBA8 ;
2007-03-21 00:06:04 +01:00
* type = GL_UNSIGNED_BYTE ;
2007-03-12 13:11:04 +01:00
* target_bpp = 4 ;
2007-03-20 23:42:37 +01:00
/* Not supported by GL_ATI_envmap_bumpmap */
2007-03-12 13:11:04 +01:00
break ;
case WINED3DFMT_V16U16 :
if ( GL_SUPPORT ( NV_TEXTURE_SHADER3 ) ) break ;
2007-03-21 00:06:04 +01:00
* convert = CONVERT_V16U16 ;
* format = GL_BGR ;
* internal = GL_RGB16 ;
2007-03-12 13:11:04 +01:00
* type = GL_SHORT ;
2007-03-21 00:06:04 +01:00
* target_bpp = 6 ;
2007-03-20 23:42:37 +01:00
/* What should I do here about GL_ATI_envmap_bumpmap?
* Convert it or allow data loss by loading it into a 8 bit / channel texture ?
*/
2007-03-12 13:11:04 +01:00
break ;
2007-08-11 20:02:01 +02:00
case WINED3DFMT_A4L4 :
/* A4L4 exists as an internal gl format, but for some reason there is not
* format + type combination to load it . Thus convert it to A8L8 , then load it
* with A4L4 internal , but A8L8 format + type
*/
* convert = CONVERT_A4L4 ;
* format = GL_LUMINANCE_ALPHA ;
* internal = GL_LUMINANCE4_ALPHA4 ;
* type = GL_UNSIGNED_BYTE ;
* target_bpp = 2 ;
break ;
2007-08-12 16:24:29 +02:00
case WINED3DFMT_R32F :
/* Can be loaded in theory with fmt=GL_RED, type=GL_FLOAT, but this fails. The reason
* is that D3D expects the undefined green , blue and alpha channels to return 1.0
* when sampling , but OpenGL sets green and blue to 0.0 instead . Thus we have to inject
* 1.0 instead .
*
* The alpha channel defaults to 1.0 in opengl , so nothing has to be done about it .
*/
* convert = CONVERT_R32F ;
* format = GL_RGB ;
* internal = GL_RGB32F_ARB ;
* type = GL_FLOAT ;
* target_bpp = 12 ;
break ;
case WINED3DFMT_R16F :
/* Simmilar to R32F */
* convert = CONVERT_R16F ;
* format = GL_RGB ;
* internal = GL_RGB16F_ARB ;
* type = GL_HALF_FLOAT_ARB ;
* target_bpp = 6 ;
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 ;
2007-04-29 15:51:47 +02:00
} else if ( pal - > Flags & WINEDDPCAPS_ALPHA ) {
table [ i ] [ 3 ] = pal - > palents [ i ] . peFlags ;
2006-05-22 23:49:09 +02:00
} 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-05-22 03:29:44 +02:00
case CONVERT_CK_5551 :
{
/* Converting X1R5G5B5 format to R5G5B5A1 to emulate color-keying. */
unsigned int x , y ;
WORD * Source ;
WORD * Dest ;
TRACE ( " Color keyed 5551 \n " ) ;
for ( y = 0 ; y < height ; y + + ) {
Source = ( WORD * ) ( src + y * pitch ) ;
Dest = ( WORD * ) ( dst + y * outpitch ) ;
for ( x = 0 ; x < width ; x + + ) {
WORD color = * Source + + ;
* Dest = color ;
if ( ( color < surf - > SrcBltCKey . dwColorSpaceLowValue ) | |
( color > surf - > SrcBltCKey . dwColorSpaceHighValue ) ) {
* Dest | = ( 1 < < 15 ) ;
}
else {
* Dest & = ~ ( 1 < < 15 ) ;
}
Dest + + ;
}
}
}
break ;
2007-02-15 03:01:44 +01:00
case CONVERT_V8U8 :
{
unsigned int x , y ;
short * Source ;
2007-03-05 21:40:47 +01:00
unsigned char * Dest ;
2007-02-15 03:01:44 +01:00
for ( y = 0 ; y < height ; y + + ) {
Source = ( short * ) ( src + y * pitch ) ;
2007-03-05 21:40:47 +01:00
Dest = ( unsigned char * ) ( dst + y * outpitch ) ;
2007-02-15 03:01:44 +01:00
for ( x = 0 ; x < width ; x + + ) {
long color = ( * Source + + ) ;
2007-03-05 21:40:47 +01:00
/* B */ Dest [ 0 ] = 0xff ;
/* G */ Dest [ 1 ] = ( color > > 8 ) + 128 ; /* V */
/* R */ Dest [ 2 ] = ( color ) + 128 ; /* U */
2007-02-15 03:01:44 +01:00
Dest + = 3 ;
}
}
break ;
}
2007-03-21 00:06:04 +01:00
case CONVERT_Q8W8V8U8 :
{
unsigned int x , y ;
DWORD * Source ;
unsigned char * Dest ;
for ( y = 0 ; y < height ; y + + ) {
Source = ( DWORD * ) ( src + y * pitch ) ;
Dest = ( unsigned char * ) ( dst + y * outpitch ) ;
for ( x = 0 ; x < width ; x + + ) {
long color = ( * Source + + ) ;
/* B */ Dest [ 0 ] = ( ( color > > 16 ) & 0xff ) + 128 ; /* W */
/* G */ Dest [ 1 ] = ( ( color > > 8 ) & 0xff ) + 128 ; /* V */
/* R */ Dest [ 2 ] = ( color & 0xff ) + 128 ; /* U */
/* A */ Dest [ 3 ] = ( ( color > > 24 ) & 0xff ) + 128 ; /* Q */
Dest + = 4 ;
}
}
break ;
}
2007-08-11 20:02:01 +02:00
case CONVERT_A4L4 :
{
unsigned int x , y ;
unsigned char * Source ;
unsigned char * Dest ;
for ( y = 0 ; y < height ; y + + ) {
Source = ( unsigned char * ) ( src + y * pitch ) ;
Dest = ( unsigned char * ) ( dst + y * outpitch ) ;
for ( x = 0 ; x < width ; x + + ) {
unsigned char color = ( * Source + + ) ;
/* A */ Dest [ 1 ] = ( color & 0xf0 ) < < 0 ;
/* L */ Dest [ 0 ] = ( color & 0x0f ) < < 4 ;
Dest + = 2 ;
}
}
break ;
}
2007-08-12 16:24:29 +02:00
case CONVERT_R32F :
{
unsigned int x , y ;
float * Source ;
float * Dest ;
for ( y = 0 ; y < height ; y + + ) {
Source = ( float * ) ( src + y * pitch ) ;
Dest = ( float * ) ( dst + y * outpitch ) ;
for ( x = 0 ; x < width ; x + + ) {
float color = ( * Source + + ) ;
Dest [ 0 ] = color ;
Dest [ 1 ] = 1.0 ;
Dest [ 2 ] = 1.0 ;
Dest + = 3 ;
}
}
break ;
}
case CONVERT_R16F :
{
unsigned int x , y ;
WORD * Source ;
WORD * Dest ;
WORD one = 0x3c00 ;
for ( y = 0 ; y < height ; y + + ) {
Source = ( WORD * ) ( src + y * pitch ) ;
Dest = ( WORD * ) ( dst + y * outpitch ) ;
for ( x = 0 ; x < width ; x + + ) {
WORD color = ( * Source + + ) ;
Dest [ 0 ] = color ;
Dest [ 1 ] = one ;
Dest [ 2 ] = one ;
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
*/
2007-03-12 16:50:54 +01:00
static void d3dfmt_p8_upload_palette ( IWineD3DSurface * iface , CONVERT_TYPES convert ) {
2006-08-03 12:39:04 +02:00
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 ;
2007-04-29 15:51:47 +02:00
} else if ( pal - > Flags & WINEDDPCAPS_ALPHA ) {
table [ i ] [ 3 ] = pal - > palents [ i ] . peFlags ;
2006-08-03 12:39:04 +02:00
} else {
table [ i ] [ 3 ] = 0xFF ;
}
}
}
GL_EXTCALL ( glColorTableEXT ( GL_TEXTURE_2D , GL_RGBA , 256 , GL_RGBA , GL_UNSIGNED_BYTE , table ) ) ;
}
2007-02-19 15:23:36 +01:00
static BOOL palette9_changed ( IWineD3DSurfaceImpl * This ) {
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
if ( This - > palette | | ( This - > resource . format ! = WINED3DFMT_P8 & & This - > resource . format ! = WINED3DFMT_A8P8 ) ) {
/* If a ddraw-style palette is attached assume no d3d9 palette change.
* Also the palette isn ' t interesting if the surface format isn ' t P8 or A8P8
*/
return FALSE ;
}
if ( This - > palette9 ) {
if ( memcmp ( This - > palette9 , & device - > palettes [ device - > currentPalette ] , sizeof ( PALETTEENTRY ) * 256 ) = = 0 ) {
return FALSE ;
}
} else {
2007-06-27 00:13:32 +02:00
This - > palette9 = HeapAlloc ( GetProcessHeap ( ) , 0 , sizeof ( PALETTEENTRY ) * 256 ) ;
2007-02-19 15:23:36 +01:00
}
memcpy ( This - > palette9 , & device - > palettes [ device - > currentPalette ] , sizeof ( PALETTEENTRY ) * 256 ) ;
return TRUE ;
}
2007-06-07 01:08:26 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_LoadTexture ( IWineD3DSurface * iface , BOOL srgb_mode ) {
2005-01-09 18:37:02 +01:00
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
2007-04-08 00:56:45 +02:00
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
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
2007-03-06 21:47:45 +01:00
if ( ! ( This - > Flags & SFLAG_INTEXTURE ) ) {
2006-07-25 16:41:41 +02:00
TRACE ( " Reloading because surface is dirty \n " ) ;
} else if ( /* Reload: gl texture has ck, now no ckey is set OR */
2007-04-14 22:44:55 +02:00
( ( This - > Flags & SFLAG_GLCKEY ) & & ( ! ( This - > CKeyFlags & WINEDDSD_CKSRCBLT ) ) ) | |
2006-07-25 16:41:41 +02:00
/* Reload: vice versa OR */
2007-04-14 22:44:55 +02:00
( ( ! ( This - > Flags & SFLAG_GLCKEY ) ) & & ( This - > CKeyFlags & WINEDDSD_CKSRCBLT ) ) | |
2006-07-25 16:41:41 +02:00
/* Also reload: Color key is active AND the color key has changed */
2007-04-14 22:44:55 +02:00
( ( This - > CKeyFlags & WINEDDSD_CKSRCBLT ) & & (
2006-07-26 23:19:04 +02:00
( 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 " ) ;
2007-02-19 15:23:36 +01:00
} else if ( palette9_changed ( This ) ) {
TRACE ( " Reloading surface because the d3d8/9 palette was changed \n " ) ;
2006-07-25 16:41:41 +02:00
} else {
2007-03-06 21:47:45 +01:00
TRACE ( " surface is already in texture \n " ) ;
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2005-07-11 16:25:54 +02:00
}
2007-03-06 21:47:45 +01:00
This - > Flags | = SFLAG_INTEXTURE ;
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 .
*/
2007-03-13 03:44:22 +01:00
if ( This - > resource . pool = = WINED3DPOOL_SCRATCH )
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
2007-06-07 01:08:26 +02:00
d3dfmt_get_conv ( This , TRUE /* We need color keying */ , TRUE /* We will use textures */ , & format , & internal , & type , & convert , & bpp , srgb_mode ) ;
2007-04-07 23:37:41 +02:00
2007-03-06 21:47:45 +01:00
if ( This - > Flags & SFLAG_INDRAWABLE ) {
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 " ) ;
2007-02-27 21:32:15 +01:00
glReadBuffer ( This - > resource . wineD3DDevice - > offscreenBuffer ) ;
2005-07-11 22:38:27 +02:00
vcheckGLcall ( " glReadBuffer " ) ;
2007-04-22 02:02:29 +02:00
if ( ! ( This - > Flags & SFLAG_ALLOCATED ) ) {
surface_allocate_surface ( This , internal , This - > pow2Width ,
This - > pow2Height , format , type ) ;
}
2007-04-07 23:37:41 +02:00
glCopyTexSubImage2D ( This - > glDescription . target ,
This - > glDescription . level ,
0 , 0 , 0 , 0 ,
This - > currentDesc . Width ,
This - > currentDesc . Height ) ;
checkGLcall ( " glCopyTexSubImage2D " ) ;
2005-07-11 22:38:27 +02:00
glReadBuffer ( prevRead ) ;
vcheckGLcall ( " glReadBuffer " ) ;
2006-09-11 18:51:18 +02:00
LEAVE_GL ( ) ;
2007-03-06 21:47:45 +01:00
TRACE ( " Updated target %d \n " , This - > glDescription . target ) ;
2005-01-09 18:37:02 +01:00
}
2006-04-07 12:51:12 +02:00
return WINED3D_OK ;
2007-04-08 00:56:45 +02:00
} else
/* The only place where LoadTexture() might get called when isInDraw=1
* is ActivateContext where lastActiveRenderTarget is preloaded .
*/
if ( iface = = device - > lastActiveRenderTarget & & device - > isInDraw )
ERR ( " Reading back render target but SFLAG_INDRAWABLE not set \n " ) ;
2007-03-06 21:47:45 +01:00
/* Otherwise: System memory copy must be most up to date */
2005-01-09 18:37:02 +01:00
2007-04-14 22:44:55 +02:00
if ( This - > CKeyFlags & WINEDDSD_CKSRCBLT ) {
2006-09-11 18:51:18 +02:00
This - > Flags | = SFLAG_GLCKEY ;
This - > glCKey = This - > SrcBltCKey ;
}
else This - > Flags & = ~ SFLAG_GLCKEY ;
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 ;
2007-06-08 22:28:04 +02:00
outpitch = ( outpitch + device - > surface_alignment - 1 ) & ~ ( device - > 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 " ) ;
2007-04-22 02:02:29 +02:00
if ( ! ( This - > Flags & SFLAG_ALLOCATED ) ) {
surface_allocate_surface ( This , internal , This - > pow2Width , This - > pow2Height , format , type ) ;
}
2006-09-11 18:51:18 +02:00
if ( mem ) {
2007-06-07 01:08:26 +02:00
surface_upload_data ( This , internal , 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 {
2007-03-31 23:02:37 +02:00
/* When making the realloc conditional, keep in mind that GL_APPLE_client_storage may be in use, and This->resource.allocatedMemory
* changed . So also keep track of memory changes . In this case the texture has to be reallocated
*/
2007-04-22 02:02:29 +02:00
if ( ! ( This - > Flags & SFLAG_ALLOCATED ) ) {
surface_allocate_surface ( This , internal , This - > glRect . right - This - > glRect . left , This - > glRect . bottom - This - > glRect . top , format , type ) ;
}
2006-09-11 18:51:18 +02:00
if ( mem ) {
2007-06-07 01:08:26 +02:00
surface_upload_data ( This , internal , This - > glRect . right - This - > glRect . left , This - > glRect . bottom - This - > glRect . top , format , type , mem ) ;
2006-09-11 18:51:18 +02:00
}
}
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 ;
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ SFLAG_INSYSMEM ;
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:
2007-06-07 11:48:55 +02:00
Textures may 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 ;
2007-06-07 11:48:55 +02:00
/* check to see if we're a 'virtual' texture, e.g. we're not a pbuffer of texture, we're a back buffer*/
2005-09-27 12:49:59 +02:00
IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DSwapChain , ( void * * ) & swapChain ) ;
2007-03-06 21:47:45 +01:00
if ( This - > Flags & SFLAG_INDRAWABLE & & ! ( This - > Flags & SFLAG_INTEXTURE ) ) {
/* if were not a real texture then read the back buffer into a real texture */
/* 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 ( ) ;
2007-03-06 21:47:45 +01:00
TRACE ( " (%p) Reading render target into texture \n " , This ) ;
2005-09-27 12:49:59 +02:00
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 " ) ;
2007-02-27 21:32:15 +01:00
glReadBuffer ( swapChain ? GL_BACK : This - > resource . wineD3DDevice - > offscreenBuffer ) ;
2005-09-27 12:49:59 +02:00
vcheckGLcall ( " glReadBuffer " ) ;
glCopyTexImage2D ( GL_TEXTURE_2D ,
0 ,
GL_RGBA ,
0 ,
0 ,
width ,
height ,
0 ) ;
checkGLcall ( " glCopyTexImage2D " ) ;
glReadBuffer ( prevRead ) ;
LEAVE_GL ( ) ;
2007-03-06 21:47:45 +01:00
} else { /* bind the real texture, and make sure it up to date */
2005-09-27 12:49:59 +02:00
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
}
2007-06-07 11:48:55 +02:00
/* Save the data out to a TGA file because 1: it's an easy raw format, 2: it supports an alpha channel */
2005-09-27 12:49:59 +02:00
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 */
2007-06-07 11:48:55 +02:00
/* 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 */
2005-09-27 12:49:59 +02:00
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
}
/**
* 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 ;
2007-06-07 01:08:26 +02:00
if ( ! ( This - > Flags & SFLAG_INSYSMEM ) & & ( This - > Flags & SFLAG_INTEXTURE ) )
surface_download_data ( This ) ;
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ ( SFLAG_INTEXTURE | SFLAG_INDRAWABLE ) ;
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 ;
}
2007-03-06 21:47:45 +01:00
TRACE ( " (%p) : Dirty: yes, Rect:(%d,%d,%d,%d) \n " , This , 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 ) {
2007-06-06 00:21:08 +02:00
TRACE ( " Passing to container \n " ) ;
2005-07-11 16:25:54 +02:00
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 ;
2007-07-23 11:13:37 +02:00
const GlPixelFormatDesc * glDesc ;
2007-07-27 13:22:54 +02:00
const StaticPixelFormatDesc * formatEntry = getFormatDescEntry ( format , & GLINFO_LOCATION , & glDesc ) ;
2005-11-02 21:01:54 +01:00
if ( This - > resource . format ! = WINED3DFMT_UNKNOWN ) {
2007-06-06 00:21:08 +02:00
FIXME ( " (%p) : The format 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
}
2007-06-06 00:21:08 +02:00
TRACE ( " (%p) : Setting texture format to (%d,%s) \n " , This , format , debug_d3dformat ( format ) ) ;
2005-11-02 21:01:54 +01:00
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 {
2007-06-08 22:28:04 +02:00
unsigned char alignment = This - > resource . wineD3DDevice - > surface_alignment ;
This - > resource . size = ( ( This - > pow2Width * formatEntry - > bpp ) + alignment - 1 ) & ~ ( 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 */
2007-07-23 11:13:37 +02:00
This - > glDescription . glFormat = glDesc - > glFormat ;
This - > glDescription . glFormatInternal = glDesc - > glInternal ;
This - > glDescription . glType = glDesc - > 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 ;
2007-04-22 02:02:29 +02:00
This - > Flags & = ~ SFLAG_ALLOCATED ;
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 ;
2007-06-07 11:48:55 +02:00
/* Render targets depend on their hdc, and we can't create an hdc on a user pointer */
2006-07-25 16:38:02 +02:00
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 ) {
2007-03-31 23:02:37 +02:00
void * release = NULL ;
2006-07-25 16:38:02 +02:00
/* 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 ) ) {
2007-03-31 23:02:37 +02:00
release = This - > resource . allocatedMemory ;
2006-07-25 16:38:02 +02:00
}
This - > resource . allocatedMemory = Mem ;
2007-03-31 23:02:37 +02:00
This - > Flags | = SFLAG_USERPTR | SFLAG_INSYSMEM ;
/* Now the surface memory is most up do date. Invalidate drawable and texture */
This - > Flags & = ~ ( SFLAG_INDRAWABLE | SFLAG_INTEXTURE ) ;
/* For client textures opengl has to be notified */
if ( This - > Flags & SFLAG_CLIENT ) {
2007-04-22 02:02:29 +02:00
This - > Flags & = ~ SFLAG_ALLOCATED ;
2007-03-31 23:02:37 +02:00
IWineD3DSurface_PreLoad ( iface ) ;
/* And hope that the app behaves correctly and did not free the old surface memory before setting a new pointer */
}
/* Now free the old memory if any */
HeapFree ( GetProcessHeap ( ) , 0 , release ) ;
2006-07-25 16:38:02 +02:00
} 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 ;
2007-03-31 23:02:37 +02:00
if ( This - > Flags & SFLAG_CLIENT ) {
2007-04-22 02:02:29 +02:00
This - > Flags & = ~ SFLAG_ALLOCATED ;
2007-03-31 23:02:37 +02:00
/* This respecifies an empty texture and opengl knows that the old memory is gone */
IWineD3DSurface_PreLoad ( iface ) ;
}
2006-07-25 16:38:02 +02:00
}
return WINED3D_OK ;
}
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 ;
2007-06-03 13:20:27 +02:00
IWineD3DSwapChainImpl * swapchain = NULL ;
HRESULT hr ;
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 */
2007-04-14 22:44:55 +02:00
if ( ! ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) ) return WINEDDERR_NOTFLIPPABLE ;
2006-05-09 19:49:24 +02:00
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
*/
}
2007-06-03 13:20:27 +02:00
IWineD3DSurface_GetContainer ( iface , & IID_IWineD3DSwapChain , ( void * * ) & swapchain ) ;
if ( ! swapchain ) {
ERR ( " Flipped surface is not on a swapchain \n " ) ;
return WINEDDERR_NOTFLIPPABLE ;
}
/* Just overwrite the swapchain presentation interval. This is ok because only ddraw apps can call Flip,
* and only d3d8 and d3d9 apps specify the presentation interval
*/
if ( ( Flags & ( WINEDDFLIP_NOVSYNC | WINEDDFLIP_INTERVAL2 | WINEDDFLIP_INTERVAL3 | WINEDDFLIP_INTERVAL4 ) ) = = 0 ) {
/* Most common case first to avoid wasting time on all the other cases */
swapchain - > presentParms . PresentationInterval = WINED3DPRESENT_INTERVAL_ONE ;
} else if ( Flags & WINEDDFLIP_NOVSYNC ) {
swapchain - > presentParms . PresentationInterval = WINED3DPRESENT_INTERVAL_IMMEDIATE ;
} else if ( Flags & WINEDDFLIP_INTERVAL2 ) {
swapchain - > presentParms . PresentationInterval = WINED3DPRESENT_INTERVAL_TWO ;
} else if ( Flags & WINEDDFLIP_INTERVAL3 ) {
swapchain - > presentParms . PresentationInterval = WINED3DPRESENT_INTERVAL_THREE ;
} else {
swapchain - > presentParms . PresentationInterval = WINED3DPRESENT_INTERVAL_FOUR ;
}
2006-05-09 19:49:24 +02:00
/* Flipping a OpenGL surface -> Use WineD3DDevice::Present */
2007-06-03 13:20:27 +02:00
hr = IWineD3DSwapChain_Present ( ( IWineD3DSwapChain * ) swapchain , NULL , NULL , 0 , NULL , 0 ) ;
IWineD3DSwapChain_Release ( ( IWineD3DSwapChain * ) swapchain ) ;
return hr ;
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
*/
2007-04-03 12:00:15 +02:00
static inline void fb_copy_to_texture_direct ( IWineD3DSurfaceImpl * This , IWineD3DSurface * SrcSurface , IWineD3DSwapChainImpl * swapchain , WINED3DRECT * srect , WINED3DRECT * drect , BOOL upsidedown , WINED3DTEXTUREFILTERTYPE Filter ) {
2007-02-12 19:18:46 +01:00
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
float xrel , yrel ;
UINT row ;
IWineD3DSurfaceImpl * Src = ( IWineD3DSurfaceImpl * ) SrcSurface ;
ActivateContext ( myDevice , SrcSurface , CTXUSAGE_BLIT ) ;
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
2007-06-09 14:59:15 +02:00
IWineD3DSurface_PreLoad ( ( IWineD3DSurface * ) This ) ;
2007-02-12 19:18:46 +01:00
/* Bind the target texture */
glBindTexture ( GL_TEXTURE_2D , This - > glDescription . textureName ) ;
checkGLcall ( " glBindTexture " ) ;
2007-02-27 21:32:15 +01:00
if ( ! swapchain ) {
glReadBuffer ( myDevice - > offscreenBuffer ) ;
2007-02-12 19:18:46 +01:00
} else {
2007-04-16 21:20:32 +02:00
GLenum buffer = surface_get_gl_buffer ( SrcSurface , ( IWineD3DSwapChain * ) swapchain ) ;
glReadBuffer ( buffer ) ;
2007-02-12 19:18:46 +01:00
}
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 " ) ;
2007-04-03 12:00:15 +02:00
2007-05-10 17:42:41 +02:00
if ( Filter ! = WINED3DTEXF_NONE & & Filter ! = WINED3DTEXF_POINT ) {
2007-04-03 12:00:15 +02:00
ERR ( " Texture filtering not supported in direct blit \n " ) ;
}
2007-05-10 17:42:41 +02:00
} else if ( ( Filter ! = WINED3DTEXF_NONE & & Filter ! = WINED3DTEXF_POINT ) & & ( ( yrel - 1.0 < - eps ) | | ( yrel - 1.0 > eps ) ) ) {
2007-04-03 12:00:15 +02:00
ERR ( " Texture filtering not supported in direct blit \n " ) ;
2007-02-12 19:18:46 +01:00
}
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,
2007-06-06 00:21:08 +02:00
* otherwise it would be upside down , so stretching in y direction
2007-02-12 19:18:46 +01:00
* doesn ' t cost extra time
*
2007-06-06 00:21:08 +02:00
* However , stretching in x direction can be avoided if not necessary
2007-02-12 19:18:46 +01:00
*/
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 ;
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 */
2007-04-03 12:00:15 +02:00
static inline void fb_copy_to_texture_hwstretch ( IWineD3DSurfaceImpl * This , IWineD3DSurface * SrcSurface , IWineD3DSwapChainImpl * swapchain , WINED3DRECT * srect , WINED3DRECT * drect , BOOL upsidedown , WINED3DTEXTUREFILTERTYPE Filter ) {
2007-03-01 01:42:50 +01:00
GLuint src , backup = 0 ;
2007-02-12 19:18:46 +01:00
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 ;
2007-03-04 23:47:57 +01:00
GLenum drawBuffer = GL_BACK ;
2007-02-12 19:18:46 +01:00
TRACE ( " Using hwstretch blit \n " ) ;
/* Activate the Proper context for reading from the source surface, set it up for blitting */
ActivateContext ( myDevice , SrcSurface , CTXUSAGE_BLIT ) ;
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
2007-06-09 14:59:15 +02:00
IWineD3DSurface_PreLoad ( ( IWineD3DSurface * ) This ) ;
2007-02-12 19:18:46 +01:00
2007-03-04 23:47:57 +01:00
/* Try to use an aux buffer for drawing the rectangle. This way it doesn't need restoring.
* This way we don ' t have to wait for the 2 nd readback to finish to leave this function .
*/
if ( GL_LIMITS ( aux_buffers ) > = 2 ) {
/* Got more than one aux buffer? Use the 2nd aux buffer */
drawBuffer = GL_AUX1 ;
} else if ( ( swapchain | | myDevice - > offscreenBuffer = = GL_BACK ) & & GL_LIMITS ( aux_buffers ) > = 1 ) {
/* Only one aux buffer, but it isn't used (Onscreen rendering, or non-aux orm)? Use it! */
drawBuffer = GL_AUX0 ;
}
2007-03-01 01:42:50 +01:00
if ( ! swapchain & & wined3d_settings . offscreen_rendering_mode = = ORM_FBO ) {
glGenTextures ( 1 , & backup ) ;
checkGLcall ( " glGenTextures \n " ) ;
glBindTexture ( GL_TEXTURE_2D , backup ) ;
checkGLcall ( " glBindTexture(Src->glDescription.target, Src->glDescription.textureName) " ) ;
} else {
/* 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
*/
if ( Src - > glDescription . textureName = = 0 ) {
/* Get it a description */
IWineD3DSurface_PreLoad ( SrcSurface ) ;
}
glBindTexture ( GL_TEXTURE_2D , Src - > glDescription . textureName ) ;
checkGLcall ( " glBindTexture(Src->glDescription.target, Src->glDescription.textureName) " ) ;
2007-03-06 21:47:45 +01:00
/* For now invalidate the texture copy of the back buffer. Drawable and sysmem copy are untouched */
Src - > Flags & = ~ SFLAG_INTEXTURE ;
2007-03-01 01:42:50 +01:00
}
2007-02-12 19:18:46 +01:00
glReadBuffer ( GL_BACK ) ;
checkGLcall ( " glReadBuffer(GL_BACK) " ) ;
/* TODO: Only back up the part that will be overwritten */
glCopyTexSubImage2D ( GL_TEXTURE_2D , 0 ,
0 , 0 /* read offsets */ ,
0 , 0 ,
fbwidth ,
fbheight ) ;
checkGLcall ( " glCopyTexSubImage2D " ) ;
2007-03-01 01:42:50 +01:00
/* No issue with overriding these - the sampler is dirty due to blit usage */
2007-04-03 12:00:15 +02:00
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER ,
stateLookup [ WINELOOKUP_MAGFILTER ] [ Filter - minLookup [ WINELOOKUP_MAGFILTER ] ] ) ;
2007-02-12 19:18:46 +01:00
checkGLcall ( " glTexParameteri " ) ;
2007-04-03 12:00:15 +02:00
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER ,
minMipLookup [ Filter ] [ WINED3DTEXF_NONE ] ) ;
2007-02-12 19:18:46 +01:00
checkGLcall ( " glTexParameteri " ) ;
2007-03-01 01:42:50 +01:00
if ( ! swapchain | | ( IWineD3DSurface * ) Src = = swapchain - > backBuffer [ 0 ] ) {
src = backup ? backup : Src - > glDescription . textureName ;
2007-02-12 19:18:46 +01:00
} 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 ) ;
2007-03-04 23:47:57 +01:00
glDrawBuffer ( drawBuffer ) ;
glReadBuffer ( drawBuffer ) ;
2007-02-12 19:18:46 +01:00
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 */
2007-03-01 01:42:50 +01:00
glBindTexture ( GL_TEXTURE_2D , backup ? backup : Src - > glDescription . textureName ) ;
checkGLcall ( " glBindTexture(GL_TEXTURE_2D, Src->glDescription.textureName) " ) ;
2007-02-12 19:18:46 +01:00
2007-03-04 23:47:57 +01:00
if ( drawBuffer = = GL_BACK ) {
glBegin ( GL_QUADS ) ;
/* top left */
glTexCoord2f ( 0.0 , ( float ) fbheight / ( float ) Src - > pow2Height ) ;
glVertex2i ( 0 , 0 ) ;
2007-02-12 19:18:46 +01:00
2007-03-04 23:47:57 +01:00
/* bottom left */
glTexCoord2f ( 0.0 , 0.0 ) ;
glVertex2i ( 0 , fbheight ) ;
2007-02-12 19:18:46 +01:00
2007-03-04 23:47:57 +01:00
/* bottom right */
glTexCoord2f ( ( float ) fbwidth / ( float ) Src - > pow2Width , 0.0 ) ;
glVertex2i ( fbwidth , Src - > currentDesc . Height ) ;
2007-02-12 19:18:46 +01:00
2007-03-04 23:47:57 +01:00
/* top right */
glTexCoord2f ( ( float ) fbwidth / ( float ) Src - > pow2Width , ( float ) fbheight / ( float ) Src - > pow2Height ) ;
glVertex2i ( fbwidth , 0 ) ;
glEnd ( ) ;
} else {
/* Restore the old draw buffer */
glDrawBuffer ( GL_BACK ) ;
}
2007-02-12 19:18:46 +01:00
/* Cleanup */
2007-03-01 01:42:50 +01:00
if ( src ! = Src - > glDescription . textureName & & src ! = backup ) {
2007-02-12 19:18:46 +01:00
glDeleteTextures ( 1 , & src ) ;
checkGLcall ( " glDeleteTextures(1, &src) " ) ;
}
2007-03-01 01:42:50 +01:00
if ( backup ) {
glDeleteTextures ( 1 , & backup ) ;
checkGLcall ( " glDeleteTextures(1, &backup) " ) ;
}
2007-02-12 19:18:46 +01:00
LEAVE_GL ( ) ;
}
2006-05-23 00:48:54 +02:00
/* Not called from the VTable */
2007-04-14 22:44:55 +02:00
static HRESULT IWineD3DSurfaceImpl_BltOverride ( IWineD3DSurfaceImpl * This , RECT * DestRect , IWineD3DSurface * SrcSurface , RECT * SrcRect , DWORD Flags , WINEDDBLTFX * DDBltFx , WINED3DTEXTUREFILTERTYPE Filter ) {
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 ;
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 */
2007-04-14 22:44:55 +02:00
if ( Flags & ( WINEDDBLT_KEYDEST | WINEDDBLT_KEYDESTOVERRIDE ) ) {
2007-02-15 03:03:05 +01:00
/* 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 */
2007-04-29 12:48:34 +02:00
if ( dstSwapchain & & dstSwapchain = = srcSwapchain & & dstSwapchain - > backBuffer & &
( ( IWineD3DSurface * ) This = = dstSwapchain - > frontBuffer ) & & SrcSurface = = dstSwapchain - > backBuffer [ 0 ] ) {
2007-02-15 03:02:39 +01:00
/* Half-life does a Blt from the back buffer to the front buffer,
* Full surface size , no flags . . . Use present instead
2007-04-29 12:48:34 +02:00
*
* This path will only be entered for d3d7 and ddraw apps , because d3d8 / 9 offer no way to blit TO the front buffer
2007-02-15 03:02:39 +01:00
*/
2006-05-23 00:48:54 +02:00
/* Check rects - IWineD3DDevice_Present doesn't handle them */
2007-04-29 12:48:34 +02:00
while ( 1 )
{
RECT mySrcRect ;
2007-05-10 18:51:00 +02:00
TRACE ( " Looking if a Present can be done... \n " ) ;
2007-04-29 12:48:34 +02:00
/* Source Rectangle must be full surface */
if ( SrcRect ) {
if ( SrcRect - > left ! = 0 | | SrcRect - > top ! = 0 | |
SrcRect - > right ! = Src - > currentDesc . Width | | SrcRect - > bottom ! = Src - > currentDesc . Height ) {
TRACE ( " No, Source rectangle doesn't match \n " ) ;
break ;
}
}
mySrcRect . left = 0 ;
mySrcRect . top = 0 ;
mySrcRect . right = Src - > currentDesc . Width ;
mySrcRect . bottom = Src - > currentDesc . Height ;
/* No stretching may occur */
if ( mySrcRect . right ! = rect . x2 - rect . x1 | |
mySrcRect . bottom ! = rect . y2 - rect . y1 ) {
TRACE ( " No, stretching is done \n " ) ;
break ;
2006-05-23 00:48:54 +02:00
}
2007-04-29 12:48:34 +02:00
/* Destination must be full surface or match the clipping rectangle */
if ( This - > clipper & & ( ( IWineD3DClipperImpl * ) This - > clipper ) - > hWnd )
{
RECT cliprect ;
POINT pos [ 2 ] ;
GetClientRect ( ( ( IWineD3DClipperImpl * ) This - > clipper ) - > hWnd , & cliprect ) ;
pos [ 0 ] . x = rect . x1 ;
pos [ 0 ] . y = rect . y1 ;
pos [ 1 ] . x = rect . x2 ;
pos [ 1 ] . y = rect . y2 ;
MapWindowPoints ( GetDesktopWindow ( ) , ( ( IWineD3DClipperImpl * ) This - > clipper ) - > hWnd ,
pos , 2 ) ;
if ( pos [ 0 ] . x ! = cliprect . left | | pos [ 0 ] . y ! = cliprect . top | |
pos [ 1 ] . x ! = cliprect . right | | pos [ 1 ] . y ! = cliprect . bottom )
{
TRACE ( " No, dest rectangle doesn't match(clipper) \n " ) ;
TRACE ( " Clip rect at (%d,%d)-(%d,%d) \n " , cliprect . left , cliprect . top , cliprect . right , cliprect . bottom ) ;
TRACE ( " Blt dest: (%d,%d)-(%d,%d) \n " , rect . x1 , rect . y1 , rect . x2 , rect . y2 ) ;
break ;
}
}
else
{
if ( rect . x1 ! = 0 | | rect . y1 ! = 0 | |
rect . x2 ! = This - > currentDesc . Width | | rect . y2 ! = This - > currentDesc . Height ) {
TRACE ( " No, dest rectangle doesn't match(surface size) \n " ) ;
break ;
}
}
2006-05-23 00:48:54 +02:00
2007-04-29 12:48:34 +02:00
TRACE ( " Yes \n " ) ;
2006-05-23 00:48:54 +02:00
2007-04-29 12:48:34 +02:00
/* These flags are unimportant for the flag check, remove them */
if ( ( Flags & ~ ( WINEDDBLT_DONOTWAIT | WINEDDBLT_WAIT ) ) = = 0 ) {
WINED3DSWAPEFFECT orig_swap = dstSwapchain - > presentParms . SwapEffect ;
/* 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-06-06 00:21:08 +02:00
* Windowed Direct3D < 7 apps do the same . The D3D7 sdk demos are nice
2007-04-29 12:48:34 +02:00
* testcases .
*/
dstSwapchain - > presentParms . SwapEffect = WINED3DSWAPEFFECT_COPY ;
2007-06-03 13:20:27 +02:00
dstSwapchain - > presentParms . PresentationInterval = WINED3DPRESENT_INTERVAL_IMMEDIATE ;
2007-04-29 12:48:34 +02:00
TRACE ( " Full screen back buffer -> front buffer blt, performing a flip instead \n " ) ;
2007-06-03 13:20:27 +02:00
IWineD3DSwapChain_Present ( ( IWineD3DSwapChain * ) dstSwapchain , NULL , NULL , 0 , NULL , 0 ) ;
2007-04-29 12:48:34 +02:00
dstSwapchain - > presentParms . SwapEffect = orig_swap ;
return WINED3D_OK ;
2006-05-23 00:48:54 +02:00
}
2007-04-29 12:48:34 +02:00
break ;
2006-05-23 00:48:54 +02:00
}
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-04-14 22:44:55 +02:00
if ( Flags & ( WINEDDBLT_KEYSRC | WINEDDBLT_KEYSRCOVERRIDE ) ) {
2007-02-15 03:03:05 +01:00
TRACE ( " Color keying not supported by frame buffer to texture blit \n " ) ;
return WINED3DERR_INVALIDCALL ;
/* Destination color key is checked above */
}
2007-06-06 00:21:08 +02:00
/* Make sure that the top pixel is always above the bottom pixel, and keep a separate upside down flag
2007-06-09 14:59:15 +02:00
* glCopyTexSubImage is a bit picky about the parameters we pass to it
*/
2007-02-15 03:02:39 +01:00
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
2007-04-09 01:54:07 +02:00
* flip the image nor scale it .
2007-02-15 03:02:39 +01:00
*
* - > 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
2007-04-09 01:54:07 +02:00
*
* If EXT_framebuffer_blit is supported that can be used instead . Note that EXT_framebuffer_blit implies
* FBO support , so it doesn ' t really make sense to try and make it work with different offscreen rendering
* backends .
2007-02-15 03:02:39 +01:00
*/
2007-04-09 01:54:07 +02:00
if ( wined3d_settings . offscreen_rendering_mode = = ORM_FBO & & GL_SUPPORT ( EXT_FRAMEBUFFER_BLIT ) ) {
stretch_rect_fbo ( ( IWineD3DDevice * ) myDevice , SrcSurface , & srect ,
( IWineD3DSurface * ) This , & rect , Filter , upsideDown ) ;
2007-02-15 03:02:39 +01:00
} 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 " ) ;
2007-04-03 12:00:15 +02:00
fb_copy_to_texture_direct ( This , SrcSurface , srcSwapchain , & srect , & rect , upsideDown , Filter ) ;
2007-02-15 03:02:39 +01:00
} else {
TRACE ( " Using hardware stretching to flip / stretch the texture \n " ) ;
2007-04-03 12:00:15 +02:00
fb_copy_to_texture_hwstretch ( This , SrcSurface , srcSwapchain , & srect , & rect , upsideDown , Filter ) ;
2007-02-15 03:02:39 +01:00
}
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 {
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ SFLAG_INSYSMEM ;
2007-02-15 03:02:39 +01:00
}
2007-03-06 21:47:45 +01:00
/* The texture is now most up to date - If the surface is a render target and has a drawable, this
* path is never entered
*/
This - > Flags | = SFLAG_INTEXTURE ;
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 ;
2007-04-14 22:44:55 +02:00
WINEDDCOLORKEY oldBltCKey = This - > SrcBltCKey ;
2007-02-15 03:02:39 +01:00
RECT SourceRectangle ;
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 .
2007-06-06 00:21:08 +02:00
* PreLoad will catch the change to the flags and color key and reload if necessary .
2007-02-15 03:02:39 +01:00
*/
2007-04-14 22:44:55 +02:00
if ( Flags & WINEDDBLT_KEYSRC ) {
2007-02-15 03:03:05 +01:00
/* Use color key from surface */
2007-04-14 22:44:55 +02:00
} else if ( Flags & WINEDDBLT_KEYSRCOVERRIDE ) {
2007-02-15 03:03:05 +01:00
/* Use color key from DDBltFx */
2007-04-14 22:44:55 +02:00
Src - > CKeyFlags | = WINEDDSD_CKSRCBLT ;
2007-02-15 03:03:05 +01:00
This - > SrcBltCKey = DDBltFx - > ddckSrcColorkey ;
} else {
/* Do not use color key */
2007-04-14 22:44:55 +02:00
Src - > CKeyFlags & = ~ WINEDDSD_CKSRCBLT ;
2007-02-15 03:02:39 +01:00
}
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
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 ) ;
2007-08-06 16:27:08 +02:00
ENTER_GL ( ) ;
2006-05-23 00:48:54 +02:00
2007-03-14 22:00:31 +01:00
if ( ! dstSwapchain ) {
TRACE ( " Drawing to offscreen buffer \n " ) ;
glDrawBuffer ( myDevice - > offscreenBuffer ) ;
2007-06-05 04:42:04 +02:00
checkGLcall ( " glDrawBuffer " ) ;
2007-03-14 22:00:31 +01:00
} else {
2007-04-16 21:20:32 +02:00
GLenum buffer = surface_get_gl_buffer ( ( IWineD3DSurface * ) This , ( IWineD3DSwapChain * ) dstSwapchain ) ;
TRACE ( " Drawing to %#x buffer \n " , buffer ) ;
glDrawBuffer ( buffer ) ;
checkGLcall ( " glDrawBuffer " ) ;
2007-02-15 03:02:39 +01:00
}
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 " ) ;
2007-04-03 12:00:15 +02:00
/* Filtering for StretchRect */
2007-02-15 03:02:39 +01:00
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER ,
2007-04-03 12:00:15 +02:00
stateLookup [ WINELOOKUP_MAGFILTER ] [ Filter - minLookup [ WINELOOKUP_MAGFILTER ] ] ) ;
2007-02-15 03:02:39 +01:00
checkGLcall ( " glTexParameteri " ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER ,
2007-04-03 12:00:15 +02:00
minMipLookup [ Filter ] [ WINED3DTEXF_NONE ] ) ;
2007-02-15 03:02:39 +01:00
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-04-14 22:44:55 +02:00
if ( Flags & ( WINEDDBLT_KEYSRC | WINEDDBLT_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
2007-03-14 22:00:31 +01:00
*/
2007-02-15 03:02:39 +01:00
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-04-14 22:44:55 +02:00
if ( Flags & ( WINEDDBLT_KEYSRC | WINEDDBLT_KEYSRCOVERRIDE ) ) {
2007-02-15 03:04:40 +01:00
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-03-14 22:00:31 +01:00
/* The draw buffer should only need to be restored if we were drawing to the front buffer, and there is a back buffer.
* otherwise the context manager should choose between GL_BACK / offscreenDrawBuffer
*/
if ( dstSwapchain & & This = = ( IWineD3DSurfaceImpl * ) dstSwapchain - > frontBuffer & & dstSwapchain - > backBuffer ) {
glDrawBuffer ( GL_BACK ) ;
2007-06-05 04:42:04 +02:00
checkGLcall ( " glDrawBuffer " ) ;
2007-02-15 03:02:39 +01:00
}
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 */
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ SFLAG_INSYSMEM ;
/* The surface is now in the drawable. On onscreen surfaces or without fbos the texture
* is outdated now
*/
if ( dstSwapchain | | wined3d_settings . offscreen_rendering_mode ! = ORM_FBO ) {
This - > Flags | = SFLAG_INDRAWABLE ;
This - > Flags & = ~ SFLAG_INTEXTURE ;
} else {
This - > Flags | = SFLAG_INTEXTURE ;
}
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 */
2007-04-14 22:44:55 +02:00
if ( Flags & WINEDDBLT_COLORFILL ) {
2007-02-15 03:02:39 +01:00
/* 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 ] ) {
2007-02-27 21:32:15 +01:00
glDrawBuffer ( myDevice - > offscreenBuffer ) ;
checkGLcall ( " glDrawBuffer(myDevice->offscreenBuffer3) " ) ;
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 */
2007-02-27 21:32:15 +01:00
if ( ! dstSwapchain ) {
glDrawBuffer ( myDevice - > offscreenBuffer ) ;
} else if ( dstSwapchain - > backBuffer & & dstSwapchain - > backBuffer [ 0 ] ) {
2007-02-15 03:02:39 +01:00
glDrawBuffer ( GL_BACK ) ;
}
2007-02-27 21:32:15 +01:00
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-20 15:57:10 +01:00
/* Default: Fall back to the generic blt. Not an error, a TRACE is enough */
2007-02-15 03:02:39 +01:00
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 ;
}
2007-05-10 21:10:49 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_BltZ ( IWineD3DSurfaceImpl * This , RECT * DestRect , IWineD3DSurface * SrcSurface , RECT * SrcRect , DWORD Flags , WINEDDBLTFX * DDBltFx )
{
IWineD3DDeviceImpl * myDevice = This - > resource . wineD3DDevice ;
2007-07-24 22:29:16 +02:00
if ( Flags & WINEDDBLT_DEPTHFILL )
2007-05-10 21:10:49 +02:00
return IWineD3DDevice_Clear ( ( IWineD3DDevice * ) myDevice ,
DestRect = = NULL ? 0 : 1 ,
( WINED3DRECT * ) DestRect ,
WINED3DCLEAR_ZBUFFER ,
0x00000000 ,
( float ) DDBltFx - > u5 . dwFillDepth / ( float ) MAXDWORD ,
0x00000000 ) ;
FIXME ( " (%p): Unsupp depthstencil blit \n " , This ) ;
return WINED3DERR_INVALIDCALL ;
}
2007-04-14 22:44:55 +02:00
static HRESULT WINAPI IWineD3DSurfaceImpl_Blt ( IWineD3DSurface * iface , RECT * DestRect , IWineD3DSurface * SrcSurface , RECT * SrcRect , DWORD Flags , WINEDDBLTFX * DDBltFx , WINED3DTEXTUREFILTERTYPE Filter ) {
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-06-08 20:59:57 +02:00
/* Accessing the depth stencil is supposed to fail between a BeginScene and EndScene pair,
* except depth blits , which seem to work
*/
2007-05-10 21:10:49 +02:00
if ( iface = = myDevice - > stencilBufferTarget | | ( SrcSurface & & SrcSurface = = myDevice - > stencilBufferTarget ) ) {
2007-06-08 20:59:57 +02:00
if ( myDevice - > inScene & & ! ( Flags & WINEDDBLT_DEPTHFILL ) ) {
2007-05-10 21:10:49 +02:00
TRACE ( " Attempt to access the depth stencil surface in a BeginScene / EndScene pair, returning WINED3DERR_INVALIDCALL \n " ) ;
return WINED3DERR_INVALIDCALL ;
} else if ( IWineD3DSurfaceImpl_BltZ ( This , DestRect , SrcSurface , SrcRect , Flags , DDBltFx ) = = WINED3D_OK ) {
TRACE ( " Z Blit override handled the blit \n " ) ;
return WINED3D_OK ;
}
2007-02-09 16:35:45 +01:00
}
2006-05-23 00:48:54 +02:00
/* Special cases for RenderTargets */
if ( ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) | |
( Src & & ( Src - > resource . usage & WINED3DUSAGE_RENDERTARGET ) ) ) {
2007-04-03 12:00:15 +02:00
if ( IWineD3DSurfaceImpl_BltOverride ( This , DestRect , SrcSurface , SrcRect , Flags , DDBltFx , Filter ) = = WINED3D_OK ) return WINED3D_OK ;
2006-05-23 00:48:54 +02:00
}
/* For the rest call the X11 surface implementation.
* For RenderTargets this should be implemented OpenGL accelerated in BltOverride ,
* other Blts are rather rare
*/
2007-04-03 12:00:15 +02:00
return IWineGDISurfaceImpl_Blt ( iface , DestRect , SrcSurface , SrcRect , Flags , DDBltFx , Filter ) ;
2006-05-23 00:48:54 +02:00
}
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 )
{
2007-04-14 22:44:55 +02:00
case WINEDDGBS_CANBLT :
case WINEDDGBS_ISBLTDONE :
return WINED3D_OK ;
2006-05-09 20:07:40 +02:00
default :
2007-04-14 22:44:55 +02:00
return WINED3DERR_INVALIDCALL ;
2006-05-09 20:07:40 +02:00
}
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 ) {
2007-04-14 22:44:55 +02:00
case WINEDDGFS_CANFLIP :
case WINEDDGFS_ISFLIPDONE :
return WINED3D_OK ;
2006-05-09 20:07:40 +02:00
default :
2007-04-14 22:44:55 +02:00
return WINED3DERR_INVALIDCALL ;
2006-05-09 20:07:40 +02:00
}
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 ) ;
2007-04-14 22:44:55 +02:00
/* D3D8 and 9 loose full devices, ddraw only surfaces */
return This - > Flags & SFLAG_LOST ? WINED3DERR_DEVICELOST : 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 */
2007-04-14 22:44:55 +02:00
if ( trans & WINEDDBLTFAST_SRCCOLORKEY )
Flags | = WINEDDBLT_KEYSRC ;
if ( trans & WINEDDBLTFAST_DESTCOLORKEY )
Flags | = WINEDDBLT_KEYDEST ;
if ( trans & WINEDDBLTFAST_WAIT )
Flags | = WINEDDBLT_WAIT ;
if ( trans & WINEDDBLTFAST_DONOTWAIT )
Flags | = WINEDDBLT_DONOTWAIT ;
2006-07-26 23:19:04 +02:00
2007-05-10 17:42:41 +02:00
if ( IWineD3DSurfaceImpl_BltOverride ( This , & DstRect , Source , & SrcRect , Flags , NULL , WINED3DTEXF_POINT ) = = 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 ;
2007-04-14 22:44:55 +02:00
return WINED3D_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 )
{
2007-03-06 21:47:45 +01:00
if ( ! This - > Flags & SFLAG_INSYSMEM ) {
FIXME ( " Palette changed with surface that does not have an up to date system memory copy \n " ) ;
}
2006-05-09 20:28:23 +02:00
TRACE ( " Dirtifying surface \n " ) ;
2007-03-06 21:47:45 +01:00
This - > Flags & = ~ ( SFLAG_INTEXTURE | SFLAG_INDRAWABLE ) ;
2006-05-09 20:28:23 +02:00
}
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 )
2007-04-14 22:44:55 +02:00
This - > palette - > Flags & = ~ WINEDDPCAPS_PRIMARYSURFACE ;
2006-05-09 20:28:23 +02:00
if ( PalImpl ! = NULL ) {
if ( This - > resource . usage & WINED3DUSAGE_RENDERTARGET ) {
/* Set the device's main palette if the palette
* wasn ' t a primary palette before
*/
2007-04-14 22:44:55 +02:00
if ( ! ( PalImpl - > Flags & WINEDDPCAPS_PRIMARYSURFACE ) ) {
2006-05-09 20:28:23 +02:00
IWineD3DDeviceImpl * device = This - > resource . wineD3DDevice ;
unsigned int i ;
for ( i = 0 ; i < 256 ; i + + ) {
device - > palettes [ device - > currentPalette ] [ i ] = PalImpl - > palents [ i ] ;
}
}
2007-04-14 22:44:55 +02:00
( PalImpl ) - > Flags | = WINEDDPCAPS_PRIMARYSURFACE ;
2006-05-09 20:28:23 +02:00
}
}
This - > palette = PalImpl ;
return IWineD3DSurface_RealizePalette ( iface ) ;
2006-04-17 17:25:55 +02:00
}
2007-04-14 22:44:55 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_SetColorKey ( IWineD3DSurface * iface , DWORD Flags , WINEDDCOLORKEY * 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
2007-04-14 22:44:55 +02:00
if ( ( Flags & WINEDDCKEY_COLORSPACE ) ! = 0 ) {
2006-10-01 05:20:10 +02:00
FIXME ( " colorkey value not supported (%08x) ! \n " , Flags ) ;
2007-04-14 22:44:55 +02:00
return WINED3DERR_INVALIDCALL ;
2006-05-09 19:37:38 +02:00
}
/* Dirtify the surface, but only if a key was changed */
if ( CKey ) {
2007-04-14 22:44:55 +02:00
switch ( Flags & ~ WINEDDCKEY_COLORSPACE ) {
case WINEDDCKEY_DESTBLT :
2006-05-09 19:37:38 +02:00
This - > DestBltCKey = * CKey ;
2007-04-14 22:44:55 +02:00
This - > CKeyFlags | = WINEDDSD_CKDESTBLT ;
2006-05-09 19:37:38 +02:00
break ;
2007-04-14 22:44:55 +02:00
case WINEDDCKEY_DESTOVERLAY :
2006-05-09 19:37:38 +02:00
This - > DestOverlayCKey = * CKey ;
2007-04-14 22:44:55 +02:00
This - > CKeyFlags | = WINEDDSD_CKDESTOVERLAY ;
2006-05-09 19:37:38 +02:00
break ;
2007-04-14 22:44:55 +02:00
case WINEDDCKEY_SRCOVERLAY :
2006-05-09 19:37:38 +02:00
This - > SrcOverlayCKey = * CKey ;
2007-04-14 22:44:55 +02:00
This - > CKeyFlags | = WINEDDSD_CKSRCOVERLAY ;
2006-05-09 19:37:38 +02:00
break ;
2007-04-14 22:44:55 +02:00
case WINEDDCKEY_SRCBLT :
2006-05-09 19:37:38 +02:00
This - > SrcBltCKey = * CKey ;
2007-04-14 22:44:55 +02:00
This - > CKeyFlags | = WINEDDSD_CKSRCBLT ;
2006-05-09 19:37:38 +02:00
break ;
}
}
else {
2007-04-14 22:44:55 +02:00
switch ( Flags & ~ WINEDDCKEY_COLORSPACE ) {
case WINEDDCKEY_DESTBLT :
This - > CKeyFlags & = ~ WINEDDSD_CKDESTBLT ;
2006-05-09 19:37:38 +02:00
break ;
2007-04-14 22:44:55 +02:00
case WINEDDCKEY_DESTOVERLAY :
This - > CKeyFlags & = ~ WINEDDSD_CKDESTOVERLAY ;
2006-05-09 19:37:38 +02:00
break ;
2007-04-14 22:44:55 +02:00
case WINEDDCKEY_SRCOVERLAY :
This - > CKeyFlags & = ~ WINEDDSD_CKSRCOVERLAY ;
2006-05-09 19:37:38 +02:00
break ;
2007-04-14 22:44:55 +02:00
case WINEDDCKEY_SRCBLT :
This - > CKeyFlags & = ~ WINEDDSD_CKSRCBLT ;
2006-05-09 19:37:38 +02:00
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 ;
2007-07-16 20:42:16 +02:00
unsigned int pow2Width , pow2Height ;
2007-07-27 13:22:54 +02:00
const GlPixelFormatDesc * glDesc ;
2007-07-30 14:14:01 +02:00
getFormatDescEntry ( This - > resource . format , & GLINFO_LOCATION , & glDesc ) ;
2007-07-27 13:22:54 +02:00
/* Setup some glformat defaults */
This - > glDescription . glFormat = glDesc - > glFormat ;
This - > glDescription . glFormatInternal = glDesc - > glInternal ;
This - > glDescription . glType = glDesc - > glType ;
This - > glDescription . textureName = 0 ;
This - > glDescription . target = GL_TEXTURE_2D ;
2007-07-16 20:42:16 +02:00
/* Non-power2 support */
if ( GL_SUPPORT ( ARB_TEXTURE_NON_POWER_OF_TWO ) ) {
pow2Width = This - > currentDesc . Width ;
pow2Height = This - > currentDesc . Height ;
} else {
/* Find the nearest pow2 match */
pow2Width = pow2Height = 1 ;
while ( pow2Width < This - > currentDesc . Width ) pow2Width < < = 1 ;
while ( pow2Height < This - > currentDesc . Height ) pow2Height < < = 1 ;
}
This - > pow2Width = pow2Width ;
This - > pow2Height = pow2Height ;
if ( pow2Width > This - > currentDesc . Width | | pow2Height > This - > currentDesc . Height ) {
WINED3DFORMAT Format = This - > resource . format ;
/** TODO: add support for non power two compressed textures **/
if ( Format = = WINED3DFMT_DXT1 | | Format = = WINED3DFMT_DXT2 | | Format = = WINED3DFMT_DXT3
| | Format = = WINED3DFMT_DXT4 | | Format = = WINED3DFMT_DXT5 ) {
FIXME ( " (%p) Compressed non-power-two textures are not supported w(%d) h(%d) \n " ,
This , This - > currentDesc . Width , This - > currentDesc . Height ) ;
return WINED3DERR_NOTAVAILABLE ;
}
}
if ( pow2Width ! = This - > currentDesc . Width | |
pow2Height ! = This - > currentDesc . Height ) {
This - > Flags | = SFLAG_NONPOW2 ;
}
2006-05-19 00:13:29 +02:00
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 )
2007-06-06 00:21:08 +02:00
2 : Set the texture to the maximum size ( bad idea )
2006-05-19 00:13:29 +02:00
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 ;
}
2007-07-08 23:33:48 +02:00
if ( This - > resource . allocatedMemory = = NULL ) {
/* Make sure memory exists from the start, and it is initialized properly. D3D initializes surfaces,
* gl does not , so we need to upload zeroes to init the gl texture .
2007-03-31 23:02:37 +02:00
*/
This - > resource . allocatedMemory = HeapAlloc ( GetProcessHeap ( ) , HEAP_ZERO_MEMORY , This - > resource . size + 4 ) ;
}
2007-07-08 23:33:48 +02:00
This - > Flags | = SFLAG_INSYSMEM ;
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-06-08 22:28:04 +02:00
unsigned char alignment = This - > resource . wineD3DDevice - > surface_alignment ;
2007-02-13 20:21:48 +01:00
ret = This - > bytesPerPixel * This - > currentDesc . Width ; /* Bytes / row */
2007-06-08 22:28:04 +02:00
ret = ( ret + alignment - 1 ) & ~ ( 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 ) ;
2007-04-14 22:44:55 +02:00
return WINEDDERR_NOTAOVERLAYSURFACE ;
2006-10-05 17:56:54 +02:00
}
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 ) ;
2007-04-14 22:44:55 +02:00
return WINEDDERR_NOTAOVERLAYSURFACE ;
2006-10-05 17:56:54 +02:00
}
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 ) ;
2007-04-14 22:44:55 +02:00
return WINEDDERR_NOTAOVERLAYSURFACE ;
2006-10-05 17:56:54 +02:00
}
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 ) ;
2007-04-14 22:44:55 +02:00
return WINEDDERR_NOTAOVERLAYSURFACE ;
2006-10-05 17:56:54 +02:00
}
2006-10-05 16:43:26 +02:00
return WINED3D_OK ;
}
2007-04-28 17:59:37 +02:00
HRESULT WINAPI IWineD3DSurfaceImpl_SetClipper ( IWineD3DSurface * iface , IWineD3DClipper * clipper )
{
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p)->(%p) \n " , This , clipper ) ;
This - > clipper = clipper ;
return WINED3D_OK ;
}
HRESULT WINAPI IWineD3DSurfaceImpl_GetClipper ( IWineD3DSurface * iface , IWineD3DClipper * * clipper )
{
IWineD3DSurfaceImpl * This = ( IWineD3DSurfaceImpl * ) iface ;
TRACE ( " (%p)->(%p) \n " , This , clipper ) ;
* clipper = This - > clipper ;
2007-05-10 21:45:51 +02:00
if ( * clipper ) {
IWineD3DClipper_AddRef ( * clipper ) ;
}
2007-04-28 17:59:37 +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 ,
2007-04-28 17:59:37 +02:00
IWineD3DSurfaceImpl_SetClipper ,
IWineD3DSurfaceImpl_GetClipper ,
2005-01-09 18:37:02 +01:00
/* Internal use: */
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_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
} ;