Mercurial > pidgin.yaz
view pidgin/win32/winpidgin.c @ 29978:8b9359833987
Somebody said something about a ChangeLog
author | Elliott Sales de Andrade <qulogic@pidgin.im> |
---|---|
date | Thu, 11 Mar 2010 02:48:12 +0000 |
parents | b248178bc7b0 |
children | d4ebabbfdeb6 |
line wrap: on
line source
/* * winpidgin.c * * Date: June, 2002 * Description: Entry point for win32 pidgin, and various win32 dependant * routines. * * Pidgin is the legal property of its developers, whose names are too numerous * to list here. Please refer to the COPYRIGHT file distributed with this * source distribution. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA * */ /* This is for ATTACH_PARENT_PROCESS */ #define UNICODE #define _UNICODE #ifndef _WIN32_WINNT #define _WIN32_WINNT 0x501 #endif #include <windows.h> #include <tchar.h> #include <fcntl.h> #include <stdlib.h> #include <string.h> #include <stdio.h> #include <sys/types.h> #include <sys/stat.h> #include "config.h" typedef int (CALLBACK* LPFNPIDGINMAIN)(HINSTANCE, int, char**); typedef void (CALLBACK* LPFNSETDLLDIRECTORY)(LPCWSTR); typedef BOOL (CALLBACK* LPFNATTACHCONSOLE)(DWORD); static BOOL portable_mode = FALSE; /* * PROTOTYPES */ static LPFNPIDGINMAIN pidgin_main = NULL; static LPFNSETDLLDIRECTORY MySetDllDirectory = NULL; static const TCHAR *get_win32_error_message(DWORD err) { static TCHAR err_msg[512]; FormatMessage( FORMAT_MESSAGE_FROM_SYSTEM, NULL, err, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &err_msg, sizeof(err_msg) / sizeof(TCHAR), NULL); return err_msg; } static BOOL read_reg_string(HKEY key, TCHAR *sub_key, TCHAR *val_name, LPBYTE data, LPDWORD data_len) { HKEY hkey; BOOL ret = FALSE; LONG retv; if (ERROR_SUCCESS == (retv = RegOpenKeyEx(key, sub_key, 0, KEY_QUERY_VALUE, &hkey))) { if (ERROR_SUCCESS == (retv = RegQueryValueEx(hkey, val_name, NULL, NULL, data, data_len))) ret = TRUE; else { const TCHAR *err_msg = get_win32_error_message(retv); _tprintf(_T("Could not read reg key '%s' subkey '%s' value: '%s'.\nMessage: (%ld) %s\n"), (key == HKEY_LOCAL_MACHINE) ? _T("HKLM") : ((key == HKEY_CURRENT_USER) ? _T("HKCU") : _T("???")), sub_key, val_name, retv, err_msg); } RegCloseKey(hkey); } else { TCHAR szBuf[80]; FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, retv, 0, (LPTSTR) &szBuf, sizeof(szBuf) / sizeof(TCHAR), NULL); _tprintf(_T("Could not open reg subkey: %s\nError: (%ld) %s\n"), sub_key, retv, szBuf); } return ret; } static BOOL common_dll_prep(const TCHAR *path) { HMODULE hmod; HKEY hkey; struct _stat stat_buf; TCHAR test_path[MAX_PATH + 1]; _sntprintf(test_path, sizeof(test_path) / sizeof(TCHAR), _T("%s\\libgtk-win32-2.0-0.dll"), path); test_path[sizeof(test_path) / sizeof(TCHAR) - 1] = _T('\0'); if (_tstat(test_path, &stat_buf) != 0) { printf("Unable to determine GTK+ path. \n" "Assuming GTK+ is in the PATH.\n"); return FALSE; } _tprintf(_T("GTK+ path found: %s\n"), path); if ((hmod = GetModuleHandle(_T("kernel32.dll")))) { MySetDllDirectory = (LPFNSETDLLDIRECTORY) GetProcAddress( hmod, "SetDllDirectoryW"); if (!MySetDllDirectory) printf("SetDllDirectory not supported\n"); } else printf("Error getting kernel32.dll module handle\n"); /* For Windows XP SP1+ / Server 2003 we use SetDllDirectory to avoid dll hell */ if (MySetDllDirectory) { printf("Using SetDllDirectory\n"); MySetDllDirectory(path); } /* For the rest, we set the current directory and make sure * SafeDllSearch is set to 0 where needed. */ else { OSVERSIONINFO osinfo; printf("Setting current directory to GTK+ dll directory\n"); SetCurrentDirectory(path); /* For Windows 2000 (SP3+) / WinXP (No SP): * If SafeDllSearchMode is set to 1, Windows system directories are * searched for dlls before the current directory. Therefore we set it * to 0. */ osinfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); GetVersionEx(&osinfo); if ((osinfo.dwMajorVersion == 5 && osinfo.dwMinorVersion == 0 && _tcscmp(osinfo.szCSDVersion, _T("Service Pack 3")) >= 0) || (osinfo.dwMajorVersion == 5 && osinfo.dwMinorVersion == 1 && _tcscmp(osinfo.szCSDVersion, _T("")) >= 0) ) { DWORD regval = 1; DWORD reglen = sizeof(DWORD); printf("Using Win2k (SP3+) / WinXP (No SP)... Checking SafeDllSearch\n"); read_reg_string(HKEY_LOCAL_MACHINE, _T("System\\CurrentControlSet\\Control\\Session Manager"), _T("SafeDllSearchMode"), (LPBYTE) ®val, ®len); if (regval != 0) { printf("Trying to set SafeDllSearchMode to 0\n"); regval = 0; if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("System\\CurrentControlSet\\Control\\Session Manager"), 0, KEY_SET_VALUE, &hkey ) == ERROR_SUCCESS) { if (RegSetValueEx(hkey, _T("SafeDllSearchMode"), 0, REG_DWORD, (LPBYTE) ®val, sizeof(DWORD) ) != ERROR_SUCCESS) printf("Error writing SafeDllSearchMode. Error: %u\n", (UINT) GetLastError()); RegCloseKey(hkey); } else printf("Error opening Session Manager key for writing. Error: %u\n", (UINT) GetLastError()); } else printf("SafeDllSearchMode is set to 0\n"); }/*end else*/ } return TRUE; } static BOOL dll_prep(const TCHAR *pidgin_dir) { TCHAR path[MAX_PATH + 1]; path[0] = _T('\0'); if (*pidgin_dir) { _sntprintf(path, sizeof(path) / sizeof(TCHAR), _T("%s\\Gtk\\bin"), pidgin_dir); path[sizeof(path) / sizeof(TCHAR)] = _T('\0'); } return common_dll_prep(path); } static void portable_mode_dll_prep(const TCHAR *pidgin_dir) { /* need to be able to fit MAX_PATH + "PIDGIN_ASPELL_DIR=\\Aspell\\bin" in path2 */ TCHAR path[MAX_PATH + 1]; TCHAR path2[MAX_PATH + 33]; const TCHAR *prev = NULL; /* We assume that GTK+ is installed under \\path\to\Pidgin\..\GTK * First we find \\path\to */ if (*pidgin_dir) /* pidgin_dir points to \\path\to\Pidgin */ prev = _tcsrchr(pidgin_dir, _T('\\')); if (prev) { int cnt = (prev - pidgin_dir); _tcsncpy(path, pidgin_dir, cnt); path[cnt] = _T('\0'); } else { printf("Unable to determine current executable path. \n" "This will prevent the settings dir from being set.\n" "Assuming GTK+ is in the PATH.\n"); return; } /* Set $HOME so that the GTK+ settings get stored in the right place */ _sntprintf(path2, sizeof(path2) / sizeof(TCHAR), _T("HOME=%s"), path); _tputenv(path2); /* Set up the settings dir base to be \\path\to * The actual settings dir will be \\path\to\.purple */ _sntprintf(path2, sizeof(path2) / sizeof(TCHAR), _T("PURPLEHOME=%s"), path); _tprintf(_T("Setting settings dir: %s\n"), path2); _tputenv(path2); _sntprintf(path2, sizeof(path2) / sizeof(TCHAR), _T("PIDGIN_ASPELL_DIR=%s\\Aspell\\bin"), path); _tprintf(_T("%s\n"), path2); _tputenv(path2); if (!dll_prep(pidgin_dir)) { /* set the GTK+ path to be \\path\to\GTK\bin */ _tcscat(path, _T("\\GTK\\bin")); common_dll_prep(path); } } static TCHAR* winpidgin_lcid_to_posix(LCID lcid) { TCHAR *posix = NULL; int lang_id = PRIMARYLANGID(lcid); int sub_id = SUBLANGID(lcid); switch (lang_id) { case LANG_AFRIKAANS: posix = _T("af"); break; case LANG_ARABIC: posix = _T("ar"); break; case LANG_AZERI: posix = _T("az"); break; case LANG_BENGALI: posix = _T("bn"); break; case LANG_BULGARIAN: posix = _T("bg"); break; case LANG_CATALAN: posix = _T("ca"); break; case LANG_CZECH: posix = _T("cs"); break; case LANG_DANISH: posix = _T("da"); break; case LANG_ESTONIAN: posix = _T("et"); break; case LANG_PERSIAN: posix = _T("fa"); break; case LANG_GERMAN: posix = _T("de"); break; case LANG_GREEK: posix = _T("el"); break; case LANG_ENGLISH: switch (sub_id) { case SUBLANG_ENGLISH_UK: posix = _T("en_GB"); break; case SUBLANG_ENGLISH_AUS: posix = _T("en_AU"); break; case SUBLANG_ENGLISH_CAN: posix = _T("en_CA"); break; default: posix = _T("en"); break; } break; case LANG_SPANISH: posix = _T("es"); break; case LANG_BASQUE: posix = _T("eu"); break; case LANG_FINNISH: posix = _T("fi"); break; case LANG_FRENCH: posix = _T("fr"); break; case LANG_GALICIAN: posix = _T("gl"); break; case LANG_GUJARATI: posix = _T("gu"); break; case LANG_HEBREW: posix = _T("he"); break; case LANG_HINDI: posix = _T("hi"); break; case LANG_HUNGARIAN: posix = _T("hu"); break; case LANG_ICELANDIC: break; case LANG_INDONESIAN: posix = _T("id"); break; case LANG_ITALIAN: posix = _T("it"); break; case LANG_JAPANESE: posix = _T("ja"); break; case LANG_GEORGIAN: posix = _T("ka"); break; case LANG_KANNADA: posix = _T("kn"); break; case LANG_KOREAN: posix = _T("ko"); break; case LANG_LITHUANIAN: posix = _T("lt"); break; case LANG_MACEDONIAN: posix = _T("mk"); break; case LANG_DUTCH: posix = _T("nl"); break; case LANG_NEPALI: posix = _T("ne"); break; case LANG_NORWEGIAN: switch (sub_id) { case SUBLANG_NORWEGIAN_BOKMAL: posix = _T("nb"); break; case SUBLANG_NORWEGIAN_NYNORSK: posix = _T("nn"); break; } break; case LANG_PUNJABI: posix = _T("pa"); break; case LANG_POLISH: posix = _T("pl"); break; case LANG_PASHTO: posix = _T("ps"); break; case LANG_PORTUGUESE: switch (sub_id) { case SUBLANG_PORTUGUESE_BRAZILIAN: posix = _T("pt_BR"); break; default: posix = _T("pt"); break; } break; case LANG_ROMANIAN: posix = _T("ro"); break; case LANG_RUSSIAN: posix = _T("ru"); break; case LANG_SLOVAK: posix = _T("sk"); break; case LANG_SLOVENIAN: posix = _T("sl"); break; case LANG_ALBANIAN: posix = _T("sq"); break; /* LANG_CROATIAN == LANG_SERBIAN == LANG_BOSNIAN */ case LANG_SERBIAN: switch (sub_id) { case SUBLANG_SERBIAN_LATIN: posix = _T("sr@Latn"); break; case SUBLANG_SERBIAN_CYRILLIC: posix = _T("sr"); break; case SUBLANG_BOSNIAN_BOSNIA_HERZEGOVINA_CYRILLIC: case SUBLANG_BOSNIAN_BOSNIA_HERZEGOVINA_LATIN: posix = _T("bs"); break; case SUBLANG_CROATIAN_BOSNIA_HERZEGOVINA_LATIN: posix = _T("hr"); break; } break; case LANG_SWEDISH: posix = _T("sv"); break; case LANG_TAMIL: posix = _T("ta"); break; case LANG_TELUGU: posix = _T("te"); break; case LANG_THAI: posix = _T("th"); break; case LANG_TURKISH: posix = _T("tr"); break; case LANG_UKRAINIAN: posix = _T("uk"); break; case LANG_VIETNAMESE: posix = _T("vi"); break; case LANG_XHOSA: posix = _T("xh"); break; case LANG_CHINESE: switch (sub_id) { case SUBLANG_CHINESE_SIMPLIFIED: posix = _T("zh_CN"); break; case SUBLANG_CHINESE_TRADITIONAL: posix = _T("zh_TW"); break; default: posix = _T("zh"); break; } break; case LANG_URDU: break; case LANG_BELARUSIAN: break; case LANG_LATVIAN: break; case LANG_ARMENIAN: break; case LANG_FAEROESE: break; case LANG_MALAY: break; case LANG_KAZAK: break; case LANG_KYRGYZ: break; case LANG_SWAHILI: break; case LANG_UZBEK: break; case LANG_TATAR: break; case LANG_ORIYA: break; case LANG_MALAYALAM: break; case LANG_ASSAMESE: break; case LANG_MARATHI: break; case LANG_SANSKRIT: break; case LANG_MONGOLIAN: break; case LANG_KONKANI: break; case LANG_MANIPURI: break; case LANG_SINDHI: break; case LANG_SYRIAC: break; case LANG_KASHMIRI: break; case LANG_DIVEHI: break; } /* Deal with exceptions */ if (posix == NULL) { switch (lcid) { case 0x0455: posix = _T("my_MM"); break; /* Myanmar (Burmese) */ case 9999: posix = _T("ku"); break; /* Kurdish (from NSIS) */ } } return posix; } /* Determine and set Pidgin locale as follows (in order of priority): - Check PIDGINLANG env var - Check NSIS Installer Language reg value - Use default user locale */ static const TCHAR *winpidgin_get_locale() { const TCHAR *locale = NULL; LCID lcid; TCHAR data[10]; DWORD datalen = sizeof(data) / sizeof(TCHAR); /* Check if user set PIDGINLANG env var */ if ((locale = _tgetenv(_T("PIDGINLANG")))) return locale; if (!portable_mode && read_reg_string(HKEY_CURRENT_USER, _T("SOFTWARE\\pidgin"), _T("Installer Language"), (LPBYTE) &data, &datalen)) { if ((locale = winpidgin_lcid_to_posix(_ttoi(data)))) return locale; } lcid = GetUserDefaultLCID(); if ((locale = winpidgin_lcid_to_posix(lcid))) return locale; return _T("en"); } static void winpidgin_set_locale() { const TCHAR *locale; TCHAR envstr[25]; locale = winpidgin_get_locale(); _sntprintf(envstr, sizeof(envstr) / sizeof(TCHAR), _T("LANG=%s"), locale); _tprintf(_T("Setting locale: %s\n"), envstr); _tputenv(envstr); } static void winpidgin_add_stuff_to_path() { TCHAR perl_path[MAX_PATH + 1]; TCHAR *ppath = NULL; TCHAR mit_kerberos_path[MAX_PATH + 1]; TCHAR *mpath = NULL; DWORD plen; printf("%s", "Looking for Perl... "); plen = sizeof(perl_path) / sizeof(TCHAR); if (read_reg_string(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Perl"), _T(""), (LPBYTE) &perl_path, &plen)) { /* We *could* check for perl510.dll, but it seems unnecessary. */ _tprintf(_T("found in '%s'.\n"), perl_path); if (perl_path[_tcslen(perl_path) - 1] != _T('\\')) _tcscat(perl_path, _T("\\")); _tcscat(perl_path, _T("bin")); ppath = perl_path; } else printf("%s", "not found.\n"); printf("%s", "Looking for MIT Kerberos... "); plen = sizeof(mit_kerberos_path) / sizeof(TCHAR); if (read_reg_string(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\MIT\\Kerberos"), _T("InstallDir"), (LPBYTE) &mit_kerberos_path, &plen)) { /* We *could* check for gssapi32.dll */ _tprintf(_T("found in '%s'.\n"), mit_kerberos_path); if (mit_kerberos_path[_tcslen(mit_kerberos_path) - 1] != _T('\\')) _tcscat(mit_kerberos_path, _T("\\")); _tcscat(mit_kerberos_path, _T("bin")); mpath = mit_kerberos_path; } else printf("%s", "not found.\n"); if (ppath != NULL || mpath != NULL) { const TCHAR *path = _tgetenv(_T("PATH")); BOOL add_ppath = ppath != NULL && (path == NULL || !_tcsstr(path, ppath)); BOOL add_mpath = mpath != NULL && (path == NULL || !_tcsstr(path, mpath)); TCHAR *newpath; int newlen; if (add_ppath || add_mpath) { /* Enough to add "PATH=" + path + ";" + ppath + ";" + mpath + \0 */ newlen = 6 + (path ? _tcslen(path) + 1 : 0); if (add_ppath) newlen += _tcslen(ppath) + 1; if (add_mpath) newlen += _tcslen(mpath) + 1; newpath = malloc(newlen * sizeof(TCHAR)); _sntprintf(newpath, newlen, _T("PATH=%s%s%s%s%s%s"), path ? path : _T(""), path ? _T(";") : _T(""), add_ppath ? ppath : _T(""), add_ppath ? _T(";") : _T(""), add_mpath ? mpath : _T(""), add_mpath ? _T(";") : _T("")); _tprintf(_T("New PATH: %s\n"), newpath); _tputenv(newpath); free(newpath); } } } #define PIDGIN_WM_FOCUS_REQUEST (WM_APP + 13) #define PIDGIN_WM_PROTOCOL_HANDLE (WM_APP + 14) static BOOL winpidgin_set_running(BOOL fail_if_running) { HANDLE h; if ((h = CreateMutex(NULL, FALSE, _T("pidgin_is_running")))) { DWORD err = GetLastError(); if (err == ERROR_ALREADY_EXISTS) { if (fail_if_running) { HWND msg_win; printf("An instance of Pidgin is already running.\n"); if((msg_win = FindWindowEx(NULL, NULL, _T("WinpidginMsgWinCls"), NULL))) if(SendMessage(msg_win, PIDGIN_WM_FOCUS_REQUEST, (WPARAM) NULL, (LPARAM) NULL)) return FALSE; /* If we get here, the focus request wasn't successful */ MessageBox(NULL, _T("An instance of Pidgin is already running"), NULL, MB_OK | MB_TOPMOST); return FALSE; } } else if (err != ERROR_SUCCESS) printf("Error (%u) accessing \"pidgin_is_running\" mutex.\n", (UINT) err); } return TRUE; } #define PROTO_HANDLER_SWITCH L"--protocolhandler=" static void handle_protocol(wchar_t *cmd) { char *remote_msg, *utf8msg; wchar_t *tmp1, *tmp2; int len, wlen; SIZE_T len_written; HWND msg_win; DWORD pid; HANDLE process; /* The start of the message */ tmp1 = cmd + wcslen(PROTO_HANDLER_SWITCH); /* The end of the message */ if ((tmp2 = wcschr(tmp1, L' '))) wlen = (tmp2 - tmp1); else wlen = wcslen(tmp1); if (wlen == 0) { printf("No protocol message specified.\n"); return; } if (!(msg_win = FindWindowEx(NULL, NULL, _T("WinpidginMsgWinCls"), NULL))) { printf("Unable to find an instance of Pidgin to handle protocol message.\n"); return; } len = WideCharToMultiByte(CP_UTF8, 0, tmp1, wlen, NULL, 0, NULL, NULL); if (len) { utf8msg = malloc(len * sizeof(char)); len = WideCharToMultiByte(CP_UTF8, 0, tmp1, wlen, utf8msg, len, NULL, NULL); } if (len == 0) { printf("No protocol message specified.\n"); return; } GetWindowThreadProcessId(msg_win, &pid); if (!(process = OpenProcess(PROCESS_VM_OPERATION | PROCESS_VM_WRITE, FALSE, pid))) { DWORD dw = GetLastError(); const TCHAR *err_msg = get_win32_error_message(dw); _tprintf(_T("Unable to open Pidgin process. (%u) %s\n"), (UINT) dw, err_msg); return; } wprintf(L"Trying to handle protocol message:\n'%.*s'\n", wlen, tmp1); /* MEM_COMMIT initializes the memory to zero * so we don't need to worry that our section of utf8msg isn't nul-terminated */ if ((remote_msg = (char*) VirtualAllocEx(process, NULL, len + 1, MEM_COMMIT, PAGE_READWRITE))) { if (WriteProcessMemory(process, remote_msg, utf8msg, len, &len_written)) { if (!SendMessageA(msg_win, PIDGIN_WM_PROTOCOL_HANDLE, len_written, (LPARAM) remote_msg)) printf("Unable to send protocol message to Pidgin instance.\n"); } else { DWORD dw = GetLastError(); const TCHAR *err_msg = get_win32_error_message(dw); _tprintf(_T("Unable to write to remote memory. (%u) %s\n"), (UINT) dw, err_msg); } VirtualFreeEx(process, remote_msg, 0, MEM_RELEASE); } else { DWORD dw = GetLastError(); const TCHAR *err_msg = get_win32_error_message(dw); _tprintf(_T("Unable to allocate remote memory. (%u) %s\n"), (UINT) dw, err_msg); } CloseHandle(process); free(utf8msg); } int _stdcall WinMain (struct HINSTANCE__ *hInstance, struct HINSTANCE__ *hPrevInstance, char *lpszCmdLine, int nCmdShow) { TCHAR errbuf[512]; TCHAR pidgin_dir[MAX_PATH]; TCHAR exe_name[MAX_PATH]; HMODULE hmod; TCHAR *tmp; wchar_t *wtmp; int pidgin_argc; char **pidgin_argv; /* This is in utf-8 */ int i, j, k; BOOL debug = FALSE, help = FALSE, version = FALSE, multiple = FALSE, success; LPWSTR *szArglist; LPWSTR cmdLine; /* If debug or help or version flag used, create console for output */ for (i = 1; i < __argc; i++) { if (strlen(__argv[i]) > 1 && __argv[i][0] == '-') { /* check if we're looking at -- or - option */ if (__argv[i][1] == '-') { if (strstr(__argv[i], "--debug") == __argv[i]) debug = TRUE; else if (strstr(__argv[i], "--help") == __argv[i]) help = TRUE; else if (strstr(__argv[i], "--version") == __argv[i]) version = TRUE; else if (strstr(__argv[i], "--multiple") == __argv[i]) multiple = TRUE; } else { if (strchr(__argv[i], 'd')) debug = TRUE; else if (strchr(__argv[i], 'h')) help = TRUE; else if (strchr(__argv[i], 'v')) version = TRUE; else if (strchr(__argv[i], 'm')) multiple = TRUE; } } } if (debug || help || version) { /* If stdout hasn't been redirected to a file, alloc a console * (_istty() doesn't work for stuff using the GUI subsystem) */ if (_fileno(stdout) == -1 || _fileno(stdout) == -2) { LPFNATTACHCONSOLE MyAttachConsole = NULL; if ((hmod = GetModuleHandle(_T("kernel32.dll")))) { MyAttachConsole = (LPFNATTACHCONSOLE) GetProcAddress(hmod, "AttachConsole"); } if ((MyAttachConsole && MyAttachConsole(ATTACH_PARENT_PROCESS)) || AllocConsole()) { freopen("CONOUT$", "w", stdout); freopen("CONOUT$", "w", stderr); } } } cmdLine = GetCommandLineW(); /* If this is a protocol handler invocation, deal with it accordingly */ if ((wtmp = wcsstr(cmdLine, PROTO_HANDLER_SWITCH)) != NULL) { handle_protocol(wtmp); return 0; } /* Load exception handler if we have it */ if (GetModuleFileName(NULL, pidgin_dir, MAX_PATH) != 0) { /* primitive dirname() */ tmp = _tcsrchr(pidgin_dir, _T('\\')); if (tmp) { HMODULE hmod; tmp[0] = _T('\0'); /* tmp++ will now point to the executable file name */ _tcscpy(exe_name, tmp + 1); _tcscat(pidgin_dir, _T("\\exchndl.dll")); if ((hmod = LoadLibrary(pidgin_dir))) { FARPROC proc; /* exchndl.dll is built without UNICODE */ char debug_dir[MAX_PATH]; printf("Loaded exchndl.dll\n"); /* Temporarily override exchndl.dll's logfile * to something sane (Pidgin will override it * again when it initializes) */ proc = GetProcAddress(hmod, "SetLogFile"); if (proc) { if (GetTempPathA(sizeof(debug_dir) * sizeof(char), debug_dir) != 0) { strcat(debug_dir, "pidgin.RPT"); printf(" Setting exchndl.dll LogFile to %s\n", debug_dir); (proc)(debug_dir); } } proc = GetProcAddress(hmod, "SetDebugInfoDir"); if (proc) { char *pidgin_dir_ansi = NULL; tmp[0] = _T('\0'); #ifdef _UNICODE i = WideCharToMultiByte(CP_ACP, 0, pidgin_dir, -1, NULL, 0, NULL, NULL); if (i != 0) { pidgin_dir_ansi = malloc(i * sizeof(char)); i = WideCharToMultiByte(CP_ACP, 0, pidgin_dir, -1, pidgin_dir_ansi, i, NULL, NULL); if (i == 0) { free(pidgin_dir_ansi); pidgin_dir_ansi = NULL; } } #else pidgin_dir_ansi = pidgin_dir; #endif if (pidgin_dir_ansi != NULL) { _snprintf(debug_dir, sizeof(debug_dir) / sizeof(char), "%s\\pidgin-%s-dbgsym", pidgin_dir_ansi, VERSION); debug_dir[sizeof(debug_dir) / sizeof(char) - 1] = '\0'; printf(" Setting exchndl.dll DebugInfoDir to %s\n", debug_dir); (proc)(debug_dir); #ifdef _UNICODE free(pidgin_dir_ansi); #endif } } } tmp[0] = _T('\0'); } } else { DWORD dw = GetLastError(); const TCHAR *err_msg = get_win32_error_message(dw); _sntprintf(errbuf, 512, _T("Error getting module filename.\nError: (%u) %s"), (UINT) dw, err_msg); _tprintf(_T("%s\n"), errbuf); MessageBox(NULL, errbuf, NULL, MB_OK | MB_TOPMOST); pidgin_dir[0] = _T('\0'); } /* Determine if we're running in portable mode */ if (wcsstr(cmdLine, L"--portable-mode") || (exe_name != NULL && _tcsstr(exe_name, _T("-portable.exe")))) { printf("Running in PORTABLE mode.\n"); portable_mode = TRUE; } if (portable_mode) portable_mode_dll_prep(pidgin_dir); else if (!getenv("PIDGIN_NO_DLL_CHECK")) dll_prep(pidgin_dir); winpidgin_set_locale(); winpidgin_add_stuff_to_path(); /* If help, version or multiple flag used, do not check Mutex */ if (!help && !version) if (!winpidgin_set_running(getenv("PIDGIN_MULTI_INST") == NULL && !multiple)) return 0; /* Now we are ready for Pidgin .. */ if ((hmod = LoadLibrary(_T("pidgin.dll")))) pidgin_main = (LPFNPIDGINMAIN) GetProcAddress(hmod, "pidgin_main"); if (!pidgin_main) { DWORD dw = GetLastError(); BOOL mod_not_found = (dw == ERROR_MOD_NOT_FOUND || dw == ERROR_DLL_NOT_FOUND); const TCHAR *err_msg = get_win32_error_message(dw); _sntprintf(errbuf, 512, _T("Error loading pidgin.dll.\nError: (%u) %s%s%s"), (UINT) dw, err_msg, mod_not_found ? _T("\n") : _T(""), mod_not_found ? _T("This probably means that GTK+ can't be found.") : _T("")); _tprintf(_T("%s\n"), errbuf); MessageBox(NULL, errbuf, _T("Error"), MB_OK | MB_TOPMOST); return 0; } /* Convert argv to utf-8*/ szArglist = CommandLineToArgvW(cmdLine, &j); pidgin_argc = j; pidgin_argv = malloc(pidgin_argc* sizeof(char*)); k = 0; for (i = 0; i < j; i++) { success = FALSE; /* Remove the --portable-mode arg from the args passed to pidgin so it doesn't choke */ if (wcsstr(szArglist[i], L"--portable-mode") == NULL) { int len = WideCharToMultiByte(CP_UTF8, 0, szArglist[i], -1, NULL, 0, NULL, NULL); if (len != 0) { char *arg = malloc(len * sizeof(char)); len = WideCharToMultiByte(CP_UTF8, 0, szArglist[i], -1, arg, len, NULL, NULL); if (len != 0) { pidgin_argv[k++] = arg; success = TRUE; } } if (!success) wprintf(L"Error converting argument '%s' to UTF-8\n", szArglist[i]); } if (!success) pidgin_argc--; } LocalFree(szArglist); return pidgin_main(hInstance, pidgin_argc, pidgin_argv); }