| // Copyright (c) 2007, Google 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. |
| // * Neither the name of Google Inc. nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // 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 |
| // OWNER 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. |
| |
| // --- |
| // Author: Craig Silverstein |
| // |
| // A few routines that are useful for multiple tests in this directory. |
| |
| #include "config_for_unittests.h" |
| #include <stdlib.h> // for NULL, abort() |
| // On FreeBSD, if you #include <sys/resource.h>, you have to get stdint first. |
| #ifdef HAVE_STDINT_H |
| #include <stdint.h> |
| #endif |
| #ifdef HAVE_SYS_RESOURCE_H |
| #include <sys/resource.h> |
| #endif |
| #include "tests/testutil.h" |
| |
| |
| // When compiled 64-bit and run on systems with swap several unittests will end |
| // up trying to consume all of RAM+swap, and that can take quite some time. By |
| // limiting the address-space size we get sufficient coverage without blowing |
| // out job limits. |
| void SetTestResourceLimit() { |
| #ifdef HAVE_SYS_RESOURCE_H |
| // The actual resource we need to set varies depending on which flavour of |
| // unix. On Linux we need RLIMIT_AS because that covers the use of mmap. |
| // Otherwise hopefully RLIMIT_RSS is good enough. (Unfortunately 64-bit |
| // and 32-bit headers disagree on the type of these constants!) |
| #ifdef RLIMIT_AS |
| #define USE_RESOURCE RLIMIT_AS |
| #else |
| #define USE_RESOURCE RLIMIT_RSS |
| #endif |
| |
| // Restrict the test to 1GiB, which should fit comfortably well on both |
| // 32-bit and 64-bit hosts, and executes in ~1s. |
| const rlim_t kMaxMem = 1<<30; |
| |
| struct rlimit rlim; |
| if (getrlimit(USE_RESOURCE, &rlim) == 0) { |
| if (rlim.rlim_cur == RLIM_INFINITY || rlim.rlim_cur > kMaxMem) { |
| rlim.rlim_cur = kMaxMem; |
| setrlimit(USE_RESOURCE, &rlim); // ignore result |
| } |
| } |
| #endif /* HAVE_SYS_RESOURCE_H */ |
| } |
| |
| |
| struct FunctionAndId { |
| void (*ptr_to_function)(int); |
| int id; |
| }; |
| |
| #if defined(NO_THREADS) || !(defined(HAVE_PTHREAD) || defined(_WIN32)) |
| |
| extern "C" void RunThread(void (*fn)()) { |
| (*fn)(); |
| } |
| |
| extern "C" void RunManyThreads(void (*fn)(), int count) { |
| // I guess the best we can do is run fn sequentially, 'count' times |
| for (int i = 0; i < count; i++) |
| (*fn)(); |
| } |
| |
| extern "C" void RunManyThreadsWithId(void (*fn)(int), int count, int) { |
| for (int i = 0; i < count; i++) |
| (*fn)(i); // stacksize doesn't make sense in a non-threaded context |
| } |
| |
| #elif defined(_WIN32) |
| |
| #ifndef WIN32_LEAN_AND_MEAN |
| #define WIN32_LEAN_AND_MEAN /* We always want minimal includes */ |
| #endif |
| #include <windows.h> |
| |
| extern "C" { |
| // This helper function has the signature that pthread_create wants. |
| DWORD WINAPI RunFunctionInThread(LPVOID ptr_to_ptr_to_fn) { |
| (**static_cast<void (**)()>(ptr_to_ptr_to_fn))(); // runs fn |
| return 0; |
| } |
| |
| DWORD WINAPI RunFunctionInThreadWithId(LPVOID ptr_to_fnid) { |
| FunctionAndId* fn_and_id = static_cast<FunctionAndId*>(ptr_to_fnid); |
| (*fn_and_id->ptr_to_function)(fn_and_id->id); // runs fn |
| return 0; |
| } |
| |
| void RunManyThreads(void (*fn)(), int count) { |
| DWORD dummy; |
| HANDLE* hThread = new HANDLE[count]; |
| for (int i = 0; i < count; i++) { |
| hThread[i] = CreateThread(NULL, 0, RunFunctionInThread, &fn, 0, &dummy); |
| if (hThread[i] == NULL) ExitProcess(i); |
| } |
| WaitForMultipleObjects(count, hThread, TRUE, INFINITE); |
| for (int i = 0; i < count; i++) { |
| CloseHandle(hThread[i]); |
| } |
| delete[] hThread; |
| } |
| |
| void RunThread(void (*fn)()) { |
| RunManyThreads(fn, 1); |
| } |
| |
| void RunManyThreadsWithId(void (*fn)(int), int count, int stacksize) { |
| DWORD dummy; |
| HANDLE* hThread = new HANDLE[count]; |
| FunctionAndId* fn_and_ids = new FunctionAndId[count]; |
| for (int i = 0; i < count; i++) { |
| fn_and_ids[i].ptr_to_function = fn; |
| fn_and_ids[i].id = i; |
| hThread[i] = CreateThread(NULL, stacksize, RunFunctionInThreadWithId, |
| &fn_and_ids[i], 0, &dummy); |
| if (hThread[i] == NULL) ExitProcess(i); |
| } |
| WaitForMultipleObjects(count, hThread, TRUE, INFINITE); |
| for (int i = 0; i < count; i++) { |
| CloseHandle(hThread[i]); |
| } |
| delete[] fn_and_ids; |
| delete[] hThread; |
| } |
| } |
| |
| #else // not NO_THREADS, not !HAVE_PTHREAD, not _WIN32 |
| |
| #include <pthread.h> |
| |
| #define SAFE_PTHREAD(fncall) do { if ((fncall) != 0) abort(); } while (0) |
| |
| extern "C" { |
| // This helper function has the signature that pthread_create wants. |
| static void* RunFunctionInThread(void *ptr_to_ptr_to_fn) { |
| (**static_cast<void (**)()>(ptr_to_ptr_to_fn))(); // runs fn |
| return NULL; |
| } |
| |
| static void* RunFunctionInThreadWithId(void *ptr_to_fnid) { |
| FunctionAndId* fn_and_id = static_cast<FunctionAndId*>(ptr_to_fnid); |
| (*fn_and_id->ptr_to_function)(fn_and_id->id); // runs fn |
| return NULL; |
| } |
| |
| // Run a function in a thread of its own and wait for it to finish. |
| // This is useful for tcmalloc testing, because each thread is |
| // handled separately in tcmalloc, so there's interesting stuff to |
| // test even if the threads are not running concurrently. |
| void RunThread(void (*fn)()) { |
| pthread_t thr; |
| // Even though fn is on the stack, it's safe to pass a pointer to it, |
| // because we pthread_join immediately (ie, before RunInThread exits). |
| SAFE_PTHREAD(pthread_create(&thr, NULL, RunFunctionInThread, &fn)); |
| SAFE_PTHREAD(pthread_join(thr, NULL)); |
| } |
| |
| void RunManyThreads(void (*fn)(), int count) { |
| pthread_t* thr = new pthread_t[count]; |
| for (int i = 0; i < count; i++) { |
| SAFE_PTHREAD(pthread_create(&thr[i], NULL, RunFunctionInThread, &fn)); |
| } |
| for (int i = 0; i < count; i++) { |
| SAFE_PTHREAD(pthread_join(thr[i], NULL)); |
| } |
| delete[] thr; |
| } |
| |
| void RunManyThreadsWithId(void (*fn)(int), int count, int stacksize) { |
| pthread_attr_t attr; |
| pthread_attr_init(&attr); |
| pthread_attr_setstacksize(&attr, stacksize); |
| |
| pthread_t* thr = new pthread_t[count]; |
| FunctionAndId* fn_and_ids = new FunctionAndId[count]; |
| for (int i = 0; i < count; i++) { |
| fn_and_ids[i].ptr_to_function = fn; |
| fn_and_ids[i].id = i; |
| SAFE_PTHREAD(pthread_create(&thr[i], &attr, |
| RunFunctionInThreadWithId, &fn_and_ids[i])); |
| } |
| for (int i = 0; i < count; i++) { |
| SAFE_PTHREAD(pthread_join(thr[i], NULL)); |
| } |
| delete[] fn_and_ids; |
| delete[] thr; |
| |
| pthread_attr_destroy(&attr); |
| } |
| } |
| |
| #endif |