www.pudn.com > fsm.rar > port_fifo.c
/* The oSIP library implements the Session Initiation Protocol (SIP -rfc2543-) Copyright (C) 2001 Aymeric MOIZARD jack@atosc.org This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include#include #include #include /* always use this method to initiate fifo_t. */ void fifo_init(fifo_t *ff) { #ifdef OSIP_MT ff->qislocked = smutex_init(); /*INIT SEMA TO BLOCK ON GET() WHEN QUEUE IS EMPTY */ ff->qisempty = ssem_init( 0 ); #endif ff->queue = (list_t *)smalloc(sizeof(list_t)); list_init(ff->queue); /* ff->nb_elt = 0; */ ff->etat = vide; } int fifo_add(fifo_t *ff , void *el) { #ifdef OSIP_MT smutex_lock(ff->qislocked); #endif if (ff->etat != plein) { /* ff->nb_elt++; */ list_add(ff->queue,el,-1); /* insert at end of queue */ } else { fprintf(stdout," (fifo_t:%x) WARNING STACK IS OVERLOADED\n",(int)ff); #ifdef OSIP_MT smutex_unlock(ff->qislocked); #endif return -1; /* stack is full */ } /* if (ff->nb_elt >= MAX_LEN) */ if (list_size(ff->queue) >= MAX_LEN) ff->etat = plein; else ff->etat = ok; #ifdef OSIP_MT ssem_post(ff->qisempty); smutex_unlock(ff->qislocked); #endif return 0; } #ifdef OSIP_MT void* fifo_get(fifo_t *ff) { void *el; ssem_wait(ff->qisempty); smutex_lock(ff->qislocked); if (ff->etat != vide) { el = list_get(ff->queue,0); list_remove(ff->queue,0); /* ff->nb_elt--; */ } else { fprintf(stdout," (fifo_t:%x) No element available?\n",(int)ff); smutex_unlock(ff->qislocked); return 0; /* pile vide */ } /* if (ff->nb_elt <= 0) */ if (list_size(ff->queue) <= 0) ff->etat = vide; else ff->etat = ok; smutex_unlock(ff->qislocked); return el; } #endif void * fifo_tryget(fifo_t *ff) { void *el; #ifdef OSIP_MT if (0>ssem_trywait(ff->qisempty)) {/* no elements... */ return NULL; } smutex_lock(ff->qislocked); #else if (ff->etat == vide) return NULL; #endif if (ff->etat != vide) { el = list_get(ff->queue,0); list_remove(ff->queue,0); /* ff->nb_elt--; */ } #ifdef OSIP_MT else { /* this case MUST never happen... */ fprintf(stdout," (fifo_t:%x) No element available?\n",(int)ff); smutex_unlock(ff->qislocked); return 0; } #endif /* if (ff->nb_elt <= 0) */ if (list_size(ff->queue) <= 0) ff->etat = vide; else ff->etat = ok; #ifdef OSIP_MT smutex_unlock(ff->qislocked); #endif return el; } void fifo_free(fifo_t *ff) { #ifdef OSIP_MT smutex_destroy(ff->qislocked); #ifndef __VXWORKS_OS__ sfree(ff->qislocked); #endif /* seems that pthread_mutex_destroy does not free space by itself */ ssem_destroy(ff->qisempty); #ifndef __VXWORKS_OS__ sfree(ff->qisempty); #endif #endif sfree(ff->queue); }