/* * Copyright 2010 Piotr Caban for CodeWeavers * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include "wine/test.h" /* basic_string, allocator> */ #define BUF_SIZE_CHAR 16 typedef struct _basic_string_char { void *allocator; union _data { char buf[BUF_SIZE_CHAR]; char *ptr; } data; size_t size; size_t res; } basic_string_char; static void* (__cdecl *p_set_invalid_parameter_handler)(void*); #ifdef __i386__ static basic_string_char* (WINAPI *p_basic_string_char_ctor)(void); static basic_string_char* (WINAPI *p_basic_string_char_copy_ctor)(basic_string_char*); static basic_string_char* (WINAPI *p_basic_string_char_ctor_cstr)(const char*); static void (WINAPI *p_basic_string_char_dtor)(void); static basic_string_char* (WINAPI *p_basic_string_char_erase)(size_t, size_t); static basic_string_char* (WINAPI *p_basic_string_char_assign_cstr_len)(const char*, size_t); static const char* (WINAPI *p_basic_string_char_cstr)(void); #else static basic_string_char* (__cdecl *p_basic_string_char_ctor)(basic_string_char*); static basic_string_char* (__cdecl *p_basic_string_char_copy_ctor)(basic_string_char*, basic_string_char*); static basic_string_char* (__cdecl *p_basic_string_char_ctor_cstr)(basic_string_char*, const char*); static void (__cdecl *p_basic_string_char_dtor)(basic_string_char*); static basic_string_char* (__cdecl *p_basic_string_char_erase)(basic_string_char*, size_t, size_t); static basic_string_char* (__cdecl *p_basic_string_char_assign_cstr_len)(basic_string_char*, const char*, size_t); static const char* (__cdecl *p_basic_string_char_cstr)(basic_string_char*); #endif static int invalid_parameter = 0; static void __cdecl test_invalid_parameter_handler(const wchar_t *expression, const wchar_t *function, const wchar_t *file, unsigned line, uintptr_t arg) { ok(expression == NULL, "expression is not NULL\n"); ok(function == NULL, "function is not NULL\n"); ok(file == NULL, "file is not NULL\n"); ok(line == 0, "line = %u\n", line); ok(arg == 0, "arg = %lx\n", (UINT_PTR)arg); invalid_parameter++; } /* Emulate a __thiscall */ #ifdef __i386__ #ifdef _MSC_VER static inline void* do_call_func1(void *func, void *_this) { volatile void* retval = 0; __asm { push ecx mov ecx, _this call func mov retval, eax pop ecx } return (void*)retval; } static inline void* do_call_func2(void *func, void *_this, const void *arg) { volatile void* retval = 0; __asm { push ecx push arg mov ecx, _this call func mov retval, eax pop ecx } return (void*)retval; } static inline void* do_call_func3(void *func, void *_this, const void *arg1, const void *arg2) { volatile void* retval = 0; __asm { push ecx push arg1 push arg2 mov ecx, _this call func mov retval, eax pop ecx } return (void*)retval; } #else static void* do_call_func1(void *func, void *_this) { void *ret, *dummy; __asm__ __volatile__ ( "call *%2" : "=a" (ret), "=c" (dummy) : "g" (func), "1" (_this) : "edx", "memory" ); return ret; } static void* do_call_func2(void *func, void *_this, const void *arg) { void *ret, *dummy; __asm__ __volatile__ ( "pushl %3\n\tcall *%2" : "=a" (ret), "=c" (dummy) : "r" (func), "r" (arg), "1" (_this) : "edx", "memory" ); return ret; } static void* do_call_func3(void *func, void *_this, const void *arg1, const void *arg2) { void *ret, *dummy; __asm__ __volatile__ ( "pushl %4\n\tpushl %3\n\tcall *%2" : "=a" (ret), "=c" (dummy) : "r" (func), "r" (arg1), "r" (arg2), "1" (_this) : "edx", "memory" ); return ret; } #endif #define call_func1(func,_this) do_call_func1(func,_this) #define call_func2(func,_this,a) do_call_func2(func,_this,(const void*)a) #define call_func3(func,_this,a,b) do_call_func3(func,_this,(const void*)a,(const void*)b) #else #define call_func1(func,_this) func(_this) #define call_func2(func,_this,a) func(_this,a) #define call_func3(func,_this,a,b) func(_this,a,b) #endif /* __i386__ */ static BOOL init(void) { HMODULE msvcr = LoadLibraryA("msvcr90.dll"); HMODULE msvcp = LoadLibraryA("msvcp90.dll"); if(!msvcr || !msvcp) { win_skip("msvcp90.dll or msvcrt90.dll not installed\n"); return FALSE; } p_set_invalid_parameter_handler = (void*)GetProcAddress(msvcr, "_set_invalid_parameter_handler"); if(!p_set_invalid_parameter_handler) { win_skip("Error setting tests environment\n"); return FALSE; } p_set_invalid_parameter_handler(test_invalid_parameter_handler); if(sizeof(void*) == 8) { /* 64-bit initialization */ p_basic_string_char_ctor = (void*)GetProcAddress(msvcp, "??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@XZ"); p_basic_string_char_copy_ctor = (void*)GetProcAddress(msvcp, "??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@AEBV01@@Z"); p_basic_string_char_ctor_cstr = (void*)GetProcAddress(msvcp, "??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@PEBD@Z"); p_basic_string_char_dtor = (void*)GetProcAddress(msvcp, "??1?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@XZ"); p_basic_string_char_erase = (void*)GetProcAddress(msvcp, "?erase@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@_K0@Z"); p_basic_string_char_assign_cstr_len = (void*)GetProcAddress(msvcp, "?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@PEBD_K@Z"); p_basic_string_char_cstr = (void*)GetProcAddress(msvcp, "?c_str@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAPEBDXZ"); } else { p_basic_string_char_ctor = (void*)GetProcAddress(msvcp, "??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@XZ"); p_basic_string_char_copy_ctor = (void*)GetProcAddress(msvcp, "??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@ABV01@@Z"); p_basic_string_char_ctor_cstr = (void*)GetProcAddress(msvcp, "??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@PBD@Z"); p_basic_string_char_dtor = (void*)GetProcAddress(msvcp, "??1?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@XZ"); p_basic_string_char_erase = (void*)GetProcAddress(msvcp, "?erase@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@II@Z"); p_basic_string_char_assign_cstr_len = (void*)GetProcAddress(msvcp, "?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@PBDI@Z"); p_basic_string_char_cstr = (void*)GetProcAddress(msvcp, "?c_str@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEPBDXZ"); } return TRUE; } static void test_basic_string_char(void) { basic_string_char str1, str2, *pstr; const char *str; if(!p_basic_string_char_ctor || !p_basic_string_char_copy_ctor || !p_basic_string_char_ctor_cstr || !p_basic_string_char_dtor || !p_basic_string_char_erase || !p_basic_string_char_assign_cstr_len || !p_basic_string_char_cstr) { win_skip("basic_string unavailable\n"); return; } call_func1(p_basic_string_char_ctor, &str1); str = NULL; str = call_func1(p_basic_string_char_cstr, &str1); ok(str != NULL, "str = NULL\n"); ok(*str == '\0', "*str = %c\n", *str); call_func1(p_basic_string_char_dtor, &str1); pstr = call_func2(p_basic_string_char_ctor_cstr, &str1, "test"); ok(pstr == &str1, "pstr != &str1\n"); str = call_func1(p_basic_string_char_cstr, &str1); ok(!memcmp(str, "test", 5), "str = %s\n", str); pstr = call_func2(p_basic_string_char_copy_ctor, &str2, &str1); ok(pstr == &str2, "pstr != &str2\n"); str = call_func1(p_basic_string_char_cstr, &str2); ok(!memcmp(str, "test", 5), "str = %s\n", str); call_func3(p_basic_string_char_erase, &str2, 1, 2); str = call_func1(p_basic_string_char_cstr, &str2); ok(!memcmp(str, "tt", 3), "str = %s\n", str); call_func3(p_basic_string_char_erase, &str2, 1, 100); str = call_func1(p_basic_string_char_cstr, &str2); ok(!memcmp(str, "t", 2), "str = %s\n", str); call_func3(p_basic_string_char_assign_cstr_len, &str2, "test", 4); str = call_func1(p_basic_string_char_cstr, &str2); ok(!memcmp(str, "test", 5), "str = %s\n", str); call_func3(p_basic_string_char_assign_cstr_len, &str2, (str+1), 2); str = call_func1(p_basic_string_char_cstr, &str2); ok(!memcmp(str, "es", 3), "str = %s\n", str); call_func1(p_basic_string_char_dtor, &str1); call_func1(p_basic_string_char_dtor, &str2); } START_TEST(string) { if(!init()) return; test_basic_string_char(); ok(!invalid_parameter, "invalid_parameter_handler was invoked too many times\n"); }