/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (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.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is the Netscape Portable Runtime (NSPR). * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1998-2000 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ /* ** File: foreign.c ** Description: Testing various functions w/ foreign threads ** ** We create a thread and get it to call exactly one runtime function. ** The thread is allowed to be created by some other environment that ** NSPR, but it does not announce itself to the runtime prior to calling ** in. ** ** The goal: try to survive. ** */ #include "prcvar.h" #include "prenv.h" #include "prerror.h" #include "prinit.h" #include "prinrval.h" #include "prio.h" #include "prlock.h" #include "prlog.h" #include "prmem.h" #include "prthread.h" #include "prtypes.h" #include "prprf.h" #include "plgetopt.h" #include #include static enum { thread_nspr, thread_pthread, thread_uithread, thread_sproc, thread_win32 } thread_provider; typedef void (*StartFn)(void*); typedef struct StartObject { StartFn start; void *arg; } StartObject; static PRFileDesc *output; static int _debug_on = 0; #define DEFAULT_THREAD_COUNT 10 #define DPRINTF(arg) if (_debug_on) PR_fprintf arg #if defined(_PR_PTHREADS) && !defined(_PR_DCETHREADS) #include #include "md/_pth.h" static void *pthread_start(void *arg) { StartFn start = ((StartObject*)arg)->start; void *data = ((StartObject*)arg)->arg; PR_Free(arg); start(data); return NULL; } /* pthread_start */ #endif /* defined(_PR_PTHREADS) && !defined(_PR_DCETHREADS) */ #if defined(SOLARIS) && defined(_PR_GLOBAL_THREADS_ONLY) #include static void *uithread_start(void *arg) { StartFn start = ((StartObject*)arg)->start; void *data = ((StartObject*)arg)->arg; PR_Free(arg); start(data); return NULL; } /* uithread_start */ #endif /* defined(SOLARIS) && defined(_PR_GLOBAL_THREADS_ONLY) */ #if defined(IRIX) && !defined(_PR_PTHREADS) #include #include static void sproc_start(void *arg, PRSize size) { StartObject *so = (StartObject*)arg; StartFn start = so->start; void *data = so->arg; PR_Free(so); start(data); } /* sproc_start */ #endif /* defined(IRIX) && !defined(_PR_PTHREADS) */ #if defined(WIN32) #include /* for _beginthreadex() */ static PRUintn __stdcall windows_start(void *arg) { StartObject *so = (StartObject*)arg; StartFn start = so->start; void *data = so->arg; PR_Free(so); start(data); return 0; } /* windows_start */ #endif /* defined(WIN32) */ static PRStatus CreateThread(StartFn start, void *arg) { PRStatus rv; switch (thread_provider) { case thread_nspr: { PRThread *thread = PR_CreateThread( PR_USER_THREAD, start, arg, PR_PRIORITY_NORMAL, PR_GLOBAL_THREAD, PR_UNJOINABLE_THREAD, 0); rv = (NULL == thread) ? PR_FAILURE : PR_SUCCESS; } break; case thread_pthread: #if defined(_PR_PTHREADS) && !defined(_PR_DCETHREADS) { int rv; pthread_t id; pthread_attr_t tattr; StartObject *start_object; start_object = PR_NEW(StartObject); PR_ASSERT(NULL != start_object); start_object->start = start; start_object->arg = arg; rv = _PT_PTHREAD_ATTR_INIT(&tattr); PR_ASSERT(0 == rv); rv = pthread_attr_setdetachstate(&tattr, PTHREAD_CREATE_DETACHED); PR_ASSERT(0 == rv); #if !defined(LINUX) rv = pthread_attr_setstacksize(&tattr, 64 * 1024); PR_ASSERT(0 == rv); #endif rv = _PT_PTHREAD_CREATE(&id, tattr, pthread_start, start_object); (void)_PT_PTHREAD_ATTR_DESTROY(&tattr); return (0 == rv) ? PR_SUCCESS : PR_FAILURE; } #else PR_SetError(PR_NOT_IMPLEMENTED_ERROR, 0); rv = PR_FAILURE; break; #endif /* defined(_PR_PTHREADS) && !defined(_PR_DCETHREADS) */ case thread_uithread: #if defined(SOLARIS) && defined(_PR_GLOBAL_THREADS_ONLY) { int rv; thread_t id; long flags; StartObject *start_object; start_object = PR_NEW(StartObject); PR_ASSERT(NULL != start_object); start_object->start = start; start_object->arg = arg; flags = THR_DETACHED; rv = thr_create(NULL, NULL, uithread_start, start_object, flags, &id); return (0 == rv) ? PR_SUCCESS : PR_FAILURE; } #else PR_SetError(PR_NOT_IMPLEMENTED_ERROR, 0); rv = PR_FAILURE; break; #endif /* defined(SOLARIS) && defined(_PR_GLOBAL_THREADS_ONLY) */ case thread_sproc: #if defined(IRIX) && !defined(_PR_PTHREADS) { PRInt32 pid; StartObject *start_object; start_object = PR_NEW(StartObject); PR_ASSERT(NULL != start_object); start_object->start = start; start_object->arg = arg; pid = sprocsp( sproc_start, PR_SALL, start_object, NULL, 64 * 1024); rv = (0 < pid) ? PR_SUCCESS : PR_FAILURE; } #else PR_SetError(PR_NOT_IMPLEMENTED_ERROR, 0); rv = PR_FAILURE; #endif /* defined(IRIX) && !defined(_PR_PTHREADS) */ break; case thread_win32: #if defined(WIN32) { void *th; PRUintn id; StartObject *start_object; start_object = PR_NEW(StartObject); PR_ASSERT(NULL != start_object); start_object->start = start; start_object->arg = arg; th = (void*)_beginthreadex( NULL, /* LPSECURITY_ATTRIBUTES - pointer to thread security attributes */ 0U, /* DWORD - initial thread stack size, in bytes */ windows_start, /* LPTHREAD_START_ROUTINE - pointer to thread function */ start_object, /* LPVOID - argument for new thread */ 0U, /*DWORD dwCreationFlags - creation flags */ &id /* LPDWORD - pointer to returned thread identifier */ ); rv = (NULL == th) ? PR_FAILURE : PR_SUCCESS; } #else PR_SetError(PR_NOT_IMPLEMENTED_ERROR, 0); rv = PR_FAILURE; #endif break; default: PR_SetError(PR_NOT_IMPLEMENTED_ERROR, 0); rv = PR_FAILURE; } return rv; } /* CreateThread */ static void PR_CALLBACK lazyEntry(void *arg) { PR_ASSERT(NULL == arg); } /* lazyEntry */ static void OneShot(void *arg) { PRUintn pdkey; PRLock *lock; PRFileDesc *fd; PRDir *dir; PRFileDesc *pair[2]; PRIntn test = (PRIntn)arg; for (test = 0; test < 12; ++test) { switch (test) { case 0: lock = PR_NewLock(); DPRINTF((output,"Thread[0x%x] called PR_NewLock\n", PR_GetCurrentThread())); PR_DestroyLock(lock); break; case 1: (void)PR_SecondsToInterval(1); DPRINTF((output,"Thread[0x%x] called PR_SecondsToInterval\n", PR_GetCurrentThread())); break; case 2: (void)PR_CreateThread( PR_USER_THREAD, lazyEntry, NULL, PR_PRIORITY_NORMAL, PR_LOCAL_THREAD, PR_UNJOINABLE_THREAD, 0); DPRINTF((output,"Thread[0x%x] called PR_CreateThread\n", PR_GetCurrentThread())); break; case 3: fd = PR_Open("foreign.tmp", PR_CREATE_FILE | PR_RDWR, 0666); DPRINTF((output,"Thread[0x%x] called PR_Open\n", PR_GetCurrentThread())); PR_Close(fd); break; case 4: fd = PR_NewUDPSocket(); DPRINTF((output,"Thread[0x%x] called PR_NewUDPSocket\n", PR_GetCurrentThread())); PR_Close(fd); break; case 5: fd = PR_NewTCPSocket(); DPRINTF((output,"Thread[0x%x] called PR_NewTCPSocket\n", PR_GetCurrentThread())); PR_Close(fd); break; case 6: dir = PR_OpenDir("/tmp/"); DPRINTF((output,"Thread[0x%x] called PR_OpenDir\n", PR_GetCurrentThread())); PR_CloseDir(dir); break; case 7: (void)PR_NewThreadPrivateIndex(&pdkey, NULL); DPRINTF((output,"Thread[0x%x] called PR_NewThreadPrivateIndex\n", PR_GetCurrentThread())); break; case 8: (void)PR_GetEnv("PATH"); DPRINTF((output,"Thread[0x%x] called PR_GetEnv\n", PR_GetCurrentThread())); break; case 9: (void)PR_NewTCPSocketPair(pair); DPRINTF((output,"Thread[0x%x] called PR_NewTCPSocketPair\n", PR_GetCurrentThread())); PR_Close(pair[0]); PR_Close(pair[1]); break; case 10: PR_SetConcurrency(2); DPRINTF((output,"Thread[0x%x] called PR_SetConcurrency\n", PR_GetCurrentThread())); break; case 11: PR_SetThreadPriority(PR_GetCurrentThread(), PR_PRIORITY_HIGH); DPRINTF((output,"Thread[0x%x] called PR_SetThreadPriority\n", PR_GetCurrentThread())); break; default: break; } /* switch() */ } } /* OneShot */ PRIntn main(PRIntn argc, char **argv) { PRStatus rv; PRInt32 thread_cnt = DEFAULT_THREAD_COUNT; PLOptStatus os; PLOptState *opt = PL_CreateOptState(argc, argv, "dt:"); #if defined(WIN32) thread_provider = thread_win32; #elif defined(_PR_PTHREADS) thread_provider = thread_pthread; #elif defined(SOLARIS) && defined(_PR_GLOBAL_THREADS_ONLY) thread_provider = thread_uithread; #elif defined(IRIX) thread_provider = thread_sproc; #else thread_provider = thread_nspr; #endif while (PL_OPT_EOL != (os = PL_GetNextOpt(opt))) { if (PL_OPT_BAD == os) continue; switch (opt->option) { case 'd': /* debug mode */ _debug_on = 1; break; case 't': /* thread count */ thread_cnt = atoi(opt->value); break; default: break; } } PL_DestroyOptState(opt); PR_SetConcurrency(2); output = PR_GetSpecialFD(PR_StandardOutput); while (thread_cnt-- > 0) { rv = CreateThread(OneShot, (void*)thread_cnt); PR_ASSERT(PR_SUCCESS == rv); PR_Sleep(PR_MillisecondsToInterval(5)); } PR_Sleep(PR_SecondsToInterval(3)); return (PR_SUCCESS == PR_Cleanup()) ? 0 : 1; } /* main */ /* foreign.c */