/* Boost Software License - Version 1.0 - August 17th, 2003 Permission is hereby granted, free of charge, to any person or organization obtaining a copy of the software and accompanying documentation covered by this license (the "Software") to use, reproduce, display, distribute, execute, and transmit the Software, and to prepare derivative works of the Software, and to permit third-parties to whom the Software is furnished to do so, all subject to the following: The copyright notices in the Software and this entire statement, including the above license grant, this restriction and the following disclaimer, must be included in all copies of the Software, in whole or in part, and all derivative works of the Software, unless such copies or derivative works are solely in the form of machine-executable object code generated by a source language processor. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #ifndef LIBINTL_INTERNAL_UTIL_HPP_ #define LIBINTL_INTERNAL_UTIL_HPP_ #if defined(WIN32) || defined(WINCE) typedef unsigned int uint32_t; #else #include #endif namespace libintllite { namespace internal { // Helper functions for handling numbers and char array conversions: static inline bool isBigEndian() { uint32_t checkNumber = 0x1100; return (*reinterpret_cast(&checkNumber) != 0); } static inline uint32_t swapUInt32Bytes(uint32_t number) { const char* numberAsCharArray = reinterpret_cast(&number); uint32_t swappedNumber; char* swappedNumberAsCharArray = reinterpret_cast(&swappedNumber); swappedNumberAsCharArray[0] = numberAsCharArray[3]; swappedNumberAsCharArray[1] = numberAsCharArray[2]; swappedNumberAsCharArray[2] = numberAsCharArray[1]; swappedNumberAsCharArray[3] = numberAsCharArray[0]; return swappedNumber; } static inline uint32_t charArrayToUInt32(const char uint32CharArray[4]) { return *reinterpret_cast(uint32CharArray); } static inline bool readUIn32FromFile(FILE* fileHandle, bool needsBeToLeConversion, uint32_t& outReadUInt32) { char uint32CharArray[4]; if ((fread(uint32CharArray, 1, 4, fileHandle)) != 4) { return false; } if (needsBeToLeConversion) { outReadUInt32 = swapUInt32Bytes(charArrayToUInt32(uint32CharArray)); return true; } else { outReadUInt32 = charArrayToUInt32(uint32CharArray); return true; } } // RAII classes: template class ArrayGurard { private: ArrayGurard(const ArrayGurard&); ArrayGurard& operator=(const ArrayGurard&); T*& arrayRef; bool released; public: explicit ArrayGurard(T*& arrayRef) : arrayRef(arrayRef), released(false) { } ~ArrayGurard() { if (!this->released) { delete[] this->arrayRef; } } const T* release() { this->released = true; return this->arrayRef; } }; class CloseFileHandleGuard { private: CloseFileHandleGuard(const CloseFileHandleGuard&); CloseFileHandleGuard& operator=(const CloseFileHandleGuard&); FILE*& fileHandleRef; public: explicit CloseFileHandleGuard(FILE*& fileHandleRef) : fileHandleRef(fileHandleRef) { } ~CloseFileHandleGuard() { if (this->fileHandleRef) { fclose(this->fileHandleRef); } } }; // Helper function to load strings from a .mo file and stores them in a given array static bool loadMoFileStringsToArray(FILE* moFile, uint32_t numberOfStrings, uint32_t stringsTableOffsetFromFileBegin, bool needsBeToLeConversion, std::string* outStringsFromMoFileArray) { if (fseek(moFile, stringsTableOffsetFromFileBegin, SEEK_SET) != 0) return false; uint32_t* stringsLengthsArray = NULL; ArrayGurard stringsLengthsArrayGuard(stringsLengthsArray); stringsLengthsArray = new uint32_t[numberOfStrings]; if (!stringsLengthsArray) { return false; } uint32_t firstStringOffset; uint32_t lastStringOffset; { uint32_t currentStringLength; uint32_t currentStringOffset; for (uint32_t i = 0; i < numberOfStrings; i++) { if (!readUIn32FromFile(moFile, needsBeToLeConversion, currentStringLength)) return false; if (!readUIn32FromFile(moFile, needsBeToLeConversion, currentStringOffset)) return false; stringsLengthsArray[i] = currentStringLength; if (i == 0) { firstStringOffset = currentStringOffset; } if (i == (numberOfStrings - 1)) { lastStringOffset = currentStringOffset; } } } { char* stringCharsArray = NULL; ArrayGurard stringCharsArrayGuard(stringCharsArray); uint32_t stringCharsArraySize = lastStringOffset + stringsLengthsArray[numberOfStrings - 1] + 1 - firstStringOffset; if (stringCharsArraySize == 0) { return false; } if (fseek(moFile, firstStringOffset, SEEK_SET) != 0) return false; stringCharsArray = new char[stringCharsArraySize]; if (!stringCharsArray) { return false; } if (fread(stringCharsArray, 1, stringCharsArraySize, moFile) != stringCharsArraySize) return false; const char* stringsCharsArrayIter = stringCharsArray; for (uint32_t i = 0; i < numberOfStrings; i++) { const char* currentStrEndIter = stringsCharsArrayIter + stringsLengthsArray[i]; outStringsFromMoFileArray[i] = std::string(stringsCharsArrayIter, currentStrEndIter); stringsCharsArrayIter = currentStrEndIter + 1 /* skip the NULL char at the end of the string */ ; } } return true; } } // namespace internal } // namespace libintllite #endif // LIBINTL_INTERNAL_UTIL_HPP_