186 lines
4.7 KiB
C
186 lines
4.7 KiB
C
/***********************************************************************
|
|
* Copyright (c) 2009, Secure Endpoints Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* - Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
*
|
|
* - Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
* COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
|
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
**********************************************************************/
|
|
|
|
#include <config.h>
|
|
#include <windows.h>
|
|
#include <dlfcn.h>
|
|
#include <strsafe.h>
|
|
|
|
#define ERR_STR_LEN 256
|
|
|
|
static volatile LONG dlfcn_tls = TLS_OUT_OF_INDEXES;
|
|
|
|
static DWORD get_tl_error_slot(void)
|
|
{
|
|
if (dlfcn_tls == TLS_OUT_OF_INDEXES) {
|
|
DWORD slot = TlsAlloc();
|
|
DWORD old_slot;
|
|
|
|
if (slot == TLS_OUT_OF_INDEXES)
|
|
return dlfcn_tls;
|
|
|
|
if ((old_slot = InterlockedCompareExchange(&dlfcn_tls, slot,
|
|
TLS_OUT_OF_INDEXES)) !=
|
|
TLS_OUT_OF_INDEXES) {
|
|
|
|
/* Lost a race */
|
|
TlsFree(slot);
|
|
return old_slot;
|
|
} else {
|
|
return slot;
|
|
}
|
|
}
|
|
|
|
return dlfcn_tls;
|
|
}
|
|
|
|
static void set_error(const char * e)
|
|
{
|
|
char * s;
|
|
char * old_s;
|
|
size_t len;
|
|
|
|
DWORD slot = get_tl_error_slot();
|
|
|
|
if (slot == TLS_OUT_OF_INDEXES)
|
|
return;
|
|
|
|
len = strlen(e) * sizeof(char) + sizeof(char);
|
|
s = LocalAlloc(LMEM_FIXED, len);
|
|
if (s == NULL)
|
|
return;
|
|
|
|
old_s = (char *) TlsGetValue(slot);
|
|
TlsSetValue(slot, (LPVOID) s);
|
|
|
|
if (old_s != NULL)
|
|
LocalFree(old_s);
|
|
}
|
|
|
|
static void set_error_from_last(void) {
|
|
DWORD slot = get_tl_error_slot();
|
|
char * s = NULL;
|
|
char * old_s;
|
|
|
|
if (slot == TLS_OUT_OF_INDEXES)
|
|
return;
|
|
|
|
FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER,
|
|
0, GetLastError(), 0,
|
|
(LPTSTR) &s, 0,
|
|
NULL);
|
|
if (s == NULL)
|
|
return;
|
|
|
|
old_s = (char *) TlsGetValue(slot);
|
|
TlsSetValue(slot, (LPVOID) s);
|
|
|
|
if (old_s != NULL)
|
|
LocalFree(old_s);
|
|
}
|
|
|
|
ROKEN_LIB_FUNCTION int ROKEN_LIB_CALL
|
|
dlclose(void * vhm)
|
|
{
|
|
BOOL brv;
|
|
|
|
brv = FreeLibrary((HMODULE) vhm);
|
|
if (!brv) {
|
|
set_error_from_last();
|
|
}
|
|
return !brv;
|
|
}
|
|
|
|
ROKEN_LIB_FUNCTION char * ROKEN_LIB_CALL
|
|
dlerror(void)
|
|
{
|
|
DWORD slot = get_tl_error_slot();
|
|
|
|
if (slot == TLS_OUT_OF_INDEXES)
|
|
return NULL;
|
|
|
|
return (char *) TlsGetValue(slot);
|
|
}
|
|
|
|
ROKEN_LIB_FUNCTION void * ROKEN_LIB_CALL
|
|
dlopen(const char *fn, int flags)
|
|
{
|
|
HMODULE hm;
|
|
UINT old_error_mode;
|
|
|
|
/* We don't support dlopen(0, ...) on Windows.*/
|
|
if ( fn == NULL ) {
|
|
set_error("Not implemented");
|
|
return NULL;
|
|
}
|
|
|
|
old_error_mode = SetErrorMode(SEM_FAILCRITICALERRORS);
|
|
|
|
hm = LoadLibraryEx(fn, 0, LOAD_WITH_ALTERED_SEARCH_PATH);
|
|
|
|
if (hm == NULL) {
|
|
set_error_from_last();
|
|
}
|
|
|
|
SetErrorMode(old_error_mode);
|
|
|
|
return (void *) hm;
|
|
}
|
|
|
|
ROKEN_LIB_FUNCTION DLSYM_RET_TYPE ROKEN_LIB_CALL
|
|
dlsym(void * vhm, const char * func_name)
|
|
{
|
|
HMODULE hm = (HMODULE) vhm;
|
|
|
|
return (DLSYM_RET_TYPE)(ULONG_PTR)GetProcAddress(hm, func_name);
|
|
}
|
|
|
|
ROKEN_LIB_FUNCTION int ROKEN_LIB_CALL
|
|
dladdr(void *addr, Dl_info *dli)
|
|
{
|
|
HMODULE hm;
|
|
DWORD nsize;
|
|
|
|
memset(dli, 0, sizeof(*dli));
|
|
dli->dli_fname = dli->_dli_buf;
|
|
|
|
if (!GetModuleHandleEx(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS |
|
|
GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,
|
|
(LPCTSTR)(ULONG_PTR)addr, &hm))
|
|
return 0;
|
|
|
|
nsize = GetModuleFileName(hm, dli->_dli_buf, sizeof(dli->_dli_buf));
|
|
dli->_dli_buf[sizeof(dli->_dli_buf) - 1] = '\0';
|
|
if (nsize >= sizeof(dli->_dli_buf))
|
|
return 0; /* truncated? can't be... */
|
|
return 1;
|
|
}
|