1999-04-05 21:03:22 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
*
|
1999-11-06 03:43:54 +00:00
|
|
|
* The contents of this file are subject to the Netscape 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/NPL/
|
1999-04-05 21:03:22 +00:00
|
|
|
*
|
1999-11-06 03:43:54 +00:00
|
|
|
* 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.
|
1999-04-05 21:03:22 +00:00
|
|
|
*
|
1999-11-06 03:43:54 +00:00
|
|
|
* The Original Code is mozilla.org code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is Netscape
|
1999-04-05 21:03:22 +00:00
|
|
|
* Communications Corporation. Portions created by Netscape are
|
1999-11-06 03:43:54 +00:00
|
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All
|
|
|
|
* Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
1999-12-01 00:21:53 +00:00
|
|
|
* Pierre Phaneuf <pp@ludusdesign.com>
|
1999-04-05 21:03:22 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "nsIThread.h"
|
1999-09-15 22:28:27 +00:00
|
|
|
#include "nsIThreadPool.h"
|
|
|
|
#include "nsIRunnable.h"
|
1999-04-05 21:03:22 +00:00
|
|
|
#include <stdio.h>
|
2000-03-05 21:26:01 +00:00
|
|
|
#include "nsCOMPtr.h"
|
|
|
|
#include "nsIServiceManager.h"
|
1999-04-05 21:03:22 +00:00
|
|
|
|
|
|
|
class nsRunner : public nsIRunnable {
|
|
|
|
public:
|
|
|
|
NS_DECL_ISUPPORTS
|
|
|
|
|
|
|
|
NS_IMETHOD Run() {
|
2000-03-05 21:26:01 +00:00
|
|
|
nsCOMPtr<nsIThread> thread;
|
|
|
|
nsresult rv = nsIThread::GetCurrent(getter_AddRefs(thread));
|
1999-04-05 21:03:22 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
printf("failed to get current thread\n");
|
|
|
|
return rv;
|
|
|
|
}
|
2000-07-11 09:46:51 +00:00
|
|
|
printf("running %d on thread %x\n", mNum, thread.get());
|
1999-04-05 21:03:22 +00:00
|
|
|
|
|
|
|
// if we don't do something slow, we'll never see the other
|
|
|
|
// worker threads run
|
2000-04-21 05:12:57 +00:00
|
|
|
PR_Sleep(PR_MillisecondsToInterval(100));
|
1999-04-05 21:03:22 +00:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsRunner(int num) : mNum(num) {
|
|
|
|
NS_INIT_REFCNT();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
int mNum;
|
|
|
|
};
|
|
|
|
|
2000-03-05 21:26:01 +00:00
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS1(nsRunner, nsIRunnable);
|
1999-04-05 21:03:22 +00:00
|
|
|
|
|
|
|
nsresult
|
|
|
|
TestThreads()
|
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
2000-03-05 21:26:01 +00:00
|
|
|
nsCOMPtr<nsIThread> runner;
|
|
|
|
rv = NS_NewThread(getter_AddRefs(runner), new nsRunner(0), 0, PR_JOINABLE_THREAD);
|
1999-04-05 21:03:22 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
printf("failed to create thread\n");
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2000-03-05 21:26:01 +00:00
|
|
|
nsCOMPtr<nsIThread> thread;
|
|
|
|
rv = nsIThread::GetCurrent(getter_AddRefs(thread));
|
1999-04-05 21:03:22 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
printf("failed to get current thread\n");
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRThreadScope scope;
|
|
|
|
rv = runner->GetScope(&scope);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
printf("runner already exited\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = runner->Join(); // wait for the runner to die before quitting
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
printf("join failed\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = runner->GetScope(&scope); // this should fail after Join
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
printf("get scope failed\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = runner->Interrupt(); // this should fail after Join
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
printf("interrupt failed\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
// try an unjoinable thread
|
2000-03-05 21:26:01 +00:00
|
|
|
rv = NS_NewThread(getter_AddRefs(runner), new nsRunner(1));
|
1999-04-05 21:03:22 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
printf("failed to create thread\n");
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = runner->Join(); // wait for the runner to die before quitting
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
printf("shouldn't have been able to join an unjoinable thread\n");
|
|
|
|
}
|
2000-03-05 21:26:01 +00:00
|
|
|
|
2000-04-21 05:12:57 +00:00
|
|
|
PR_Sleep(PR_MillisecondsToInterval(100)); // hopefully the runner will quit here
|
1999-04-05 21:03:22 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2000-04-21 05:12:57 +00:00
|
|
|
TestThreadPools(PRUint32 poolMinSize, PRUint32 poolMaxSize,
|
|
|
|
PRUint32 nRequests, PRIntervalTime dispatchWaitInterval = 0)
|
1999-04-05 21:03:22 +00:00
|
|
|
{
|
2000-03-05 21:26:01 +00:00
|
|
|
nsCOMPtr<nsIThreadPool> pool;
|
2000-04-21 05:12:57 +00:00
|
|
|
nsresult rv = NS_NewThreadPool(getter_AddRefs(pool), poolMinSize, poolMaxSize);
|
1999-04-05 21:03:22 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
printf("failed to create thead pool\n");
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2000-04-21 05:12:57 +00:00
|
|
|
for (PRUint32 i = 0; i < nRequests; i++) {
|
1999-04-05 21:03:22 +00:00
|
|
|
rv = pool->DispatchRequest(new nsRunner(i+2));
|
2000-04-21 05:12:57 +00:00
|
|
|
if (dispatchWaitInterval && i % poolMaxSize == poolMaxSize - 1) {
|
|
|
|
PR_Sleep(dispatchWaitInterval);
|
|
|
|
}
|
1999-04-05 21:03:22 +00:00
|
|
|
}
|
1999-04-13 18:15:27 +00:00
|
|
|
rv = pool->Shutdown();
|
1999-04-05 21:03:22 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main()
|
|
|
|
{
|
2000-03-05 21:26:01 +00:00
|
|
|
nsresult rv;
|
|
|
|
rv = NS_InitXPCOM(nsnull, nsnull);
|
|
|
|
if (NS_FAILED(rv)) return -1;
|
|
|
|
|
|
|
|
rv = TestThreads();
|
1999-04-05 21:03:22 +00:00
|
|
|
if (NS_FAILED(rv)) return -1;
|
|
|
|
|
2000-04-21 05:12:57 +00:00
|
|
|
rv = TestThreadPools(1, 4, 100);
|
|
|
|
if (NS_FAILED(rv)) return -1;
|
|
|
|
|
|
|
|
rv = TestThreadPools(4, 16, 100);
|
|
|
|
if (NS_FAILED(rv)) return -1;
|
|
|
|
|
|
|
|
// this test delays between each request to give threads a chance to
|
|
|
|
// decide to go away:
|
|
|
|
rv = TestThreadPools(4, 8, 32, PR_MillisecondsToInterval(1000));
|
1999-04-05 21:03:22 +00:00
|
|
|
if (NS_FAILED(rv)) return -1;
|
|
|
|
|
2000-03-05 21:26:01 +00:00
|
|
|
rv = NS_ShutdownXPCOM(nsnull);
|
|
|
|
if (NS_FAILED(rv)) return -1;
|
1999-04-05 21:03:22 +00:00
|
|
|
return 0;
|
|
|
|
}
|