зеркало из https://github.com/mozilla/gecko-dev.git
322 строки
9.2 KiB
C
322 строки
9.2 KiB
C
/* -*- 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: 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 <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
static enum {
|
|
thread_nspr, thread_pthread, 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)
|
|
#include <pthread.h>
|
|
#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) */
|
|
|
|
#if defined(WIN32)
|
|
#include <windows.h>
|
|
#include <process.h> /* 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 NSPRPUB_TESTS_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)
|
|
{
|
|
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);
|
|
|
|
rv = pthread_attr_setstacksize(&tattr, 64 * 1024);
|
|
PR_ASSERT(0 == rv);
|
|
|
|
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) */
|
|
|
|
case thread_sproc:
|
|
PR_SetError(PR_NOT_IMPLEMENTED_ERROR, 0);
|
|
rv = PR_FAILURE;
|
|
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 */
|
|
STACK_SIZE_PARAM_IS_A_RESERVATION, /*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;
|
|
} /* NSPRPUB_TESTS_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:
|
|
#define TEMP_DIR "./tmp"
|
|
PR_MkDir(TEMP_DIR, 0700);
|
|
dir = PR_OpenDir(TEMP_DIR);
|
|
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 */
|
|
|
|
int main(int 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;
|
|
#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 = NSPRPUB_TESTS_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 */
|