//---------------------------------------------------------------------------- // // drawprim.cpp // // Implements DrawOnePrimitive, DrawOneIndexedPrimitive and // DrawPrimitives. // // Copyright (C) Microsoft Corporation, 1997. // //---------------------------------------------------------------------------- #include "pch.cpp" #pragma hdrstop //---------------------------------------------------------------------------- // // CheckFVF // // Check a FVF control word and then init m_fvfData accordingly // //---------------------------------------------------------------------------- HRESULT FASTCALL D3DContext::CheckFVF(DWORD dwFVF) { // check if FVF controls have changed if ( (m_fvfData.preFVF == dwFVF) && (m_fvfData.TexIdx[0] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[0][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.TexIdx[1] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[1][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.TexIdx[2] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[2][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.TexIdx[3] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[3][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.TexIdx[4] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[4][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.TexIdx[5] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[5][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.TexIdx[6] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[6][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.TexIdx[7] == (INT)(0xffff&m_RastCtx.pdwTextureStageState[7][D3DTSS_TEXCOORDINDEX])) && (m_fvfData.cActTex == m_RastCtx.cActTex) ) { return D3D_OK; } #if DBG // This is added here per Iouri's request. It will make it easier for him // to test his code for legacy drivers. if (dwFVF == 0) { dwFVF = D3DFVF_TLVERTEX; } #endif memset(&m_fvfData, 0, sizeof(FVFDATA)); m_fvfData.preFVF = dwFVF; INT32 i; for ( i = 0; i < D3DHAL_TSS_MAXSTAGES; i++) { m_fvfData.TexIdx[i] = 0xffff&m_RastCtx.pdwTextureStageState[i][D3DTSS_TEXCOORDINDEX]; } m_fvfData.cActTex = m_RastCtx.cActTex; #if DBG // We only support max 8 texture coords if (m_fvfData.TexIdx[0] > 7 || m_fvfData.TexIdx[1] > 7) { D3D_WARN(0, "(Rast) Texture coord index bigger than max supported."); return DDERR_INVALIDPARAMS; } #endif // Update the copy of wrap states in RastCtx for ( i = 0; i < D3DHAL_TSS_MAXSTAGES; i++) { m_RastCtx.pdwWrap[i] = m_RastCtx.pdwRenderState[ (D3DRENDERSTATETYPE)(D3DRENDERSTATE_WRAP0+m_fvfData.TexIdx[i])]; } // do either true FVF parsing or legacy TLVERTEX handling if ( (m_RastCtx.BeadSet != D3DIBS_RAMP) && ( (dwFVF != D3DFVF_TLVERTEX) || (0 != m_fvfData.TexIdx[0]) || (m_RastCtx.cActTex > 1) ) ) { // New (non TL)FVF vertex // XYZ if ( (dwFVF & (D3DFVF_RESERVED0 | D3DFVF_RESERVED1 | D3DFVF_RESERVED2 | D3DFVF_NORMAL)) || ((dwFVF & (D3DFVF_XYZ | D3DFVF_XYZRHW)) == 0) ) { // can't set reserved bits, shouldn't have normals in // output to rasterizers, and must have coordinates return DDERR_INVALIDPARAMS; } m_fvfData.stride = sizeof(D3DVALUE) * 3; if (dwFVF & D3DFVF_XYZRHW) { m_fvfData.offsetRHW = m_fvfData.stride; m_fvfData.stride += sizeof(D3DVALUE); } if (dwFVF & D3DFVF_DIFFUSE) { m_fvfData.offsetDiff = m_fvfData.stride; m_fvfData.stride += sizeof(D3DCOLOR); } if (dwFVF & D3DFVF_SPECULAR) { m_fvfData.offsetSpec = m_fvfData.stride; m_fvfData.stride += sizeof(D3DCOLOR); } INT iTexCount = (dwFVF & D3DFVF_TEXCOUNT_MASK) >> D3DFVF_TEXCOUNT_SHIFT; #if DBG INT iTexIdx0 = m_fvfData.TexIdx[0], iTexIdx1 = m_fvfData.TexIdx[1]; if (iTexCount > 0) { // set offset for Textures for ( i = 0; i < D3DHAL_TSS_MAXSTAGES; i ++) { INT iTexIdx = m_fvfData.TexIdx[i]; if ( iTexIdx >= iTexCount) { D3D_WARN(1, "(Rast)Texture coord index bigger than texture coord count."); iTexIdx = 0; } m_fvfData.offsetTex[i] = (SHORT)(m_fvfData.stride + 2*sizeof(D3DVALUE)*iTexIdx); } // update stride m_fvfData.stride += (USHORT)(iTexCount * (sizeof(D3DVALUE) * 2)); } #else if (iTexCount > 0) { // set offset for Textures for ( i = 0; i < D3DHAL_TSS_MAXSTAGES; i ++) { m_fvfData.offsetTex[i] = (SHORT)(m_fvfData.stride + 2*sizeof(D3DVALUE)*m_fvfData.TexIdx[i]); } // update stride m_fvfData.stride += (USHORT)(iTexCount * (sizeof(D3DVALUE) * 2)); } #endif m_fvfData.vtxType = RAST_GENVERTEX; } else { // (Legacy) TL vertex if (0 < m_fvfData.TexIdx[0]) { D3D_ERR("(Rast) Texture coord index bigger than 0 for legacy TL vertex."); return DDERR_INVALIDPARAMS; } m_fvfData.stride = sizeof(D3DTLVERTEX); m_fvfData.vtxType = RAST_TLVERTEX; } UpdatePrimFunctionTbl(); return D3D_OK; } //---------------------------------------------------------------------------- // // PackGenVertex // // Pack a FvFVertex into RAST_GENERIC_VERTEX. This is called for every non TL // FVF vertex. It can be optimized for speed later. // //---------------------------------------------------------------------------- void FASTCALL D3DContext::PackGenVertex(PUINT8 pFvfVtx, RAST_GENERIC_VERTEX *pGenVtx) { pGenVtx->sx = *((D3DVALUE *)pFvfVtx); pGenVtx->sy = *((D3DVALUE *)pFvfVtx + 1); pGenVtx->sz = *((D3DVALUE *)pFvfVtx + 2); if (m_fvfData.offsetRHW) { pGenVtx->rhw = *((D3DVALUE *)(pFvfVtx + m_fvfData.offsetRHW)); } else { pGenVtx->rhw = 1.0f; } if (m_fvfData.offsetDiff) { pGenVtx->color = *((D3DCOLOR *)(pFvfVtx + m_fvfData.offsetDiff)); } else { pGenVtx->color = __DEFAULT_DIFFUSE; } if (m_fvfData.offsetSpec) { pGenVtx->specular = *((D3DCOLOR *)(pFvfVtx + m_fvfData.offsetSpec)); } else { pGenVtx->specular = __DEFAULT_SPECULAR; } for (INT32 i = 0; i < (INT32)m_fvfData.cActTex; i++) { if (m_fvfData.offsetTex[i]) { pGenVtx->texCoord[i].tu = *((D3DVALUE *)(pFvfVtx + m_fvfData.offsetTex[i])); pGenVtx->texCoord[i].tv = *((D3DVALUE *)(pFvfVtx + m_fvfData.offsetTex[i]) + 1); } else { pGenVtx->texCoord[i].tu = 0.0f; pGenVtx->texCoord[i].tv = 0.0f; } } } //---------------------------------------------------------------------------- // // DoDrawOnePrimitive // // Draw one list of primitives. It's called by both RastDrawOnePrimitive and // RastDrawPrimitives. // //---------------------------------------------------------------------------- HRESULT FASTCALL DoDrawOnePrimitive(LPVOID pCtx, PRIMITIVE_FUNTIONS *pfnPrims, UINT16 FvfStride, PUINT8 pVtx, D3DPRIMITIVETYPE PrimType, UINT cVertices) { INT i; PUINT8 pV0, pV1, pV2; HRESULT hr; switch (PrimType) { case D3DPT_POINTLIST: for (i = (INT)cVertices; i > 0; i--) { HR_RET(pfnPrims->pfnPoint(pCtx, pVtx)); pVtx += FvfStride; } break; case D3DPT_LINELIST: for (i = (INT)cVertices / 2; i > 0; i--) { pV0 = pVtx; pVtx += FvfStride; pV1 = pVtx; pVtx += FvfStride; HR_RET(pfnPrims->pfnLine(pCtx, pV0, pV1)); } break; case D3DPT_LINESTRIP: { pV1 = pVtx; // Disable last-pixel setting for shared verties and store prestate. pfnPrims->pfnStoreLastPixelState(pCtx, 1); // Initial pV0. for (i = (INT)cVertices - 1; i > 1; i--) { pV0 = pV1; pVtx += FvfStride; pV1 = pVtx; HR_RET(pfnPrims->pfnLine(pCtx, pV0, pV1)); } // Restore last-pixel setting. pfnPrims->pfnStoreLastPixelState(pCtx, 0); // Draw last line with last-pixel setting from state. if (i == 1) { pV0 = pVtx + FvfStride; HR_RET(pfnPrims->pfnLine(pCtx, pV1, pV0)); } } break; case D3DPT_TRIANGLELIST: for (i = (INT)cVertices; i > 0; i -= 3) { pV0 = pVtx; pVtx += FvfStride; pV1 = pVtx; pVtx += FvfStride; pV2 = pVtx; pVtx += FvfStride; HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); } break; case D3DPT_TRIANGLESTRIP: { // Get initial vertex values. pV1 = pVtx; pVtx += FvfStride; pV2 = pVtx; pVtx += FvfStride; for (i = (INT)cVertices - 2; i > 1; i -= 2) { pV0 = pV1; pV1 = pV2; pV2 = pVtx; pVtx += FvfStride; HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); pV0 = pV1; pV1 = pV2; pV2 = pVtx; pVtx += FvfStride; HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV2, pV1, PFN_TRIANGLE_5ARG_DEFAULT)); } if (i > 0) { pV0 = pV1; pV1 = pV2; pV2 = pVtx; HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); } } break; case D3DPT_TRIANGLEFAN: { pV2 = pVtx; pVtx += FvfStride; // Preload initial pV0. pV1 = pVtx; pVtx += FvfStride; for (i = (INT)cVertices - 2; i > 0; i--) { pV0 = pV1; pV1 = pVtx; pVtx += FvfStride; HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); } } break; default: D3D_ERR("(Rast) Unknown or unsupported primitive type " "requested of DrawOnePrimitive"); return DDERR_INVALIDPARAMS; } return D3D_OK; } //---------------------------------------------------------------------------- // // DoDrawOneIndexedPrimitive // // Draw one list of indexed primitives. It's called by // RastDrawOneIndexedPrimitive. // //---------------------------------------------------------------------------- HRESULT FASTCALL DoDrawOneIndexedPrimitive(LPVOID pCtx, PRIMITIVE_FUNTIONS *pfnPrims, UINT16 FvfStride, PUINT8 pVtx, LPWORD puIndices, D3DPRIMITIVETYPE PrimType, UINT cIndices) { INT i; PUINT8 pV0, pV1, pV2; HRESULT hr; switch(PrimType) { case D3DPT_POINTLIST: for (i = (INT)cIndices; i > 0; i--) { pV0 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnPoint(pCtx, pV0)); } break; case D3DPT_LINELIST: for (i = (INT)cIndices / 2; i > 0; i--) { pV0 = pVtx + FvfStride * (*puIndices++); pV1 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnLine(pCtx, pV0, pV1)); } break; case D3DPT_LINESTRIP: { // Disable last-pixel setting for shared verties and store prestate. pfnPrims->pfnStoreLastPixelState(pCtx, 1); // Initial pV1. pV1 = pVtx + FvfStride * (*puIndices++); for (i = (INT)cIndices - 1; i > 1; i--) { pV0 = pV1; pV1 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnLine(pCtx, pV0, pV1)); } // Restore last-pixel setting. pfnPrims->pfnStoreLastPixelState(pCtx, 0); // Draw last line with last-pixel setting from state. if (i == 1) { pV0 = pVtx + FvfStride * (*puIndices); HR_RET(pfnPrims->pfnLine(pCtx, pV1, pV0)); } } break; case D3DPT_TRIANGLELIST: for (i = (INT)cIndices; i > 0; i -= 3) { pV0 = pVtx + FvfStride * (*puIndices++); pV1 = pVtx + FvfStride * (*puIndices++); pV2 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); } break; case D3DPT_TRIANGLESTRIP: { // Get initial vertex values. pV1 = pVtx + FvfStride * (*puIndices++); pV2 = pVtx + FvfStride * (*puIndices++); for (i = (INT)cIndices - 2; i > 1; i -= 2) { pV0 = pV1; pV1 = pV2; pV2 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); pV0 = pV1; pV1 = pV2; pV2 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV2, pV1, PFN_TRIANGLE_5ARG_DEFAULT)); } if (i > 0) { pV0 = pV1; pV1 = pV2; pV2 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); } } break; case D3DPT_TRIANGLEFAN: { pV2 = pVtx + FvfStride * (*puIndices++); // Preload initial pV0. pV1 = pVtx + FvfStride * (*puIndices++); for (i = (INT)cIndices - 2; i > 0; i--) { pV0 = pV1; pV1 = pVtx + FvfStride * (*puIndices++); HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, PFN_TRIANGLE_5ARG_DEFAULT)); } } break; default: D3D_ERR("(Rast) Unknown or unsupported primitive type " "requested of DrawOneIndexedPrimitive"); return DDERR_INVALIDPARAMS; } return D3D_OK; } //---------------------------------------------------------------------------- // // DoDrawOneEdgeFlagTriangleFan // // Draw one list of triangle fans. It's called by both RastDrawOnePrimitive and // RastDrawPrimitives. // //---------------------------------------------------------------------------- HRESULT FASTCALL DoDrawOneEdgeFlagTriangleFan(LPVOID pCtx, PRIMITIVE_FUNTIONS *pfnPrims, UINT16 FvfStride, PUINT8 pVtx, UINT cVertices, UINT32 dwEdgeFlags) { INT i; PUINT8 pV0, pV1, pV2; HRESULT hr; pV2 = pVtx; pVtx += FvfStride; pV0 = pVtx; pVtx += FvfStride; pV1 = pVtx; pVtx += FvfStride; WORD wFlags = 0; if(dwEdgeFlags & 0x2) wFlags |= D3DTRIFLAG_EDGEENABLE1; if(dwEdgeFlags & 0x1) wFlags |= D3DTRIFLAG_EDGEENABLE3; if(cVertices == 3) { if(dwEdgeFlags & 0x4) wFlags |= D3DTRIFLAG_EDGEENABLE2; HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, wFlags)); return D3D_OK; } HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, wFlags)); UINT32 dwMask = 0x4; for (i = (INT)cVertices - 4; i > 0; i--) { pV0 = pV1; pV1 = pVtx; pVtx += FvfStride; if(dwEdgeFlags & dwMask) { HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, D3DTRIFLAG_EDGEENABLE1)); } else { HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, 0)); } dwMask <<= 1; } pV0 = pV1; pV1 = pVtx; wFlags = 0; if(dwEdgeFlags & dwMask) wFlags |= D3DTRIFLAG_EDGEENABLE1; dwMask <<= 1; if(dwEdgeFlags & dwMask) wFlags |= D3DTRIFLAG_EDGEENABLE2; HR_RET(pfnPrims->pfnTri(pCtx, pV0, pV1, pV2, wFlags)); return D3D_OK; } #if DBG //---------------------------------------------------------------------------- // // ValidatePrimType // // Check if the primitive type is supported. We could remove this function // after we have implemented all primitive types and then depend on D3DIM // to check if the primitive type is valid. // //---------------------------------------------------------------------------- inline HRESULT D3DContext::ValidatePrimType(D3DPRIMITIVETYPE PrimitiveType) { switch(PrimitiveType) { case D3DPT_POINTLIST: case D3DPT_LINELIST: case D3DPT_LINESTRIP: case D3DPT_TRIANGLELIST: case D3DPT_TRIANGLEFAN: case D3DPT_TRIANGLESTRIP: break; default: D3D_ERR("(Rast) PrimitiveType not supported by the new rasterizer."); return DDERR_INVALIDPARAMS; } return D3D_OK; } #endif //---------------------------------------------------------------------------- // // CheckDrawOnePrimitive // // Check if the DRAWONEPRIMITIVEDATA is valid. // //---------------------------------------------------------------------------- inline HRESULT D3DContext::CheckDrawOnePrimitive(LPD3DHAL_DRAWONEPRIMITIVEDATA pOnePrimData) { #if DBG HRESULT hr; if (pOnePrimData == NULL || pOnePrimData->dwhContext == 0 || pOnePrimData->lpvVertices == NULL) { D3D_ERR("(Rast) Invalid data passed to the new rasterizer."); return DDERR_INVALIDPARAMS; } HR_RET(ValidatePrimType(pOnePrimData->PrimitiveType)); #endif return D3D_OK; } //---------------------------------------------------------------------------- // // CheckDrawOneIndexedPrimitive // // Check if the DRAWONEINDEXEDPRIMITIVEDATA is valid. // //---------------------------------------------------------------------------- inline HRESULT D3DContext::CheckDrawOneIndexedPrimitive( LPD3DHAL_DRAWONEINDEXEDPRIMITIVEDATA pOneIdxPrimData) { #if DBG HRESULT hr; if (pOneIdxPrimData == NULL || pOneIdxPrimData->dwhContext == 0 || pOneIdxPrimData->lpvVertices == NULL || pOneIdxPrimData->lpwIndices == NULL) { D3D_ERR("(Rast) Invalid data passed to the new rasterizer."); return DDERR_INVALIDPARAMS; } HR_RET(ValidatePrimType(pOneIdxPrimData->PrimitiveType)); #endif return D3D_OK; } //---------------------------------------------------------------------------- // // RastDrawOnePrimitive // // Draw one list of primitives. This is called by D3DIM for API DrawPrimitive. // //---------------------------------------------------------------------------- DWORD __stdcall RastDrawOnePrimitive(LPD3DHAL_DRAWONEPRIMITIVEDATA pOnePrimData) { HRESULT hr; D3DContext *pDCtx; VALIDATE_D3DCONTEXT("RastDrawOnePrimitive", pOnePrimData); if ((pOnePrimData->ddrval = pDCtx->CheckDrawOnePrimitive(pOnePrimData)) != DD_OK) { return DDHAL_DRIVER_HANDLED; } // Check for FVF vertex, and init FVF related fileds if necessary // Assume the control word is passed in through dwFlags CHECK_FVF(pOnePrimData->ddrval, pDCtx, (DWORD)pOnePrimData->dwFVFControl); pOnePrimData->ddrval = pDCtx->Begin(); if (pOnePrimData->ddrval != D3D_OK) { return DDHAL_DRIVER_HANDLED; } pOnePrimData->ddrval = pDCtx->DrawOnePrimitive((PUINT8)pOnePrimData->lpvVertices, pOnePrimData->PrimitiveType, pOnePrimData->dwNumVertices); hr = pDCtx->End(); if (pOnePrimData->ddrval == D3D_OK) { pOnePrimData->ddrval = hr; } return DDHAL_DRIVER_HANDLED; } //---------------------------------------------------------------------------- // // RastDrawOneIndexedPrimitive // // Draw one list of primitives. This is called by D3DIM for API // DrawIndexedPrimitive. // //---------------------------------------------------------------------------- DWORD __stdcall RastDrawOneIndexedPrimitive(LPD3DHAL_DRAWONEINDEXEDPRIMITIVEDATA pOneIdxPrimData) { HRESULT hr; D3DContext *pDCtx; VALIDATE_D3DCONTEXT("RastDrawOneIndexedPrimitive", pOneIdxPrimData); if ((pOneIdxPrimData->ddrval = pDCtx->CheckDrawOneIndexedPrimitive(pOneIdxPrimData)) != DD_OK) { return DDHAL_DRIVER_HANDLED; } // Check for FVF vertex, and init FVF related fileds if necessary // Assume the control word is passed in through dwFlags CHECK_FVF(pOneIdxPrimData->ddrval, pDCtx, (DWORD)pOneIdxPrimData->dwFVFControl); pOneIdxPrimData->ddrval = pDCtx->Begin(); if (pOneIdxPrimData->ddrval != D3D_OK) { return DDHAL_DRIVER_HANDLED; } pOneIdxPrimData->ddrval = pDCtx->DrawOneIndexedPrimitive((PUINT8)pOneIdxPrimData->lpvVertices, pOneIdxPrimData->lpwIndices, pOneIdxPrimData->PrimitiveType, pOneIdxPrimData->dwNumIndices); hr = pDCtx->End(); if (pOneIdxPrimData->ddrval == D3D_OK) { pOneIdxPrimData->ddrval = hr; } return DDHAL_DRIVER_HANDLED; } //---------------------------------------------------------------------------- // // RastDrawPrimitives // // This is called by D3DIM for a list of batched API DrawPrimitive calls. // //---------------------------------------------------------------------------- DWORD __stdcall RastDrawPrimitives(LPD3DHAL_DRAWPRIMITIVESDATA pDrawPrimData) { PUINT8 pData = (PUINT8)pDrawPrimData->lpvData; LPD3DHAL_DRAWPRIMCOUNTS pDrawPrimitiveCounts; D3DContext *pDCtx; VALIDATE_D3DCONTEXT("RastDrawPrimitives", pDrawPrimData); pDrawPrimitiveCounts = (LPD3DHAL_DRAWPRIMCOUNTS)pData; // Check for FVF vertex when there are actually something to be drawn, and // init FVF related fileds if necessary Assume the control word is passed // in through dwReserved if (pDrawPrimitiveCounts->wNumVertices > 0) { CHECK_FVF(pDrawPrimData->ddrval, pDCtx, pDrawPrimData->dwFVFControl); } // Skip state check and texture lock if the first thing is state change if (pDrawPrimitiveCounts->wNumStateChanges == 0) { pDrawPrimData->ddrval =pDCtx->Begin(); if (pDrawPrimData->ddrval != D3D_OK) { goto EH_Exit; } } // Loop through the data, update render states // and then draw the primitive for (;;) { pDrawPrimitiveCounts = (LPD3DHAL_DRAWPRIMCOUNTS)pData; pData += sizeof(D3DHAL_DRAWPRIMCOUNTS); // // Update render states // if (pDrawPrimitiveCounts->wNumStateChanges > 0) { // Flush the prim proc before any state changs pDrawPrimData->ddrval = pDCtx->End(FALSE); if (pDrawPrimData->ddrval != D3D_OK) { return DDHAL_DRIVER_HANDLED; } pDrawPrimData->ddrval = pDCtx->UpdateRenderStates((LPDWORD)pData, pDrawPrimitiveCounts->wNumStateChanges); if (pDrawPrimData->ddrval != D3D_OK) { goto EH_Exit; } pData += pDrawPrimitiveCounts->wNumStateChanges * sizeof(DWORD) * 2; } // Check for exit if (pDrawPrimitiveCounts->wNumVertices == 0) { break; } // Align pointer to vertex data pData = (PUINT8) ((UINT_PTR)(pData + (DP_VTX_ALIGN - 1)) & ~(DP_VTX_ALIGN - 1)); // Delayed change until we really need to render something if (pDrawPrimitiveCounts->wNumStateChanges > 0) { // We might have a new texture so lock. pDrawPrimData->ddrval = pDCtx->Begin(); if (pDrawPrimData->ddrval != D3D_OK) { goto EH_Exit; } } // // Primitives // pDrawPrimData->ddrval = pDCtx->DrawOnePrimitive((PUINT8)pData, (D3DPRIMITIVETYPE)pDrawPrimitiveCounts->wPrimitiveType, pDrawPrimitiveCounts->wNumVertices); if (pDrawPrimData->ddrval != DD_OK) { goto EH_Exit; } pData += pDrawPrimitiveCounts->wNumVertices * pDCtx->GetFvfStride(); } EH_Exit: HRESULT hr; hr = pDCtx->End(); if (pDrawPrimData->ddrval == D3D_OK) { pDrawPrimData->ddrval = hr; } return DDHAL_DRIVER_HANDLED; }