mirror of https://github.com/tongzx/nt5src
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.
1354 lines
56 KiB
1354 lines
56 KiB
/*==========================================================================;
|
|
*
|
|
* Copyright (C) 1998 Microsoft Corporation. All Rights Reserved.
|
|
*
|
|
* File: loops.mcp
|
|
* Content: Generates code for multiple loop geometry pipeline
|
|
*
|
|
***************************************************************************/
|
|
#include "pch.cpp"
|
|
#pragma hdrstop
|
|
|
|
include(`pvvid.mh') dnl
|
|
#include "stdio.h"
|
|
|
|
dnl//------------------------------------------------------------------
|
|
dnl// d_ComputeSpecular
|
|
dnl//
|
|
dnl// Generates code to compute specular component based on a dot product
|
|
dnl//
|
|
dnl// Arguments:
|
|
dnl// $1 - margin count
|
|
dnl// $2 - if present, equal to the attenuation factor
|
|
dnl// dot - dot product
|
|
dnl// pv - process vertices structure
|
|
dnl// d_Op - operation "=" or "+="
|
|
dnl// d_LightingFlags - DWORD
|
|
dnl// d_SPECULARCOMPUTED - bit
|
|
dnl// d_pInpSpecular - vertex specular color (DWORD*)
|
|
dnl// d_OutSpecular - output specular color, (D3DFE_COLOR)
|
|
dnl//
|
|
define(`d_ComputeSpecular',`dnl
|
|
d_empty_($1)if (FLOAT_CMP_POS(dot, >=, pv->lighting.specThreshold))
|
|
d_margin($1){
|
|
d_margin($1) d_LightingFlags |= __LIGHT_SPECULARCOMPUTED;
|
|
d_margin($1) // Compute power = dot**SpecularExponent;
|
|
d_margin($1) D3DVALUE power;
|
|
d_margin($1) if (FLOAT_CMP_PONE(dot, <))
|
|
d_margin($1) {
|
|
d_margin($1) int indx;
|
|
d_margin($1) float v;
|
|
d_margin($1) float dot_floor;
|
|
d_margin($1) dot *= 255.0f;
|
|
d_margin($1) dot_floor = (float)floor(dot);
|
|
d_margin($1) indx = FTOI(dot_floor);
|
|
d_margin($1) dot -= dot_floor;
|
|
d_margin($1) v = pv->lighting.currentSpecTable[indx];
|
|
d_margin($1) power = v + (pv->lighting.currentSpecTable[indx+1] - v)*dot;
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) power = pv->lighting.currentSpecTable[255];
|
|
dnl
|
|
ifelse($#,2,`d_margin($1+1)power*= $2;')dnl// If parameter 2 (attenuation) is present, use it
|
|
|
|
d_margin($1) // Update specular component
|
|
d_margin($1) if (!(dwFlags & D3DPV_COLORVERTEX_S))
|
|
d_margin($1) {
|
|
d_margin($1) d_OutSpecular.r d_Op light->specularMat.r * power;
|
|
d_margin($1) d_OutSpecular.g d_Op light->specularMat.g * power;
|
|
d_margin($1) d_OutSpecular.b d_Op light->specularMat.b * power;
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) const D3DVALUE r = (D3DVALUE)RGBA_GETRED(*d_pInpSpecular);
|
|
d_margin($1) const D3DVALUE g = (D3DVALUE)RGBA_GETGREEN(*d_pInpSpecular);
|
|
d_margin($1) const D3DVALUE b = (D3DVALUE)RGBA_GETBLUE(*d_pInpSpecular);
|
|
d_margin($1) d_OutSpecular.r d_Op light->specular.r * r * power;
|
|
d_margin($1) d_OutSpecular.g d_Op light->specular.g * g * power;
|
|
d_margin($1) d_OutSpecular.b d_Op light->specular.b * b * power;
|
|
d_margin($1) }
|
|
d_margin($1)}')dnl
|
|
dnl//------------------------------------------------------------------
|
|
dnl// d_UpdateDiffuseColor
|
|
dnl//
|
|
dnl// Generates code to compute diffuse component, based on a dot product
|
|
dnl//
|
|
dnl// Arguments:
|
|
dnl// $1 - margin count
|
|
dnl// $2 - operation "=" or "+="
|
|
dnl// dot - dot product
|
|
dnl// d_LightingFlags - DWORD
|
|
dnl// d_pInpDiffuse - vertex specular color (DWORD*)
|
|
dnl// d_OutDiffuse - output specular color, (D3DFE_COLOR)
|
|
dnl//
|
|
define(`d_UpdateDiffuseColor',`dnl
|
|
d_empty_($1)if (!(dwFlags & D3DPV_COLORVERTEX_D))
|
|
d_margin($1){
|
|
d_margin($1) d_OutDiffuse.r $2 light->diffuseMat.r * dot;
|
|
d_margin($1) d_OutDiffuse.g $2 light->diffuseMat.g * dot;
|
|
d_margin($1) d_OutDiffuse.b $2 light->diffuseMat.b * dot;
|
|
d_margin($1)}
|
|
d_margin($1)else
|
|
d_margin($1){
|
|
d_margin($1) const D3DVALUE r = (D3DVALUE)RGBA_GETRED(*d_pInpDiffuse);
|
|
d_margin($1) const D3DVALUE g = (D3DVALUE)RGBA_GETGREEN(*d_pInpDiffuse);
|
|
d_margin($1) const D3DVALUE b = (D3DVALUE)RGBA_GETBLUE(*d_pInpDiffuse);
|
|
d_margin($1) d_OutDiffuse.r $2 light->diffuse.r * r * dot;
|
|
d_margin($1) d_OutDiffuse.g $2 light->diffuse.g * g * dot;
|
|
d_margin($1) d_OutDiffuse.b $2 light->diffuse.b * b * dot;
|
|
d_margin($1)}
|
|
d_margin($1)d_LightingFlags |= __LIGHT_DIFFUSECOMPUTED;')dnl
|
|
dnl//------------------------------------------------------------------
|
|
dnl// d_UpdateAmbientColor
|
|
dnl//
|
|
dnl// Generates code to compute ambient component
|
|
dnl//
|
|
dnl// Arguments:
|
|
dnl// $1 - margin count
|
|
dnl// $2 - "* att" or empty
|
|
dnl// dot - dot product
|
|
dnl// d_Op - operation "=" or "+="
|
|
dnl// d_LightingFlags - DWORD
|
|
dnl// d_OutDiffuse - output specular color, (D3DFE_COLOR)
|
|
dnl//
|
|
define(`d_UpdateAmbientColor',`dnl
|
|
d_empty_($1)if (!(light->flags & D3DLIGHTI_AMBIENT_IS_ZERO))
|
|
d_margin($1){
|
|
d_margin($1) if (!(dwFlags & D3DPV_COLORVERTEX_A))
|
|
d_margin($1) {
|
|
d_margin($1) d_OutDiffuse.r d_Op light->ambientMat.r $2;
|
|
d_margin($1) d_OutDiffuse.g d_Op light->ambientMat.g $2;
|
|
d_margin($1) d_OutDiffuse.b d_Op light->ambientMat.b $2;
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) const D3DVALUE r = (D3DVALUE)RGBA_GETRED(*d_pInpAmbient);
|
|
d_margin($1) const D3DVALUE g = (D3DVALUE)RGBA_GETGREEN(*d_pInpAmbient);
|
|
d_margin($1) const D3DVALUE b = (D3DVALUE)RGBA_GETBLUE(*d_pInpAmbient);
|
|
d_margin($1) d_OutDiffuse.r d_Op light->ambient.r * r $2;
|
|
d_margin($1) d_OutDiffuse.g d_Op light->ambient.g * g $2;
|
|
d_margin($1) d_OutDiffuse.b d_Op light->ambient.b * b $2;
|
|
d_margin($1) }
|
|
d_margin($1) d_LightingFlags |= __LIGHT_DIFFUSECOMPUTED;
|
|
d_margin($1)}')dnl
|
|
dnl//------------------------------------------------------------------
|
|
dnl// d_Directional7
|
|
dnl//
|
|
dnl// Generate code to light a vertex using directional or parallel point light.
|
|
dnl// Model space and camera space lighting are handled
|
|
dnl//
|
|
dnl// Arguments:
|
|
dnl/ $1 - margin count
|
|
dnl// d_pInpPosition - input position pointer (D3DVERTEX*)
|
|
dnl// d_TmpPosition - temporary position buffer (D3DVECTOR).
|
|
dnl// Used in camera space lighting
|
|
dnl// d_pInpNormal - input normal pointer (D3DVECTOR*)
|
|
dnl// d_TmpNormal - temporary normal buffer (D3DVECTOR)
|
|
dnl// Used in camera space lighting
|
|
dnl// d_Space - Defines the coordinate system: modelSpace or cameraSpace
|
|
dnl// d_LightingFlags - DWORD where __LIGHT_ bits are defined
|
|
dnl//
|
|
dnl// For camera space lighting vertex normal is assumed to be already transformed
|
|
dnl//
|
|
define(`d_Directional7',`dnl
|
|
d_empty_($1)D3DVALUE dot;
|
|
d_margin($1)d_UpdateAmbientColor($1)
|
|
d_margin($1)if (!(pv->dwVIDIn & D3DFVF_NORMAL))
|
|
d_margin($1) goto l_exit;
|
|
d_margin($1)
|
|
ifelse(d_Space,modelSpace,`
|
|
d_margin($1)dot = VecDot(light->model_direction, (*d_pInpNormal));',`
|
|
d_margin($1)dot = VecDot(light->model_direction, d_TmpNormal);')
|
|
dnl// endif
|
|
d_margin($1)
|
|
d_margin($1)if (FLOAT_GTZ(dot))
|
|
d_margin($1){
|
|
ifelse(d_Op,+=,`dnl
|
|
d_margin($1) d_UpdateDiffuseColor($1+1,+=)',`
|
|
d_margin($1) if (!(d_LightingFlags & __LIGHT_DIFFUSECOMPUTED))
|
|
d_margin($1) {
|
|
d_margin($1) d_UpdateDiffuseColor($1+2, d_Op)
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) d_UpdateDiffuseColor($1+2,+=)
|
|
d_margin($1) }')
|
|
|
|
d_margin($1) if (light->flags & D3DLIGHTI_COMPUTE_SPECULAR)
|
|
d_margin($1) {
|
|
d_margin($1) D3DVECTOR h; // halfway vector
|
|
d_margin($1) D3DVECTOR eye; // incident vector ie vector from eye
|
|
d_margin($1)ifelse(d_Space,modelSpace,`
|
|
d_margin($1) if (pv->dwDeviceFlags & D3DDEV_LOCALVIEWER)
|
|
d_margin($1) {
|
|
d_margin($1) // calc vector from vertex to the camera
|
|
d_margin($1) VecSub(pv->lighting.model_eye, (*(D3DVECTOR*)d_pInpPosition), eye);
|
|
d_margin($1) VecNormalizeFast(eye);
|
|
d_margin($1) VecAdd(light->model_direction, eye, h); // calc halfway vector
|
|
d_margin($1) dot = VecDot(h, (*d_pInpNormal));
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) dot = VecDot(light->halfway, (*d_pInpNormal));
|
|
d_margin($1) }',`
|
|
dnl// else
|
|
d_margin($1)ifelse(d_Op,+=,`
|
|
d_margin($1) if (!(d_LightingFlags & __LIGHT_VERTEXTRANSFORMED))')
|
|
dnl// endif
|
|
d_margin($1) {
|
|
d_margin($1) // For tweening vertex position is already blended
|
|
d_margin($1) d_TransformVertexToCameraSpace($1+3, d_pInpPosition, (&d_TmpPosition), pWeights, pMatrixIndices)
|
|
d_margin($1) d_LightingFlags |= __LIGHT_VERTEXTRANSFORMED;
|
|
d_margin($1) }
|
|
d_margin($1) if (pv->dwDeviceFlags & D3DDEV_LOCALVIEWER)
|
|
d_margin($1) {
|
|
d_margin($1) // calc vector from vertex to the camera
|
|
d_margin($1) VecSub(pv->lighting.model_eye, d_TmpPosition, eye);
|
|
d_margin($1) VecNormalizeFast(eye);
|
|
d_margin($1) VecAdd(light->model_direction, eye, h); // calc halfway vector
|
|
d_margin($1) dot = VecDot(h, d_TmpNormal);
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) dot = VecDot(light->halfway, d_TmpNormal);
|
|
d_margin($1) }')
|
|
dnl// endif
|
|
d_margin($1) if (FLOAT_GTZ(dot))
|
|
d_margin($1) {
|
|
d_margin($1) if (pv->dwDeviceFlags & D3DDEV_LOCALVIEWER)
|
|
d_margin($1) dot *= ISQRTF(VecLenSq(h));
|
|
d_margin($1) d_ComputeSpecular($1+3);
|
|
d_margin($1) }
|
|
d_margin($1) }
|
|
d_margin($1)}
|
|
d_margin($1)l_exit:;
|
|
d_margin($1)')dnl
|
|
dnl//------------------------------------------------------------------
|
|
dnl// d_PointSpot7
|
|
dnl//
|
|
dnl// Generate code to light a vertex using point spot light.
|
|
dnl// Model space and camera space lighting are handled
|
|
dnl//
|
|
dnl// Arguments:
|
|
dnl/ $1 - margin count
|
|
dnl// d_pInpPosition - input position pointer (D3DVERTEX*)
|
|
dnl// d_TmpPosition - temporary position buffer (D3DVECTOR).
|
|
dnl// Used in camera space lighting
|
|
dnl// d_pInpNormal - input normal pointer (D3DVECTOR*)
|
|
dnl// d_TmpNormal - temporary normal buffer (D3DVECTOR)
|
|
dnl// Used in camera space lighting
|
|
dnl// d_Space - Defines the coordinate system: modelSpace or cameraSpace
|
|
dnl// d_LightingFlags - DWORD where __LIGHT_ bits are defined
|
|
dnl//
|
|
dnl// For camera space lighting vertex position is assumed to be already transformed
|
|
dnl//
|
|
define(`d_PointSpot7',`dnl
|
|
d_margin($1)D3DVALUE dot; // dot product
|
|
d_margin($1)D3DVALUE dist; // Distance from light to the vertex
|
|
d_margin($1)D3DVALUE dist2; // Square of the dist
|
|
d_margin($1)D3DVECTOR d; // Direction to light
|
|
d_margin($1)D3DVALUE att; // attenuation
|
|
|
|
ifelse(d_Space,modelSpace,`dnl
|
|
d_margin($1)VecSub(light->model_position, (*(D3DVECTOR*)d_pInpPosition), d);',`dnl
|
|
d_margin($1)VecSub(light->model_position, d_TmpPosition, d);')dnl
|
|
dnl// endif
|
|
|
|
d_margin($1)// early out if out of range or exactly on the vertex
|
|
d_margin($1)dist2 = d.x*d.x + d.y*d.y + d.z*d.z;
|
|
d_margin($1)if (FLOAT_CMP_POS(dist2, >=, light->range_squared) || FLOAT_EQZ(dist2))
|
|
d_margin($1) goto l_exit;
|
|
|
|
d_margin($1)dot = 0; // It is possible not to have normals (ambient component only)
|
|
d_margin($1) // So we set dot to zero for this case
|
|
d_margin($1)// Calc dot product of light dir with normal. Note that since we
|
|
d_margin($1)// did not normalize the direction the result is scaled by the distance.
|
|
ifelse(d_Space,modelSpace,`dnl
|
|
d_margin($1)if (pv->dwVIDIn & D3DFVF_NORMAL)
|
|
d_margin($1){
|
|
d_margin($1) dot = VecDot(d, (*d_pInpNormal));
|
|
d_margin($1)}',`
|
|
d_margin($1)if (pv->dwVIDIn & D3DFVF_NORMAL)
|
|
d_margin($1){
|
|
ifelse(d_Op,+=,`dnl Normal should be transformed by the first light. So do not check.
|
|
d_margin($1) if (!(d_LightingFlags & __LIGHT_NORMALTRANSFORMED))')
|
|
d_margin($1) {
|
|
d_margin($1) // For tweening normal should be already blended
|
|
d_margin($1) d_TransformNormalToCameraSpace($1+1, d_pInpNormal, (&d_TmpNormal), pWeights, pMatrixIndices)
|
|
d_margin($1) d_LightingFlags |= __LIGHT_NORMALTRANSFORMED;
|
|
d_margin($1) }
|
|
d_margin($1) dot = VecDot(d, d_TmpNormal);
|
|
d_margin($1)}')dnl
|
|
|
|
d_margin($1)if (!(light->flags & D3DLIGHTI_AMBIENT_IS_ZERO) || FLOAT_GTZ(dot))
|
|
d_margin($1){
|
|
d_margin($1) dist = SQRTF(dist2);
|
|
d_margin($1) att = light->attenuation0 +
|
|
d_margin($1) light->attenuation1 * dist +
|
|
d_margin($1) light->attenuation2 * dist2;
|
|
|
|
d_margin($1) if (!FLOAT_EQZ(att))
|
|
d_margin($1) att = (D3DVALUE)1.0/att;
|
|
d_margin($1) else
|
|
d_margin($1) att = (D3DVALUE)FLT_MAX;
|
|
|
|
d_margin($1) dist = D3DVAL(1)/dist;
|
|
d_margin($1) if (light->type == D3DLIGHT_SPOT)
|
|
d_margin($1) {
|
|
d_margin($1) D3DVALUE cone_dot;
|
|
d_margin($1) // Calc dot product of direction to light with light direction to
|
|
d_margin($1) // be compared anganst the cone angles to see if we are in the light.
|
|
d_margin($1) // Note that cone_dot is still scaled by dist
|
|
d_margin($1) cone_dot = VecDot(d, light->model_direction)*dist;
|
|
|
|
d_margin($1) if (FLOAT_CMP_POS(cone_dot, <=, light->cos_phi_by_2))
|
|
d_margin($1) goto l_exit;
|
|
|
|
d_margin($1) // modify att if in the region between phi and theta
|
|
d_margin($1) if (FLOAT_CMP_POS(cone_dot, <, light->cos_theta_by_2))
|
|
d_margin($1) {
|
|
d_margin($1) D3DVALUE val;
|
|
d_margin($1) val = (cone_dot - light->cos_phi_by_2) * light->inv_theta_minus_phi;
|
|
d_margin($1) if (!(light->flags & D3DLIGHTI_LINEAR_FALLOFF))
|
|
d_margin($1) {
|
|
d_margin($1) val = POWF(val, light->falloff);
|
|
d_margin($1) }
|
|
d_margin($1) att *= val;
|
|
d_margin($1) }
|
|
d_margin($1) }
|
|
d_margin($1) d_UpdateAmbientColor($1+1,* att)
|
|
d_margin($1) if (FLOAT_LEZ(dot))
|
|
d_margin($1) goto l_exit;
|
|
|
|
d_margin($1) dot *= dist*att;
|
|
ifelse(d_Op,+=,`dnl
|
|
d_margin($1) d_UpdateDiffuseColor($1+1,+=)',`
|
|
d_margin($1) if (!(d_LightingFlags & __LIGHT_DIFFUSECOMPUTED))
|
|
d_margin($1) {
|
|
d_margin($1) d_UpdateDiffuseColor($1+2, d_Op)
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) d_UpdateDiffuseColor($1+2,+=)
|
|
d_margin($1) }')
|
|
|
|
d_margin($1) if (light->flags & D3DLIGHTI_COMPUTE_SPECULAR)
|
|
d_margin($1) {
|
|
d_margin($1) D3DVECTOR eye;
|
|
d_margin($1) D3DVECTOR h;
|
|
d_margin($1) // normalize light direction
|
|
d_margin($1) d.x *= dist;
|
|
d_margin($1) d.y *= dist;
|
|
d_margin($1) d.z *= dist;
|
|
|
|
d_margin($1) // calc vector from vertex to the camera
|
|
dnl
|
|
ifelse(d_Space,modelSpace,`dnl
|
|
dnl
|
|
d_margin($1) if (pv->dwDeviceFlags & D3DDEV_LOCALVIEWER)
|
|
d_margin($1) {
|
|
d_margin($1) VecSub(pv->lighting.model_eye, (*(D3DVECTOR*)d_pInpPosition), eye);
|
|
d_margin($1) VecNormalizeFast(eye);
|
|
d_margin($1) VecAdd(d, eye, h); // halfway vector
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) VecAdd(d, pv->lighting.directionToCamera, h);
|
|
d_margin($1) }
|
|
d_margin($1) VecNormalizeFast(h);
|
|
d_margin($1) dot = VecDot(h, *d_pInpNormal);',`dnl
|
|
dnl
|
|
dnl// else
|
|
dnl
|
|
d_margin($1) if (pv->dwDeviceFlags & D3DDEV_LOCALVIEWER)
|
|
d_margin($1) {
|
|
d_margin($1) VecSub(pv->lighting.model_eye, d_TmpPosition, eye);
|
|
d_margin($1) VecNormalizeFast(eye);
|
|
d_margin($1) VecAdd(d, eye, h); // halfway vector
|
|
d_margin($1) }
|
|
d_margin($1) else
|
|
d_margin($1) {
|
|
d_margin($1) h.x = d.x;
|
|
d_margin($1) h.y = d.y;
|
|
d_margin($1) h.z = d.z - 1.0f;
|
|
d_margin($1) }
|
|
d_margin($1) VecNormalizeFast(h);
|
|
d_margin($1) dot = VecDot(h, d_TmpNormal);')dnl
|
|
dnl
|
|
dnl// endif
|
|
|
|
d_margin($1) d_ComputeSpecular($1+2,att)
|
|
d_margin($1) }
|
|
d_margin($1)l_exit:;
|
|
d_margin($1)}')dnl
|
|
dnl//------------------------------------------------------------------
|
|
dnl// d_LightVertices
|
|
dnl//
|
|
dnl// Generate code to light vertices in a small batch using directional or
|
|
dnl// parallel point light.
|
|
dnl// Handles strided and non-strided cases
|
|
dnl//
|
|
dnl// Arguments:
|
|
dnl// $1 - function name
|
|
dnl// $2 - Light type: d_Directional7 or d_PointSpot7
|
|
dnl//
|
|
define(`d_LightVertices',`dnl
|
|
//---------------------------------------------------------------------
|
|
void $1(LPD3DFE_PROCESSVERTICES pv,
|
|
DWORD dwVerCount,
|
|
BATCHBUFFER *pBatchBuffer,
|
|
D3DI_LIGHT *light,
|
|
D3DVERTEX *pCoord,
|
|
D3DVALUE* pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DVECTOR *pNormal,
|
|
DWORD *pDiffuse,
|
|
DWORD *pSpecular)
|
|
{
|
|
// Setup vertex data pointers
|
|
DWORD dwFlags = pv->dwFlags;
|
|
DWORD *pColors[2] = {pDiffuse, pSpecular};
|
|
DWORD **ppEmissiveSource = pColors + pv->lighting.dwEmissiveSrcIndex;
|
|
DWORD **ppAmbientSource = pColors + pv->lighting.dwAmbientSrcIndex;
|
|
DWORD **ppSpecularSource = pColors + pv->lighting.dwSpecularSrcIndex;;
|
|
DWORD **ppDiffuseSource = pColors + pv->lighting.dwDiffuseSrcIndex;
|
|
for (DWORD i = dwVerCount; i; i--)
|
|
{
|
|
$2(2)
|
|
NEXT(pCoord, pv->position.dwStride, D3DVERTEX);
|
|
NEXT(pNormal, pv->normal.dwStride, D3DVECTOR);
|
|
NEXT(pWeights, pv->weights.dwStride, D3DVALUE);
|
|
NEXT(pMatrixIndices, pv->matrixIndices.dwStride, BYTE);
|
|
if (dwFlags & D3DPV_DOCOLORVERTEX)
|
|
{
|
|
NEXT(pColors[0], pv->diffuse.dwStride, DWORD);
|
|
NEXT(pColors[1], pv->specular.dwStride, DWORD);
|
|
|
|
}
|
|
pBatchBuffer++;
|
|
}
|
|
}') dnl
|
|
|
|
//--------------------------------------------------------------------------
|
|
// This batch buffer used to hold temporary vertex data for every small loop
|
|
//
|
|
const DWORD BATCH_SIZE = 10; // Number of vertices in the batch
|
|
struct BATCHBUFFER
|
|
{
|
|
D3DVALUE sx,sy,sz,rhw; // Screen coordinates
|
|
D3DFE_COLOR diffuse;
|
|
D3DFE_COLOR specular;
|
|
D3DVECTOR position; // Vertex position in the camera space
|
|
D3DVECTOR normal; // Vertex normal in the camera space
|
|
DWORD dwFlags; // 8 low bits are the same as lighting
|
|
// flags from D3DFE
|
|
};
|
|
dnl//======================================================================
|
|
dnl// Generate light functions for batch processing
|
|
dnl//
|
|
dnl
|
|
define(`d_LightingFlags',pBatchBuffer->dwFlags)dnl
|
|
define(`d_pInpAmbient',*ppAmbientSource)dnl
|
|
define(`d_pInpDiffuse',*ppDiffuseSource)dnl
|
|
define(`d_pInpSpecular',*ppSpecularSource)dnl
|
|
define(`d_OutDiffuse',pBatchBuffer->diffuse)dnl
|
|
define(`d_OutSpecular',pBatchBuffer->specular)dnl
|
|
define(`d_pInpPosition',`pCoord')dnl
|
|
define(`d_TmpPosition',`'pBatchBuffer->position)dnl
|
|
define(`d_pInpNormal',`pNormal')dnl
|
|
define(`d_TmpNormal',`pBatchBuffer->normal')dnl
|
|
dnl
|
|
define(`d_Op',=)dnl
|
|
define(`d_Space',cameraSpace)dnl
|
|
d_LightVertices(DirectionalFirst,`d_Directional7')
|
|
d_LightVertices(PointSpotFirst,`d_PointSpot7')
|
|
dnl
|
|
define(`d_Space',modelSpace)dnl
|
|
d_LightVertices(DirectionalFirstModel,`d_Directional7')
|
|
d_LightVertices(PointSpotFirstModel,`d_PointSpot7')
|
|
dnl
|
|
define(`d_Op',+=)dnl
|
|
dnl
|
|
define(`d_Space',cameraSpace)dnl
|
|
d_LightVertices(DirectionalNext,`d_Directional7')
|
|
d_LightVertices(PointSpotNext,`d_PointSpot7')
|
|
dnl
|
|
define(`d_Space',modelSpace)dnl
|
|
d_LightVertices(DirectionalNextModel,`d_Directional7')
|
|
d_LightVertices(PointSpotNextModel,`d_PointSpot7')
|
|
dnl//======================================================================
|
|
dnl// Generate light functions for one vertex processing
|
|
//-------------------------------------------------------------------------
|
|
// Directional light, computed in the camera space
|
|
//
|
|
define(`d_LightingFlags',pv->lighting.dwLightingFlags)dnl
|
|
define(`d_pInpAmbient',(&pv->lighting.vertexAmbient))dnl
|
|
define(`d_pInpDiffuse',(&pv->lighting.vertexDiffuse))dnl
|
|
define(`d_pInpSpecular',(&pv->lighting.vertexSpecular))dnl
|
|
define(`d_OutDiffuse',pv->lighting.diffuse)dnl
|
|
define(`d_OutSpecular',pv->lighting.specular)dnl
|
|
define(`d_pInpPosition',`(pInpCoord)')dnl
|
|
define(`d_TmpPosition',`'*(D3DVERTEX*)pEyeSpaceData)dnl
|
|
define(`d_pInpNormal',`(pInpNormal)')dnl
|
|
define(`d_TmpNormal',`pEyeSpaceData->dvNormal')dnl
|
|
define(`d_Space',cameraSpace)dnl
|
|
dnl
|
|
void Directional7(LPD3DFE_PROCESSVERTICES pv,
|
|
D3DI_LIGHT *light,
|
|
D3DVERTEX *pInpCoord,
|
|
D3DVALUE *pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DVECTOR *pInpNormal,
|
|
D3DLIGHTINGELEMENT *pEyeSpaceData)
|
|
{
|
|
DWORD dwFlags = pv->dwFlags;
|
|
d_Directional7(1)
|
|
}
|
|
//---------------------------------------------------------------------
|
|
// Directional light, computed in the model space
|
|
//
|
|
define(`d_Space',modelSpace)dnl
|
|
dnl
|
|
void Directional7Model(LPD3DFE_PROCESSVERTICES pv,
|
|
D3DI_LIGHT *light,
|
|
D3DVERTEX *pInpCoord,
|
|
D3DVALUE *pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DVECTOR *pInpNormal,
|
|
D3DLIGHTINGELEMENT *pEyeSpaceData)
|
|
{
|
|
DWORD dwFlags = pv->dwFlags;
|
|
d_Directional7(1)
|
|
}
|
|
//---------------------------------------------------------------------
|
|
// Point-spot light, computed in the camera space
|
|
//
|
|
define(`d_Space',cameraSpace)dnl
|
|
void PointSpot7(LPD3DFE_PROCESSVERTICES pv,
|
|
D3DI_LIGHT *light,
|
|
D3DVERTEX *pInpCoord,
|
|
D3DVALUE *pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DVECTOR *pInpNormal,
|
|
D3DLIGHTINGELEMENT *pEyeSpaceData)
|
|
{
|
|
DWORD dwFlags = pv->dwFlags;
|
|
d_PointSpot7(1)
|
|
}
|
|
//---------------------------------------------------------------------
|
|
// Point-spot light, computed in the model space
|
|
//
|
|
define(`d_Space',modelSpace)dnl
|
|
void PointSpot7Model(LPD3DFE_PROCESSVERTICES pv,
|
|
D3DI_LIGHT *light,
|
|
D3DVERTEX *pInpCoord,
|
|
D3DVALUE *pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DVECTOR *pInpNormal,
|
|
D3DLIGHTINGELEMENT *pEyeSpaceData)
|
|
{
|
|
DWORD dwFlags = pv->dwFlags;
|
|
d_PointSpot7(1)
|
|
}
|
|
//--------------------------------------------------------------------------
|
|
// Prototype to transform vertices in batches
|
|
//
|
|
typedef DWORD (*PFN_TRANSFORMLOOP)(LPD3DFE_PROCESSVERTICES pv,
|
|
DWORD dwVerCount,
|
|
D3DVERTEX *in,
|
|
D3DVALUE* pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DTLVERTEX **ppOut,
|
|
D3DFE_CLIPCODE **ppClipCodes);
|
|
//---------------------------------------------------------------------
|
|
// Transform vertices in a batch with clipping
|
|
//
|
|
// Arguments:
|
|
// dwVerCount - number of vertices in the batch
|
|
// in - pointer to the input coordinates
|
|
// ppOut - pointer to the output vertices
|
|
// ppClipVodes - pointer to the clip code buffer
|
|
// Returns:
|
|
// Number of processed vertices
|
|
// Notes:
|
|
// ppOut and ppClipCodes will be set to the next vertex after the batch
|
|
//
|
|
DWORD TransformClip(LPD3DFE_PROCESSVERTICES pv,
|
|
DWORD dwVerCount,
|
|
D3DVERTEX *in,
|
|
D3DVALUE* pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DTLVERTEX **ppOut,
|
|
D3DFE_CLIPCODE **ppClipCodes)
|
|
{
|
|
float x, y, z, w;
|
|
D3DMATRIX *m = (D3DMATRIX*)&pv->mCTM[0];
|
|
DWORD dwInpVerSize = pv->position.dwStride;
|
|
DWORD dwOutVerSize = pv->dwOutputSize;
|
|
D3DFE_CLIPCODE *pClipCodes = *ppClipCodes;
|
|
D3DTLVERTEX *out = *ppOut;
|
|
DWORD dwDeviceFlags = pv->dwDeviceFlags;
|
|
|
|
for (DWORD i = dwVerCount; i; i--)
|
|
{
|
|
// Transform vertex to the clipping space
|
|
d_TransformVertex(2, in, m, x, y, z, w, pWeights, pMatrixIndices)
|
|
|
|
DWORD clip;
|
|
// Compute clip code
|
|
d_ComputeClipCode(2)
|
|
if (clip == 0)
|
|
{
|
|
pv->dwClipIntersection = 0;
|
|
*pClipCodes++ = 0;
|
|
w = D3DVAL(1)/w;
|
|
}
|
|
else
|
|
{
|
|
if (dwDeviceFlags & D3DDEV_GUARDBAND)
|
|
{
|
|
// We do guardband check in the projection space, so
|
|
// we transform X and Y of the vertex there
|
|
d_ComputeClipCodeGB(4)
|
|
if ((clip & ~__D3DCS_INGUARDBAND) == 0)
|
|
{
|
|
// If vertex is inside the guardband we have to compute
|
|
// screen coordinates
|
|
w = D3DVAL(1)/w;
|
|
*pClipCodes++ = (D3DFE_CLIPCODE)clip;
|
|
pv->dwClipIntersection &= clip;
|
|
pv->dwClipUnion |= clip;
|
|
goto l_DoScreenCoord;
|
|
}
|
|
}
|
|
if (pv->dwFlags & D3DPV_ONEPASSCLIPPING)
|
|
{
|
|
return dwVerCount - i;
|
|
}
|
|
pv->dwClipIntersection &= clip;
|
|
pv->dwClipUnion |= clip;
|
|
*pClipCodes++ = (D3DFE_CLIPCODE)clip;
|
|
// If vertex is outside the frustum we can not compute screen
|
|
// coordinates
|
|
out->sx = x;
|
|
out->sy = y;
|
|
out->sz = z;
|
|
out->rhw = w;
|
|
goto l_Continue;
|
|
}
|
|
|
|
l_DoScreenCoord:
|
|
|
|
d_ComputeScreenCoordinates(2, x, y, z, w, out)
|
|
|
|
l_Continue:
|
|
NEXT(in, dwInpVerSize, D3DVERTEX);
|
|
NEXT(out, dwOutVerSize, D3DTLVERTEX);
|
|
NEXT(pWeights, pv->weights.dwStride, D3DVALUE);
|
|
NEXT(pMatrixIndices, pv->matrixIndices.dwStride, BYTE);
|
|
}
|
|
*ppClipCodes = pClipCodes;
|
|
*ppOut = out;
|
|
return dwVerCount;
|
|
}
|
|
//---------------------------------------------------------------------
|
|
// Transform vertices in a batch without clipping
|
|
//
|
|
// Arguments:
|
|
// dwVerCount - number of vertices in the batch
|
|
// in - pointer to the input coordinates
|
|
// ppOut - pointer to the output vertices
|
|
// ppClipVodes - pointer to the clip code buffer
|
|
// Returns:
|
|
// Number of processed vertices
|
|
// Notes:
|
|
// ppOut and ppClipCodes will be set to the next vertex after the batch
|
|
//
|
|
DWORD TransformNoClip(LPD3DFE_PROCESSVERTICES pv,
|
|
DWORD dwVerCount,
|
|
D3DVERTEX *in,
|
|
D3DVALUE* pWeights,
|
|
BYTE* pMatrixIndices,
|
|
D3DTLVERTEX **ppOut,
|
|
D3DFE_CLIPCODE **pClipCodes)
|
|
{
|
|
float x, y, z, w;
|
|
D3DMATRIX *m = (D3DMATRIX*)&pv->mCTM[0];
|
|
DWORD dwInpVerSize = pv->position.dwStride;
|
|
DWORD dwOutVerSize = pv->dwOutputSize;
|
|
D3DTLVERTEX *out = *ppOut;
|
|
|
|
for (DWORD i = dwVerCount; i; i--)
|
|
{
|
|
// Transform vertex to the clipping space
|
|
d_TransformVertex(2, in, m, x, y, z, w, pWeights, pMatrixIndices)
|
|
|
|
// We have to check this only for DONOTCLIP case, because otherwise
|
|
// the vertex with "we = 0" will be clipped and screen coordinates
|
|
// will not be computed
|
|
// "clip" is not zero, if "w" is zero.
|
|
if (!FLOAT_EQZ(w))
|
|
w = D3DVAL(1)/w;
|
|
else
|
|
w = __HUGE_PWR2;
|
|
|
|
d_ComputeScreenCoordinates(2, x, y, z, w, out)
|
|
|
|
NEXT(in, dwInpVerSize, D3DVERTEX);
|
|
NEXT(pWeights, pv->weights.dwStride, D3DVALUE);
|
|
NEXT(pMatrixIndices, pv->matrixIndices.dwStride, BYTE);
|
|
NEXT(out, dwOutVerSize, D3DTLVERTEX);
|
|
}
|
|
*ppOut = out;
|
|
return dwVerCount;
|
|
}
|
|
//---------------------------------------------------------------------
|
|
// Transforms, lights vertices, computes clip codes
|
|
// Processing is done in small batches (BATCH_SIZE).
|
|
//
|
|
// The following fields from pv are used:
|
|
// dwFlags
|
|
// dwNumVertices
|
|
// all pointer and strides
|
|
// position.lpvStrides
|
|
// dwVIDIn
|
|
// dwVIDOut
|
|
// lpvOut
|
|
// lpClipFlags
|
|
// nTexCoord
|
|
// Returns:
|
|
// returns dwClipIntersection or 0 (if D3DDEV_DONOTCLIP is set)
|
|
// Side effects:
|
|
// dwClipUnion, dwClipIntersection are set only if D3DDEV_DONOTCLIP is not set
|
|
//
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "ProcessVerticesLoops"
|
|
DWORD ProcessVerticesLoop(LPD3DFE_PROCESSVERTICES pv)
|
|
{
|
|
D3DFE_CLIPCODE *hout = pv->lpClipFlags;
|
|
D3DTLVERTEX *out = (D3DTLVERTEX*)pv->lpvOut;
|
|
D3DMATRIXI *m = &pv->mCTM[0];
|
|
DWORD dwNumVertices = pv->dwNumVertices;
|
|
D3DVALUE *pOutTexture = (D3DVALUE*)((char*)out + pv->texOffsetOut);
|
|
DWORD *pOutDiffuse = (DWORD*)((char*)out + pv->diffuseOffsetOut);
|
|
DWORD *pOutSpecular = (DWORD*)((char*)out + pv->specularOffsetOut);
|
|
float* pOutPointSize = (float*)((char*)out + pv->pointSizeOffsetOut);
|
|
DWORD dwNumTexCoord = pv->nOutTexCoord;
|
|
DWORD *pOutFogFactor = pOutSpecular;
|
|
PFN_TRANSFORMLOOP pfnTransform;
|
|
float PointSizeRs = *(float*)&pv->lpdwRStates[D3DRS_POINTSIZE];
|
|
float PointSizeMin = *(float*)&pv->lpdwRStates[D3DRS_POINTSIZE_MIN];
|
|
float A, B, C; // Point size scales
|
|
BOOL bDoPointScale = FALSE;
|
|
|
|
d_Setup()
|
|
|
|
if (pv->lpdwRStates[D3DRS_POINTSCALEENABLE] != 0)
|
|
{
|
|
bDoPointScale = TRUE;
|
|
A = *(float*)&pv->lpdwRStates[D3DRS_POINTSCALE_A];
|
|
B = *(float*)&pv->lpdwRStates[D3DRS_POINTSCALE_B];
|
|
C = *(float*)&pv->lpdwRStates[D3DRS_POINTSCALE_C];
|
|
}
|
|
|
|
if (pv->dwFlags & D3DPV_DONOTCOPYTEXTURE)
|
|
dwNumTexCoord = 0;
|
|
|
|
BATCHBUFFER batchBuffer[BATCH_SIZE];
|
|
DWORD dwInpVerSizeBatch = dwInpVerSize * BATCH_SIZE;
|
|
DWORD dwOutVerSizeBatch = dwOutVerSize * BATCH_SIZE;
|
|
DWORD dwNormalStrideBatch = pv->normal.dwStride * BATCH_SIZE;
|
|
DWORD dwWeightsStrideBatch = pv->weights.dwStride * BATCH_SIZE;
|
|
DWORD dwMatrixIndicesStrideBatch = pv->matrixIndices.dwStride * BATCH_SIZE;
|
|
if (!(dwDeviceFlags & D3DDEV_DONOTCLIP))
|
|
{
|
|
pfnTransform = TransformClip;
|
|
pv->dwClipIntersection = ~0;
|
|
pv->dwClipUnion = 0;
|
|
}
|
|
else
|
|
{
|
|
pfnTransform = TransformNoClip;
|
|
pv->dwClipIntersection = 0;
|
|
pv->dwClipUnion = 0;
|
|
}
|
|
|
|
// When we do tweening we make "in" and "inNormal" pointers to point
|
|
// to the tweened value. We also change position and normal stride.
|
|
// But need to restore the strides later
|
|
UINT oldPositionStride = pv->position.dwStride;
|
|
UINT oldNormalStride = pv->normal.dwStride;
|
|
|
|
if (pv->dwFlags & (D3DPV_POSITION_TWEENING |
|
|
D3DPV_NORMAL_TWEENING))
|
|
{
|
|
pv->tweenFactor = *(float*)&pv->lpdwRStates[D3DRS_TWEENFACTOR];
|
|
// Replace strides because we will use blended positions and normals
|
|
if (pv->dwFlags & D3DPV_POSITION_TWEENING)
|
|
pv->position.dwStride = sizeof(D3DVECTOR);
|
|
if (pv->dwFlags & D3DPV_NORMAL_TWEENING)
|
|
pv->normal.dwStride = sizeof(D3DVECTOR);
|
|
}
|
|
|
|
// Input vertex pointers for tweening
|
|
D3DVECTOR* inT = in;
|
|
D3DVECTOR* inNormalT = inNormal;
|
|
|
|
// These two arrays are used when we do tweening.
|
|
// We blend positions and normals in model space using tweenFactor
|
|
// and then transform then to the camera (clipping) space
|
|
//
|
|
D3DVECTOR posT[BATCH_SIZE]; // Blended position in model space
|
|
D3DVECTOR normT[BATCH_SIZE]; // Blended normal in model space
|
|
|
|
do
|
|
{
|
|
DWORD count1 = min(dwNumVertices, BATCH_SIZE);
|
|
|
|
// Count of vertices to process after transformation. It could be less
|
|
// than "count1" because of clipping
|
|
DWORD count;
|
|
|
|
if (pv->dwFlags & D3DPV_POSITION_TWEENING)
|
|
{
|
|
// Blend vertices in the model space
|
|
for (UINT i=0; i < count1; i++)
|
|
{
|
|
DoBlending(pv->tweenFactor, inT, in2, &posT[i]);
|
|
inT = (D3DVECTOR*)((BYTE*)inT + oldPositionStride);
|
|
in2 = (D3DVECTOR*)((BYTE*)in2 + pv->position2.dwStride);
|
|
}
|
|
// Substitute input pointer
|
|
in = posT;
|
|
}
|
|
if (pv->dwFlags & D3DPV_NORMAL_TWEENING)
|
|
{
|
|
for (UINT i=0; i < count1; i++)
|
|
{
|
|
DoBlending(pv->tweenFactor, inNormalT, inNormal2, &normT[i]);
|
|
inNormalT = (D3DVECTOR*)((BYTE*)inNormalT + oldNormalStride);
|
|
inNormal2 = (D3DVECTOR*)((BYTE*)inNormal2 + pv->normal2.dwStride);
|
|
}
|
|
// Substitute input pointer
|
|
inNormal = normT;
|
|
}
|
|
|
|
count = (*pfnTransform)(pv, count1, (D3DVERTEX*)in,
|
|
inWeights, inMatrixIndices, &out, &hout);
|
|
|
|
if (pv->dwFlags & (D3DPV_FOG | D3DPV_LIGHTING) ||
|
|
bDoPointScale ||
|
|
pv->dwDeviceFlags & (D3DDEV_POSITIONINCAMERASPACE | D3DDEV_NORMALINCAMERASPACE))
|
|
{
|
|
memset(batchBuffer, 0, sizeof(batchBuffer));
|
|
}
|
|
// Compute camera space position if needed
|
|
if (pv->dwDeviceFlags & (D3DDEV_POSITIONINCAMERASPACE | D3DDEV_NORMALINCAMERASPACE) ||
|
|
bDoPointScale)
|
|
{
|
|
BATCHBUFFER *buf = batchBuffer;
|
|
D3DVECTOR* pVertex = in;
|
|
D3DVECTOR* pNormal = inNormal;
|
|
D3DVALUE* pWeights = inWeights;
|
|
BYTE* pMatrixIndices = inMatrixIndices;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
if (pv->dwDeviceFlags & D3DDEV_POSITIONINCAMERASPACE ||
|
|
bDoPointScale)
|
|
{
|
|
d_TransformVertexToCameraSpace(5, pVertex, ((D3DVERTEX*)&buf->position), pWeights, pMatrixIndices)
|
|
buf->dwFlags |= __LIGHT_VERTEXTRANSFORMED;
|
|
}
|
|
if (pv->dwDeviceFlags & D3DDEV_NORMALINCAMERASPACE)
|
|
{
|
|
d_TransformNormalToCameraSpace(5, pNormal, ((D3DVERTEX*)&buf->normal), pWeights, pMatrixIndices)
|
|
buf->dwFlags |= __LIGHT_NORMALTRANSFORMED;
|
|
NEXT(pNormal, pv->normal.dwStride, D3DVECTOR);
|
|
}
|
|
NEXT(pVertex, pv->position.dwStride, D3DVECTOR);
|
|
NEXT(pWeights, pv->weights.dwStride, D3DVALUE);
|
|
NEXT(pMatrixIndices, pv->matrixIndices.dwStride, BYTE);
|
|
buf++;
|
|
}
|
|
}
|
|
if (pv->dwFlags & D3DPV_LIGHTING)
|
|
{
|
|
// Light vertices. Output goes to the batch buffer
|
|
D3DI_LIGHT *light = pv->lighting.activeLights;
|
|
if (light)
|
|
{
|
|
light->pfnLightFirst(pv, count, batchBuffer, light, (D3DVERTEX*)in,
|
|
inWeights, inMatrixIndices, inNormal,
|
|
inDiffuse, inSpecular);
|
|
while(light = light->next)
|
|
{
|
|
light->pfnLightNext(pv, count, batchBuffer, light, (D3DVERTEX*)in,
|
|
inWeights, inMatrixIndices, inNormal,
|
|
inDiffuse, inSpecular);
|
|
}
|
|
}
|
|
// Copy vertices from the batch buffer to the output
|
|
BATCHBUFFER *buf = batchBuffer;
|
|
dnl
|
|
define(`d_OutDiffuse',buf->diffuse)dnl
|
|
define(`d_OutSpecular',buf->specular)dnl
|
|
define(`d_dwOutSpecular',*pOutSpecular)dnl
|
|
define(`d_dwOutDiffuse',*pOutDiffuse)dnl
|
|
define(`d_LightingFlags',buf->dwFlags)dnl
|
|
dnl
|
|
if (pv->dwFlags & D3DPV_DOCOLORVERTEX)
|
|
{
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
d_MakeOutputColors(5)
|
|
buf++;
|
|
NEXT(pOutSpecular, dwOutVerSize, DWORD);
|
|
NEXT(pOutDiffuse, dwOutVerSize, DWORD);
|
|
NEXT(inDiffuse, pv->diffuse.dwStride, DWORD);
|
|
NEXT(inSpecular, pv->specular.dwStride, DWORD);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
d_MakeOutputColorsNoColorVertex(5)
|
|
buf++;
|
|
NEXT(pOutSpecular, dwOutVerSize, DWORD);
|
|
NEXT(pOutDiffuse, dwOutVerSize, DWORD);
|
|
NEXT(inDiffuse, pv->diffuse.dwStride, DWORD);
|
|
NEXT(inSpecular, pv->specular.dwStride, DWORD);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If there is no lighting, we have to copy vertex color or
|
|
// default color to the output
|
|
if (!(pv->dwFlags & D3DPV_DONOTCOPYDIFFUSE))
|
|
{
|
|
if (pv->dwVIDIn & D3DFVF_DIFFUSE)
|
|
{
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
*pOutDiffuse = *inDiffuse;
|
|
NEXT(pOutDiffuse, dwOutVerSize, DWORD);
|
|
NEXT(inDiffuse, pv->diffuse.dwStride, DWORD);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
*pOutDiffuse = __DEFAULT_DIFFUSE;
|
|
NEXT(pOutDiffuse, dwOutVerSize, DWORD);
|
|
}
|
|
}
|
|
}
|
|
if (!(pv->dwFlags & D3DPV_DONOTCOPYSPECULAR))
|
|
{
|
|
if (pv->dwVIDIn & D3DFVF_SPECULAR)
|
|
{
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
*pOutSpecular = *inSpecular;
|
|
NEXT(pOutSpecular, dwOutVerSize, DWORD);
|
|
NEXT(inSpecular, pv->specular.dwStride, DWORD);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
*pOutSpecular = __DEFAULT_SPECULAR;
|
|
NEXT(pOutSpecular, dwOutVerSize, DWORD);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pv->dwFlags & D3DPV_FOG)
|
|
{
|
|
BATCHBUFFER* buf = batchBuffer;
|
|
D3DVECTOR* pVertex = in;
|
|
D3DVALUE* pWeights = inWeights;
|
|
BYTE* pMatrixIndices = inMatrixIndices;
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
D3DVALUE dist;
|
|
// Vertex is already transformed to the camera space
|
|
if (dwDeviceFlags & D3DDEV_RANGEBASEDFOG)
|
|
dist = SQRTF(buf->position.x*buf->position.x +
|
|
buf->position.y*buf->position.y +
|
|
buf->position.z*buf->position.z);
|
|
else
|
|
dist = ABSF(buf->position.z);
|
|
|
|
ComputeFogFactor(pv, dist, pOutFogFactor);
|
|
|
|
NEXT(pVertex, pv->position.dwStride, D3DVECTOR);
|
|
NEXT(pWeights, pv->weights.dwStride, D3DVALUE);
|
|
NEXT(pMatrixIndices, pv->matrixIndices.dwStride, BYTE);
|
|
NEXT(pOutFogFactor, dwOutVerSize, DWORD);
|
|
buf++;
|
|
}
|
|
}
|
|
|
|
if (pv->dwVIDOut & D3DFVF_PSIZE)
|
|
{
|
|
float PointSize;
|
|
BATCHBUFFER *buf = batchBuffer;
|
|
for (DWORD i = count; i; i--)
|
|
{
|
|
if (pv->dwVIDIn & D3DFVF_PSIZE)
|
|
PointSize = *inPointSize;
|
|
else
|
|
PointSize = PointSizeRs;
|
|
|
|
if (bDoPointScale)
|
|
{
|
|
float dist = SQRTF(buf->position.x*buf->position.x +
|
|
buf->position.y*buf->position.y +
|
|
buf->position.z*buf->position.z);
|
|
float v = A + B*dist + C*dist*dist;
|
|
if (v <= 0)
|
|
{
|
|
PointSize = pv->PointSizeMax;
|
|
}
|
|
else
|
|
{
|
|
// Clamping of the point size to [PointSizeMin, PointSizeMax]
|
|
// will be done by hardware or when we expand points
|
|
float PointSizeScale = pv->vcache.dvHeight * (float)sqrt(1.0/v);
|
|
PointSize *= PointSizeScale;
|
|
}
|
|
buf++;
|
|
}
|
|
*pOutPointSize = PointSize;
|
|
NEXT(pOutPointSize, dwOutVerSize, float);
|
|
NEXT(inPointSize, pv->psize.dwStride, float);
|
|
}
|
|
}
|
|
|
|
// Process texture coordinates
|
|
if (dwNumTexCoord != 0)
|
|
{
|
|
if (pv->dwDeviceFlags & D3DDEV_STRIDE)
|
|
{
|
|
if (!(pv->dwDeviceFlags & (D3DDEV_TEXTURETRANSFORM | D3DDEV_REMAPTEXTUREINDICES)))
|
|
{
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
D3DVALUE *pTexture = pOutTexture;
|
|
for (DWORD k=0; k < dwNumTexCoord; k++)
|
|
{
|
|
const DWORD dwSize = pv->dwTextureCoordSize[k];
|
|
memcpy(pTexture, inTexture[k], dwSize);
|
|
pTexture = (D3DVALUE*)((char*)pTexture + dwSize);
|
|
NEXT(inTexture[k], pv->textures[k].dwStride, D3DVALUE);
|
|
}
|
|
NEXT(pOutTexture, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!(pv->dwDeviceFlags & D3DDEV_REMAPTEXTUREINDICES))
|
|
{
|
|
D3DVALUE *pOut = pOutTexture;
|
|
for (DWORD k=0; k < dwNumTexCoord; k++)
|
|
{
|
|
const DWORD dwSize = pv->dwTextureCoordSize[k];
|
|
const DWORD dwInpSize = pv->dwInpTextureCoordSize[k];
|
|
const DWORD dwStride = pv->textures[k].dwStride;
|
|
D3DVALUE *pInpTexture = inTexture[k];
|
|
if (pv->pmTexture[k] == NULL)
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
memcpy(pOutTmp, pInpTexture, dwSize);
|
|
NEXT(pInpTexture, dwStride, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
const DWORD n = dwSize >> 2; // Number of input tex. coord.
|
|
const DWORD m = dwInpSize >> 2; // Number of input tex. coord.
|
|
(*(g_pfnTextureTransformLoop[MakeTexTransformFuncIndex(m, n)]))
|
|
(pInpTexture, pOut, pv->pmTexture[k], count,
|
|
dwStride, dwOutVerSize);
|
|
}
|
|
NEXT(pOut, dwSize, D3DVALUE);
|
|
NEXT(inTexture[k], dwStride*BATCH_SIZE, D3DVALUE);
|
|
}
|
|
NEXT(pOutTexture, dwOutVerSizeBatch, D3DVALUE);
|
|
}
|
|
else
|
|
{
|
|
D3DVALUE *pOut = pOutTexture;
|
|
for (DWORD k=0; k < pv->dwNumTextureStages; k++)
|
|
{
|
|
const LPD3DFE_TEXTURESTAGE pStage = &pv->textureStage[k];
|
|
const DWORD dwOutTexSize = pv->dwTextureCoordSize[k];
|
|
DWORD dwStride;
|
|
D3DVALUE *pIn;
|
|
D3DVECTOR reflectionVector[BATCH_SIZE];
|
|
if (pStage->dwTexGenMode == 0)
|
|
{
|
|
const DWORD dwInpIndex = pStage->dwInpCoordIndex;
|
|
pIn = inTexture[dwInpIndex];
|
|
dwStride = pv->textures[dwInpIndex].dwStride;
|
|
}
|
|
else
|
|
if (pStage->dwTexGenMode == D3DTSS_TCI_CAMERASPACEPOSITION)
|
|
{
|
|
pIn = (D3DVALUE*)&batchBuffer[0].position;
|
|
dwStride = sizeof(BATCHBUFFER);
|
|
}
|
|
else
|
|
if (pStage->dwTexGenMode == D3DTSS_TCI_CAMERASPACENORMAL)
|
|
{
|
|
pIn = (D3DVALUE*)&batchBuffer[0].normal;
|
|
dwStride = sizeof(BATCHBUFFER);
|
|
}
|
|
else // D3DTSS_TCI_CAMERASPACEREFLECTIONVECTOR
|
|
{
|
|
if (pv->dwDeviceFlags & D3DDEV_LOCALVIEWER)
|
|
{
|
|
for (DWORD i=0; i < count; i++)
|
|
{
|
|
ComputeReflectionVector(&batchBuffer[i].position,
|
|
&batchBuffer[i].normal,
|
|
&reflectionVector[i]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (DWORD i=0; i < count; i++)
|
|
{
|
|
ComputeReflectionVectorInfiniteViewer(&batchBuffer[i].normal,
|
|
&reflectionVector[i]);
|
|
}
|
|
}
|
|
pIn = (D3DVALUE*)reflectionVector;
|
|
dwStride = sizeof(D3DVECTOR);
|
|
}
|
|
if (pStage->bDoTextureProjection)
|
|
{
|
|
// We need to do emulation of texture projection
|
|
if (pStage->pmTextureTransform == NULL)
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
DoTextureProjection(pIn, pOutTmp, dwOutTexSize);
|
|
NEXT(pIn, dwStride, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
float TmpOutputTexture[4];
|
|
(*(g_pfnTextureTransform[pStage->dwTexTransformFuncIndex]))
|
|
(pIn, TmpOutputTexture, pStage->pmTextureTransform);
|
|
DoTextureProjection(TmpOutputTexture, pOutTmp, dwOutTexSize);
|
|
NEXT(pIn, dwStride, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
if (pStage->pmTextureTransform == NULL)
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
memcpy(pOutTmp, pIn, dwOutTexSize);
|
|
NEXT(pIn, dwStride, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
(*(g_pfnTextureTransformLoop[pStage->dwTexTransformFuncIndex]))
|
|
(pIn, pOut, pStage->pmTextureTransform, count,
|
|
dwStride, dwOutVerSize);
|
|
}
|
|
NEXT(pOut, dwOutTexSize, D3DVALUE);
|
|
}
|
|
NEXT(pOutTexture, dwOutVerSizeBatch, D3DVALUE);
|
|
for (DWORD m=0; m < pv->nTexCoord; m++)
|
|
{
|
|
NEXT(inTexture[m], pv->textures[m].dwStride*BATCH_SIZE, D3DVALUE);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!(pv->dwDeviceFlags & (D3DDEV_TEXTURETRANSFORM | D3DDEV_REMAPTEXTUREINDICES)))
|
|
{
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
memcpy(pOutTexture, inTexture[0], pv->dwTextureCoordSizeTotal);
|
|
NEXT(pOutTexture, dwOutVerSize, D3DVALUE);
|
|
NEXT(inTexture[0], dwInpVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
if (!(pv->dwDeviceFlags & D3DDEV_REMAPTEXTUREINDICES))
|
|
{
|
|
D3DVALUE *pIn = inTexture[0];
|
|
D3DVALUE *pOut = pOutTexture;
|
|
for (DWORD k=0; k < dwNumTexCoord; k++)
|
|
{
|
|
const DWORD dwSize = pv->dwTextureCoordSize[k];
|
|
const DWORD dwInpSize = pv->dwInpTextureCoordSize[k];
|
|
if (pv->pmTexture[k] == NULL)
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
D3DVALUE *pInpTmp = pIn;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
memcpy(pOutTmp, pInpTmp, dwSize);
|
|
NEXT(pInpTmp, dwInpVerSize, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
const DWORD n = dwSize >> 2; // Number of output tex. coord.
|
|
const DWORD m = dwInpSize >> 2; // Number of input tex. coord.
|
|
(*(g_pfnTextureTransformLoop[MakeTexTransformFuncIndex(m, n)]))
|
|
(pIn, pOut, pv->pmTexture[k], count, dwInpVerSize, dwOutVerSize);
|
|
}
|
|
NEXT(pIn, dwInpSize, D3DVALUE);
|
|
NEXT(pOut, dwSize, D3DVALUE);
|
|
}
|
|
NEXT(inTexture[0], dwInpVerSizeBatch, D3DVALUE);
|
|
NEXT(pOutTexture, dwOutVerSizeBatch, D3DVALUE);
|
|
}
|
|
else
|
|
{
|
|
D3DVALUE *pOut = pOutTexture;
|
|
for (DWORD i=0; i < pv->dwNumTextureStages; i++)
|
|
{
|
|
LPD3DFE_TEXTURESTAGE pStage = &pv->textureStage[i];
|
|
const DWORD dwSize = pv->dwTextureCoordSize[i];
|
|
D3DVALUE *pIn;
|
|
DWORD dwStride;
|
|
D3DVECTOR reflectionVector[BATCH_SIZE];
|
|
if (pStage->dwTexGenMode == 0)
|
|
{
|
|
pIn = (D3DVALUE*)((BYTE*)inTexture[0] + pStage->dwInpOffset);
|
|
dwStride = dwInpVerSize;
|
|
}
|
|
else
|
|
if (pStage->dwTexGenMode == D3DTSS_TCI_CAMERASPACEPOSITION)
|
|
{
|
|
pIn = (D3DVALUE*)&batchBuffer[0].position;
|
|
dwStride = sizeof(BATCHBUFFER);
|
|
}
|
|
else
|
|
if (pStage->dwTexGenMode == D3DTSS_TCI_CAMERASPACENORMAL)
|
|
{
|
|
pIn = (D3DVALUE*)&batchBuffer[0].normal;
|
|
dwStride = sizeof(BATCHBUFFER);
|
|
}
|
|
else // D3DTSS_TCI_CAMERASPACEREFLECTIONVECTOR
|
|
{
|
|
if (pv->dwDeviceFlags & D3DDEV_LOCALVIEWER)
|
|
{
|
|
for (DWORD i=0; i < count; i++)
|
|
{
|
|
ComputeReflectionVector(&batchBuffer[i].position,
|
|
&batchBuffer[i].normal,
|
|
&reflectionVector[i]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (DWORD i=0; i < count; i++)
|
|
{
|
|
ComputeReflectionVectorInfiniteViewer(&batchBuffer[i].normal,
|
|
&reflectionVector[i]);
|
|
}
|
|
}
|
|
pIn = (D3DVALUE*)reflectionVector;
|
|
dwStride = sizeof(D3DVECTOR);
|
|
}
|
|
if (pStage->bDoTextureProjection)
|
|
{
|
|
// We need to do emulation of texture projection
|
|
if (pStage->pmTextureTransform == NULL)
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
DoTextureProjection(pIn, pOutTmp, dwSize);
|
|
NEXT(pIn, dwStride, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
float TmpOutputTexture[4];
|
|
(*(g_pfnTextureTransform[pStage->dwTexTransformFuncIndex]))
|
|
(pIn, TmpOutputTexture, pStage->pmTextureTransform);
|
|
DoTextureProjection(TmpOutputTexture, pOutTmp, dwSize);
|
|
NEXT(pIn, dwStride, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
if (pStage->pmTextureTransform == NULL)
|
|
{
|
|
D3DVALUE *pOutTmp = pOut;
|
|
for (DWORD i=count; i; i--)
|
|
{
|
|
memcpy(pOutTmp, pIn, dwSize);
|
|
NEXT(pIn, dwStride, D3DVALUE);
|
|
NEXT(pOutTmp, dwOutVerSize, D3DVALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
(*(g_pfnTextureTransformLoop[pStage->dwTexTransformFuncIndex]))
|
|
(pIn, pOut, pStage->pmTextureTransform, count, dwStride, dwOutVerSize);
|
|
}
|
|
NEXT(pOut, dwSize, D3DVALUE);
|
|
}
|
|
NEXT(inTexture[0], dwInpVerSizeBatch, D3DVALUE);
|
|
NEXT(pOutTexture, dwOutVerSizeBatch, D3DVALUE);
|
|
}
|
|
}
|
|
}
|
|
if (count != count1)
|
|
{
|
|
pv->dwFirstClippedVertex = pv->dwNumVertices - dwNumVertices + count;
|
|
break;
|
|
}
|
|
|
|
NEXT(inNormal, dwNormalStrideBatch, D3DVECTOR);
|
|
NEXT(in, dwInpVerSizeBatch, D3DVECTOR);
|
|
NEXT(inWeights, dwWeightsStrideBatch, D3DVALUE);
|
|
NEXT(inMatrixIndices, dwMatrixIndicesStrideBatch, BYTE);
|
|
|
|
dwNumVertices -= count;
|
|
} while (dwNumVertices);
|
|
|
|
// Restore original strides, because they could changed for tweening
|
|
pv->position.dwStride = oldPositionStride;
|
|
pv->normal.dwStride = oldNormalStride;
|
|
|
|
return pv->dwClipIntersection;
|
|
}
|
|
|