| /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ |
| /* This Source Code Form is subject to the terms of the Mozilla Public |
| * License, v. 2.0. If a copy of the MPL was not distributed with this |
| * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ |
| |
| /* |
| * File: primblok.c |
| * Purpose: testing whether the primordial thread can block in a |
| * native blocking function without affecting the correct |
| * functioning of NSPR I/O functions (Bugzilla bug #30746) |
| */ |
| |
| #if !defined(WINNT) |
| |
| #include <stdio.h> |
| |
| int main(int argc, char **argv) |
| { |
| printf("This test is not relevant on this platform\n"); |
| return 0; |
| } |
| |
| #else /* WINNT */ |
| |
| #include "nspr.h" |
| |
| #include <windows.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| #define TEST_FILE_NAME "primblok.dat" |
| |
| /* use InterlockedExchange to update this variable */ |
| static LONG iothread_done; |
| |
| static void PR_CALLBACK IOThread(void *arg) |
| { |
| PRFileDesc *fd; |
| char buf[32]; |
| PRInt32 nbytes; |
| |
| /* Give the primordial thread one second to block */ |
| Sleep(1000); |
| |
| /* |
| * See if our PR_Write call will hang when the primordial |
| * thread is blocking in a native blocking function. |
| */ |
| fd = PR_Open(TEST_FILE_NAME, PR_WRONLY|PR_CREATE_FILE, 0666); |
| if (NULL == fd) { |
| fprintf(stderr, "PR_Open failed\n"); |
| exit(1); |
| } |
| memset(buf, 0xaf, sizeof(buf)); |
| fprintf(stderr, "iothread: calling PR_Write\n"); |
| nbytes = PR_Write(fd, buf, sizeof(buf)); |
| fprintf(stderr, "iothread: PR_Write returned\n"); |
| if (nbytes != sizeof(buf)) { |
| fprintf(stderr, "PR_Write returned %d\n", nbytes); |
| exit(1); |
| } |
| if (PR_Close(fd) == PR_FAILURE) { |
| fprintf(stderr, "PR_Close failed\n"); |
| exit(1); |
| } |
| if (PR_Delete(TEST_FILE_NAME) == PR_FAILURE) { |
| fprintf(stderr, "PR_Delete failed\n"); |
| exit(1); |
| } |
| |
| /* Tell the main thread that we are done */ |
| InterlockedExchange(&iothread_done, 1); |
| } |
| |
| int main(int argc, char **argv) |
| { |
| PRThread *iothread; |
| |
| /* Must be a global thread */ |
| iothread = PR_CreateThread( |
| PR_USER_THREAD, IOThread, NULL, PR_PRIORITY_NORMAL, |
| PR_GLOBAL_THREAD, PR_JOINABLE_THREAD, 0); |
| if (iothread == NULL) { |
| fprintf(stderr, "cannot create thread\n"); |
| exit(1); |
| } |
| |
| /* |
| * Block in a native blocking function. |
| * Give iothread 5 seconds to finish its task. |
| */ |
| Sleep(5000); |
| |
| /* |
| * Is iothread done or is it hung? |
| * |
| * I'm actually only interested in reading the value |
| * of iothread_done. I'm using InterlockedExchange as |
| * a thread-safe way to read iothread_done. |
| */ |
| if (InterlockedExchange(&iothread_done, 1) == 0) { |
| fprintf(stderr, "iothread is hung\n"); |
| fprintf(stderr, "FAILED\n"); |
| exit(1); |
| } |
| |
| if (PR_JoinThread(iothread) == PR_FAILURE) { |
| fprintf(stderr, "PR_JoinThread failed\n"); |
| exit(1); |
| } |
| printf("PASSED\n"); |
| return 0; |
| } /* main */ |
| |
| #endif /* WINNT */ |