252 lines
5.4 KiB
C
252 lines
5.4 KiB
C
/*
|
|
* $Id$
|
|
*
|
|
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions
|
|
are met:
|
|
1. Redistributions of source code must retain the above copyright
|
|
notice, this list of conditions and the following disclaimer.
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
notice, this list of conditions and the following disclaimer in the
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
|
|
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <qse/si/rwl.h>
|
|
#include "../cmn/mem-prv.h"
|
|
|
|
qse_rwl_t* qse_rwl_open (qse_mmgr_t* mmgr, qse_size_t xtnsize, int flags)
|
|
{
|
|
qse_rwl_t* rwl;
|
|
|
|
rwl = (qse_rwl_t*) QSE_MMGR_ALLOC (mmgr, QSE_SIZEOF(qse_rwl_t) + xtnsize);
|
|
if (rwl)
|
|
{
|
|
if (qse_rwl_init (rwl, mmgr, flags) <= -1)
|
|
{
|
|
QSE_MMGR_FREE (mmgr, rwl);
|
|
return QSE_NULL;
|
|
}
|
|
else QSE_MEMSET (QSE_XTN(rwl), 0, xtnsize);
|
|
}
|
|
|
|
return rwl;
|
|
}
|
|
|
|
void qse_rwl_close (qse_rwl_t* rwl)
|
|
{
|
|
qse_rwl_fini (rwl);
|
|
QSE_MMGR_FREE (rwl->mmgr, rwl);
|
|
}
|
|
|
|
int qse_rwl_init (qse_rwl_t* rwl, qse_mmgr_t* mmgr, int flags)
|
|
{
|
|
QSE_MEMSET (rwl, 0, QSE_SIZEOF(*rwl));
|
|
rwl->mmgr = mmgr;
|
|
rwl->flags = flags;
|
|
|
|
if (qse_mtx_init (&rwl->mtx, mmgr) <= -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
if (qse_cnd_init (&rwl->rcnd, mmgr) <= -1)
|
|
{
|
|
qse_mtx_fini (&rwl->mtx);
|
|
return -1;
|
|
}
|
|
|
|
if (qse_cnd_init (&rwl->wcnd, mmgr) <= -1)
|
|
{
|
|
qse_cnd_fini (&rwl->rcnd);
|
|
qse_mtx_fini (&rwl->mtx);
|
|
return -1;
|
|
}
|
|
|
|
rwl->rwait_count = 0;
|
|
rwl->wwait_count = 0;
|
|
rwl->ractive_count = 0;
|
|
rwl->wactive_count = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void qse_rwl_fini (qse_rwl_t* rwl)
|
|
{
|
|
qse_cnd_fini (&rwl->wcnd);
|
|
qse_cnd_fini (&rwl->rcnd);
|
|
qse_mtx_fini (&rwl->mtx);
|
|
}
|
|
|
|
qse_mmgr_t* qse_rwl_getmmgr (qse_rwl_t* rwl)
|
|
{
|
|
return rwl->mmgr;
|
|
}
|
|
|
|
void* qse_rwl_getxtn (qse_rwl_t* rwl)
|
|
{
|
|
return QSE_XTN (rwl);
|
|
}
|
|
|
|
int qse_rwl_lockr (qse_rwl_t* rwl, const qse_ntime_t* waiting_time)
|
|
{
|
|
qse_ntime_t dead_line, now, rem, zero;
|
|
|
|
if (waiting_time)
|
|
{
|
|
qse_clear_ntime (&zero);
|
|
qse_gettime (&now);
|
|
qse_add_ntime (&dead_line, &now, waiting_time);
|
|
}
|
|
if (qse_mtx_lock (&rwl->mtx, waiting_time) <= -1) return -1;
|
|
|
|
if (rwl->wactive_count > 0 || ((rwl->flags & QSE_RWL_PREFER_WRITER) && rwl->wwait_count > 0))
|
|
{
|
|
rwl->rwait_count++;
|
|
while (rwl->wactive_count > 0 || ((rwl->flags & QSE_RWL_PREFER_WRITER) && rwl->wwait_count > 0))
|
|
{
|
|
if (waiting_time)
|
|
{
|
|
qse_gettime (&now);
|
|
qse_sub_ntime (&rem, &dead_line, &now);
|
|
if (qse_cmp_ntime(&rem, &zero) <= 0)
|
|
{
|
|
/* timed out */
|
|
rwl->rwait_count--;
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
return -1;
|
|
}
|
|
qse_cnd_wait (&rwl->rcnd, &rwl->mtx, &rem);
|
|
}
|
|
else
|
|
{
|
|
qse_cnd_wait (&rwl->rcnd, &rwl->mtx, QSE_NULL);
|
|
}
|
|
}
|
|
rwl->rwait_count--;
|
|
}
|
|
|
|
rwl->ractive_count++;
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int qse_rwl_unlockr (qse_rwl_t* rwl)
|
|
{
|
|
if (qse_mtx_lock (&rwl->mtx, QSE_NULL) <= -1) return -1;
|
|
|
|
if (rwl->ractive_count <= 0)
|
|
{
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
return -1;
|
|
}
|
|
|
|
rwl->ractive_count--;
|
|
if (rwl->ractive_count == 0 && rwl->wwait_count > 0)
|
|
{
|
|
qse_cnd_signal (&rwl->wcnd);
|
|
}
|
|
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
return 0;
|
|
}
|
|
|
|
int qse_rwl_lockw (qse_rwl_t* rwl, const qse_ntime_t* waiting_time)
|
|
{
|
|
qse_ntime_t dead_line, now, rem, zero;
|
|
|
|
if (waiting_time)
|
|
{
|
|
qse_clear_ntime (&zero);
|
|
qse_gettime (&now);
|
|
qse_add_ntime (&dead_line, &now, waiting_time);
|
|
}
|
|
if (qse_mtx_lock (&rwl->mtx, waiting_time) <= -1) return -1;
|
|
|
|
if (rwl->wactive_count > 0 || rwl->ractive_count > 0)
|
|
{
|
|
rwl->wwait_count++;
|
|
while (rwl->wactive_count > 0 || rwl->ractive_count > 0)
|
|
{
|
|
if (waiting_time)
|
|
{
|
|
qse_gettime (&now);
|
|
qse_sub_ntime (&rem, &dead_line, &now);
|
|
if (qse_cmp_ntime(&rem, &zero) <= 0)
|
|
{
|
|
/* timed out */
|
|
rwl->wwait_count--;
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
return -1;
|
|
}
|
|
qse_cnd_wait (&rwl->wcnd, &rwl->mtx, &rem);
|
|
}
|
|
else
|
|
{
|
|
qse_cnd_wait (&rwl->wcnd, &rwl->mtx, QSE_NULL);
|
|
}
|
|
}
|
|
rwl->wwait_count--;
|
|
}
|
|
|
|
rwl->wactive_count++;
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
return 0;
|
|
}
|
|
|
|
int qse_rwl_unlockw (qse_rwl_t* rwl)
|
|
{
|
|
if (qse_mtx_lock (&rwl->mtx, QSE_NULL) <= -1) return -1;
|
|
|
|
|
|
if (rwl->wactive_count <= 0)
|
|
{
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
return -1;
|
|
}
|
|
|
|
rwl->wactive_count--;
|
|
|
|
#if 0
|
|
if (rwl->flags & QSE_RWL_PREFER_WRITER)
|
|
{
|
|
if (rwl->wwait_count > 0)
|
|
{
|
|
qse_cnd_signal (&rwl->wcnd);
|
|
}
|
|
else if (rwl->rwait_count > 0)
|
|
{
|
|
qse_cnd_broadcast (&rwl->rcnd);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#endif
|
|
if (rwl->rwait_count > 0)
|
|
{
|
|
qse_cnd_broadcast (&rwl->rcnd);
|
|
}
|
|
/*else*/ if (rwl->wwait_count > 0)
|
|
{
|
|
qse_cnd_signal (&rwl->wcnd);
|
|
}
|
|
#if 0
|
|
}
|
|
#endif
|
|
qse_mtx_unlock (&rwl->mtx);
|
|
return 0;
|
|
}
|