|  | // Copyright 2014 PDFium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | 
|  |  | 
|  | #include "core/fxcrt/include/fx_basic.h" | 
|  | #include "core/fxcrt/include/fx_ext.h" | 
|  |  | 
|  | #if _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_ | 
|  | #include <dirent.h> | 
|  | #include <sys/types.h> | 
|  | #else | 
|  | #include <direct.h> | 
|  | #endif | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <cctype> | 
|  | #include <limits> | 
|  | #include <memory> | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const int kDefaultIntValue = 0; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | bool FX_atonum(const CFX_ByteStringC& strc, void* pData) { | 
|  | if (strc.Find('.') != -1) { | 
|  | FX_FLOAT* pFloat = static_cast<FX_FLOAT*>(pData); | 
|  | *pFloat = FX_atof(strc); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Note, numbers in PDF are typically of the form 123, -123, etc. But, | 
|  | // for things like the Permissions on the encryption hash the number is | 
|  | // actually an unsigned value. We use a uint32_t so we can deal with the | 
|  | // unsigned and then check for overflow if the user actually signed the value. | 
|  | // The Permissions flag is listed in Table 3.20 PDF 1.7 spec. | 
|  | pdfium::base::CheckedNumeric<uint32_t> integer = 0; | 
|  | bool bNegative = false; | 
|  | bool bSigned = false; | 
|  | int cc = 0; | 
|  | if (strc[0] == '+') { | 
|  | cc++; | 
|  | bSigned = true; | 
|  | } else if (strc[0] == '-') { | 
|  | bNegative = true; | 
|  | bSigned = true; | 
|  | cc++; | 
|  | } | 
|  |  | 
|  | while (cc < strc.GetLength() && std::isdigit(strc[cc])) { | 
|  | integer = integer * 10 + FXSYS_toDecimalDigit(strc.CharAt(cc)); | 
|  | if (!integer.IsValid()) | 
|  | break; | 
|  | cc++; | 
|  | } | 
|  |  | 
|  | // We have a sign, and the value was greater then a regular integer | 
|  | // we've overflowed, reset to the default value. | 
|  | if (bSigned) { | 
|  | if (bNegative) { | 
|  | if (integer.ValueOrDefault(kDefaultIntValue) > | 
|  | static_cast<uint32_t>(std::numeric_limits<int>::max()) + 1) { | 
|  | integer = kDefaultIntValue; | 
|  | } | 
|  | } else if (integer.ValueOrDefault(kDefaultIntValue) > | 
|  | static_cast<uint32_t>(std::numeric_limits<int>::max())) { | 
|  | integer = kDefaultIntValue; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Switch back to the int space so we can flip to a negative if we need. | 
|  | int value = static_cast<int>(integer.ValueOrDefault(kDefaultIntValue)); | 
|  | if (bNegative) | 
|  | value = -value; | 
|  |  | 
|  | int* pInt = static_cast<int*>(pData); | 
|  | *pInt = value; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static const FX_FLOAT fraction_scales[] = { | 
|  | 0.1f,         0.01f,         0.001f,        0.0001f, | 
|  | 0.00001f,     0.000001f,     0.0000001f,    0.00000001f, | 
|  | 0.000000001f, 0.0000000001f, 0.00000000001f}; | 
|  | int FXSYS_FractionalScaleCount() { | 
|  | return FX_ArraySize(fraction_scales); | 
|  | } | 
|  |  | 
|  | FX_FLOAT FXSYS_FractionalScale(size_t scale_factor, int value) { | 
|  | return fraction_scales[scale_factor] * value; | 
|  | } | 
|  |  | 
|  | FX_FLOAT FX_atof(const CFX_ByteStringC& strc) { | 
|  | if (strc.IsEmpty()) | 
|  | return 0.0; | 
|  |  | 
|  | int cc = 0; | 
|  | bool bNegative = false; | 
|  | int len = strc.GetLength(); | 
|  | if (strc[0] == '+') { | 
|  | cc++; | 
|  | } else if (strc[0] == '-') { | 
|  | bNegative = TRUE; | 
|  | cc++; | 
|  | } | 
|  | while (cc < len) { | 
|  | if (strc[cc] != '+' && strc[cc] != '-') { | 
|  | break; | 
|  | } | 
|  | cc++; | 
|  | } | 
|  | FX_FLOAT value = 0; | 
|  | while (cc < len) { | 
|  | if (strc[cc] == '.') { | 
|  | break; | 
|  | } | 
|  | value = value * 10 + FXSYS_toDecimalDigit(strc.CharAt(cc)); | 
|  | cc++; | 
|  | } | 
|  | int scale = 0; | 
|  | if (cc < len && strc[cc] == '.') { | 
|  | cc++; | 
|  | while (cc < len) { | 
|  | value += | 
|  | FXSYS_FractionalScale(scale, FXSYS_toDecimalDigit(strc.CharAt(cc))); | 
|  | scale++; | 
|  | if (scale == FXSYS_FractionalScaleCount()) | 
|  | break; | 
|  | cc++; | 
|  | } | 
|  | } | 
|  | return bNegative ? -value : value; | 
|  | } | 
|  |  | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ && _MSC_VER < 1900 | 
|  | void FXSYS_snprintf(char* str, | 
|  | size_t size, | 
|  | _Printf_format_string_ const char* fmt, | 
|  | ...) { | 
|  | va_list ap; | 
|  | va_start(ap, fmt); | 
|  | FXSYS_vsnprintf(str, size, fmt, ap); | 
|  | va_end(ap); | 
|  | } | 
|  | void FXSYS_vsnprintf(char* str, size_t size, const char* fmt, va_list ap) { | 
|  | (void)_vsnprintf(str, size, fmt, ap); | 
|  | if (size) { | 
|  | str[size - 1] = 0; | 
|  | } | 
|  | } | 
|  | #endif  // _FXM_PLATFORM_WINDOWS_ && _MSC_VER < 1900 | 
|  |  | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ | 
|  | class CFindFileData { | 
|  | public: | 
|  | virtual ~CFindFileData() {} | 
|  | HANDLE m_Handle; | 
|  | FX_BOOL m_bEnd; | 
|  | }; | 
|  |  | 
|  | class CFindFileDataA : public CFindFileData { | 
|  | public: | 
|  | ~CFindFileDataA() override {} | 
|  | WIN32_FIND_DATAA m_FindData; | 
|  | }; | 
|  |  | 
|  | class CFindFileDataW : public CFindFileData { | 
|  | public: | 
|  | ~CFindFileDataW() override {} | 
|  | WIN32_FIND_DATAW m_FindData; | 
|  | }; | 
|  | #endif | 
|  |  | 
|  | void* FX_OpenFolder(const FX_CHAR* path) { | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ | 
|  | std::unique_ptr<CFindFileDataA> pData(new CFindFileDataA); | 
|  | pData->m_Handle = FindFirstFileExA((CFX_ByteString(path) + "/*.*").c_str(), | 
|  | FindExInfoStandard, &pData->m_FindData, | 
|  | FindExSearchNameMatch, nullptr, 0); | 
|  | if (pData->m_Handle == INVALID_HANDLE_VALUE) | 
|  | return nullptr; | 
|  |  | 
|  | pData->m_bEnd = FALSE; | 
|  | return pData.release(); | 
|  | #else | 
|  | DIR* dir = opendir(path); | 
|  | return dir; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void* FX_OpenFolder(const FX_WCHAR* path) { | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ | 
|  | std::unique_ptr<CFindFileDataW> pData(new CFindFileDataW); | 
|  | pData->m_Handle = FindFirstFileExW((CFX_WideString(path) + L"/*.*").c_str(), | 
|  | FindExInfoStandard, &pData->m_FindData, | 
|  | FindExSearchNameMatch, nullptr, 0); | 
|  | if (pData->m_Handle == INVALID_HANDLE_VALUE) | 
|  | return nullptr; | 
|  |  | 
|  | pData->m_bEnd = FALSE; | 
|  | return pData.release(); | 
|  | #else | 
|  | DIR* dir = opendir(CFX_ByteString::FromUnicode(path).c_str()); | 
|  | return dir; | 
|  | #endif | 
|  | } | 
|  | FX_BOOL FX_GetNextFile(void* handle, | 
|  | CFX_ByteString& filename, | 
|  | FX_BOOL& bFolder) { | 
|  | if (!handle) { | 
|  | return FALSE; | 
|  | } | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ | 
|  | CFindFileDataA* pData = (CFindFileDataA*)handle; | 
|  | if (pData->m_bEnd) | 
|  | return FALSE; | 
|  |  | 
|  | filename = pData->m_FindData.cFileName; | 
|  | bFolder = pData->m_FindData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY; | 
|  | if (!FindNextFileA(pData->m_Handle, &pData->m_FindData)) | 
|  | pData->m_bEnd = TRUE; | 
|  | return TRUE; | 
|  | #elif defined(__native_client__) | 
|  | abort(); | 
|  | return FALSE; | 
|  | #else | 
|  | struct dirent* de = readdir((DIR*)handle); | 
|  | if (!de) { | 
|  | return FALSE; | 
|  | } | 
|  | filename = de->d_name; | 
|  | bFolder = de->d_type == DT_DIR; | 
|  | return TRUE; | 
|  | #endif | 
|  | } | 
|  | FX_BOOL FX_GetNextFile(void* handle, | 
|  | CFX_WideString& filename, | 
|  | FX_BOOL& bFolder) { | 
|  | if (!handle) { | 
|  | return FALSE; | 
|  | } | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ | 
|  | CFindFileDataW* pData = (CFindFileDataW*)handle; | 
|  | if (pData->m_bEnd) { | 
|  | return FALSE; | 
|  | } | 
|  | filename = pData->m_FindData.cFileName; | 
|  | bFolder = pData->m_FindData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY; | 
|  | if (!FindNextFileW(pData->m_Handle, &pData->m_FindData)) { | 
|  | pData->m_bEnd = TRUE; | 
|  | } | 
|  | return TRUE; | 
|  | #elif defined(__native_client__) | 
|  | abort(); | 
|  | return FALSE; | 
|  | #else | 
|  | struct dirent* de = readdir((DIR*)handle); | 
|  | if (!de) { | 
|  | return FALSE; | 
|  | } | 
|  | filename = CFX_WideString::FromLocal(de->d_name); | 
|  | bFolder = de->d_type == DT_DIR; | 
|  | return TRUE; | 
|  | #endif | 
|  | } | 
|  | void FX_CloseFolder(void* handle) { | 
|  | if (!handle) { | 
|  | return; | 
|  | } | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ | 
|  | CFindFileData* pData = (CFindFileData*)handle; | 
|  | FindClose(pData->m_Handle); | 
|  | delete pData; | 
|  | #else | 
|  | closedir((DIR*)handle); | 
|  | #endif | 
|  | } | 
|  | FX_WCHAR FX_GetFolderSeparator() { | 
|  | #if _FXM_PLATFORM_ == _FXM_PLATFORM_WINDOWS_ | 
|  | return '\\'; | 
|  | #else | 
|  | return '/'; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | CFX_Matrix_3by3 CFX_Matrix_3by3::Inverse() { | 
|  | FX_FLOAT det = | 
|  | a * (e * i - f * h) - b * (i * d - f * g) + c * (d * h - e * g); | 
|  | if (FXSYS_fabs(det) < 0.0000001) | 
|  | return CFX_Matrix_3by3(); | 
|  |  | 
|  | return CFX_Matrix_3by3( | 
|  | (e * i - f * h) / det, -(b * i - c * h) / det, (b * f - c * e) / det, | 
|  | -(d * i - f * g) / det, (a * i - c * g) / det, -(a * f - c * d) / det, | 
|  | (d * h - e * g) / det, -(a * h - b * g) / det, (a * e - b * d) / det); | 
|  | } | 
|  |  | 
|  | CFX_Matrix_3by3 CFX_Matrix_3by3::Multiply(const CFX_Matrix_3by3& m) { | 
|  | return CFX_Matrix_3by3( | 
|  | a * m.a + b * m.d + c * m.g, a * m.b + b * m.e + c * m.h, | 
|  | a * m.c + b * m.f + c * m.i, d * m.a + e * m.d + f * m.g, | 
|  | d * m.b + e * m.e + f * m.h, d * m.c + e * m.f + f * m.i, | 
|  | g * m.a + h * m.d + i * m.g, g * m.b + h * m.e + i * m.h, | 
|  | g * m.c + h * m.f + i * m.i); | 
|  | } | 
|  |  | 
|  | CFX_Vector_3by1 CFX_Matrix_3by3::TransformVector(const CFX_Vector_3by1& v) { | 
|  | return CFX_Vector_3by1(a * v.a + b * v.b + c * v.c, | 
|  | d * v.a + e * v.b + f * v.c, | 
|  | g * v.a + h * v.b + i * v.c); | 
|  | } | 
|  |  | 
|  | uint32_t GetBits32(const uint8_t* pData, int bitpos, int nbits) { | 
|  | ASSERT(0 < nbits && nbits <= 32); | 
|  | const uint8_t* dataPtr = &pData[bitpos / 8]; | 
|  | int bitShift; | 
|  | int bitMask; | 
|  | int dstShift; | 
|  | int bitCount = bitpos & 0x07; | 
|  | if (nbits < 8 && nbits + bitCount <= 8) { | 
|  | bitShift = 8 - nbits - bitCount; | 
|  | bitMask = (1 << nbits) - 1; | 
|  | dstShift = 0; | 
|  | } else { | 
|  | bitShift = 0; | 
|  | int bitOffset = 8 - bitCount; | 
|  | bitMask = (1 << std::min(bitOffset, nbits)) - 1; | 
|  | dstShift = nbits - bitOffset; | 
|  | } | 
|  | uint32_t result = (uint32_t)(*dataPtr++ >> bitShift & bitMask) << dstShift; | 
|  | while (dstShift >= 8) { | 
|  | dstShift -= 8; | 
|  | result |= *dataPtr++ << dstShift; | 
|  | } | 
|  | if (dstShift > 0) { | 
|  | bitShift = 8 - dstShift; | 
|  | bitMask = (1 << dstShift) - 1; | 
|  | result |= *dataPtr++ >> bitShift & bitMask; | 
|  | } | 
|  | return result; | 
|  | } |