#include "VxMath.hpp" #include "stb_image.h" #include "stb_image_resize.h" namespace LibCmo::VxMath { #pragma region Structure copying void VxFillStructure(CK2::CKDWORD Count, void* Dst, CK2::CKDWORD Stride, CK2::CKDWORD SizeSrc, const void* Src) { VxCopyStructure(Count, Dst, Stride, SizeSrc, Src, SizeSrc); } void VxCopyStructure(CK2::CKDWORD Count, void* Dst, CK2::CKDWORD OutStride, CK2::CKDWORD SizeSrc, const void* Src, CK2::CKDWORD InStride) { if (Dst == nullptr || Src == nullptr) return; char* cdst = reinterpret_cast(Dst); const char* csrc = reinterpret_cast(Src); for (CK2::CKDWORD i = 0; i < Count; ++i) { std::memcpy(cdst, csrc, SizeSrc); cdst += OutStride; csrc += InStride; } } #pragma endregion #pragma region Graphic Utilities void VxDoBlit(const VxImageDescEx* origin, VxImageDescEx* dst) { if (dst == nullptr || origin == nullptr) return; if (!dst->IsValid() || !origin->IsValid()) return; // if have same size, directly copy it if (dst->IsHWEqual(*origin)) { std::memcpy(dst->GetMutableImage(), origin->GetImage(), dst->GetImageSize()); return; } // perform resize by stb stbir_resize( origin->GetImage(), static_cast(origin->GetWidth()), static_cast(origin->GetHeight()), 0, dst->GetMutableImage(), static_cast(dst->GetWidth()), static_cast(dst->GetHeight()), 0, STBIR_TYPE_UINT8, 4, STBIR_ALPHA_CHANNEL_NONE, 0, // no alpha channel, mean we treat alpha channel as a normal color factor. STBIR_EDGE_CLAMP, STBIR_EDGE_CLAMP, STBIR_FILTER_BOX, STBIR_FILTER_BOX, STBIR_COLORSPACE_SRGB, nullptr ); } void VxDoBlitUpsideDown(const VxImageDescEx* origin, VxImageDescEx* dst) { if (dst == nullptr || origin == nullptr) return; if (!dst->IsValid() || !origin->IsValid()) return; // if size is not matched, return if (!dst->IsHWEqual(*origin)) { return; } // copy and swap data by line CK2::CKDWORD height = dst->GetHeight(), rowsize = VxImageDescEx::PixelSize * dst->GetWidth(); for (CK2::CKDWORD row = 0; row < height; ++row) { std::memcpy( dst->GetMutableImage() + (row * rowsize), origin->GetImage() + ((height - row - 1) * rowsize), rowsize ); } } CK2::CKDWORD VxGetBitCount(CK2::CKDWORD dwMask) { if (dwMask == 0u) return 0; dwMask >>= VxGetBitShift(dwMask); CK2::CKDWORD count = 0; while ((dwMask & 1u) == 0u) { dwMask >>= 1u; ++count; } return count; } CK2::CKDWORD VxGetBitShift(CK2::CKDWORD dwMask) { if (dwMask == 0u) return 0; CK2::CKDWORD count = 0; while ((dwMask & 1u) != 0u) { dwMask >>= 1u; ++count; } return count; } //CK2::CKDWORD VxScaleFactor(CK2::CKDWORD val, CK2::CKDWORD srcBitCount, CK2::CKDWORD dstBitCount) { // if (srcBitCount == dstBitCount) return val; // if (srcBitCount > dstBitCount) { // return val >> (srcBitCount - dstBitCount); // } else { // return val << (dstBitCount - srcBitCount); // } //} void VxDoAlphaBlit(VxImageDescEx* dst_desc, CK2::CKBYTE AlphaValue) { if (dst_desc == nullptr) return; CK2::CKDWORD* pixels = dst_desc->GetMutablePixels(); CK2::CKDWORD pixelcount = dst_desc->GetPixelCount(); for (CK2::CKDWORD i = 0; i < pixelcount; ++i) { *pixels = (*pixels) & 0x00FFFFFF | (static_cast(AlphaValue) << 24); ++pixels; } //CK2::CKDWORD* pixels = dst_desc->GetPixels(); //CK2::CKDWORD pixelcount = dst_desc->GetPixelCount(), // alphashift = VxGetBitShift(dst_desc->m_AlphaMask), // alphacount = VxGetBitCount(dst_desc->m_AlphaMask); //CK2::CKDWORD av = VxScaleFactor(AlphaValue, static_cast(sizeof(CK2::CKBYTE) * 8), alphacount) << alphashift; //for (CK2::CKDWORD i = 0; i < pixelcount; ++i) { // *pixels = (*pixels) & (~dst_desc->m_AlphaMask) | av; // ++pixels; //} } void VxDoAlphaBlit(VxImageDescEx* dst_desc, CK2::CKBYTE* AlphaValues) { if (dst_desc == nullptr) return; CK2::CKDWORD* pixels = dst_desc->GetMutablePixels(); CK2::CKDWORD pixelcount = dst_desc->GetPixelCount(); for (CK2::CKDWORD i = 0; i < pixelcount; ++i) { *pixels = (*pixels) & 0x00FFFFFF | (static_cast(*AlphaValues) << 24); ++pixels; ++AlphaValues; } //CK2::CKDWORD* pixels = dst_desc->GetPixels(); //CK2::CKDWORD pixelcount = dst_desc->GetPixelCount(), // alphashift = VxGetBitShift(dst_desc->m_AlphaMask), // alphacount = VxGetBitCount(dst_desc->m_AlphaMask); //for (CK2::CKDWORD i = 0; i < pixelcount; ++i) { // *pixels = (*pixels) & (~dst_desc->m_AlphaMask) | (VxScaleFactor(*AlphaValues, static_cast(sizeof(CK2::CKBYTE) * 8), alphacount) << alphashift); // ++pixels; // ++AlphaValues; //} } #pragma endregion }