⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 readwritesem.cpp

📁 Pegasus is an open-source implementationof the DMTF CIM and WBEM standards. It is designed to be por
💻 CPP
📖 第 1 页 / 共 2 页
字号:
//%2006//////////////////////////////////////////////////////////////////////////// Copyright (c) 2000, 2001, 2002 BMC Software; Hewlett-Packard Development// Company, L.P.; IBM Corp.; The Open Group; Tivoli Systems.// Copyright (c) 2003 BMC Software; Hewlett-Packard Development Company, L.P.;// IBM Corp.; EMC Corporation, The Open Group.// Copyright (c) 2004 BMC Software; Hewlett-Packard Development Company, L.P.;// IBM Corp.; EMC Corporation; VERITAS Software Corporation; The Open Group.// Copyright (c) 2005 Hewlett-Packard Development Company, L.P.; IBM Corp.;// EMC Corporation; VERITAS Software Corporation; The Open Group.// Copyright (c) 2006 Hewlett-Packard Development Company, L.P.; IBM Corp.;// EMC Corporation; Symantec Corporation; The Open Group.//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions:// // THE ABOVE COPYRIGHT NOTICE AND THIS PERMISSION NOTICE SHALL BE INCLUDED IN// ALL COPIES OR SUBSTANTIAL PORTIONS OF THE SOFTWARE. THE SOFTWARE IS PROVIDED// "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT// LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN// ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.////==============================================================================////%/////////////////////////////////////////////////////////////////////////////#include "ReadWriteSem.h"#include "Time.h"#include "PegasusAssert.h"#include "Threads.h"PEGASUS_NAMESPACE_BEGIN//==============================================================================//// PEGASUS_USE_POSIX_RWLOCK////==============================================================================#ifdef PEGASUS_USE_POSIX_RWLOCKReadWriteSem::ReadWriteSem():_readers(0), _writers(0){    pthread_rwlock_init(&_rwlock.rwlock, NULL);    Threads::clear(_rwlock.owner);}ReadWriteSem::~ReadWriteSem(){    int r = 0;    while (r=pthread_rwlock_destroy(&_rwlock.rwlock) == EBUSY ||           (r == -1 && errno == EBUSY))    {        Threads::yield();    }}void ReadWriteSem::wait(Uint32 mode, ThreadType caller){    int errorcode;    if (mode == PEG_SEM_READ)    {        if (0 == (errorcode = pthread_rwlock_rdlock(&_rwlock.rwlock)))        {            _readers++;            return;        }    }    else if (mode == PEG_SEM_WRITE)    {        if (0 == (errorcode = pthread_rwlock_wrlock(&_rwlock.rwlock)))        {            _rwlock.owner = caller;            _writers++;            return;        }    }    else        throw(Permission(Threads::self()));    if (errorcode == EDEADLK)        throw(Deadlock(_rwlock.owner));    else        throw(WaitFailed(Threads::self()));}void ReadWriteSem::try_wait(Uint32 mode, ThreadType caller){    int errorcode = 0;    if (mode == PEG_SEM_READ)    {        if (0 == (errorcode = pthread_rwlock_tryrdlock(&_rwlock.rwlock)))        {            _readers++;            return;        }    }    else if (mode == PEG_SEM_WRITE)    {        if (0 == (errorcode = pthread_rwlock_trywrlock(&_rwlock.rwlock)))        {            _writers++;            _rwlock.owner = caller;            return;        }    }    else        throw(Permission(Threads::self()));    if (errorcode == -1)        errorcode = errno;    if (errorcode == EBUSY)        throw(AlreadyLocked(_rwlock.owner));    else if (errorcode == EDEADLK)        throw(Deadlock(_rwlock.owner));    else        throw(WaitFailed(Threads::self()));}// timedrdlock and timedwrlock are not supported on HPUX// mdday Sun Aug  5 14:21:00 2001void ReadWriteSem::timed_wait(Uint32 mode,                              ThreadType caller, int milliseconds){    int errorcode = 0, timeout = 0;    struct timeval now, finish, remaining;    Uint32 usec;    gettimeofday(&finish, NULL);    finish.tv_sec += (milliseconds / 1000);    milliseconds %= 1000;    usec = finish.tv_usec + (milliseconds * 1000);    finish.tv_sec += (usec / 1000000);    finish.tv_usec = usec % 1000000;    if (mode == PEG_SEM_READ)    {        do        {            errorcode = pthread_rwlock_tryrdlock(&_rwlock.rwlock);            if (errorcode == -1)                errorcode = errno;            gettimeofday(&now, NULL);        }        while (errorcode == EBUSY &&               (0 == (timeout = Time::subtract(&remaining, &finish, &now))));        if (0 == errorcode)        {            _readers++;            return;        }    }    else if (mode == PEG_SEM_WRITE)    {        do        {            errorcode = pthread_rwlock_trywrlock(&_rwlock.rwlock);            if (errorcode == -1)                errorcode = errno;            gettimeofday(&now, NULL);        }        while (errorcode == EBUSY &&               (0 == (timeout = Time::subtract(&remaining, &finish, &now))));        if (0 == errorcode)        {            _writers++;            _rwlock.owner = caller;            return;        }    }    else        throw(Permission(Threads::self()));    if (timeout != 0)        throw(TimeOut(_rwlock.owner));    else if (errorcode == EDEADLK)        throw(Deadlock(_rwlock.owner));    else        throw(WaitFailed(Threads::self()));}void ReadWriteSem::unlock(Uint32 mode, ThreadType caller){    ThreadType owner;    if (mode == PEG_SEM_WRITE)    {        owner = _rwlock.owner;        Threads::clear(_rwlock.owner);    }    if (0 != pthread_rwlock_unlock(&_rwlock.rwlock))    {        _rwlock.owner = owner;        throw(Permission(Threads::self()));    }    if (mode == PEG_SEM_READ && _readers.get() != 0)        _readers--;    else if (_writers.get() != 0)        _writers--;}int ReadWriteSem::read_count() const{    return _readers.get();}int ReadWriteSem::write_count() const{    return _writers.get();}#endif /* PEGASUS_USE_POSIX_RWLOCK *///==============================================================================//// PEGASUS_USE_SEMAPHORE_RWLOCK////==============================================================================#if defined(PEGASUS_USE_SEMAPHORE_RWLOCK)// // If i get cancelled, I MUST ensure:// 1) I do not hold the internal mutex// 2) I do not hold the write lock// 3) I am not using a reader slot#if 0void extricate_read_write(void *parm){    ReadWriteSem *rws = (ReadWriteSem *) parm;    ThreadType myself = Threads::self();    if (Threads::equal(rws->_rwlock._wlock.get_owner(), myself))        rws->_rwlock._wlock.unlock();    else if (rws->_readers.get() > 0)        rws->_rwlock._rlock.signal();    if (Threads::equal(rws->_rwlock._internal_lock.get_owner(), myself))        rws->_rwlock._internal_lock.unlock();}#endifReadWriteSem::ReadWriteSem():_readers(0), _writers(0), _rwlock(){}ReadWriteSem::~ReadWriteSem(){    // lock everyone out of this object    try    {        _rwlock._internal_lock.lock();    }    catch (Deadlock &)    {        // no problem - we own the lock, which is what we want    }    catch (IPCException &)    {        PEGASUS_ASSERT(0);    }    while (_readers.get() > 0 || _writers.get() > 0)    {        Threads::yield();    }    _rwlock._internal_lock.unlock();}//-----------------------------------------------------------------// if milliseconds == -1, wait indefinately// if milliseconds == 0, fast wait//-----------------------------------------------------------------void ReadWriteSem::timed_wait(Uint32 mode, ThreadType caller,

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -