/*==LICENSE==* CyanWorlds.com Engine - MMOG client, server and tools Copyright (C) 2011 Cyan Worlds, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. Additional permissions under GNU GPL version 3 section 7 If you modify this Program, or any covered work, by linking or combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK (or a modified version of those libraries), containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the licensors of this Program grant you additional permission to convey the resulting work. Corresponding Source for a non-source form of such a combination shall include the source code for the parts of OpenSSL and IJG JPEG Library used as well as that of the covered work. You can contact Cyan Worlds, Inc. by email legal@cyan.com or by snail mail at: Cyan Worlds, Inc. 14617 N Newport Hwy Mead, WA 99021 *==LICENSE==*/ #include "hsTypes.h" #include "plPipeline.h" #include "plTweak.h" #include "hsFastMath.h" #include "plPerspDirSlave.h" #include <float.h> void plPerspDirSlave::Init() { plShadowSlave::Init(); fFlags |= kCastInCameraSpace; } hsPoint3 plPerspDirSlave::IProject(const hsMatrix44& world2NDC, const hsPoint3& pos, hsScalar w) const { hsPoint3 retVal; retVal.fX = world2NDC.fMap[0][0] * pos.fX + world2NDC.fMap[0][1] * pos.fY + world2NDC.fMap[0][2] * pos.fZ + world2NDC.fMap[0][3] * w; retVal.fY = world2NDC.fMap[1][0] * pos.fX + world2NDC.fMap[1][1] * pos.fY + world2NDC.fMap[1][2] * pos.fZ + world2NDC.fMap[1][3] * w; retVal.fZ = world2NDC.fMap[2][0] * pos.fX + world2NDC.fMap[2][1] * pos.fY + world2NDC.fMap[2][2] * pos.fZ + world2NDC.fMap[2][3] * w; hsScalar invW = 1.f / ( world2NDC.fMap[3][0] * pos.fX + world2NDC.fMap[3][1] * pos.fY + world2NDC.fMap[3][2] * pos.fZ + world2NDC.fMap[3][3] * w); retVal *= invW; return retVal; } hsBounds3Ext plPerspDirSlave::IGetPerspCasterBound(const hsMatrix44& world2NDC) const { hsPoint3 corners[8]; fCasterWorldBounds.GetCorners(corners); hsPoint3 perspCorners[8]; int i; for( i = 0; i < 8; i++ ) { perspCorners[i] = IProject(world2NDC, corners[i]); if( perspCorners[i].fX < -1.f ) perspCorners[i].fX = -1.f; else if( perspCorners[i].fX > 1.f ) perspCorners[i].fX = 1.f; if( perspCorners[i].fY < -1.f ) perspCorners[i].fY = -1.f; else if( perspCorners[i].fY > 1.f ) perspCorners[i].fY = 1.f; if( perspCorners[i].fZ < 0 ) perspCorners[i].Set(0.f, 0.f, 0.f); } hsBounds3Ext bnd; bnd.MakeEmpty(); for( i = 0; i < 8; i++ ) bnd.Union(&perspCorners[i]); return bnd; } bool plPerspDirSlave::SetupViewTransform(plPipeline* pipe) { plViewTransform pipeView = pipe->GetViewTransform(); plConst(hsScalar) kYon(100.f); plConst(hsScalar) kHither(30.f); pipeView.SetHither(kHither); pipeView.SetYon(kYon); hsMatrix44 cam2NDC = pipeView.GetCameraToNDC(); hsMatrix44 world2NDC = cam2NDC * pipeView.GetWorldToCamera(); fLightDir = fLightToWorld.GetAxis(hsMatrix44::kUp); hsVector3 worldLiDir = fLightDir; hsPoint3 pWorldLiDir(worldLiDir.fX, worldLiDir.fY, worldLiDir.fZ); hsPoint3 perspLiPos = IProject(world2NDC, pWorldLiDir, 0); hsBool reverseZ = fLightDir.InnerProduct(pipe->GetViewDirWorld()) > 0; SetFlag(kReverseZ, reverseZ); SetFlag(kReverseCull, reverseZ); hsPoint3 lookAt; plConst(hsBool) kUsePerspCenter(true); plConst(hsBool) kUseFrustCenter(true); if( kUsePerspCenter ) { hsPoint3 lookAtCam = pipeView.GetWorldToCamera() * fCasterWorldBounds.GetCenter(); hsPoint3 lookAtNDC = IProject(pipeView.GetCameraToNDC(), lookAtCam); lookAt = IProject(world2NDC, fCasterWorldBounds.GetCenter()); } else if( kUseFrustCenter ) { plConst(hsScalar) kDist(50.f); hsPoint3 camFrustCenter(0.f, 0.f, kDist); lookAt = IProject(cam2NDC, camFrustCenter); } else { hsBounds3Ext ndcBnd(IGetPerspCasterBound(world2NDC)); lookAt = ndcBnd.GetCenter(); } hsMatrix44 camNDC2Li; hsMatrix44 li2CamNDC; IComputeCamNDCToLight(perspLiPos, lookAt, camNDC2Li, li2CamNDC); hsScalar minZ, maxZ; hsScalar cotX, cotY; plConst(hsBool) kFixedPersp(true); if( !kFixedPersp ) { hsBounds3Ext bnd(IGetPerspCasterBound(camNDC2Li * world2NDC)); hsBounds3Ext bnd2(IGetPerspCasterBound(world2NDC)); bnd2.Transform(&camNDC2Li); plConst(hsBool) kUseBnd2(false); if( kUseBnd2 ) bnd = bnd2; minZ = bnd.GetMins().fZ; maxZ = bnd.GetMaxs().fZ; // THIS IS WRONG // EAP // This is my hack to get the Nexus age working. The real problem // is probably data-side. I take full responsibility for this // hack-around breaking the entire system, loosing data, causing // unauthorized credit card transactions, etc. if (_isnan(bnd.GetMins().fX) || _isnan(bnd.GetMins().fY)) return false; if (_isnan(bnd.GetMaxs().fX) || _isnan(bnd.GetMaxs().fY)) return false; // THIS IS EVEN MORE WRONG plConst(hsBool) kFakeDepth(false); if( kFakeDepth ) { plConst(hsScalar) kMin(1.f); plConst(hsScalar) kMax(30.f); minZ = kMin; maxZ = kMax; } plConst(hsScalar) kMinMinZ(1.f); if( minZ < kMinMinZ ) minZ = kMinMinZ; if( -bnd.GetMins().fX > bnd.GetMaxs().fX ) { hsAssert(bnd.GetMins().fX < 0, "Empty shadow caster bounds?"); cotX = -minZ / bnd.GetMins().fX; } else { hsAssert(bnd.GetMaxs().fX > 0, "Empty shadow caster bounds?"); cotX = minZ / bnd.GetMaxs().fX; } if( -bnd.GetMins().fY > bnd.GetMaxs().fY ) { hsAssert(bnd.GetMins().fY < 0, "Empty shadow caster bounds?"); cotY = -minZ / bnd.GetMins().fY; } else { hsAssert(bnd.GetMaxs().fY > 0, "Empty shadow caster bounds?"); cotY = minZ / bnd.GetMaxs().fY; } } else { plConst(hsScalar) kHi(1.f); hsBounds3Ext bnd; const hsPoint3 lo(-1.f, -1.f, 0.f); const hsPoint3 hi(1.f, 1.f, kHi); bnd.MakeEmpty(); bnd.Union(&lo); bnd.Union(&hi); bnd.Transform(&camNDC2Li); minZ = bnd.GetMins().fZ; maxZ = bnd.GetMaxs().fZ; // THIS IS WRONG // EAP // This is my hack to get the Nexus age working. The real problem // is probably data-side. I take full responsibility for this // hack-around breaking the entire system, loosing data, causing // unauthorized credit card transactions, etc. if (_isnan(bnd.GetMins().fX) || _isnan(bnd.GetMins().fY)) return false; if (_isnan(bnd.GetMaxs().fX) || _isnan(bnd.GetMaxs().fY)) return false; plConst(hsScalar) kMinMinZ(1.f); if( minZ < kMinMinZ ) minZ = kMinMinZ; if( -bnd.GetMins().fX > bnd.GetMaxs().fX ) { hsAssert(bnd.GetMins().fX < 0, "Empty shadow caster bounds?"); cotX = -minZ / bnd.GetMins().fX; } else { hsAssert(bnd.GetMaxs().fX > 0, "Empty shadow caster bounds?"); cotX = minZ / bnd.GetMaxs().fX; } if( -bnd.GetMins().fY > bnd.GetMaxs().fY ) { hsAssert(bnd.GetMins().fY < 0, "Empty shadow caster bounds?"); cotY = -minZ / bnd.GetMins().fY; } else { hsAssert(bnd.GetMaxs().fY > 0, "Empty shadow caster bounds?"); cotY = minZ / bnd.GetMaxs().fY; } } hsMatrix44 proj; proj.Reset(); proj.NotIdentity(); // LightToTexture // First the LightToTexture, which uses the above pretty much as is. // Note the remapping to range [0.5..width-0.5] etc. Also, the perspective // divide is by the 3rd output (not the fourth), so we make the 3rd // output be W (instead of Z). // This also means that our translate goes into [i][2] instead of [i][3]. proj.fMap[0][0] = cotX * 0.5f; proj.fMap[0][2] = 0.5f * (1.f + 0.5f/fWidth); proj.fMap[1][1] = -cotY * 0.5f; proj.fMap[1][2] = 0.5f * (1.f + 0.5f/fHeight); #if 1 // This computes correct Z, but we really just want W in 3rd component. HACKFISH proj.fMap[2][2] = maxZ / (maxZ - minZ); proj.fMap[2][3] = -minZ * maxZ / (maxZ - minZ); #elif 1 proj.fMap[2][2] = 1.f; proj.fMap[2][3] = 0; #endif proj.fMap[3][2] = 1.f; proj.fMap[3][3] = 0; // fWorldToTexture = proj * camNDC2Li * pipeView.GetWorldToNDC(); fWorldToTexture = proj * camNDC2Li * world2NDC; fWorldToTexture.fMap[2][0] = fWorldToTexture.fMap[3][0]; fWorldToTexture.fMap[2][1] = fWorldToTexture.fMap[3][1]; fWorldToTexture.fMap[2][2] = fWorldToTexture.fMap[3][2]; fWorldToTexture.fMap[2][3] = fWorldToTexture.fMap[3][3]; // Now the LightToNDC. This one's a little trickier, because we want to compensate for // having brought in the viewport to keep our border constant, so we can clamp the // projected texture and not have the edges smear off to infinity. cotX -= cotX / (fWidth * 0.5f); cotY -= cotY / (fHeight * 0.5f); hsScalar tanX = 1.f / cotX; hsScalar tanY = 1.f / cotY; fView.SetScreenSize((UInt16)fWidth, (UInt16)fHeight); fView.SetCameraTransform(pipe->GetViewTransform().GetWorldToCamera(), pipe->GetViewTransform().GetCameraToWorld()); fView.SetPerspective(true); fView.SetViewPort(0, 0, (float)fWidth, (float)fHeight, false); fView.SetView(hsPoint3(-tanX, -tanY, minZ), hsPoint3(tanX, tanY, maxZ)); hsMatrix44 cam2Light = camNDC2Li * pipeView.GetCameraToNDC(); fView.PostMultCameraToNDC(cam2Light); fLightPos = fLightToWorld.GetTranslate(); SetFlag(kPositional, false); return true; } static inline hsVector3 CrossProd(const hsVector3& a, const hsPoint3& b) { return hsVector3(a.fY*b.fZ - a.fZ*b.fY, a.fZ*b.fX - a.fX*b.fZ, a.fX*b.fY - a.fY*b.fX); } static inline void InverseOfPureRotTran(const hsMatrix44& src, hsMatrix44& inv) { inv = src; // We know this is a pure rotation and translation matrix, so // we won't have to do a full inverse. Okay kids, don't try this // at home. inv.fMap[0][1] = src.fMap[1][0]; inv.fMap[0][2] = src.fMap[2][0]; inv.fMap[1][0] = src.fMap[0][1]; inv.fMap[1][2] = src.fMap[2][1]; inv.fMap[2][0] = src.fMap[0][2]; inv.fMap[2][1] = src.fMap[1][2]; hsPoint3 newTran(-src.fMap[0][3], -src.fMap[1][3], -src.fMap[2][3]); inv.fMap[0][3] = newTran.InnerProduct((hsVector3*)&inv.fMap[0][0]); inv.fMap[1][3] = newTran.InnerProduct((hsVector3*)&inv.fMap[1][0]); inv.fMap[2][3] = newTran.InnerProduct((hsVector3*)&inv.fMap[2][0]); } void plPerspDirSlave::IComputeCamNDCToLight(const hsPoint3& from, const hsPoint3& at, hsMatrix44& camNDC2Li, hsMatrix44& li2CamNDC) { hsVector3 atToFrom(&from, &at); hsScalar distSq = atToFrom.MagnitudeSquared(); atToFrom *= hsFastMath::InvSqrtAppr(distSq); const hsScalar kMinMag = 0.5f; hsVector3 up(0,0,1.f); if( CrossProd(up, (at - from)).MagnitudeSquared() < kMinMag ) { up.Set(0, 1.f, 0); } hsMatrix44 w2light; w2light.MakeCamera(&from, &at, &up); hsMatrix44 light2w; InverseOfPureRotTran(w2light, light2w); #ifdef CHECK_INVERSE hsMatrix44 inv; w2light.GetInverse(&inv); #endif // CHECK_INVERSE camNDC2Li = w2light; li2CamNDC = light2w; }