commit
5f9939070e
@ -1,428 +0,0 @@
|
|||||||
// --------------------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Written by Zoltan Csizmadia, zoltan_csizmadia@yahoo.com
|
|
||||||
// For companies(Austin,TX): If you would like to get my resume, send an email.
|
|
||||||
//
|
|
||||||
// The source is free, but if you want to use it, mention my name and e-mail address
|
|
||||||
//
|
|
||||||
// History:
|
|
||||||
// 1.0 Initial version Zoltan Csizmadia
|
|
||||||
// 1.1 WhineCube version Masken
|
|
||||||
// 1.2 Dolphin version Masken
|
|
||||||
//
|
|
||||||
// --------------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
#if defined(WIN32)
|
|
||||||
#include <cstdio>
|
|
||||||
#include <windows.h>
|
|
||||||
#include "common/extended_trace.h"
|
|
||||||
#include "common/string_util.h"
|
|
||||||
using namespace std;
|
|
||||||
|
|
||||||
#include <tchar.h>
|
|
||||||
#include <ImageHlp.h>
|
|
||||||
|
|
||||||
#define BUFFERSIZE 0x200
|
|
||||||
#pragma warning(disable:4996)
|
|
||||||
|
|
||||||
// Unicode safe char* -> TCHAR* conversion
|
|
||||||
void PCSTR2LPTSTR( PCSTR lpszIn, LPTSTR lpszOut )
|
|
||||||
{
|
|
||||||
#if defined(UNICODE)||defined(_UNICODE)
|
|
||||||
ULONG index = 0;
|
|
||||||
PCSTR lpAct = lpszIn;
|
|
||||||
|
|
||||||
for( ; ; lpAct++ )
|
|
||||||
{
|
|
||||||
lpszOut[index++] = (TCHAR)(*lpAct);
|
|
||||||
if ( *lpAct == 0 )
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
// This is trivial :)
|
|
||||||
strcpy( lpszOut, lpszIn );
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Let's figure out the path for the symbol files
|
|
||||||
// Search path= ".;%_NT_SYMBOL_PATH%;%_NT_ALTERNATE_SYMBOL_PATH%;%SYSTEMROOT%;%SYSTEMROOT%\System32;" + lpszIniPath
|
|
||||||
// Note: There is no size check for lpszSymbolPath!
|
|
||||||
static void InitSymbolPath( PSTR lpszSymbolPath, PCSTR lpszIniPath )
|
|
||||||
{
|
|
||||||
CHAR lpszPath[BUFFERSIZE];
|
|
||||||
|
|
||||||
// Creating the default path
|
|
||||||
// ".;%_NT_SYMBOL_PATH%;%_NT_ALTERNATE_SYMBOL_PATH%;%SYSTEMROOT%;%SYSTEMROOT%\System32;"
|
|
||||||
strcpy( lpszSymbolPath, "." );
|
|
||||||
|
|
||||||
// environment variable _NT_SYMBOL_PATH
|
|
||||||
if ( GetEnvironmentVariableA( "_NT_SYMBOL_PATH", lpszPath, BUFFERSIZE ) )
|
|
||||||
{
|
|
||||||
strcat( lpszSymbolPath, ";" );
|
|
||||||
strcat( lpszSymbolPath, lpszPath );
|
|
||||||
}
|
|
||||||
|
|
||||||
// environment variable _NT_ALTERNATE_SYMBOL_PATH
|
|
||||||
if ( GetEnvironmentVariableA( "_NT_ALTERNATE_SYMBOL_PATH", lpszPath, BUFFERSIZE ) )
|
|
||||||
{
|
|
||||||
strcat( lpszSymbolPath, ";" );
|
|
||||||
strcat( lpszSymbolPath, lpszPath );
|
|
||||||
}
|
|
||||||
|
|
||||||
// environment variable SYSTEMROOT
|
|
||||||
if ( GetEnvironmentVariableA( "SYSTEMROOT", lpszPath, BUFFERSIZE ) )
|
|
||||||
{
|
|
||||||
strcat( lpszSymbolPath, ";" );
|
|
||||||
strcat( lpszSymbolPath, lpszPath );
|
|
||||||
strcat( lpszSymbolPath, ";" );
|
|
||||||
|
|
||||||
// SYSTEMROOT\System32
|
|
||||||
strcat( lpszSymbolPath, lpszPath );
|
|
||||||
strcat( lpszSymbolPath, "\\System32" );
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add user defined path
|
|
||||||
if ( lpszIniPath != nullptr )
|
|
||||||
if ( lpszIniPath[0] != '\0' )
|
|
||||||
{
|
|
||||||
strcat( lpszSymbolPath, ";" );
|
|
||||||
strcat( lpszSymbolPath, lpszIniPath );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Uninitialize the loaded symbol files
|
|
||||||
BOOL UninitSymInfo() {
|
|
||||||
return SymCleanup( GetCurrentProcess() );
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initializes the symbol files
|
|
||||||
BOOL InitSymInfo( PCSTR lpszInitialSymbolPath )
|
|
||||||
{
|
|
||||||
CHAR lpszSymbolPath[BUFFERSIZE];
|
|
||||||
DWORD symOptions = SymGetOptions();
|
|
||||||
|
|
||||||
symOptions |= SYMOPT_LOAD_LINES;
|
|
||||||
symOptions &= ~SYMOPT_UNDNAME;
|
|
||||||
SymSetOptions( symOptions );
|
|
||||||
InitSymbolPath( lpszSymbolPath, lpszInitialSymbolPath );
|
|
||||||
|
|
||||||
return SymInitialize( GetCurrentProcess(), lpszSymbolPath, TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the module name from a given address
|
|
||||||
static BOOL GetModuleNameFromAddress( UINT address, LPTSTR lpszModule )
|
|
||||||
{
|
|
||||||
BOOL ret = FALSE;
|
|
||||||
IMAGEHLP_MODULE moduleInfo;
|
|
||||||
|
|
||||||
::ZeroMemory( &moduleInfo, sizeof(moduleInfo) );
|
|
||||||
moduleInfo.SizeOfStruct = sizeof(moduleInfo);
|
|
||||||
|
|
||||||
if ( SymGetModuleInfo( GetCurrentProcess(), (DWORD)address, &moduleInfo ) )
|
|
||||||
{
|
|
||||||
// Got it!
|
|
||||||
PCSTR2LPTSTR( moduleInfo.ModuleName, lpszModule );
|
|
||||||
ret = TRUE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
// Not found :(
|
|
||||||
_tcscpy( lpszModule, _T("?") );
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get function prototype and parameter info from ip address and stack address
|
|
||||||
static BOOL GetFunctionInfoFromAddresses( ULONG fnAddress, ULONG stackAddress, LPTSTR lpszSymbol )
|
|
||||||
{
|
|
||||||
BOOL ret = FALSE;
|
|
||||||
DWORD dwSymSize = 10000;
|
|
||||||
TCHAR lpszUnDSymbol[BUFFERSIZE]=_T("?");
|
|
||||||
CHAR lpszNonUnicodeUnDSymbol[BUFFERSIZE]="?";
|
|
||||||
LPTSTR lpszParamSep = nullptr;
|
|
||||||
LPTSTR lpszParsed = lpszUnDSymbol;
|
|
||||||
PIMAGEHLP_SYMBOL pSym = (PIMAGEHLP_SYMBOL)GlobalAlloc( GMEM_FIXED, dwSymSize );
|
|
||||||
|
|
||||||
::ZeroMemory( pSym, dwSymSize );
|
|
||||||
pSym->SizeOfStruct = dwSymSize;
|
|
||||||
pSym->MaxNameLength = dwSymSize - sizeof(IMAGEHLP_SYMBOL);
|
|
||||||
|
|
||||||
// Set the default to unknown
|
|
||||||
_tcscpy( lpszSymbol, _T("?") );
|
|
||||||
|
|
||||||
// Get symbol info for IP
|
|
||||||
#ifndef _M_X64
|
|
||||||
DWORD dwDisp = 0;
|
|
||||||
if ( SymGetSymFromAddr( GetCurrentProcess(), (ULONG)fnAddress, &dwDisp, pSym ) )
|
|
||||||
#else
|
|
||||||
//makes it compile but hell im not sure if this works...
|
|
||||||
DWORD64 dwDisp = 0;
|
|
||||||
if ( SymGetSymFromAddr( GetCurrentProcess(), (ULONG)fnAddress, (PDWORD64)&dwDisp, pSym ) )
|
|
||||||
#endif
|
|
||||||
{
|
|
||||||
// Make the symbol readable for humans
|
|
||||||
UnDecorateSymbolName( pSym->Name, lpszNonUnicodeUnDSymbol, BUFFERSIZE,
|
|
||||||
UNDNAME_COMPLETE |
|
|
||||||
UNDNAME_NO_THISTYPE |
|
|
||||||
UNDNAME_NO_SPECIAL_SYMS |
|
|
||||||
UNDNAME_NO_MEMBER_TYPE |
|
|
||||||
UNDNAME_NO_MS_KEYWORDS |
|
|
||||||
UNDNAME_NO_ACCESS_SPECIFIERS );
|
|
||||||
|
|
||||||
// Symbol information is ANSI string
|
|
||||||
PCSTR2LPTSTR( lpszNonUnicodeUnDSymbol, lpszUnDSymbol );
|
|
||||||
|
|
||||||
// I am just smarter than the symbol file :)
|
|
||||||
if (_tcscmp(lpszUnDSymbol, _T("_WinMain@16")) == 0)
|
|
||||||
_tcscpy(lpszUnDSymbol, _T("WinMain(HINSTANCE,HINSTANCE,LPCTSTR,int)"));
|
|
||||||
else if (_tcscmp(lpszUnDSymbol, _T("_main")) == 0)
|
|
||||||
_tcscpy(lpszUnDSymbol, _T("main(int,TCHAR * *)"));
|
|
||||||
else if (_tcscmp(lpszUnDSymbol, _T("_mainCRTStartup")) == 0)
|
|
||||||
_tcscpy(lpszUnDSymbol, _T("mainCRTStartup()"));
|
|
||||||
else if (_tcscmp(lpszUnDSymbol, _T("_wmain")) == 0)
|
|
||||||
_tcscpy(lpszUnDSymbol, _T("wmain(int,TCHAR * *,TCHAR * *)"));
|
|
||||||
else if (_tcscmp(lpszUnDSymbol, _T("_wmainCRTStartup")) == 0)
|
|
||||||
_tcscpy(lpszUnDSymbol, _T("wmainCRTStartup()"));
|
|
||||||
|
|
||||||
lpszSymbol[0] = _T('\0');
|
|
||||||
|
|
||||||
// Let's go through the stack, and modify the function prototype, and insert the actual
|
|
||||||
// parameter values from the stack
|
|
||||||
if ( _tcsstr( lpszUnDSymbol, _T("(void)") ) == nullptr && _tcsstr( lpszUnDSymbol, _T("()") ) == nullptr)
|
|
||||||
{
|
|
||||||
ULONG index = 0;
|
|
||||||
for( ; ; index++ )
|
|
||||||
{
|
|
||||||
lpszParamSep = _tcschr( lpszParsed, _T(',') );
|
|
||||||
if ( lpszParamSep == nullptr )
|
|
||||||
break;
|
|
||||||
|
|
||||||
*lpszParamSep = _T('\0');
|
|
||||||
|
|
||||||
_tcscat( lpszSymbol, lpszParsed );
|
|
||||||
_stprintf( lpszSymbol + _tcslen(lpszSymbol), _T("=0x%08X,"), *((ULONG*)(stackAddress) + 2 + index) );
|
|
||||||
|
|
||||||
lpszParsed = lpszParamSep + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
lpszParamSep = _tcschr( lpszParsed, _T(')') );
|
|
||||||
if ( lpszParamSep != nullptr )
|
|
||||||
{
|
|
||||||
*lpszParamSep = _T('\0');
|
|
||||||
|
|
||||||
_tcscat( lpszSymbol, lpszParsed );
|
|
||||||
_stprintf( lpszSymbol + _tcslen(lpszSymbol), _T("=0x%08X)"), *((ULONG*)(stackAddress) + 2 + index) );
|
|
||||||
|
|
||||||
lpszParsed = lpszParamSep + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_tcscat( lpszSymbol, lpszParsed );
|
|
||||||
|
|
||||||
ret = TRUE;
|
|
||||||
}
|
|
||||||
GlobalFree( pSym );
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get source file name and line number from IP address
|
|
||||||
// The output format is: "sourcefile(linenumber)" or
|
|
||||||
// "modulename!address" or
|
|
||||||
// "address"
|
|
||||||
static BOOL GetSourceInfoFromAddress( UINT address, LPTSTR lpszSourceInfo )
|
|
||||||
{
|
|
||||||
BOOL ret = FALSE;
|
|
||||||
IMAGEHLP_LINE lineInfo;
|
|
||||||
DWORD dwDisp;
|
|
||||||
TCHAR lpszFileName[BUFFERSIZE] = _T("");
|
|
||||||
TCHAR lpModuleInfo[BUFFERSIZE] = _T("");
|
|
||||||
|
|
||||||
_tcscpy( lpszSourceInfo, _T("?(?)") );
|
|
||||||
|
|
||||||
::ZeroMemory( &lineInfo, sizeof( lineInfo ) );
|
|
||||||
lineInfo.SizeOfStruct = sizeof( lineInfo );
|
|
||||||
|
|
||||||
if ( SymGetLineFromAddr( GetCurrentProcess(), address, &dwDisp, &lineInfo ) )
|
|
||||||
{
|
|
||||||
// Got it. Let's use "sourcefile(linenumber)" format
|
|
||||||
PCSTR2LPTSTR( lineInfo.FileName, lpszFileName );
|
|
||||||
TCHAR fname[_MAX_FNAME];
|
|
||||||
TCHAR ext[_MAX_EXT];
|
|
||||||
_tsplitpath(lpszFileName, nullptr, nullptr, fname, ext);
|
|
||||||
_stprintf( lpszSourceInfo, _T("%s%s(%d)"), fname, ext, lineInfo.LineNumber );
|
|
||||||
ret = TRUE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// There is no source file information. :(
|
|
||||||
// Let's use the "modulename!address" format
|
|
||||||
GetModuleNameFromAddress( address, lpModuleInfo );
|
|
||||||
|
|
||||||
if ( lpModuleInfo[0] == _T('?') || lpModuleInfo[0] == _T('\0'))
|
|
||||||
// There is no modulename information. :((
|
|
||||||
// Let's use the "address" format
|
|
||||||
_stprintf( lpszSourceInfo, _T("0x%08X"), address );
|
|
||||||
else
|
|
||||||
_stprintf( lpszSourceInfo, _T("%s!0x%08X"), lpModuleInfo, address );
|
|
||||||
|
|
||||||
ret = FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void PrintFunctionAndSourceInfo(FILE* file, const STACKFRAME& callstack)
|
|
||||||
{
|
|
||||||
TCHAR symInfo[BUFFERSIZE] = _T("?");
|
|
||||||
TCHAR srcInfo[BUFFERSIZE] = _T("?");
|
|
||||||
|
|
||||||
GetFunctionInfoFromAddresses((ULONG)callstack.AddrPC.Offset, (ULONG)callstack.AddrFrame.Offset, symInfo);
|
|
||||||
GetSourceInfoFromAddress((ULONG)callstack.AddrPC.Offset, srcInfo);
|
|
||||||
etfprint(file, " " + Common::TStrToUTF8(srcInfo) + " : " + Common::TStrToUTF8(symInfo) + "\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
void StackTrace( HANDLE hThread, const char* lpszMessage, FILE *file )
|
|
||||||
{
|
|
||||||
STACKFRAME callStack;
|
|
||||||
BOOL bResult;
|
|
||||||
CONTEXT context;
|
|
||||||
HANDLE hProcess = GetCurrentProcess();
|
|
||||||
|
|
||||||
// If it's not this thread, let's suspend it, and resume it at the end
|
|
||||||
if ( hThread != GetCurrentThread() )
|
|
||||||
if ( SuspendThread( hThread ) == -1 )
|
|
||||||
{
|
|
||||||
// whaaat ?!
|
|
||||||
etfprint(file, "Call stack info failed\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
::ZeroMemory( &context, sizeof(context) );
|
|
||||||
context.ContextFlags = CONTEXT_FULL;
|
|
||||||
|
|
||||||
if ( !GetThreadContext( hThread, &context ) )
|
|
||||||
{
|
|
||||||
etfprint(file, "Call stack info failed\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
::ZeroMemory( &callStack, sizeof(callStack) );
|
|
||||||
#ifndef _M_X64
|
|
||||||
callStack.AddrPC.Offset = context.Eip;
|
|
||||||
callStack.AddrStack.Offset = context.Esp;
|
|
||||||
callStack.AddrFrame.Offset = context.Ebp;
|
|
||||||
#else
|
|
||||||
callStack.AddrPC.Offset = context.Rip;
|
|
||||||
callStack.AddrStack.Offset = context.Rsp;
|
|
||||||
callStack.AddrFrame.Offset = context.Rbp;
|
|
||||||
#endif
|
|
||||||
callStack.AddrPC.Mode = AddrModeFlat;
|
|
||||||
callStack.AddrStack.Mode = AddrModeFlat;
|
|
||||||
callStack.AddrFrame.Mode = AddrModeFlat;
|
|
||||||
|
|
||||||
etfprint(file, "Call stack info: \n");
|
|
||||||
etfprint(file, lpszMessage);
|
|
||||||
|
|
||||||
PrintFunctionAndSourceInfo(file, callStack);
|
|
||||||
|
|
||||||
for( ULONG index = 0; ; index++ )
|
|
||||||
{
|
|
||||||
bResult = StackWalk(
|
|
||||||
IMAGE_FILE_MACHINE_I386,
|
|
||||||
hProcess,
|
|
||||||
hThread,
|
|
||||||
&callStack,
|
|
||||||
nullptr,
|
|
||||||
nullptr,
|
|
||||||
SymFunctionTableAccess,
|
|
||||||
SymGetModuleBase,
|
|
||||||
nullptr);
|
|
||||||
|
|
||||||
if ( index == 0 )
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if( !bResult || callStack.AddrFrame.Offset == 0 )
|
|
||||||
break;
|
|
||||||
|
|
||||||
PrintFunctionAndSourceInfo(file, callStack);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( hThread != GetCurrentThread() )
|
|
||||||
ResumeThread( hThread );
|
|
||||||
}
|
|
||||||
|
|
||||||
void StackTrace(HANDLE hThread, const char* lpszMessage, FILE *file, DWORD eip, DWORD esp, DWORD ebp )
|
|
||||||
{
|
|
||||||
STACKFRAME callStack;
|
|
||||||
BOOL bResult;
|
|
||||||
TCHAR symInfo[BUFFERSIZE] = _T("?");
|
|
||||||
TCHAR srcInfo[BUFFERSIZE] = _T("?");
|
|
||||||
HANDLE hProcess = GetCurrentProcess();
|
|
||||||
|
|
||||||
// If it's not this thread, let's suspend it, and resume it at the end
|
|
||||||
if ( hThread != GetCurrentThread() )
|
|
||||||
if ( SuspendThread( hThread ) == -1 )
|
|
||||||
{
|
|
||||||
// whaaat ?!
|
|
||||||
etfprint(file, "Call stack info failed\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
::ZeroMemory( &callStack, sizeof(callStack) );
|
|
||||||
callStack.AddrPC.Offset = eip;
|
|
||||||
callStack.AddrStack.Offset = esp;
|
|
||||||
callStack.AddrFrame.Offset = ebp;
|
|
||||||
callStack.AddrPC.Mode = AddrModeFlat;
|
|
||||||
callStack.AddrStack.Mode = AddrModeFlat;
|
|
||||||
callStack.AddrFrame.Mode = AddrModeFlat;
|
|
||||||
|
|
||||||
etfprint(file, "Call stack info: \n");
|
|
||||||
etfprint(file, lpszMessage);
|
|
||||||
|
|
||||||
PrintFunctionAndSourceInfo(file, callStack);
|
|
||||||
|
|
||||||
for( ULONG index = 0; ; index++ )
|
|
||||||
{
|
|
||||||
bResult = StackWalk(
|
|
||||||
IMAGE_FILE_MACHINE_I386,
|
|
||||||
hProcess,
|
|
||||||
hThread,
|
|
||||||
&callStack,
|
|
||||||
nullptr,
|
|
||||||
nullptr,
|
|
||||||
SymFunctionTableAccess,
|
|
||||||
SymGetModuleBase,
|
|
||||||
nullptr);
|
|
||||||
|
|
||||||
if ( index == 0 )
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if( !bResult || callStack.AddrFrame.Offset == 0 )
|
|
||||||
break;
|
|
||||||
|
|
||||||
PrintFunctionAndSourceInfo(file, callStack);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( hThread != GetCurrentThread() )
|
|
||||||
ResumeThread( hThread );
|
|
||||||
}
|
|
||||||
|
|
||||||
char g_uefbuf[2048];
|
|
||||||
|
|
||||||
void etfprintf(FILE *file, const char *format, ...)
|
|
||||||
{
|
|
||||||
va_list ap;
|
|
||||||
va_start(ap, format);
|
|
||||||
int len = vsprintf(g_uefbuf, format, ap);
|
|
||||||
fwrite(g_uefbuf, 1, len, file);
|
|
||||||
va_end(ap);
|
|
||||||
}
|
|
||||||
|
|
||||||
void etfprint(FILE *file, const std::string &text)
|
|
||||||
{
|
|
||||||
size_t len = text.length();
|
|
||||||
fwrite(text.data(), 1, len, file);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif //WIN32
|
|
@ -1,50 +0,0 @@
|
|||||||
// -----------------------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Written by Zoltan Csizmadia, zoltan_csizmadia@yahoo.com
|
|
||||||
// For companies(Austin,TX): If you would like to get my resume, send an email.
|
|
||||||
//
|
|
||||||
// The source is free, but if you want to use it, mention my name and e-mail address
|
|
||||||
//
|
|
||||||
// History:
|
|
||||||
// 1.0 Initial version Zoltan Csizmadia
|
|
||||||
// 1.1 WhineCube version Masken
|
|
||||||
// 1.2 Dolphin version Masken
|
|
||||||
//
|
|
||||||
// ----------------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#if defined(WIN32)
|
|
||||||
|
|
||||||
#include <windows.h>
|
|
||||||
#include <tchar.h>
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#pragma comment( lib, "imagehlp.lib" )
|
|
||||||
|
|
||||||
#define EXTENDEDTRACEINITIALIZE( IniSymbolPath ) InitSymInfo( IniSymbolPath )
|
|
||||||
#define EXTENDEDTRACEUNINITIALIZE() UninitSymInfo()
|
|
||||||
#define STACKTRACE(file) StackTrace( GetCurrentThread(), "", file)
|
|
||||||
#define STACKTRACE2(file, eip, esp, ebp) StackTrace(GetCurrentThread(), "", file, eip, esp, ebp)
|
|
||||||
// class File;
|
|
||||||
|
|
||||||
BOOL InitSymInfo( PCSTR );
|
|
||||||
BOOL UninitSymInfo();
|
|
||||||
void StackTrace(HANDLE, char const* msg, FILE *file);
|
|
||||||
void StackTrace(HANDLE, char const* msg, FILE *file, DWORD eip, DWORD esp, DWORD ebp);
|
|
||||||
|
|
||||||
// functions by Masken
|
|
||||||
void etfprintf(FILE *file, const char *format, ...);
|
|
||||||
void etfprint(FILE *file, const std::string &text);
|
|
||||||
#define UEFBUFSIZE 2048
|
|
||||||
extern char g_uefbuf[UEFBUFSIZE];
|
|
||||||
|
|
||||||
#else // not WIN32
|
|
||||||
|
|
||||||
#define EXTENDEDTRACEINITIALIZE( IniSymbolPath ) ((void)0)
|
|
||||||
#define EXTENDEDTRACEUNINITIALIZE() ((void)0)
|
|
||||||
#define STACKTRACE(file) ((void)0)
|
|
||||||
#define STACKTRACE2(file, eip, esp, ebp) ((void)0)
|
|
||||||
|
|
||||||
#endif // WIN32
|
|
@ -1,103 +0,0 @@
|
|||||||
// Copyright 2013 Dolphin Emulator Project / 2014 Citra Emulator Project
|
|
||||||
// Licensed under GPLv2 or any later version
|
|
||||||
// Refer to the license.txt file included.
|
|
||||||
|
|
||||||
|
|
||||||
#include "common/common.h"
|
|
||||||
|
|
||||||
#ifndef _WIN32
|
|
||||||
#include <dirent.h>
|
|
||||||
#else
|
|
||||||
#include <windows.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <algorithm>
|
|
||||||
|
|
||||||
#include "common/file_search.h"
|
|
||||||
#include "common/string_util.h"
|
|
||||||
|
|
||||||
|
|
||||||
CFileSearch::CFileSearch(const CFileSearch::XStringVector& _rSearchStrings, const CFileSearch::XStringVector& _rDirectories)
|
|
||||||
{
|
|
||||||
// Reverse the loop order for speed?
|
|
||||||
for (size_t j = 0; j < _rSearchStrings.size(); j++)
|
|
||||||
{
|
|
||||||
for (size_t i = 0; i < _rDirectories.size(); i++)
|
|
||||||
{
|
|
||||||
FindFiles(_rSearchStrings[j], _rDirectories[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void CFileSearch::FindFiles(const std::string& _searchString, const std::string& _strPath)
|
|
||||||
{
|
|
||||||
std::string GCMSearchPath;
|
|
||||||
Common::BuildCompleteFilename(GCMSearchPath, _strPath, _searchString);
|
|
||||||
#ifdef _WIN32
|
|
||||||
WIN32_FIND_DATA findData;
|
|
||||||
HANDLE FindFirst = FindFirstFile(Common::UTF8ToTStr(GCMSearchPath).c_str(), &findData);
|
|
||||||
|
|
||||||
if (FindFirst != INVALID_HANDLE_VALUE)
|
|
||||||
{
|
|
||||||
bool bkeepLooping = true;
|
|
||||||
|
|
||||||
while (bkeepLooping)
|
|
||||||
{
|
|
||||||
if (findData.cFileName[0] != '.')
|
|
||||||
{
|
|
||||||
std::string strFilename;
|
|
||||||
Common::BuildCompleteFilename(strFilename, _strPath, Common::TStrToUTF8(findData.cFileName));
|
|
||||||
m_FileNames.push_back(strFilename);
|
|
||||||
}
|
|
||||||
|
|
||||||
bkeepLooping = FindNextFile(FindFirst, &findData) ? true : false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FindClose(FindFirst);
|
|
||||||
|
|
||||||
|
|
||||||
#else
|
|
||||||
// TODO: super lame/broken
|
|
||||||
|
|
||||||
auto end_match(_searchString);
|
|
||||||
|
|
||||||
// assuming we have a "*.blah"-like pattern
|
|
||||||
if (!end_match.empty() && end_match[0] == '*')
|
|
||||||
end_match.erase(0, 1);
|
|
||||||
|
|
||||||
// ugly
|
|
||||||
if (end_match == ".*")
|
|
||||||
end_match.clear();
|
|
||||||
|
|
||||||
DIR* dir = opendir(_strPath.c_str());
|
|
||||||
|
|
||||||
if (!dir)
|
|
||||||
return;
|
|
||||||
|
|
||||||
while (auto const dp = readdir(dir))
|
|
||||||
{
|
|
||||||
std::string found(dp->d_name);
|
|
||||||
|
|
||||||
if ((found != ".") && (found != "..")
|
|
||||||
&& (found.size() >= end_match.size())
|
|
||||||
&& std::equal(end_match.rbegin(), end_match.rend(), found.rbegin()))
|
|
||||||
{
|
|
||||||
std::string full_name;
|
|
||||||
if (_strPath.c_str()[_strPath.size()-1] == DIR_SEP_CHR)
|
|
||||||
full_name = _strPath + found;
|
|
||||||
else
|
|
||||||
full_name = _strPath + DIR_SEP + found;
|
|
||||||
|
|
||||||
m_FileNames.push_back(full_name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
closedir(dir);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
const CFileSearch::XStringVector& CFileSearch::GetFileNames() const
|
|
||||||
{
|
|
||||||
return m_FileNames;
|
|
||||||
}
|
|
@ -1,23 +0,0 @@
|
|||||||
// Copyright 2013 Dolphin Emulator Project / 2014 Citra Emulator Project
|
|
||||||
// Licensed under GPLv2 or any later version
|
|
||||||
// Refer to the license.txt file included.
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
class CFileSearch
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
typedef std::vector<std::string>XStringVector;
|
|
||||||
|
|
||||||
CFileSearch(const XStringVector& _rSearchStrings, const XStringVector& _rDirectories);
|
|
||||||
const XStringVector& GetFileNames() const;
|
|
||||||
|
|
||||||
private:
|
|
||||||
|
|
||||||
void FindFiles(const std::string& _searchString, const std::string& _strPath);
|
|
||||||
|
|
||||||
XStringVector m_FileNames;
|
|
||||||
};
|
|
@ -1,107 +0,0 @@
|
|||||||
// Copyright 2013 Dolphin Emulator Project / 2014 Citra Emulator Project
|
|
||||||
// Licensed under GPLv2 or any later version
|
|
||||||
// Refer to the license.txt file included.
|
|
||||||
|
|
||||||
#include <cstdio>
|
|
||||||
|
|
||||||
#include "common/common.h" // Local
|
|
||||||
#include "common/string_util.h"
|
|
||||||
|
|
||||||
bool DefaultMsgHandler(const char* caption, const char* text, bool yes_no, int Style);
|
|
||||||
static MsgAlertHandler msg_handler = DefaultMsgHandler;
|
|
||||||
static bool AlertEnabled = true;
|
|
||||||
|
|
||||||
std::string DefaultStringTranslator(const char* text);
|
|
||||||
static StringTranslator str_translator = DefaultStringTranslator;
|
|
||||||
|
|
||||||
// Select which of these functions that are used for message boxes. If
|
|
||||||
// wxWidgets is enabled we will use wxMsgAlert() that is defined in Main.cpp
|
|
||||||
void RegisterMsgAlertHandler(MsgAlertHandler handler)
|
|
||||||
{
|
|
||||||
msg_handler = handler;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Select translation function. For wxWidgets use wxStringTranslator in Main.cpp
|
|
||||||
void RegisterStringTranslator(StringTranslator translator)
|
|
||||||
{
|
|
||||||
str_translator = translator;
|
|
||||||
}
|
|
||||||
|
|
||||||
// enable/disable the alert handler
|
|
||||||
void SetEnableAlert(bool enable)
|
|
||||||
{
|
|
||||||
AlertEnabled = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
// This is the first stop for gui alerts where the log is updated and the
|
|
||||||
// correct window is shown
|
|
||||||
bool MsgAlert(bool yes_no, int Style, const char* format, ...)
|
|
||||||
{
|
|
||||||
// Read message and write it to the log
|
|
||||||
std::string caption;
|
|
||||||
char buffer[2048];
|
|
||||||
|
|
||||||
static std::string info_caption;
|
|
||||||
static std::string warn_caption;
|
|
||||||
static std::string ques_caption;
|
|
||||||
static std::string crit_caption;
|
|
||||||
|
|
||||||
if (!info_caption.length())
|
|
||||||
{
|
|
||||||
info_caption = str_translator(_trans("Information"));
|
|
||||||
ques_caption = str_translator(_trans("Question"));
|
|
||||||
warn_caption = str_translator(_trans("Warning"));
|
|
||||||
crit_caption = str_translator(_trans("Critical"));
|
|
||||||
}
|
|
||||||
|
|
||||||
switch(Style)
|
|
||||||
{
|
|
||||||
case INFORMATION:
|
|
||||||
caption = info_caption;
|
|
||||||
break;
|
|
||||||
case QUESTION:
|
|
||||||
caption = ques_caption;
|
|
||||||
break;
|
|
||||||
case WARNING:
|
|
||||||
caption = warn_caption;
|
|
||||||
break;
|
|
||||||
case CRITICAL:
|
|
||||||
caption = crit_caption;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, format);
|
|
||||||
Common::CharArrayFromFormatV(buffer, sizeof(buffer)-1, str_translator(format).c_str(), args);
|
|
||||||
va_end(args);
|
|
||||||
|
|
||||||
LOG_INFO(Common, "%s: %s", caption.c_str(), buffer);
|
|
||||||
|
|
||||||
// Don't ignore questions, especially AskYesNo, PanicYesNo could be ignored
|
|
||||||
if (msg_handler && (AlertEnabled || Style == QUESTION || Style == CRITICAL))
|
|
||||||
return msg_handler(caption.c_str(), buffer, yes_no, Style);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default non library dependent panic alert
|
|
||||||
bool DefaultMsgHandler(const char* caption, const char* text, bool yes_no, int Style)
|
|
||||||
{
|
|
||||||
//#ifdef _WIN32
|
|
||||||
// int STYLE = MB_ICONINFORMATION;
|
|
||||||
// if (Style == QUESTION) STYLE = MB_ICONQUESTION;
|
|
||||||
// if (Style == WARNING) STYLE = MB_ICONWARNING;
|
|
||||||
//
|
|
||||||
// return IDYES == MessageBox(0, UTF8ToTStr(text).c_str(), UTF8ToTStr(caption).c_str(), STYLE | (yes_no ? MB_YESNO : MB_OK));
|
|
||||||
//#else
|
|
||||||
printf("%s\n", text);
|
|
||||||
return true;
|
|
||||||
//#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default (non) translator
|
|
||||||
std::string DefaultStringTranslator(const char* text)
|
|
||||||
{
|
|
||||||
return text;
|
|
||||||
}
|
|
||||||
|
|
@ -1,56 +0,0 @@
|
|||||||
// Copyright 2013 Dolphin Emulator Project / 2014 Citra Emulator Project
|
|
||||||
// Licensed under GPLv2 or any later version
|
|
||||||
// Refer to the license.txt file included.
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
// Message alerts
|
|
||||||
enum MSG_TYPE
|
|
||||||
{
|
|
||||||
INFORMATION,
|
|
||||||
QUESTION,
|
|
||||||
WARNING,
|
|
||||||
CRITICAL
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef bool (*MsgAlertHandler)(const char* caption, const char* text,
|
|
||||||
bool yes_no, int Style);
|
|
||||||
typedef std::string (*StringTranslator)(const char* text);
|
|
||||||
|
|
||||||
void RegisterMsgAlertHandler(MsgAlertHandler handler);
|
|
||||||
void RegisterStringTranslator(StringTranslator translator);
|
|
||||||
|
|
||||||
extern bool MsgAlert(bool yes_no, int Style, const char* format, ...)
|
|
||||||
#ifdef __GNUC__
|
|
||||||
__attribute__((format(printf, 3, 4)))
|
|
||||||
#endif
|
|
||||||
;
|
|
||||||
void SetEnableAlert(bool enable);
|
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
#define SuccessAlert(format, ...) MsgAlert(false, INFORMATION, format, __VA_ARGS__)
|
|
||||||
#define PanicAlert(format, ...) MsgAlert(false, WARNING, format, __VA_ARGS__)
|
|
||||||
#define PanicYesNo(format, ...) MsgAlert(true, WARNING, format, __VA_ARGS__)
|
|
||||||
#define AskYesNo(format, ...) MsgAlert(true, QUESTION, format, __VA_ARGS__)
|
|
||||||
#define CriticalAlert(format, ...) MsgAlert(false, CRITICAL, format, __VA_ARGS__)
|
|
||||||
// Use these macros (that do the same thing) if the message should be translated.
|
|
||||||
#define SuccessAlertT(format, ...) MsgAlert(false, INFORMATION, format, __VA_ARGS__)
|
|
||||||
#define PanicAlertT(format, ...) MsgAlert(false, WARNING, format, __VA_ARGS__)
|
|
||||||
#define PanicYesNoT(format, ...) MsgAlert(true, WARNING, format, __VA_ARGS__)
|
|
||||||
#define AskYesNoT(format, ...) MsgAlert(true, QUESTION, format, __VA_ARGS__)
|
|
||||||
#define CriticalAlertT(format, ...) MsgAlert(false, CRITICAL, format, __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define SuccessAlert(format, ...) MsgAlert(false, INFORMATION, format, ##__VA_ARGS__)
|
|
||||||
#define PanicAlert(format, ...) MsgAlert(false, WARNING, format, ##__VA_ARGS__)
|
|
||||||
#define PanicYesNo(format, ...) MsgAlert(true, WARNING, format, ##__VA_ARGS__)
|
|
||||||
#define AskYesNo(format, ...) MsgAlert(true, QUESTION, format, ##__VA_ARGS__)
|
|
||||||
#define CriticalAlert(format, ...) MsgAlert(false, CRITICAL, format, ##__VA_ARGS__)
|
|
||||||
// Use these macros (that do the same thing) if the message should be translated.
|
|
||||||
#define SuccessAlertT(format, ...) MsgAlert(false, INFORMATION, format, ##__VA_ARGS__)
|
|
||||||
#define PanicAlertT(format, ...) MsgAlert(false, WARNING, format, ##__VA_ARGS__)
|
|
||||||
#define PanicYesNoT(format, ...) MsgAlert(true, WARNING, format, ##__VA_ARGS__)
|
|
||||||
#define AskYesNoT(format, ...) MsgAlert(true, QUESTION, format, ##__VA_ARGS__)
|
|
||||||
#define CriticalAlertT(format, ...) MsgAlert(false, CRITICAL, format, ##__VA_ARGS__)
|
|
||||||
#endif
|
|
@ -1,459 +0,0 @@
|
|||||||
/*
|
|
||||||
Basic UTF-8 manipulation routines
|
|
||||||
by Jeff Bezanson
|
|
||||||
placed in the public domain Fall 2005
|
|
||||||
|
|
||||||
This code is designed to provide the utilities you need to manipulate
|
|
||||||
UTF-8 as an internal string encoding. These functions do not perform the
|
|
||||||
error checking normally needed when handling UTF-8 data, so if you happen
|
|
||||||
to be from the Unicode Consortium you will want to flay me alive.
|
|
||||||
I do this because error checking can be performed at the boundaries (I/O),
|
|
||||||
with these routines reserved for higher performance on data known to be
|
|
||||||
valid.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
#include <windows.h>
|
|
||||||
#undef min
|
|
||||||
#undef max
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <cstdlib>
|
|
||||||
#include <cstring>
|
|
||||||
#include <algorithm>
|
|
||||||
|
|
||||||
#include "common/common_types.h"
|
|
||||||
#include "common/utf8.h"
|
|
||||||
|
|
||||||
// is start of UTF sequence
|
|
||||||
inline bool isutf(char c) {
|
|
||||||
return (c & 0xC0) != 0x80;
|
|
||||||
}
|
|
||||||
|
|
||||||
static const u32 offsetsFromUTF8[6] = {
|
|
||||||
0x00000000UL, 0x00003080UL, 0x000E2080UL,
|
|
||||||
0x03C82080UL, 0xFA082080UL, 0x82082080UL
|
|
||||||
};
|
|
||||||
|
|
||||||
static const u8 trailingBytesForUTF8[256] = {
|
|
||||||
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
||||||
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
||||||
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
||||||
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
||||||
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
||||||
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
||||||
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
||||||
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2, 3,3,3,3,3,3,3,3,4,4,4,4,5,5,5,5,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* returns length of next utf-8 sequence */
|
|
||||||
int u8_seqlen(const char *s)
|
|
||||||
{
|
|
||||||
return trailingBytesForUTF8[(unsigned int)(unsigned char)s[0]] + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* conversions without error checking
|
|
||||||
only works for valid UTF-8, i.e. no 5- or 6-byte sequences
|
|
||||||
srcsz = source size in bytes, or -1 if 0-terminated
|
|
||||||
sz = dest size in # of wide characters
|
|
||||||
|
|
||||||
returns # characters converted
|
|
||||||
dest will always be L'\0'-terminated, even if there isn't enough room
|
|
||||||
for all the characters.
|
|
||||||
if sz = srcsz+1 (i.e. 4*srcsz+4 bytes), there will always be enough space.
|
|
||||||
*/
|
|
||||||
int u8_toucs(u32 *dest, int sz, const char *src, int srcsz)
|
|
||||||
{
|
|
||||||
u32 ch;
|
|
||||||
const char *src_end = src + srcsz;
|
|
||||||
int nb;
|
|
||||||
int i=0;
|
|
||||||
|
|
||||||
while (i < sz-1) {
|
|
||||||
nb = trailingBytesForUTF8[(unsigned char)*src];
|
|
||||||
if (srcsz == -1) {
|
|
||||||
if (*src == 0)
|
|
||||||
goto done_toucs;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
if (src + nb >= src_end)
|
|
||||||
goto done_toucs;
|
|
||||||
}
|
|
||||||
ch = 0;
|
|
||||||
switch (nb) {
|
|
||||||
/* these fall through deliberately */
|
|
||||||
case 3: ch += (unsigned char)*src++; ch <<= 6;
|
|
||||||
case 2: ch += (unsigned char)*src++; ch <<= 6;
|
|
||||||
case 1: ch += (unsigned char)*src++; ch <<= 6;
|
|
||||||
case 0: ch += (unsigned char)*src++;
|
|
||||||
}
|
|
||||||
ch -= offsetsFromUTF8[nb];
|
|
||||||
dest[i++] = ch;
|
|
||||||
}
|
|
||||||
done_toucs:
|
|
||||||
dest[i] = 0;
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* srcsz = number of source characters, or -1 if 0-terminated
|
|
||||||
sz = size of dest buffer in bytes
|
|
||||||
|
|
||||||
returns # characters converted
|
|
||||||
dest will only be '\0'-terminated if there is enough space. this is
|
|
||||||
for consistency; imagine there are 2 bytes of space left, but the next
|
|
||||||
character requires 3 bytes. in this case we could NUL-terminate, but in
|
|
||||||
general we can't when there's insufficient space. therefore this function
|
|
||||||
only NUL-terminates if all the characters fit, and there's space for
|
|
||||||
the NUL as well.
|
|
||||||
the destination string will never be bigger than the source string.
|
|
||||||
*/
|
|
||||||
int u8_toutf8(char *dest, int sz, u32 *src, int srcsz)
|
|
||||||
{
|
|
||||||
u32 ch;
|
|
||||||
int i = 0;
|
|
||||||
char *dest_end = dest + sz;
|
|
||||||
|
|
||||||
while (srcsz<0 ? src[i]!=0 : i < srcsz) {
|
|
||||||
ch = src[i];
|
|
||||||
if (ch < 0x80) {
|
|
||||||
if (dest >= dest_end)
|
|
||||||
return i;
|
|
||||||
*dest++ = (char)ch;
|
|
||||||
}
|
|
||||||
else if (ch < 0x800) {
|
|
||||||
if (dest >= dest_end-1)
|
|
||||||
return i;
|
|
||||||
*dest++ = (ch>>6) | 0xC0;
|
|
||||||
*dest++ = (ch & 0x3F) | 0x80;
|
|
||||||
}
|
|
||||||
else if (ch < 0x10000) {
|
|
||||||
if (dest >= dest_end-2)
|
|
||||||
return i;
|
|
||||||
*dest++ = (ch>>12) | 0xE0;
|
|
||||||
*dest++ = ((ch>>6) & 0x3F) | 0x80;
|
|
||||||
*dest++ = (ch & 0x3F) | 0x80;
|
|
||||||
}
|
|
||||||
else if (ch < 0x110000) {
|
|
||||||
if (dest >= dest_end-3)
|
|
||||||
return i;
|
|
||||||
*dest++ = (ch>>18) | 0xF0;
|
|
||||||
*dest++ = ((ch>>12) & 0x3F) | 0x80;
|
|
||||||
*dest++ = ((ch>>6) & 0x3F) | 0x80;
|
|
||||||
*dest++ = (ch & 0x3F) | 0x80;
|
|
||||||
}
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
if (dest < dest_end)
|
|
||||||
*dest = '\0';
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
|
|
||||||
int u8_wc_toutf8(char *dest, u32 ch)
|
|
||||||
{
|
|
||||||
if (ch < 0x80) {
|
|
||||||
dest[0] = (char)ch;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
if (ch < 0x800) {
|
|
||||||
dest[0] = (ch>>6) | 0xC0;
|
|
||||||
dest[1] = (ch & 0x3F) | 0x80;
|
|
||||||
return 2;
|
|
||||||
}
|
|
||||||
if (ch < 0x10000) {
|
|
||||||
dest[0] = (ch>>12) | 0xE0;
|
|
||||||
dest[1] = ((ch>>6) & 0x3F) | 0x80;
|
|
||||||
dest[2] = (ch & 0x3F) | 0x80;
|
|
||||||
return 3;
|
|
||||||
}
|
|
||||||
if (ch < 0x110000) {
|
|
||||||
dest[0] = (ch>>18) | 0xF0;
|
|
||||||
dest[1] = ((ch>>12) & 0x3F) | 0x80;
|
|
||||||
dest[2] = ((ch>>6) & 0x3F) | 0x80;
|
|
||||||
dest[3] = (ch & 0x3F) | 0x80;
|
|
||||||
return 4;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* charnum => byte offset */
|
|
||||||
int u8_offset(const char *str, int charnum)
|
|
||||||
{
|
|
||||||
int offs=0;
|
|
||||||
|
|
||||||
while (charnum > 0 && str[offs]) {
|
|
||||||
(void)(isutf(str[++offs]) || isutf(str[++offs]) ||
|
|
||||||
isutf(str[++offs]) || ++offs);
|
|
||||||
charnum--;
|
|
||||||
}
|
|
||||||
return offs;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* byte offset => charnum */
|
|
||||||
int u8_charnum(const char *s, int offset)
|
|
||||||
{
|
|
||||||
int charnum = 0, offs=0;
|
|
||||||
|
|
||||||
while (offs < offset && s[offs]) {
|
|
||||||
(void)(isutf(s[++offs]) || isutf(s[++offs]) ||
|
|
||||||
isutf(s[++offs]) || ++offs);
|
|
||||||
charnum++;
|
|
||||||
}
|
|
||||||
return charnum;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* number of characters */
|
|
||||||
int u8_strlen(const char *s)
|
|
||||||
{
|
|
||||||
int count = 0;
|
|
||||||
int i = 0;
|
|
||||||
|
|
||||||
while (u8_nextchar(s, &i) != 0)
|
|
||||||
count++;
|
|
||||||
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* reads the next utf-8 sequence out of a string, updating an index */
|
|
||||||
u32 u8_nextchar(const char *s, int *i)
|
|
||||||
{
|
|
||||||
u32 ch = 0;
|
|
||||||
int sz = 0;
|
|
||||||
|
|
||||||
do {
|
|
||||||
ch <<= 6;
|
|
||||||
ch += (unsigned char)s[(*i)++];
|
|
||||||
sz++;
|
|
||||||
} while (s[*i] && !isutf(s[*i]));
|
|
||||||
ch -= offsetsFromUTF8[sz-1];
|
|
||||||
|
|
||||||
return ch;
|
|
||||||
}
|
|
||||||
|
|
||||||
void u8_inc(const char *s, int *i)
|
|
||||||
{
|
|
||||||
(void)(isutf(s[++(*i)]) || isutf(s[++(*i)]) ||
|
|
||||||
isutf(s[++(*i)]) || ++(*i));
|
|
||||||
}
|
|
||||||
|
|
||||||
void u8_dec(const char *s, int *i)
|
|
||||||
{
|
|
||||||
(void)(isutf(s[--(*i)]) || isutf(s[--(*i)]) ||
|
|
||||||
isutf(s[--(*i)]) || --(*i));
|
|
||||||
}
|
|
||||||
|
|
||||||
int octal_digit(char c)
|
|
||||||
{
|
|
||||||
return (c >= '0' && c <= '7');
|
|
||||||
}
|
|
||||||
|
|
||||||
int hex_digit(char c)
|
|
||||||
{
|
|
||||||
return ((c >= '0' && c <= '9') ||
|
|
||||||
(c >= 'A' && c <= 'F') ||
|
|
||||||
(c >= 'a' && c <= 'f'));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* assumes that src points to the character after a backslash
|
|
||||||
returns number of input characters processed */
|
|
||||||
int u8_read_escape_sequence(const char *str, u32 *dest)
|
|
||||||
{
|
|
||||||
u32 ch;
|
|
||||||
char digs[9]="\0\0\0\0\0\0\0\0";
|
|
||||||
int dno=0, i=1;
|
|
||||||
|
|
||||||
ch = (u32)str[0]; /* take literal character */
|
|
||||||
if (str[0] == 'n')
|
|
||||||
ch = L'\n';
|
|
||||||
else if (str[0] == 't')
|
|
||||||
ch = L'\t';
|
|
||||||
else if (str[0] == 'r')
|
|
||||||
ch = L'\r';
|
|
||||||
else if (str[0] == 'b')
|
|
||||||
ch = L'\b';
|
|
||||||
else if (str[0] == 'f')
|
|
||||||
ch = L'\f';
|
|
||||||
else if (str[0] == 'v')
|
|
||||||
ch = L'\v';
|
|
||||||
else if (str[0] == 'a')
|
|
||||||
ch = L'\a';
|
|
||||||
else if (octal_digit(str[0])) {
|
|
||||||
i = 0;
|
|
||||||
do {
|
|
||||||
digs[dno++] = str[i++];
|
|
||||||
} while (octal_digit(str[i]) && dno < 3);
|
|
||||||
ch = strtol(digs, nullptr, 8);
|
|
||||||
}
|
|
||||||
else if (str[0] == 'x') {
|
|
||||||
while (hex_digit(str[i]) && dno < 2) {
|
|
||||||
digs[dno++] = str[i++];
|
|
||||||
}
|
|
||||||
if (dno > 0)
|
|
||||||
ch = strtol(digs, nullptr, 16);
|
|
||||||
}
|
|
||||||
else if (str[0] == 'u') {
|
|
||||||
while (hex_digit(str[i]) && dno < 4) {
|
|
||||||
digs[dno++] = str[i++];
|
|
||||||
}
|
|
||||||
if (dno > 0)
|
|
||||||
ch = strtol(digs, nullptr, 16);
|
|
||||||
}
|
|
||||||
else if (str[0] == 'U') {
|
|
||||||
while (hex_digit(str[i]) && dno < 8) {
|
|
||||||
digs[dno++] = str[i++];
|
|
||||||
}
|
|
||||||
if (dno > 0)
|
|
||||||
ch = strtol(digs, nullptr, 16);
|
|
||||||
}
|
|
||||||
*dest = ch;
|
|
||||||
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* convert a string with literal \uxxxx or \Uxxxxxxxx characters to UTF-8
|
|
||||||
example: u8_unescape(mybuf, 256, "hello\\u220e")
|
|
||||||
note the double backslash is needed if called on a C string literal */
|
|
||||||
int u8_unescape(char *buf, int sz, char *src)
|
|
||||||
{
|
|
||||||
int c=0, amt;
|
|
||||||
u32 ch;
|
|
||||||
char temp[4];
|
|
||||||
|
|
||||||
while (*src && c < sz) {
|
|
||||||
if (*src == '\\') {
|
|
||||||
src++;
|
|
||||||
amt = u8_read_escape_sequence(src, &ch);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
ch = (u32)*src;
|
|
||||||
amt = 1;
|
|
||||||
}
|
|
||||||
src += amt;
|
|
||||||
amt = u8_wc_toutf8(temp, ch);
|
|
||||||
if (amt > sz-c)
|
|
||||||
break;
|
|
||||||
memcpy(&buf[c], temp, amt);
|
|
||||||
c += amt;
|
|
||||||
}
|
|
||||||
if (c < sz)
|
|
||||||
buf[c] = '\0';
|
|
||||||
return c;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *u8_strchr(const char *s, u32 ch, int *charn)
|
|
||||||
{
|
|
||||||
int i = 0, lasti=0;
|
|
||||||
u32 c;
|
|
||||||
|
|
||||||
*charn = 0;
|
|
||||||
while (s[i]) {
|
|
||||||
c = u8_nextchar(s, &i);
|
|
||||||
if (c == ch) {
|
|
||||||
return &s[lasti];
|
|
||||||
}
|
|
||||||
lasti = i;
|
|
||||||
(*charn)++;
|
|
||||||
}
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *u8_memchr(const char *s, u32 ch, size_t sz, int *charn)
|
|
||||||
{
|
|
||||||
u32 i = 0, lasti=0;
|
|
||||||
u32 c;
|
|
||||||
int csz;
|
|
||||||
|
|
||||||
*charn = 0;
|
|
||||||
while (i < sz) {
|
|
||||||
c = csz = 0;
|
|
||||||
do {
|
|
||||||
c <<= 6;
|
|
||||||
c += (unsigned char)s[i++];
|
|
||||||
csz++;
|
|
||||||
} while (i < sz && !isutf(s[i]));
|
|
||||||
c -= offsetsFromUTF8[csz-1];
|
|
||||||
|
|
||||||
if (c == ch) {
|
|
||||||
return &s[lasti];
|
|
||||||
}
|
|
||||||
lasti = i;
|
|
||||||
(*charn)++;
|
|
||||||
}
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
int u8_is_locale_utf8(const char *locale)
|
|
||||||
{
|
|
||||||
/* this code based on libutf8 */
|
|
||||||
const char* cp = locale;
|
|
||||||
|
|
||||||
for (; *cp != '\0' && *cp != '@' && *cp != '+' && *cp != ','; cp++) {
|
|
||||||
if (*cp == '.') {
|
|
||||||
const char* encoding = ++cp;
|
|
||||||
for (; *cp != '\0' && *cp != '@' && *cp != '+' && *cp != ','; cp++)
|
|
||||||
;
|
|
||||||
if ((cp-encoding == 5 && !strncmp(encoding, "UTF-8", 5))
|
|
||||||
|| (cp-encoding == 4 && !strncmp(encoding, "utf8", 4)))
|
|
||||||
return 1; /* it's UTF-8 */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int UTF8StringNonASCIICount(const char *utf8string) {
|
|
||||||
UTF8 utf(utf8string);
|
|
||||||
int count = 0;
|
|
||||||
while (!utf.end()) {
|
|
||||||
int c = utf.next();
|
|
||||||
if (c > 127)
|
|
||||||
++count;
|
|
||||||
}
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool UTF8StringHasNonASCII(const char *utf8string) {
|
|
||||||
return UTF8StringNonASCIICount(utf8string) > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
|
|
||||||
std::string ConvertWStringToUTF8(const wchar_t *wstr) {
|
|
||||||
int len = (int)wcslen(wstr);
|
|
||||||
int size = (int)WideCharToMultiByte(CP_UTF8, 0, wstr, len, 0, 0, nullptr, nullptr);
|
|
||||||
std::string s;
|
|
||||||
s.resize(size);
|
|
||||||
if (size > 0) {
|
|
||||||
WideCharToMultiByte(CP_UTF8, 0, wstr, len, &s[0], size, nullptr, nullptr);
|
|
||||||
}
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string ConvertWStringToUTF8(const std::wstring &wstr) {
|
|
||||||
int len = (int)wstr.size();
|
|
||||||
int size = (int)WideCharToMultiByte(CP_UTF8, 0, wstr.c_str(), len, 0, 0, nullptr, nullptr);
|
|
||||||
std::string s;
|
|
||||||
s.resize(size);
|
|
||||||
if (size > 0) {
|
|
||||||
WideCharToMultiByte(CP_UTF8, 0, wstr.c_str(), len, &s[0], size, nullptr, nullptr);
|
|
||||||
}
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ConvertUTF8ToWString(wchar_t *dest, size_t destSize, const std::string &source) {
|
|
||||||
int len = (int)source.size();
|
|
||||||
int size = (int)MultiByteToWideChar(CP_UTF8, 0, source.c_str(), len, nullptr, 0);
|
|
||||||
MultiByteToWideChar(CP_UTF8, 0, source.c_str(), len, dest, std::min((int)destSize, size));
|
|
||||||
}
|
|
||||||
|
|
||||||
std::wstring ConvertUTF8ToWString(const std::string &source) {
|
|
||||||
int len = (int)source.size();
|
|
||||||
int size = (int)MultiByteToWideChar(CP_UTF8, 0, source.c_str(), len, nullptr, 0);
|
|
||||||
std::wstring str;
|
|
||||||
str.resize(size);
|
|
||||||
if (size > 0) {
|
|
||||||
MultiByteToWideChar(CP_UTF8, 0, source.c_str(), len, &str[0], size);
|
|
||||||
}
|
|
||||||
return str;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,67 +0,0 @@
|
|||||||
/*
|
|
||||||
Basic UTF-8 manipulation routines
|
|
||||||
by Jeff Bezanson
|
|
||||||
placed in the public domain Fall 2005
|
|
||||||
|
|
||||||
This code is designed to provide the utilities you need to manipulate
|
|
||||||
UTF-8 as an internal string encoding. These functions do not perform the
|
|
||||||
error checking normally needed when handling UTF-8 data, so if you happen
|
|
||||||
to be from the Unicode Consortium you will want to flay me alive.
|
|
||||||
I do this because error checking can be performed at the boundaries (I/O),
|
|
||||||
with these routines reserved for higher performance on data known to be
|
|
||||||
valid.
|
|
||||||
*/
|
|
||||||
|
|
||||||
// Further modified, and C++ stuff added, by hrydgard@gmail.com.
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "common/common_types.h"
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
u32 u8_nextchar(const char *s, int *i);
|
|
||||||
int u8_wc_toutf8(char *dest, u32 ch);
|
|
||||||
int u8_strlen(const char *s);
|
|
||||||
|
|
||||||
class UTF8 {
|
|
||||||
public:
|
|
||||||
static const u32 INVALID = (u32)-1;
|
|
||||||
UTF8(const char *c) : c_(c), index_(0) {}
|
|
||||||
bool end() const { return c_[index_] == 0; }
|
|
||||||
u32 next() {
|
|
||||||
return u8_nextchar(c_, &index_);
|
|
||||||
}
|
|
||||||
u32 peek() {
|
|
||||||
int tempIndex = index_;
|
|
||||||
return u8_nextchar(c_, &tempIndex);
|
|
||||||
}
|
|
||||||
int length() const {
|
|
||||||
return u8_strlen(c_);
|
|
||||||
}
|
|
||||||
int byteIndex() const {
|
|
||||||
return index_;
|
|
||||||
}
|
|
||||||
static int encode(char *dest, u32 ch) {
|
|
||||||
return u8_wc_toutf8(dest, ch);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
const char *c_;
|
|
||||||
int index_;
|
|
||||||
};
|
|
||||||
|
|
||||||
int UTF8StringNonASCIICount(const char *utf8string);
|
|
||||||
|
|
||||||
bool UTF8StringHasNonASCII(const char *utf8string);
|
|
||||||
|
|
||||||
|
|
||||||
// UTF8 to Win32 UTF-16
|
|
||||||
// Should be used when calling Win32 api calls
|
|
||||||
#ifdef _WIN32
|
|
||||||
|
|
||||||
std::string ConvertWStringToUTF8(const std::wstring &wstr);
|
|
||||||
std::string ConvertWStringToUTF8(const wchar_t *wstr);
|
|
||||||
void ConvertUTF8ToWString(wchar_t *dest, size_t destSize, const std::string &source);
|
|
||||||
std::wstring ConvertUTF8ToWString(const std::string &source);
|
|
||||||
|
|
||||||
#endif
|
|
Loading…
Reference in New Issue