//---------------------------------------------------------------------------- // // 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; }