1999-02-04 03:25:56 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
1999-02-04 01:44:14 +03:00
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Netscape Public License
|
|
|
|
* Version 1.0 (the "NPL"); you may not use this file except in
|
|
|
|
* compliance with the NPL. You may obtain a copy of the NPL at
|
|
|
|
* http://www.mozilla.org/NPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the NPL is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* NPL.
|
|
|
|
*
|
|
|
|
* The Initial Developer of this code under the NPL is Netscape
|
|
|
|
* Communications Corporation. Portions created by Netscape are
|
|
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
|
|
|
|
* Reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
1999-02-05 02:20:30 +03:00
|
|
|
A stack-based lock object that makes using PRLock a bit more
|
1999-02-04 01:44:14 +03:00
|
|
|
convenient. It acquires the monitor when constructed, and releases
|
|
|
|
it when it goes out of scope.
|
|
|
|
|
|
|
|
For example,
|
|
|
|
|
|
|
|
class Foo {
|
|
|
|
private:
|
|
|
|
PRLock* mLock;
|
|
|
|
|
|
|
|
public:
|
|
|
|
Foo(void) {
|
|
|
|
mLock = PR_NewLock();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual ~Foo(void) {
|
|
|
|
PR_DestroyLock(mLock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThreadSafeMethod(void) {
|
1999-02-05 02:20:30 +03:00
|
|
|
// we're don't hold the lock yet...
|
1999-02-04 01:44:14 +03:00
|
|
|
|
|
|
|
nsAutoLock lock(mLock);
|
|
|
|
// ...but now we do.
|
|
|
|
|
|
|
|
// we even can do wacky stuff like return from arbitrary places w/o
|
1999-02-05 02:20:30 +03:00
|
|
|
// worrying about forgetting to release the lock
|
1999-02-04 01:44:14 +03:00
|
|
|
if (some_weird_condition)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// otherwise do some other stuff
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThreadSafeBlockScope(void) {
|
1999-02-05 02:20:30 +03:00
|
|
|
// we're not in the lock here...
|
1999-02-04 01:44:14 +03:00
|
|
|
|
|
|
|
{
|
|
|
|
nsAutoLock lock(mLock);
|
|
|
|
// but we are now, at least until the block scope closes
|
|
|
|
}
|
|
|
|
|
1999-02-05 02:20:30 +03:00
|
|
|
// ...now we're not in the lock anymore
|
1999-02-04 01:44:14 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef nsAutoLock_h__
|
|
|
|
#define nsAutoLock_h__
|
|
|
|
|
1999-02-04 03:39:49 +03:00
|
|
|
#include "nscore.h"
|
1999-02-04 01:44:14 +03:00
|
|
|
#include "prlock.h"
|
1999-02-04 03:39:49 +03:00
|
|
|
#include "prlog.h"
|
1999-02-04 01:44:14 +03:00
|
|
|
|
|
|
|
// If you ever decide that you need to add a non-inline method to this
|
|
|
|
// class, be sure to change the class declaration to "class NS_BASE
|
|
|
|
// nsAutoLock".
|
|
|
|
|
|
|
|
class nsAutoLock {
|
|
|
|
private:
|
|
|
|
PRLock* mLock;
|
|
|
|
|
1999-02-04 03:25:56 +03:00
|
|
|
// Not meant to be implemented. This makes it a compiler error to
|
|
|
|
// construct or assign an nsAutoLock object incorrectly.
|
1999-02-04 01:44:14 +03:00
|
|
|
nsAutoLock(void) {}
|
|
|
|
nsAutoLock(nsAutoLock& aLock) {}
|
1999-02-04 03:25:56 +03:00
|
|
|
nsAutoLock& operator =(nsAutoLock& aLock) {
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Not meant to be implemented. This makes it a compiler error to
|
|
|
|
// attempt to create an nsAutoLock object on the heap.
|
|
|
|
static void* operator new(size_t size) {
|
|
|
|
return nsnull;
|
|
|
|
}
|
|
|
|
static void operator delete(void* memory) {}
|
1999-02-04 01:44:14 +03:00
|
|
|
|
|
|
|
public:
|
|
|
|
nsAutoLock(PRLock* aLock) : mLock(aLock) {
|
|
|
|
PR_ASSERT(mLock);
|
|
|
|
|
|
|
|
// This will assert deep in the bowels of NSPR if you attempt
|
|
|
|
// to re-enter the lock.
|
|
|
|
PR_Lock(mLock);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
~nsAutoLock(void) {
|
|
|
|
PR_Unlock(mLock);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#endif // nsAutoLock_h__
|
|
|
|
|