jdk-24/test/lib/native/testlib_threads.h
2023-11-22 13:43:33 +00:00

103 lines
3.0 KiB
C

/*
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code 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
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
#ifndef TEST_LIB_NATIVE_THREAD_H
#define TEST_LIB_NATIVE_THREAD_H
// Header only library for using threads in tests
#include <stdlib.h>
#include <stdio.h>
#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#include <pthread.h>
#endif
extern "C" {
typedef void(*PROCEDURE)(void*);
struct Helper {
PROCEDURE proc;
void* context;
};
static void fatal(const char* message, int code) {
fputs(message, stderr);
// exit the test with a non-zero exit code to avoid accidental false positives
exit(code);
}
// Adapt from the callback type the OS API expects to
// our OS-independent PROCEDURE type.
#ifdef _WIN32
DWORD WINAPI procedure(_In_ LPVOID ctxt) {
#else
void* procedure(void* ctxt) {
#endif
Helper* helper = (Helper*)ctxt;
helper->proc(helper->context);
return 0;
}
// Run 'proc' in a newly started thread, passing 'context' to it
// as an argument, and then join that thread.
void run_in_new_thread_and_join(PROCEDURE proc, void* context) {
struct Helper helper;
helper.proc = proc;
helper.context = context;
#ifdef _WIN32
HANDLE thread = CreateThread(NULL, 0, procedure, &helper, 0, NULL);
if (thread == NULL) {
fatal("failed to create thread", GetLastError());
}
if (WaitForSingleObject(thread, INFINITE) != WAIT_OBJECT_0) {
// Should be WAIT_FAILED, since this is not a mutex, and
// we set no timeout.
fatal("failed to join thread", GetLastError());
}
#else
pthread_t thread;
pthread_attr_t attr;
pthread_attr_init(&attr);
size_t stack_size = 0x100000;
pthread_attr_setstacksize(&attr, stack_size);
int result = pthread_create(&thread, &attr, procedure, &helper);
if (result != 0) {
fatal("failed to create thread", result);
}
pthread_attr_destroy(&attr);
result = pthread_join(thread, NULL);
if (result != 0) {
fatal("failed to join thread", result);
}
#endif
}
}
#endif // TEST_LIB_NATIVE_THREAD_H