blob: 1b2eda70c081446a2f6ca08f2387c62793cf28be [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <stdio.h>
#include "testframe.h"
#include "thread_unit_test_utils.h"
#include <jthread.h>
/*
* Test jthread_park(...)
* Test jthread_unpark(...)
*/
void JNICALL run_for_test_jthread_park_unpark(jvmtiEnv * jvmti_env, JNIEnv * jni_env, void *args){
tested_thread_sturct_t * tts = (tested_thread_sturct_t *) args;
IDATA status;
tts->phase = TT_PHASE_PARKED;
tested_thread_started(tts);
status = jthread_park();
if (status != TM_ERROR_NONE) {
log_info("Test status is %d, but expected %d", status, TM_ERROR_NONE);
tts->phase = TT_PHASE_ERROR;
} else {
tts->phase = TT_PHASE_RUNNING;
}
tested_thread_wait_for_stop_request(tts);
tts->phase = TT_PHASE_DEAD;
tested_thread_ended(tts);
}
int test_jthread_park_unpark(void) {
tested_thread_sturct_t *tts;
tested_thread_sturct_t *parked_tts;
int i;
int parked_nmb;
// Initialize tts structures and run all tested threads
tested_threads_run(run_for_test_jthread_park_unpark);
reset_tested_thread_iterator(&tts);
while(next_tested_thread(&tts)){
check_tested_thread_phase(tts, TT_PHASE_PARKED);
}
for (i = 0; i <= MAX_TESTED_THREAD_NUMBER; i++){
parked_nmb = 0;
parked_tts = NULL;
reset_tested_thread_iterator(&tts);
while(next_tested_thread(&tts)){
check_tested_thread_phase(tts, TT_PHASE_ANY); // to make thread running
if (tts->phase == TT_PHASE_PARKED){
parked_nmb++;
parked_tts = tts;
} else {
tf_assert_same(tts->phase, TT_PHASE_RUNNING);
}
}
if (MAX_TESTED_THREAD_NUMBER - parked_nmb - i != 0){
tf_fail("Wrong number of parked threads");
}
if (parked_nmb > 0){
tf_assert_same(jthread_unpark(parked_tts->java_thread), TM_ERROR_NONE);
tested_thread_wait_running(parked_tts);
check_tested_thread_phase(parked_tts, TT_PHASE_RUNNING);
}
}
// Terminate all threads and clear tts structures
tested_threads_destroy();
return TEST_PASSED;
}
/*
* Test jthread_park(...)
*/
void JNICALL run_for_test_jthread_park_interrupt(jvmtiEnv * jvmti_env, JNIEnv * jni_env, void *args){
tested_thread_sturct_t * tts = (tested_thread_sturct_t *) args;
IDATA status;
tts->phase = TT_PHASE_PARKED;
tested_thread_started(tts);
status = jthread_park();
if (status != TM_ERROR_INTERRUPT) {
log_info("Test status is %d, but expected %d", status, TM_ERROR_INTERRUPT);
tts->phase = TT_PHASE_ERROR;
} else {
tts->phase = TT_PHASE_RUNNING;
}
tested_thread_wait_for_stop_request(tts);
tts->phase = TT_PHASE_DEAD;
tested_thread_ended(tts);
}
int test_jthread_park_interrupt(void) {
tested_thread_sturct_t *tts;
tested_thread_sturct_t *parked_tts;
int count;
int parked_nmb;
// Initialize tts structures and run all tested threads
tested_threads_run(run_for_test_jthread_park_interrupt);
reset_tested_thread_iterator(&tts);
while(next_tested_thread(&tts)){
count = 0;
while (!hythread_is_parked(tts->native_thread)) {
// wait until the state is changed
hythread_sleep(SLEEP_TIME);
if (tts->phase == TT_PHASE_ERROR || ++count > (MAX_TIME_TO_WAIT/SLEEP_TIME)) {
tf_fail("thread failed to change state on PARKED");
}
}
check_tested_thread_phase(tts, TT_PHASE_PARKED);
}
for (count = 0; count <= MAX_TESTED_THREAD_NUMBER; count++){
parked_nmb = 0;
parked_tts = NULL;
reset_tested_thread_iterator(&tts);
while(next_tested_thread(&tts)){
if (tts->phase == TT_PHASE_PARKED){
parked_nmb++;
parked_tts = tts;
} else {
tf_assert_same(tts->phase, TT_PHASE_RUNNING);
}
}
if (MAX_TESTED_THREAD_NUMBER - parked_nmb - count != 0){
tf_fail("Wrong number of parked threads");
}
if (parked_nmb > 0){
tf_assert_same(jthread_interrupt(parked_tts->java_thread), TM_ERROR_NONE);
tested_thread_wait_running(parked_tts);
check_tested_thread_phase(parked_tts, TT_PHASE_RUNNING);
}
}
// Terminate all threads (not needed here) and clear tts structures
tested_threads_destroy();
return TEST_PASSED;
}
/*
* Test jthread_timed_park(...)
*/
void JNICALL run_for_test_jthread_timed_park(jvmtiEnv * jvmti_env, JNIEnv * jni_env, void *args){
tested_thread_sturct_t * tts = (tested_thread_sturct_t *) args;
IDATA status;
tts->phase = TT_PHASE_PARKED;
tested_thread_started(tts);
status = jthread_timed_park(SLEEP_TIME/2, 0);
if (status != TM_ERROR_TIMEOUT) {
log_info("Test status is %d, but expected %d", status, TM_ERROR_TIMEOUT);
tts->phase = TT_PHASE_ERROR;
} else {
tts->phase = TT_PHASE_RUNNING;
}
tested_thread_wait_for_stop_request(tts);
tts->phase = TT_PHASE_DEAD;
tested_thread_ended(tts);
}
int test_jthread_timed_park(void) {
tested_thread_sturct_t *tts;
// Initialize tts structures and run all tested threads
tested_threads_run(run_for_test_jthread_timed_park);
reset_tested_thread_iterator(&tts);
while(next_tested_thread(&tts)){
tested_thread_wait_running(tts);
check_tested_thread_phase(tts, TT_PHASE_RUNNING);
}
// Terminate all threads (not needed here) and clear tts structures
tested_threads_destroy();
return TEST_PASSED;
}
TEST_LIST_START
TEST(test_jthread_park_unpark)
TEST(test_jthread_park_interrupt)
TEST(test_jthread_timed_park)
TEST_LIST_END;