// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "std3d.h" #include "nel/3d/fasthls_modifier.h" #include "nel/misc/fast_floor.h" #include "nel/misc/bitmap.h" #include "nel/misc/system_info.h" #include "nel/misc/algo.h" using namespace std; using namespace NLMISC; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NL3D { // *************************************************************************** CFastHLSModifier *CFastHLSModifier::_Instance= NULL; // *************************************************************************** void CFastHLSModifier::releaseInstance() { if( _Instance ) { delete _Instance; _Instance = NULL; } } // *************************************************************************** CFastHLSModifier::CFastHLSModifier() { uint i; // build the HueTable. for(i=0;i>7; S+= S>>7; // H. CRGBA col= _HueTable[H]; // S. col.blendFromuiRGBOnly(gray, col, S); // L. if(L<=128) { col.modulateFromuiRGBOnly(col, L*2); } else { col.blendFromuiRGBOnly(col, CRGBA::White, (L-128)*2 ); } return col; } #if defined(NL_COMP_VC) && (NL_COMP_VC_VERSION >= 71) # pragma warning( push ) # pragma warning( disable : 4799 ) #endif #ifdef NL_OS_WINDOWS #pragma managed(push, off) #endif // *************************************************************************** uint16 CFastHLSModifier::applyHLSMod(uint16 colorIn, uint8 dHue, uint dLum, uint dSat) { static uint64 mmBlank = 0; static uint64 mmOne = INT64_CONSTANT(0x00FF00FF00FF00FF); static uint64 mmGray = INT64_CONSTANT(0x0080008000800080); /* dLum is actually 0xFFFFFF00 + realDLum dSat is actually 0xFFFFFF00 + realDSat */ uint16 retVal; #if defined(NL_OS_WINDOWS) && !defined(NL_NO_ASM) if(CSystemInfo::hasMMX()) { static uint64 mmInterpBufer[4]= {0,0,0,INT64_CONSTANT(0x00FF00FF00FF00FF)}; __asm { mov edi, offset mmInterpBufer mov ecx, this // get HLS in edx. mov eax, 0 mov ebx, 0 lea esi, [ecx]this._Color16ToHLS mov ax, colorIn mov edx, [esi+ eax*4] // apply dh to H (ie dl!). Auto-wrap. add dl, dHue // get the color into mm0 mov bl, dl lea esi, [ecx]this._HueTable movd mm0, [esi+ ebx*4] punpcklbw mm0, mmBlank // get L into eax and S into ebx mov eax, edx mov ebx, edx shr eax, 8 shr ebx, 16 and eax, 255 and ebx, 255 // add dLum/dSat and clamp to 1. add eax, dLum sbb ecx, ecx // ecx= FFFFFFFF if carry. add ebx, dSat sbb edx, edx or eax, ecx // eax= FFFFFFFF if carry was set or ebx, edx // add Magic delta, and clamp to 0. add eax, 256 sbb ecx, ecx // ecx= 0 if carry not set => result below 0. add ebx, 256 sbb edx, edx and eax, ecx // eax= 0 if result was below 0 and ebx, edx // Load Sat/(1-Sat) into MMX movd mm2, ebx movq mm3, mmOne punpckldq mm2, mm2 // mm2= 0000 00AA 0000 00AA packssdw mm2, mm2 // mm2= 00AA 00AA 00AA 00AA movq mm1, mmGray psubusw mm3, mm2 // mm3= 1-sat. // combine Color and Sat pmullw mm0, mm2 // mm0= color*sat pmullw mm1, mm3 // mm1= gray*(1-sat) paddusw mm0, mm1 // mm0= color saturated // shift and store into the buffer for Luminance interpolation psrlw mm0, 8 movq [edi+ 8], mm0 movq [edi+ 16], mm0 // use edx as index for luminance: 0: L=0 to 127. 1: L=128 to 255. mov edx, eax shl eax, 1 shr edx, 7 and eax, 255 // 0-127 and 128-255 transform auto to 0-254 // expand 0-254 to 0-255 mov ecx, eax shl edx, 4 shr ecx, 7 add eax, ecx // Combine color and Luminance into MMX. interpolate 0->col or col->white according to edx. // Load Lum/(1-Lum) into MMX movd mm2, eax movq mm3, mmOne punpckldq mm2, mm2 // mm2= 0000 00AA 0000 00AA packssdw mm2, mm2 // mm2= 00AA 00AA 00AA 00AA psubusw mm3, mm2 // mm3= 1-lum. // Combine color and Sat into MMX movq mm0, [edi+ edx] movq mm1, [edi+ edx + 8] pmullw mm0, mm3 // mm0= color0*(1-lum) pmullw mm1, mm2 // mm1= color1*lum paddusw mm0, mm1 // mm0= final color // shift and unpack psrlw mm0, 8 packuswb mm0, mm0 movd eax, mm0 // pack to 16bits. mov ebx, eax mov ecx, eax shl eax, 8 // Red shr ebx, 5 // Green shr ecx, 19 // Blue and eax, 0xF800 and ebx, 0x07E0 and ecx, 0x001F or eax, ebx or eax, ecx mov retVal, ax } } else #endif // NL_OS_WINDOWS { CHLSA hls= _Color16ToHLS[colorIn]; // apply (C version) Dhue, dLum and dSat hls.H= uint8((hls.H + dHue) & 0xFF); sint v= (sint)hls.L + (sint)(dLum-0xFFFFFF00); fastClamp8(v); hls.L= v; v= (sint)hls.S + (sint)(dSat-0xFFFFFF00); fastClamp8(v); hls.S= v; CRGBA ret= convert(hls.H, hls.L, hls.S); retVal= ret.get565(); } return retVal; } #ifdef NL_OS_WINDOWS #pragma managed(pop) #endif #if defined(NL_COMP_VC) && (NL_COMP_VC_VERSION >= 71) # pragma warning( pop ) #endif // *************************************************************************** #ifdef NL_OS_WINDOWS #pragma managed(push, off) #endif void CFastHLSModifier::convertDDSBitmapDXTC1Or1A(CBitmap &dst, const CBitmap &src, uint8 dh, uint dLum, uint dSat) { uint W= src.getWidth(); uint H= src.getHeight(); const uint8 *srcPix= &(src.getPixels()[0]); uint8 *dstPix= &(dst.getPixels()[0]); uint numBlock= (W*H)/16; /* need to swap color and bits for DXTC1 or DXTC1A. */ static uint32 bitLUT[8]= { 1,0,3,2, // reverse std order 1,0,2,3, // reverse order for "special 0/black packing" }; // Do not use alpha mask for now. for(;numBlock>0;numBlock--) { uint16 srcCol0= ((uint16*)srcPix)[0]; uint16 srcCol1= ((uint16*)srcPix)[1]; bool srcSign= srcCol0>srcCol1; // apply modifiers for 2 colors. uint16 dstCol0= applyHLSMod(srcCol0, dh,dLum,dSat); uint16 dstCol1= applyHLSMod(srcCol1, dh,dLum,dSat); bool dstSign= dstCol0>dstCol1; if((uint)dstSign!=(uint)srcSign) { swap(dstCol0,dstCol1); // must change bits too! uint32 srcBits= ((uint32*)srcPix)[1]; uint32 dstBits= 0; // take correct lut according to original sign uint32 *lut; if(srcCol0>srcCol1) lut= bitLUT; else lut= bitLUT+4; // for all bits, transpose with lut. #if defined(NL_OS_WINDOWS) && !defined(NL_NO_ASM) __asm { mov eax, srcBits mov esi, lut mov edx, 0 mov ecx, 16 // prepare 1st. rol eax, 2 mov ebx, eax and ebx, 2 // do it 16 times. myLoop: or edx, [esi+ebx*4] rol eax, 2 rol edx, 2 mov ebx, eax and ebx, 2 dec ecx jnz myLoop ror edx, 2 mov dstBits, edx } #else for(uint n=16;n>0;n--) { // transform the id. uint id= srcBits&3; id= lut[id]; // write. dstBits|= id<<30; // don't decal last if(n>1) dstBits>>=2; } #endif // store ((uint32*)dstPix)[1]= dstBits; } else // just copy bits ((uint32*)dstPix)[1]= ((uint32*)srcPix)[3]; ((uint16*)dstPix)[0]= dstCol0; ((uint16*)dstPix)[1]= dstCol1; // skip. srcPix+= 8; dstPix+= 8; } // Must end MMX, for applyHLSMod() #if defined(NL_OS_WINDOWS) && !defined(NL_NO_ASM) if(CSystemInfo::hasMMX()) _asm emms; #endif } #ifdef NL_OS_WINDOWS #pragma managed(pop) #endif // *************************************************************************** #ifdef NL_OS_WINDOWS #pragma managed(push, off) #endif void CFastHLSModifier::convertDDSBitmapDXTC3Or5(CBitmap &dst, const CBitmap &src, uint8 dh, uint dLum, uint dSat) { uint W= src.getWidth(); uint H= src.getHeight(); const uint8 *srcPix= &(src.getPixels()[0]); uint8 *dstPix= &(dst.getPixels()[0]); uint numBlock= (W*H)/16; /* NB: don't need to swap color and bits for DXTC3 or DXTC5. */ // Do not use alpha mask for now. for(;numBlock>0;numBlock--) { uint16 srcCol0= ((uint16*)srcPix)[4]; uint16 srcCol1= ((uint16*)srcPix)[5]; // apply modifiers for 2 colors. ((uint16*)dstPix)[4]= applyHLSMod(srcCol0, dh,dLum,dSat); ((uint16*)dstPix)[5]= applyHLSMod(srcCol1, dh,dLum,dSat); // just copy bits ((uint32*)dstPix)[3]= ((uint32*)srcPix)[3]; // copy alpha part. ((uint32*)dstPix)[0]= ((uint32*)srcPix)[0]; ((uint32*)dstPix)[1]= ((uint32*)srcPix)[1]; // skip bits and alpha part. srcPix+= 16; dstPix+= 16; } // Must end MMX, for applyHLSMod() #if defined(NL_OS_WINDOWS) && !defined(NL_NO_ASM) if(CSystemInfo::hasMMX()) _asm emms; #endif } #ifdef NL_OS_WINDOWS #pragma managed(pop) #endif // *************************************************************************** void CFastHLSModifier::convertDDSBitmap(CBitmap &dst, const CBitmap &src, uint8 dh, sint dl, sint ds) { nlassert(src.getPixelFormat()==dst.getPixelFormat()); nlassert(src.getWidth()==dst.getWidth() && src.getHeight()==dst.getHeight()); // Magic add clamp. uint dLum= 0xFFFFFF00 + dl; uint dSat= 0xFFFFFF00 + ds; if(src.getPixelFormat()==CBitmap::DXTC1 || src.getPixelFormat()==CBitmap::DXTC1Alpha) convertDDSBitmapDXTC1Or1A(dst, src, dh, dLum, dSat); else if(src.getPixelFormat()==CBitmap::DXTC3 || src.getPixelFormat()==CBitmap::DXTC5) convertDDSBitmapDXTC3Or5(dst, src, dh, dLum, dSat); else { nlstop; } } // *************************************************************************** void CFastHLSModifier::convertRGBABitmap(CBitmap &dst, const CBitmap &src, uint8 dh, sint dl, sint ds) { nlassert(src.getPixelFormat()==dst.getPixelFormat()); nlassert(src.getPixelFormat()==CBitmap::RGBA); uint W= src.getWidth(); uint H= src.getHeight(); const CRGBA *srcPix= (const CRGBA*)&(src.getPixels()[0]); CRGBA *dstPix= (CRGBA*)&(dst.getPixels()[0]); uint numPix= W*H; // Do not use alpha mask for now. for(;numPix>0;numPix--) { float H,L,S; srcPix->convertToHLS(H,L,S); H*= 256.f/360.f; L*= 255.f; S*= 255.f; H+= dh+0.5f; L+= dl+0.5f; S+= ds+0.5f; clamp(H, 0, 255); clamp(L, 0, 255); clamp(S, 0, 255); uint8 H8= (uint8)NLMISC::OptFastFloor(H); uint8 L8= (uint8)NLMISC::OptFastFloor(L); uint8 S8= (uint8)NLMISC::OptFastFloor(S); *dstPix= convert(H8, L8, S8); srcPix++; dstPix++; } } } // NL3D