Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

569 lines
17 KiB

//----------------------------------------------------------------------------
//
// rasttex.cpp
//
// Texture functions.
//
// Copyright (C) Microsoft Corporation, 1997.
//
//----------------------------------------------------------------------------
#include "pch.cpp"
#pragma hdrstop
//----------------------------------------------------------------------------
//
// SetSizesSpanTexture
//
// Initialize pSpanTex data using current iMaxMipLevel info, Getting the
// surfaces from pSurf. Assumes InitSpanTexture has been called first.
//
//----------------------------------------------------------------------------
HRESULT
D3DContext::SetSizesSpanTexture(PD3DI_SPANTEX pSpanTex)
{
LPDDRAWI_DDRAWSURFACE_LCL pLcl;
INT iFirstSurf = min(pSpanTex->iMaxMipLevel, pSpanTex->cLODTex);
LPDIRECTDRAWSURFACE pDDS = pSpanTex->pSurf[iFirstSurf];
INT i;
// Init
pLcl = ((LPDDRAWI_DDRAWSURFACE_INT)pDDS)->lpLcl;
pSpanTex->iSizeU = (INT16)DDSurf_Width(pLcl);
pSpanTex->iSizeV = (INT16)DDSurf_Height(pLcl);
pSpanTex->uMaskU = (INT16)(pSpanTex->iSizeU - 1);
pSpanTex->uMaskV = (INT16)(pSpanTex->iSizeV - 1);
pSpanTex->iShiftU = (INT16)IntLog2(pSpanTex->iSizeU);
if (0 != DDSurf_BitDepth(pLcl))
{
pSpanTex->iShiftPitch[0] =
(INT16)IntLog2((UINT32)(DDSurf_Pitch(pLcl) * 8)/DDSurf_BitDepth(pLcl));
}
else
{
pSpanTex->iShiftPitch[0] =
(INT16)IntLog2(((UINT32)DDSurf_Width(pLcl) * 8));
}
pSpanTex->iShiftV = (INT16)IntLog2(pSpanTex->iSizeV);
pSpanTex->uMaskV = pSpanTex->uMaskV;
// Check if the texture size is power of 2
if (!ValidTextureSize(pSpanTex->iSizeU, pSpanTex->iShiftU,
pSpanTex->iSizeV, pSpanTex->iShiftV))
{
return DDERR_INVALIDPARAMS;
}
// Check for mipmap if any.
// iPreSizeU and iPreSizeV store the size(u and v) of the previous level
// mipmap. They are init'ed with the first texture size.
INT16 iPreSizeU = pSpanTex->iSizeU, iPreSizeV = pSpanTex->iSizeV;
for ( i = iFirstSurf + 1; i <= pSpanTex->cLODTex; i++)
{
pDDS = pSpanTex->pSurf[i];
// Check for invalid mipmap texture size
pLcl = ((LPDDRAWI_DDRAWSURFACE_INT)pDDS)->lpLcl;
if (!ValidMipmapSize(iPreSizeU, (INT16)DDSurf_Width(pLcl)) ||
!ValidMipmapSize(iPreSizeV, (INT16)DDSurf_Height(pLcl)))
{
return DDERR_INVALIDPARAMS;
}
if (0 != DDSurf_BitDepth(pLcl))
{
pSpanTex->iShiftPitch[i - iFirstSurf] =
(INT16)IntLog2(((UINT32)DDSurf_Pitch(pLcl)*8)/DDSurf_BitDepth(pLcl));
}
else
{
pSpanTex->iShiftPitch[i - iFirstSurf] =
(INT16)IntLog2(((UINT32)DDSurf_Width(pLcl)*8));
}
iPreSizeU = (INT16)DDSurf_Width(pLcl);
iPreSizeV = (INT16)DDSurf_Height(pLcl);
}
pSpanTex->cLOD = pSpanTex->cLODTex - iFirstSurf;
pSpanTex->iMaxScaledLOD = ((pSpanTex->cLOD + 1) << LOD_SHIFT) - 1;
pSpanTex->uFlags &= ~D3DI_SPANTEX_MAXMIPLEVELS_DIRTY;
return D3D_OK;
}
//----------------------------------------------------------------------------
//
// InitSpanTexture
//
// Initializes the entire array of pSurf's (regardless of iMaxMipLevel) pointed
// to by the root surface of pDDS. Sets all pSpanTex state that will not ever
// change in SetSizesSpanTexture.
//
//----------------------------------------------------------------------------
HRESULT
D3DContext::InitSpanTexture(PD3DI_SPANTEX pSpanTex, LPDIRECTDRAWSURFACE pDDS)
{
HRESULT hr;
LPDDRAWI_DDRAWSURFACE_LCL pLcl;
DDSCAPS ddscaps;
static INT32 iGeneration = 0;
// Init
pSpanTex->iGeneration = iGeneration++;
// Note that all pSpanTex elements are initialized to 0
pLcl = ((LPDDRAWI_DDRAWSURFACE_INT)pDDS)->lpLcl;
// Set the transparent bit and the transparent color with pSurf[0]
// initially
if ((pLcl->dwFlags & DDRAWISURF_HASCKEYSRCBLT) != 0)
{
pSpanTex->uFlags |= D3DI_SPANTEX_HAS_TRANSPARENT;
pSpanTex->TransparentColor =
pLcl->ddckCKSrcBlt.dwColorSpaceHighValue;
}
else
{
pSpanTex->uFlags &= ~D3DI_SPANTEX_HAS_TRANSPARENT;
}
HR_RET(FindOutSurfFormat(&(DDSurf_PixFmt(pLcl)), &(pSpanTex->Format)));
if (pSpanTex->Format == D3DI_SPTFMT_PALETTE8 ||
pSpanTex->Format == D3DI_SPTFMT_PALETTE4)
{
if (pLcl->lpDDPalette)
{
LPDDRAWI_DDRAWPALETTE_GBL pPal = pLcl->lpDDPalette->lpLcl->lpGbl;
if (pPal->dwFlags & DDRAWIPAL_ALPHA)
{
pSpanTex->uFlags |= D3DI_SPANTEX_ALPHAPALETTE;
}
pSpanTex->pPalette = (PUINT32)pPal->lpColorTable;
}
if (pSpanTex->Format == D3DI_SPTFMT_PALETTE8)
{
pSpanTex->iPaletteSize = 256;
}
else
{
// PALETTE4
pSpanTex->iPaletteSize = 16;
}
}
pSpanTex->TexAddrU = D3DTADDRESS_WRAP;
pSpanTex->TexAddrV = D3DTADDRESS_WRAP;
pSpanTex->BorderColor = RGBA_MAKE(0xff, 0x00, 0xff, 0xff);
// assign first pSurf here (mipmap chain gets assigned below)
pSpanTex->pSurf[0] = pDDS;
// Check for mipmap if any.
LPDIRECTDRAWSURFACE pTmpS;
// iPreSizeU and iPreSizeV store the size(u and v) of the previous level
// mipmap. They are init'ed with the first texture size.
INT16 iPreSizeU = pSpanTex->iSizeU, iPreSizeV = pSpanTex->iSizeV;
for (;;)
{
memset(&ddscaps, 0, sizeof(DDSCAPS));
ddscaps.dwCaps = DDSCAPS_TEXTURE;
hr = pDDS->GetAttachedSurface(&ddscaps, &pTmpS); //implicit AddRef
if (hr == DDERR_NOTFOUND)
{
break;
}
else if (hr != D3D_OK)
{
return hr;
}
pDDS = pTmpS;
pSpanTex->cLODTex ++;
pSpanTex->pSurf[pSpanTex->cLODTex] = pTmpS;
}
pSpanTex->dwSize = sizeof(D3DI_SPANTEX);
return D3D_OK;
}
//----------------------------------------------------------------------------
//
// RemoveTexture
//
// Check to see if the to-be-destroyed pSpanTex is currently used by the
// context. If yes, set the according entry to be NULL to disable texture.
//
//----------------------------------------------------------------------------
void D3DContext::RemoveTexture(PD3DI_SPANTEX pSpanTex)
{
INT i;
INT cActTex = (INT)m_RastCtx.cActTex;
for (i = 0; i < cActTex; i++)
{
if (m_RastCtx.pTexture[i] == pSpanTex)
{
// NULL out the according texture and set dirty bits
m_RastCtx.cActTex --;
StateChanged(D3DRENDERSTATE_TEXTUREHANDLE);
m_RastCtx.pTexture[i] = NULL;
for (int j=pSpanTex->cLODTex;j>0;j--) //release attached surfs
{
pSpanTex->pSurf[j]->Release();
}
}
}
}
//----------------------------------------------------------------------------
//
// RastTextureCreate
//
// Creates a RAST texture and initializes it with the info passed in.
//
//----------------------------------------------------------------------------
DWORD __stdcall
RastTextureCreate(LPD3DHAL_TEXTURECREATEDATA pTexData)
{
PD3DI_SPANTEX *ppSpanTex;
PD3DI_SPANTEX pSpanTex;
D3DContext *pDCtx;
VALIDATE_D3DCONTEXT("RastTextureCreate", pTexData);
// Create the span texture
ppSpanTex = new PD3DI_SPANTEX;
pSpanTex = new D3DI_SPANTEX;
if (ppSpanTex == NULL || pSpanTex == NULL)
{
delete ppSpanTex;
delete pSpanTex;
D3D_ERR("(Rast) Out of memory in RastTextureCreate");
pTexData->ddrval = DDERR_OUTOFMEMORY;
return DDHAL_DRIVER_HANDLED;
}
memset(pSpanTex, 0, sizeof(D3DI_SPANTEX));
// Point indirector to this texture initially.
*ppSpanTex = pSpanTex;
// Init the span texture
if ((pTexData->ddrval = pDCtx->InitSpanTexture(pSpanTex, pTexData->lpDDS))
!= D3D_OK)
{
delete ppSpanTex;
delete pSpanTex;
return DDHAL_DRIVER_HANDLED;
}
if ((pTexData->ddrval = pDCtx->SetSizesSpanTexture(pSpanTex))
!= D3D_OK)
{
delete ppSpanTex;
delete pSpanTex;
return DDHAL_DRIVER_HANDLED;
}
// ppSpanTex is used as the texture handle returned to d3dim.
pTexData->dwHandle = (UINT32)(ULONG_PTR)ppSpanTex;
pTexData->ddrval = D3D_OK;
return DDHAL_DRIVER_HANDLED;
}
//----------------------------------------------------------------------------
//
// RastTextureDestroy
//
// Destroy a RAST texture.
//
//----------------------------------------------------------------------------
DWORD __stdcall
RastTextureDestroy(LPD3DHAL_TEXTUREDESTROYDATA pTexDestroyData)
{
PD3DI_SPANTEX *ppSpanTex;
PD3DI_SPANTEX pSpanTex;
D3DContext *pDCtx;
VALIDATE_D3DCONTEXT("RastTextureDestroy", pTexDestroyData);
if (!VALID_D3DI_SPANTEX_PTR_PTR(
(PD3DI_SPANTEX*)ULongToPtr(pTexDestroyData->dwHandle)))
{
D3D_ERR("(Rast) in RastTextureDestroy, invalid texture handle");
pTexDestroyData->ddrval = DDERR_INVALIDPARAMS;
return DDHAL_DRIVER_HANDLED;
}
// Find the texture
ppSpanTex = (PD3DI_SPANTEX *)ULongToPtr(pTexDestroyData->dwHandle);
pSpanTex = *ppSpanTex;
pDCtx->RemoveTexture(pSpanTex);
// Delete it
if (pSpanTex)
{
delete ppSpanTex;
delete pSpanTex;
}
else
{
pTexDestroyData->ddrval = DDERR_INVALIDPARAMS;
}
return DDHAL_DRIVER_HANDLED;
}
//----------------------------------------------------------------------------
//
// RastTextureGetSurf
//
// Returns the surface pointer associate with a texture handle.
//
//----------------------------------------------------------------------------
DWORD __stdcall
RastTextureGetSurf(LPD3DHAL_TEXTUREGETSURFDATA pTexGetSurf)
{
D3DContext *pDCtx;
VALIDATE_D3DCONTEXT("RastTextureGetSurf", pTexGetSurf);
// Check out the span texture
PD3DI_SPANTEX pSpanTex;
pSpanTex = HANDLE_TO_SPANTEX(pTexGetSurf->dwHandle);
if (pSpanTex)
{
pTexGetSurf->lpDDS = (UINT_PTR)pSpanTex->pSurf[0];
pTexGetSurf->ddrval = D3D_OK;
}
else
{
pTexGetSurf->ddrval = DDERR_INVALIDPARAMS;
}
return DDHAL_DRIVER_HANDLED;
}
//----------------------------------------------------------------------------
//
// RastLockSpanTexture
//
// Lock current texture surface before the texture bits are accessed.
//
//----------------------------------------------------------------------------
HRESULT
D3DContext::RastLockSpanTexture(void)
{
INT i, j;
PD3DI_SPANTEX pSpanTex;
HRESULT hr;
if (IsTextureOff())
{
return D3D_OK;
}
DDASSERT((m_uFlags & D3DCONTEXT_TEXTURE_LOCKED) == 0);
for (j = 0;
j < (INT)m_RastCtx.cActTex;
j++)
{
pSpanTex = m_RastCtx.pTexture[j];
if (pSpanTex->uFlags & D3DI_SPANTEX_MAXMIPLEVELS_DIRTY)
{
hr = SetSizesSpanTexture(pSpanTex);
if (hr != D3D_OK)
{
goto EH_Unlock;
}
}
INT iFirstSurf = min(pSpanTex->iMaxMipLevel, pSpanTex->cLODTex);
// Currently recursive locks are not allowed.
DDASSERT((pSpanTex->uFlags & D3DI_SPANTEX_SURFACES_LOCKED) == 0);
for (i = iFirstSurf; i <= pSpanTex->cLODTex; i++)
{
hr = LockSurface(pSpanTex->pSurf[i],
(LPVOID*)&(pSpanTex->pBits[i-iFirstSurf]));
if (hr != D3D_OK)
{
// Unlock any partial mipmap locks we've taken, as
// RastUnlock can only handle entire textures being
// locked or unlocked.
while (--i >= 0)
{
UnlockSurface(pSpanTex->pSurf[i]);
}
// Make sure that i is signed and that the above
// loop exited properly.
DDASSERT(i < 0);
goto EH_Unlock;
}
}
pSpanTex->uFlags |= D3DI_SPANTEX_SURFACES_LOCKED;
}
m_uFlags |= D3DCONTEXT_TEXTURE_LOCKED;
return D3D_OK;
EH_Unlock:
if (j > 0)
{
// Unlock complete textures we've already locked.
// RastUnlock will check the flags to figure
// out which ones to unlock.
RastUnlockSpanTexture();
}
return hr;
}
//----------------------------------------------------------------------------
//
// RastUnlockTexture
//
// Unlock texture surface after the texture bits are accessed.
// The input is a D3DI_SPANTEX. NULL texture needs to be checked before this
// function gets called.
//
//----------------------------------------------------------------------------
void
D3DContext::RastUnlockSpanTexture(void)
{
INT i, j;
PD3DI_SPANTEX pSpanTex;;
if (IsTextureOff())
{
return;
}
DDASSERT((m_uFlags & D3DCONTEXT_TEXTURE_LOCKED) != 0);
for (j = 0;
j < (INT)m_RastCtx.cActTex;
j++)
{
pSpanTex = m_RastCtx.pTexture[j];
INT iFirstSurf = min(pSpanTex->iMaxMipLevel, pSpanTex->cLODTex);
// RastUnlock is used for cleanup in RastLock so it needs to
// be able to handle partially locked mipmap chains.
if (pSpanTex->uFlags & D3DI_SPANTEX_SURFACES_LOCKED)
{
for (i = iFirstSurf; i <= pSpanTex->cLODTex; i++)
{
UnlockSurface(pSpanTex->pSurf[i]);
}
pSpanTex->uFlags &= ~D3DI_SPANTEX_SURFACES_LOCKED;
}
}
m_uFlags &= ~D3DCONTEXT_TEXTURE_LOCKED;
}
//----------------------------------------------------------------------------
//
// UpdateColorKeyAndPalette
//
// Updates the color key value and palette.
//
// Also, if the ColorKey enable for the texture has changed, set the texture handle
// dirty bit so the new mode is recognized in span init.
//
//----------------------------------------------------------------------------
void
D3DContext::UpdateColorKeyAndPalette(void)
{
INT j;
PD3DI_SPANTEX pSpanTex;
// Set the transparent bit and the transparent color with pSurf[0]
LPDDRAWI_DDRAWSURFACE_LCL pLcl;
for (j = 0;
j < (INT)m_RastCtx.cActTex;
j++)
{
pSpanTex = m_RastCtx.pTexture[j];
if ((pSpanTex != NULL) && (pSpanTex->pSurf[0] != NULL))
{
pLcl = ((LPDDRAWI_DDRAWSURFACE_INT) pSpanTex->pSurf[0])->lpLcl;
// Palette might be changed
if (pSpanTex->Format == D3DI_SPTFMT_PALETTE8 ||
pSpanTex->Format == D3DI_SPTFMT_PALETTE4)
{
if (pLcl->lpDDPalette)
{
LPDDRAWI_DDRAWPALETTE_GBL pPal = pLcl->lpDDPalette->lpLcl->lpGbl;
if (pPal->dwFlags & DDRAWIPAL_ALPHA)
{
pSpanTex->uFlags |= D3DI_SPANTEX_ALPHAPALETTE;
}
pSpanTex->pPalette = (PUINT32)pPal->lpColorTable;
}
}
if ((pLcl->dwFlags & DDRAWISURF_HASCKEYSRCBLT) != 0)
{
// texture has a ColorKey value
pSpanTex->TransparentColor =
pLcl->ddckCKSrcBlt.dwColorSpaceHighValue;
if (!(pSpanTex->uFlags & D3DI_SPANTEX_HAS_TRANSPARENT))
{
pSpanTex->uFlags |= D3DI_SPANTEX_HAS_TRANSPARENT;
// make sure this state change is recognized, and a new
// texture read function is used
StateChanged(RAST_TSS_DIRTYBIT(j, D3DTSS_TEXTUREMAP));
}
}
else
{
// texture does not have a ColorKey value
if (pSpanTex->uFlags & D3DI_SPANTEX_HAS_TRANSPARENT)
{
pSpanTex->uFlags &= ~D3DI_SPANTEX_HAS_TRANSPARENT;
// make sure this state change is recognized, and a new
// texture read function is used
StateChanged(RAST_TSS_DIRTYBIT(j, D3DTSS_TEXTUREMAP));
}
}
}
}
}
//----------------------------------------------------------------------------
//
// Dp2TextureStageState
//
// Called by Drawprim2 to set texture stage states..
//
//----------------------------------------------------------------------------
HRESULT
D3DContext::Dp2TextureStageState(LPD3DHAL_DP2COMMAND pCmd, DWORD dwFvf)
{
WORD wStateCount = pCmd->wStateCount;
INT i;
HRESULT hr;
LPD3DHAL_DP2TEXTURESTAGESTATE pTexStageState =
(D3DHAL_DP2TEXTURESTAGESTATE *)(pCmd + 1);
// Flush the prim proc before any state changs
HR_RET(End(FALSE));
for (i = 0; i < (INT)wStateCount; i++, pTexStageState++)
{
HR_RET(SetTextureStageState((DWORD)pTexStageState->wStage,
(DWORD)pTexStageState->TSState,
pTexStageState->dwValue));
}
HR_RET(CheckFVF(dwFvf));
hr = Begin();
return hr;
}