#include <pthread.h>
#include <time.h>
#include "xsem.h"
int
xsem_init(xsem_t *sem, int pshared, unsigned int value)
{
if (pshared != 0)
return (-1);
pthread_mutex_init(&sem->semMutex, NULL);
pthread_cond_init(&sem->semCV, NULL);
sem->semaphore = value;
return (0);
}
void
xsem_destroy(xsem_t *sem)
{
pthread_mutex_destroy(&sem->semMutex);
pthread_cond_destroy(&sem->semCV);
sem->semaphore = 0;
}
int
xsem_wait(xsem_t *sem)
{
pthread_mutex_lock(&sem->semMutex);
if (sem->semaphore < 0) {
sem->semaphore = 0;
pthread_mutex_unlock(&sem->semMutex);
return (XSEM_ERROR);
}
if (sem->semaphore > 0) {
sem->semaphore--;
} else {
while (sem->semaphore == 0)
pthread_cond_wait(&sem->semCV, &sem->semMutex);
if (sem->semaphore != 0) {
sem->semaphore--;
} else {
pthread_mutex_unlock(&sem->semMutex);
return (XSEM_ERROR);
}
}
pthread_mutex_unlock(&sem->semMutex);
return (0);
}
int
xsem_trywait(xsem_t *sem)
{
pthread_mutex_lock(&sem->semMutex);
if (sem->semaphore < 0) {
sem->semaphore = 0;
pthread_mutex_unlock(&sem->semMutex);
return (XSEM_ERROR);
}
if (sem->semaphore == 0) {
pthread_mutex_unlock(&sem->semMutex);
return (XSEM_EBUSY);
} else {
sem->semaphore--;
}
pthread_mutex_unlock(&sem->semMutex);
return (0);
}
int
xsem_post(xsem_t *sem)
{
pthread_mutex_lock(&sem->semMutex);
sem->semaphore++;
pthread_cond_signal(&sem->semCV);
pthread_mutex_unlock(&sem->semMutex);
return (0);
}
void
xsem_getvalue(xsem_t *sem, int *sval)
{
*sval = sem->semaphore;
}
int
xsem_xwait(xsem_t *sem, int timeout, timestruc_t *mytime)
{
int status;
timestruc_t delay;
if (timeout == 0)
return (xsem_wait(sem));
else {
pthread_mutex_lock(&sem->semMutex);
if (sem->semaphore < 0) {
sem->semaphore = 0;
pthread_mutex_unlock(&sem->semMutex);
return (XSEM_ERROR);
}
if (sem->semaphore > 0) {
sem->semaphore--;
} else {
status = 0;
delay = *mytime;
delay.tv_sec = delay.tv_sec + time(NULL);
while ((sem->semaphore == 0) && (status == 0)) {
status = pthread_cond_timedwait(&sem->semCV,
&sem->semMutex, &delay);
}
if (status != 0) {
pthread_mutex_unlock(&sem->semMutex);
return (XSEM_ETIME);
} else if (sem->semaphore != 0) {
sem->semaphore--;
} else {
pthread_mutex_unlock(&sem->semMutex);
return (XSEM_ERROR);
}
}
pthread_mutex_unlock(&sem->semMutex);
}
return (0);
}