Skip to content
Snippets Groups Projects
channel.c 34.5 KiB
Newer Older
  • Learn to ignore specific revisions
  • Mark Spencer's avatar
    Mark Spencer committed
    /*
     * Asterisk -- A telephony toolkit for Linux.
     *
     * Channel Management
     * 
    
    Mark Spencer's avatar
    Mark Spencer committed
     * Copyright (C) 1999, Mark Spencer
    
    Mark Spencer's avatar
    Mark Spencer committed
     *
     * Mark Spencer <markster@linux-support.net>
     *
     * This program is free software, distributed under the terms of
     * the GNU General Public License
     */
    
    #include <stdio.h>
    #include <stdlib.h>
    #include <pthread.h>
    #include <string.h>
    #include <sys/time.h>
    #include <signal.h>
    #include <errno.h>
    
    Mark Spencer's avatar
    Mark Spencer committed
    #include <unistd.h>
    
    Mark Spencer's avatar
    Mark Spencer committed
    #include <asterisk/frame.h>
    
    Mark Spencer's avatar
    Mark Spencer committed
    #include <asterisk/sched.h>
    #include <asterisk/options.h>
    #include <asterisk/channel.h>
    #include <asterisk/channel_pvt.h>
    #include <asterisk/logger.h>
    #include <asterisk/file.h>
    
    Mark Spencer's avatar
    Mark Spencer committed
    #include <asterisk/translate.h>
    
    Mark Spencer's avatar
    Mark Spencer committed
    /* XXX Lock appropriately in more functions XXX */
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    #ifdef DEBUG_MUTEX
    /* Convenient mutex debugging functions */
    #define PTHREAD_MUTEX_LOCK(a) __PTHREAD_MUTEX_LOCK(__FUNCTION__, a)
    #define PTHREAD_MUTEX_UNLOCK(a) __PTHREAD_MUTEX_UNLOCK(__FUNCTION__, a)
    
    static int __PTHREAD_MUTEX_LOCK(char *f, pthread_mutex_t *a) {
    	ast_log(LOG_DEBUG, "Locking %p (%s)\n", a, f); 
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return ast_pthread_mutex_lock(a);
    
    Mark Spencer's avatar
    Mark Spencer committed
    }
    
    static int __PTHREAD_MUTEX_UNLOCK(char *f, pthread_mutex_t *a) {
    	ast_log(LOG_DEBUG, "Unlocking %p (%s)\n", a, f); 
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return ast_pthread_mutex_unlock(a);
    
    Mark Spencer's avatar
    Mark Spencer committed
    }
    #else
    
    Mark Spencer's avatar
    Mark Spencer committed
    #define PTHREAD_MUTEX_LOCK(a) ast_pthread_mutex_lock(a)
    #define PTHREAD_MUTEX_UNLOCK(a) ast_pthread_mutex_unlock(a)
    
    Mark Spencer's avatar
    Mark Spencer committed
    #endif
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct chanlist {
    	char type[80];
    	char description[80];
    	int capabilities;
    	struct ast_channel * (*requester)(char *type, int format, void *data);
    	struct chanlist *next;
    } *backends = NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_channel *channels = NULL;
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    /* Protect the channel list (highly unlikely that two things would change
       it at the same time, but still! */
       
    static pthread_mutex_t chlock = PTHREAD_MUTEX_INITIALIZER;
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_check_hangup(struct ast_channel *chan)
    {
    time_t	myt;
    
    	  /* if soft hangup flag, return true */
    	if (chan->softhangup) return 1;
    	  /* if no hangup scheduled, just return here */
    	if (!chan->whentohangup) return 0;
    	time(&myt); /* get current time */
    	  /* return, if not yet */
    	if (chan->whentohangup > myt) return 0;
    	chan->softhangup = 1;
    	return 1;
    }
    
    void ast_channel_setwhentohangup(struct ast_channel *chan, time_t offset)
    {
    time_t	myt;
    
    	time(&myt);
    	chan->whentohangup = myt + offset;
    	return;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_channel_register(char *type, char *description, int capabilities,
    		struct ast_channel *(*requester)(char *type, int format, void *data))
    {
    	struct chanlist *chan, *last=NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (PTHREAD_MUTEX_LOCK(&chlock)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_WARNING, "Unable to lock channel list\n");
    		return -1;
    	}
    	chan = backends;
    	while(chan) {
    		if (!strcasecmp(type, chan->type)) {
    			ast_log(LOG_WARNING, "Already have a handler for type '%s'\n", type);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			return -1;
    		}
    		last = chan;
    		chan = chan->next;
    	}
    	chan = malloc(sizeof(struct chanlist));
    	if (!chan) {
    		ast_log(LOG_WARNING, "Out of memory\n");
    
    Mark Spencer's avatar
    Mark Spencer committed
    		PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	strncpy(chan->type, type, sizeof(chan->type)-1);
    	strncpy(chan->description, description, sizeof(chan->description)-1);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	chan->capabilities = capabilities;
    	chan->requester = requester;
    	chan->next = NULL;
    	if (last)
    		last->next = chan;
    	else
    		backends = chan;
    	if (option_debug)
    		ast_log(LOG_DEBUG, "Registered handler for '%s' (%s)\n", chan->type, chan->description);
    	else if (option_verbose > 1)
    		ast_verbose( VERBOSE_PREFIX_2 "Registered channel type '%s' (%s)\n", chan->type, chan->description);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return 0;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    char *ast_state2str(int state)
    {
    	switch(state) {
    	case AST_STATE_DOWN:
    		return "Down";
    	case AST_STATE_RESERVED:
    		return "Rsrvd";
    	case AST_STATE_OFFHOOK:
    		return "OffHook";
    	case AST_STATE_DIALING:
    		return "Dialing";
    	case AST_STATE_RING:
    		return "Ring";
    	case AST_STATE_RINGING:
    		return "Ringing";
    	case AST_STATE_UP:
    		return "Up";
    	case AST_STATE_BUSY:
    		return "Busy";
    	default:
    		return "Unknown";
    	}
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    int ast_best_codec(int fmts)
    {
    	/* This just our opinion, expressed in code.  We are asked to choose
    	   the best codec to use, given no information */
    	int x;
    	static int prefs[] = 
    	{
    		/* Okay, ulaw is used by all telephony equipment, so start with it */
    		AST_FORMAT_ULAW,
    		/* Unless of course, you're a silly European, so then prefer ALAW */
    		AST_FORMAT_ALAW,
    		/* Okay, well, signed linear is easy to translate into other stuff */
    		AST_FORMAT_SLINEAR,
    		/* ADPCM has great sound quality and is still pretty easy to translate */
    		AST_FORMAT_ADPCM,
    		/* Okay, we're down to vocoders now, so pick GSM because it's small and easier to
    		   translate and sounds pretty good */
    		AST_FORMAT_GSM,
    		/* Ick, LPC10 sounds terrible, but at least we have code for it, if you're tacky enough
    		   to use it */
    		AST_FORMAT_LPC10,
    		/* Down to G.723.1 which is proprietary but at least designed for voice */
    		AST_FORMAT_G723_1,
    		/* Last and least, MP3 which was of course never designed for real-time voice */
    		AST_FORMAT_MP3,
    	};
    	
    	
    	for (x=0;x<sizeof(prefs) / sizeof(prefs[0]); x++)
    		if (fmts & prefs[x])
    			return prefs[x];
    	ast_log(LOG_WARNING, "Don't know any of 0x%x formats\n", fmts);
    	return 0;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_channel *ast_channel_alloc(void)
    {
    	struct ast_channel *tmp;
    	struct ast_channel_pvt *pvt;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int x;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_LOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	tmp = malloc(sizeof(struct ast_channel));
    	memset(tmp, 0, sizeof(struct ast_channel));
    	if (tmp) {
    		pvt = malloc(sizeof(struct ast_channel_pvt));
    		if (pvt) {
    			memset(pvt, 0, sizeof(struct ast_channel_pvt));
    			tmp->sched = sched_context_create();
    			if (tmp->sched) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    				for (x=0;x<AST_MAX_FDS;x++)
    					tmp->fds[x] = -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    				strncpy(tmp->name, "**Unknown**", sizeof(tmp->name)-1);
    
    Mark Spencer's avatar
    Mark Spencer committed
    				tmp->pvt = pvt;
    				tmp->state = AST_STATE_DOWN;
    				tmp->stack = -1;
    				tmp->streamid = -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    				tmp->appl = NULL;
    				tmp->data = NULL;
    				pthread_mutex_init(&tmp->lock, NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    				strncpy(tmp->context, "default", sizeof(tmp->context)-1);
    				strncpy(tmp->language, defaultlanguage, sizeof(tmp->language)-1);
    				strncpy(tmp->exten, "s", sizeof(tmp->exten)-1);
    
    Mark Spencer's avatar
    Mark Spencer committed
    				tmp->priority=1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    				tmp->amaflags = ast_default_amaflags;
    				strncpy(tmp->accountcode, ast_default_accountcode, sizeof(tmp->accountcode)-1);
    
    Mark Spencer's avatar
    Mark Spencer committed
    				tmp->next = channels;
    				channels= tmp;
    
    Mark Spencer's avatar
    Mark Spencer committed
    			} else {
    				ast_log(LOG_WARNING, "Unable to create schedule context\n");
    				free(tmp);
    				tmp = NULL;
    			}
    		} else {
    			ast_log(LOG_WARNING, "Out of memory\n");
    			free(tmp);
    			tmp = NULL;
    		}
    	} else 
    		ast_log(LOG_WARNING, "Out of memory\n");
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return tmp;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_channel_defer_dtmf(struct ast_channel *chan)
    {
    	int pre = 0;
    	if (chan) {
    		pre = chan->deferdtmf;
    		chan->deferdtmf = 1;
    	}
    	return pre;
    }
    
    void ast_channel_undefer_dtmf(struct ast_channel *chan)
    {
    	if (chan)
    		chan->deferdtmf = 0;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_channel *ast_channel_walk(struct ast_channel *prev)
    {
    	struct ast_channel *l, *ret=NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_LOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	l = channels;
    	if (!prev) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return l;
    	}
    	while(l) {
    		if (l == prev)
    			ret = l->next;
    		l = l->next;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return ret;
    	
    }
    
    void ast_channel_free(struct ast_channel *chan)
    {
    	struct ast_channel *last=NULL, *cur;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_LOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	cur = channels;
    	while(cur) {
    		if (cur == chan) {
    			if (last)
    				last->next = cur->next;
    			else
    				channels = cur->next;
    			break;
    		}
    		last = cur;
    		cur = cur->next;
    	}
    	if (!cur)
    		ast_log(LOG_WARNING, "Unable to find channel in list\n");
    	if (chan->pvt->pvt)
    		ast_log(LOG_WARNING, "Channel '%s' may not have been hung up properly\n", chan->name);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Free translatosr */
    	if (chan->pvt->readtrans)
    		ast_translator_free_path(chan->pvt->readtrans);
    	if (chan->pvt->writetrans)
    		ast_translator_free_path(chan->pvt->writetrans);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->pbx) 
    		ast_log(LOG_WARNING, "PBX may not have been terminated properly on '%s'\n", chan->name);
    	if (chan->dnid)
    		free(chan->dnid);
    	if (chan->callerid)
    		free(chan->callerid);	
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->ani)
    		free(chan->ani);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	pthread_mutex_destroy(&chan->lock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	free(chan->pvt);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	free(chan);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_softhangup(struct ast_channel *chan)
    {
    	int res = 0;
    	if (option_debug)
    		ast_log(LOG_DEBUG, "Soft-Hanging up channel '%s'\n", chan->name);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Inform channel driver that we need to be hung up, if it cares */
    	chan->softhangup = 1;		
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Interrupt any select call or such */
    	if (chan->blocking)
    		pthread_kill(chan->blocker, SIGURG);
    	return res;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    static void free_translation(struct ast_channel *clone)
    {
    	if (clone->pvt->writetrans)
    		ast_translator_free_path(clone->pvt->writetrans);
    	if (clone->pvt->readtrans)
    		ast_translator_free_path(clone->pvt->readtrans);
    	clone->pvt->writetrans = NULL;
    	clone->pvt->readtrans = NULL;
    	clone->pvt->rawwriteformat = clone->nativeformats;
    	clone->pvt->rawreadformat = clone->nativeformats;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_hangup(struct ast_channel *chan)
    {
    	int res = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Don't actually hang up a channel that will masquerade as someone else, or
    	   if someone is going to masquerade as us */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	ast_pthread_mutex_lock(&chan->lock);
    	if (chan->masq) {
    		ast_log(LOG_WARNING, "We're getting hung up, but someone is trying to masq into us?!?\n");
    		ast_pthread_mutex_unlock(&chan->lock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* If this channel is one which will be masqueraded into something, 
    	   mark it as a zombie already, so we know to free it later */
    	if (chan->masqr) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_pthread_mutex_unlock(&chan->lock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		chan->zombie=1;
    		return 0;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	free_translation(chan);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->stream)
    		ast_stopstream(chan);
    	if (chan->sched)
    		sched_context_destroy(chan->sched);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->cdr) {
    		/* End the CDR if it hasn't already */
    		ast_cdr_end(chan->cdr);
    		/* Post and Free the CDR */
    		ast_cdr_post(chan->cdr);
    		ast_cdr_free(chan->cdr);
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->blocking) {
    		ast_log(LOG_WARNING, "Hard hangup called by thread %ld on %s, while fd "
    					"is blocked by thread %ld in procedure %s!  Expect a failure\n",
    					pthread_self(), chan->name, chan->blocker, chan->blockproc);
    		CRASH;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!chan->zombie) {
    		if (option_debug)
    			ast_log(LOG_DEBUG, "Hanging up channel '%s'\n", chan->name);
    		if (chan->pvt->hangup)
    			res = chan->pvt->hangup(chan);
    	} else
    		if (option_debug)
    			ast_log(LOG_DEBUG, "Hanging up zombie '%s'\n", chan->name);
    			
    
    Mark Spencer's avatar
    Mark Spencer committed
    	ast_pthread_mutex_unlock(&chan->lock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	ast_channel_free(chan);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return res;
    }
    
    void ast_channel_unregister(char *type)
    {
    	struct chanlist *chan, *last=NULL;
    	if (option_debug)
    		ast_log(LOG_DEBUG, "Unregistering channel type '%s'\n", type);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (PTHREAD_MUTEX_LOCK(&chlock)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_WARNING, "Unable to lock channel list\n");
    		return;
    	}
    	chan = backends;
    	while(chan) {
    		if (!strcasecmp(chan->type, type)) {
    			if (last)
    				last->next = chan->next;
    			else
    				backends = backends->next;
    			free(chan);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			return;
    		}
    		last = chan;
    		chan = chan->next;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    }
    
    int ast_answer(struct ast_channel *chan)
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int res = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Stop if we're a zombie or need a soft hangup */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->zombie || ast_check_hangup(chan)) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    	switch(chan->state) {
    	case AST_STATE_RINGING:
    
    Mark Spencer's avatar
    Mark Spencer committed
    	case AST_STATE_RING:
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (chan->pvt->answer)
    
    Mark Spencer's avatar
    Mark Spencer committed
    			res = chan->pvt->answer(chan);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		chan->state = AST_STATE_UP;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (chan->cdr)
    			ast_cdr_answer(chan->cdr);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return res;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		break;
    	case AST_STATE_UP:
    		break;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    	return 0;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_waitfor_n_fd(int *fds, int n, int *ms, int *exception)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	/* Wait for x amount of time on a file descriptor to have input.  */
    	struct timeval tv;
    	fd_set rfds, efds;
    	int res;
    	int x, max=-1;
    	int winner = -1;
    	
    	tv.tv_sec = *ms / 1000;
    	tv.tv_usec = (*ms % 1000) * 1000;
    	FD_ZERO(&rfds);
    	FD_ZERO(&efds);
    	for (x=0;x<n;x++) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (fds[x] > -1) {
    			FD_SET(fds[x], &rfds);
    			FD_SET(fds[x], &efds);
    			if (fds[x] > max)
    				max = fds[x];
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    	if (*ms >= 0) 
    		res = select(max + 1, &rfds, NULL, &efds, &tv);
    	else
    		res = select(max + 1, &rfds, NULL, &efds, NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    	if (res < 0) {
    		/* Simulate a timeout if we were interrupted */
    		if (errno != EINTR)
    			*ms = -1;
    		else
    			*ms = 0;
    		return -1;
    	}
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    	for (x=0;x<n;x++) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if ((fds[x] > -1) && (FD_ISSET(fds[x], &rfds) || FD_ISSET(fds[x], &efds)) && (winner < 0)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    			if (exception)
    				*exception = FD_ISSET(fds[x], &efds);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			winner = fds[x];
    
    Mark Spencer's avatar
    Mark Spencer committed
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    	*ms = tv.tv_sec * 1000 + tv.tv_usec / 1000;
    	return winner;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    static int ast_do_masquerade(struct ast_channel *original);
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_channel *ast_waitfor_nandfds(struct ast_channel **c, int n, int *fds, int nfds, 
    	int *exception, int *outfd, int *ms)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	/* Wait for x amount of time on a file descriptor to have input.  */
    	struct timeval tv;
    	fd_set rfds, efds;
    	int res;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int x, y, max=-1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	struct ast_channel *winner = NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (outfd)
    		*outfd = -1;
    	if (exception)
    		*exception = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Perform any pending masquerades */
    	for (x=0;x<n;x++) {
    		if (c[x]->masq) {
    			if (ast_do_masquerade(c[x])) {
    				ast_log(LOG_WARNING, "Masquerade failed\n");
    				*ms = -1;
    				return NULL;
    			}
    		}
    	}
    	
    
    Mark Spencer's avatar
    Mark Spencer committed
    	tv.tv_sec = *ms / 1000;
    	tv.tv_usec = (*ms % 1000) * 1000;
    	FD_ZERO(&rfds);
    	FD_ZERO(&efds);
    	for (x=0;x<n;x++) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		for (y=0;y<AST_MAX_FDS;y++) {
    			if (c[x]->fds[y] > 0) {
    				FD_SET(c[x]->fds[y], &rfds);
    				FD_SET(c[x]->fds[y], &efds);
    				if (c[x]->fds[y] > max)
    					max = c[x]->fds[y];
    			}
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    		CHECK_BLOCKING(c[x]);
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	for (x=0;x<nfds; x++) {
    		FD_SET(fds[x], &rfds);
    		FD_SET(fds[x], &efds);
    		if (fds[x] > max)
    			max = fds[x];
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (*ms >= 0) 
    		res = select(max + 1, &rfds, NULL, &efds, &tv);
    	else
    		res = select(max + 1, &rfds, NULL, &efds, NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    	if (res < 0) {
    		for (x=0;x<n;x++) 
    			c[x]->blocking = 0;
    		/* Simulate a timeout if we were interrupted */
    		if (errno != EINTR)
    			*ms = -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		else {
    			/* Just an interrupt */
    #if 0
    
    Mark Spencer's avatar
    Mark Spencer committed
    			*ms = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    #endif			
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return NULL;
    	}
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    	for (x=0;x<n;x++) {
    		c[x]->blocking = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		for (y=0;y<AST_MAX_FDS;y++) {
    			if (c[x]->fds[y] > -1) {
    				if ((FD_ISSET(c[x]->fds[y], &rfds) || FD_ISSET(c[x]->fds[y], &efds)) && !winner) {
    					/* Set exception flag if appropriate */
    					if (FD_ISSET(c[x]->fds[y], &efds))
    						c[x]->exception = 1;
    					c[x]->fdno = y;
    					winner = c[x];
    				}
    			}
    
    Mark Spencer's avatar
    Mark Spencer committed
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	for (x=0;x<nfds;x++) {
    		if ((FD_ISSET(fds[x], &rfds) || FD_ISSET(fds[x], &efds)) && !winner) {
    			if (outfd)
    				*outfd = fds[x];
    			if (FD_ISSET(fds[x], &efds) && exception)
    				*exception = 1;
    			winner = NULL;
    		}
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	*ms = tv.tv_sec * 1000 + tv.tv_usec / 1000;
    	return winner;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_channel *ast_waitfor_n(struct ast_channel **c, int n, int *ms)
    {
    	return ast_waitfor_nandfds(c, n, NULL, 0, NULL, NULL, ms);
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_waitfor(struct ast_channel *c, int ms)
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	struct ast_channel *chan;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int oldms = ms;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	chan = ast_waitfor_n(&c, 1, &ms);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (ms < 0) {
    		if (oldms < 0)
    			return 0;
    		else
    			return -1;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return ms;
    
    Mark Spencer's avatar
    Mark Spencer committed
    }
    
    char ast_waitfordigit(struct ast_channel *c, int ms)
    {
    	struct ast_frame *f;
    	char result = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Stop if we're a zombie or need a soft hangup */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (c->zombie || ast_check_hangup(c)) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Wait for a digit, no more than ms milliseconds total. */
    	while(ms && !result) {
    		ms = ast_waitfor(c, ms);
    		if (ms < 0) /* Error */
    			result = -1; 
    		else if (ms > 0) {
    			/* Read something */
    			f = ast_read(c);
    			if (f) {
    				if (f->frametype == AST_FRAME_DTMF) 
    					result = f->subclass;
    				ast_frfree(f);
    			} else
    				result = -1;
    		}
    	}
    	return result;
    }
    
    struct ast_frame *ast_read(struct ast_channel *chan)
    {
    	struct ast_frame *f = NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	static struct ast_frame null_frame = 
    	{
    		AST_FRAME_NULL,
    	};
    
    Mark Spencer's avatar
    Mark Spencer committed
    	
    	pthread_mutex_lock(&chan->lock);
    	if (chan->masq) {
    		if (ast_do_masquerade(chan)) {
    			ast_log(LOG_WARNING, "Failed to perform masquerade\n");
    			f = NULL;
    		} else
    			f =  &null_frame;
    		pthread_mutex_unlock(&chan->lock);
    		return f;
    	}
    
    	/* Stop if we're a zombie or need a soft hangup */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->zombie || ast_check_hangup(chan)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		pthread_mutex_unlock(&chan->lock);
    		return NULL;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	
    	if (!chan->deferdtmf && strlen(chan->dtmfq)) {
    		/* We have DTMF that has been deferred.  Return it now */
    		chan->dtmff.frametype = AST_FRAME_DTMF;
    		chan->dtmff.subclass = chan->dtmfq[0];
    		/* Drop first digit */
    		memmove(chan->dtmfq, chan->dtmfq + 1, sizeof(chan->dtmfq) - 1);
    		pthread_mutex_unlock(&chan->lock);
    		return &chan->dtmff;
    	}
    	
    
    Mark Spencer's avatar
    Mark Spencer committed
    	chan->blocker = pthread_self();
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->exception) {
    		if (chan->pvt->exception) 
    			f = chan->pvt->exception(chan);
    		else
    			ast_log(LOG_WARNING, "Exception flag set, but no exception handler\n");
    		/* Clear the exception flag */
    		chan->exception = 0;
    	} else
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->pvt->read)
    		f = chan->pvt->read(chan);
    	else
    		ast_log(LOG_WARNING, "No read routine on channel %s\n", chan);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (f && (f->frametype == AST_FRAME_VOICE)) {
    		if (chan->pvt->readtrans) {
    			f = ast_translate(chan->pvt->readtrans, f, 1);
    			if (!f)
    				f = &null_frame;
    		}
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Make sure we always return NULL in the future */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!f) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		chan->softhangup = 1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		/* End the CDR if appropriate */
    		if (chan->cdr)
    			ast_cdr_end(chan->cdr);
    	} else if (chan->deferdtmf && f->frametype == AST_FRAME_DTMF) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (strlen(chan->dtmfq) < sizeof(chan->dtmfq) - 2)
    			chan->dtmfq[strlen(chan->dtmfq)] = f->subclass;
    		else
    			ast_log(LOG_WARNING, "Dropping deferred DTMF digits on %s\n", chan->name);
    		f = &null_frame;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	} else if ((f->frametype == AST_FRAME_CONTROL) && (f->subclass == AST_CONTROL_ANSWER)) {
    		/* Answer the CDR */
    		ast_cdr_answer(chan->cdr);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	pthread_mutex_unlock(&chan->lock);
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return f;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_indicate(struct ast_channel *chan, int condition)
    {
    	int res = -1;
    	/* Stop if we're a zombie or need a soft hangup */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->zombie || ast_check_hangup(chan)) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    	if (chan->pvt->indicate) {
    		res = chan->pvt->indicate(chan, condition);
    		if (res)
    			ast_log(LOG_WARNING, "Driver for channel '%s' failed to indicate condition %d\n", chan->name, condition);
    	} else
    		ast_log(LOG_WARNING, "Driver for channel '%s' does not support indication\n", chan->name);
    	return res;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_recvchar(struct ast_channel *chan, int timeout)
    {
    	int res,ourto,c;
    	struct ast_frame *f;
    	
    	ourto = timeout;
    	for(;;)
    	   {
    		if (ast_check_hangup(chan)) return -1;
    		res = ast_waitfor(chan,ourto);
    		if (res <= 0) /* if timeout */
    		   {
    			return 0;
    		   }
    		ourto = res;
    		f = ast_read(chan);
    		if (f == NULL) return -1; /* if hangup */
    		if ((f->frametype == AST_FRAME_CONTROL) &&
    		    (f->subclass == AST_CONTROL_HANGUP)) return -1; /* if hangup */
    		if (f->frametype == AST_FRAME_TEXT)  /* if a text frame */
    		   {
    			c = *((char *)f->data);  /* get the data */
    			ast_frfree(f);
    			return(c);
    		   }
    		ast_frfree(f);
    	}
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_sendtext(struct ast_channel *chan, char *text)
    {
    	int res = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Stop if we're a zombie or need a soft hangup */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->zombie || ast_check_hangup(chan)) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	CHECK_BLOCKING(chan);
    	if (chan->pvt->send_text)
    		res = chan->pvt->send_text(chan, text);
    	chan->blocking = 0;
    	return res;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_write(struct ast_channel *chan, struct ast_frame *fr)
    {
    	int res = -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	struct ast_frame *f;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Stop if we're a zombie or need a soft hangup */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (chan->zombie || ast_check_hangup(chan)) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    	/* Handle any pending masquerades */
    	if (chan->masq) {
    		if (ast_do_masquerade(chan)) {
    			ast_log(LOG_WARNING, "Failed to perform masquerade\n");
    			return -1;
    		}
    	}
    	if (chan->masqr)
    		return 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	CHECK_BLOCKING(chan);
    	switch(fr->frametype) {
    	case AST_FRAME_CONTROL:
    		/* XXX Interpret control frames XXX */
    		ast_log(LOG_WARNING, "Don't know how to handle control frames yet\n");
    		break;
    	case AST_FRAME_DTMF:
    		if (chan->pvt->send_digit)
    			res = chan->pvt->send_digit(chan, fr->subclass);
    		break;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	case AST_FRAME_TEXT:
    		if (chan->pvt->send_text)
    			res = chan->pvt->send_text(chan, (char *) fr->data);
    		break;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	default:
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (chan->pvt->write) {
    			if (chan->pvt->writetrans) {
    				f = ast_translate(chan->pvt->writetrans, fr, 1);
    			} else
    				f = fr;
    			if (f)	
    				res = chan->pvt->write(chan, f);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			else
    				res = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    	chan->blocking = 0;
    	return res;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_set_write_format(struct ast_channel *chan, int fmts)
    {
    	int fmt;
    	int native;
    	int res;
    	
    	native = chan->nativeformats;
    	fmt = fmts;
    	
    	res = ast_translator_best_choice(&native, &fmt);
    	if (res < 0) {
    		ast_log(LOG_NOTICE, "Unable to find a path from %d to %d\n", fmts, chan->nativeformats);
    		return -1;
    	}
    	
    	/* Now we have a good choice for both.  We'll write using our native format. */
    	chan->pvt->rawwriteformat = native;
    	/* User perspective is fmt */
    	chan->writeformat = fmt;
    	/* Free any write translation we have right now */
    	if (chan->pvt->writetrans)
    		ast_translator_free_path(chan->pvt->writetrans);
    	/* Build a translation path from the user write format to the raw writing format */
    	chan->pvt->writetrans = ast_translator_build_path(chan->pvt->rawwriteformat, chan->writeformat);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (option_debug)
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_DEBUG, "Set channel %s to write format %d\n", chan->name, chan->writeformat);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return 0;
    }
    
    int ast_set_read_format(struct ast_channel *chan, int fmts)
    {
    	int fmt;
    	int native;
    	int res;
    	
    	native = chan->nativeformats;
    	fmt = fmts;
    	/* Find a translation path from the native read format to one of the user's read formats */
    	res = ast_translator_best_choice(&fmt, &native);
    	if (res < 0) {
    		ast_log(LOG_NOTICE, "Unable to find a path from %d to %d\n", chan->nativeformats, fmts);
    		return -1;
    	}
    	
    	/* Now we have a good choice for both.  We'll write using our native format. */
    	chan->pvt->rawreadformat = native;
    	/* User perspective is fmt */
    	chan->readformat = fmt;
    	/* Free any read translation we have right now */
    	if (chan->pvt->readtrans)
    		ast_translator_free_path(chan->pvt->readtrans);
    	/* Build a translation path from the raw read format to the user reading format */
    	chan->pvt->readtrans = ast_translator_build_path(chan->readformat, chan->pvt->rawreadformat);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (option_debug)
    		ast_log(LOG_DEBUG, "Set channel %s to read format %d\n", chan->name, chan->readformat);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return 0;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_channel *ast_request(char *type, int format, void *data)
    {
    	struct chanlist *chan;
    	struct ast_channel *c = NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int capabilities;
    	int fmt;
    	int res;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (PTHREAD_MUTEX_LOCK(&chlock)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_log(LOG_WARNING, "Unable to lock channel list\n");
    		return NULL;
    	}
    	chan = backends;
    	while(chan) {
    		if (!strcasecmp(type, chan->type)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    			capabilities = chan->capabilities;
    			fmt = format;
    			res = ast_translator_best_choice(&fmt, &capabilities);
    			if (res < 0) {
    				ast_log(LOG_WARNING, "No translator path exists for channel type %s (native %d) to %d\n", type, chan->capabilities, format);
    				PTHREAD_MUTEX_UNLOCK(&chlock);
    				return NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    			}
    
    Mark Spencer's avatar
    Mark Spencer committed
    			PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			if (chan->requester)
    
    Mark Spencer's avatar
    Mark Spencer committed
    				c = chan->requester(type, capabilities, data);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			return c;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		}
    		chan = chan->next;
    	}
    	if (!chan)
    		ast_log(LOG_WARNING, "No channel type registered for '%s'\n", type);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	PTHREAD_MUTEX_UNLOCK(&chlock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return c;
    }
    
    int ast_call(struct ast_channel *chan, char *addr, int timeout) 
    {
    	/* Place an outgoing call, but don't wait any longer than timeout ms before returning. 
    	   If the remote end does not answer within the timeout, then do NOT hang up, but 
    	   return anyway.  */
    	int res = -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Stop if we're a zombie or need a soft hangup */
    	pthread_mutex_lock(&chan->lock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!chan->zombie && !ast_check_hangup(chan)) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (chan->pvt->call)
    			res = chan->pvt->call(chan, addr, timeout);
    	pthread_mutex_unlock(&chan->lock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return res;
    }
    
    int ast_readstring(struct ast_channel *c, char *s, int len, int timeout, int ftimeout, char *enders)
    {
    	int pos=0;
    	int to = ftimeout;
    	char d;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	/* Stop if we're a zombie or need a soft hangup */
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (c->zombie || ast_check_hangup(c)) 
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!len)
    		return -1;
    	do {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (c->streamid > -1) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    			d = ast_waitstream(c, AST_DIGIT_ANY);
    			ast_stopstream(c);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			usleep(1000);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			if (!d)
    				d = ast_waitfordigit(c, to);
    		} else {
    			d = ast_waitfordigit(c, to);
    		}
    		if (d < 0)
    			return -1;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (d == 0) {
    			s[pos]='\0';
    			return 1;
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!strchr(enders, d))
    			s[pos++] = d;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (strchr(enders, d) || (pos >= len)) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    			s[pos]='\0';
    			return 0;
    		}
    		to = timeout;
    	} while(1);
    	/* Never reached */
    	return 0;
    }
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_channel_supports_html(struct ast_channel *chan)
    {
    	if (chan->pvt->send_html)
    		return 1;
    	return 0;
    }
    
    int ast_channel_sendhtml(struct ast_channel *chan, int subclass, char *data, int datalen)
    {
    	if (chan->pvt->send_html)
    		return chan->pvt->send_html(chan, subclass, data, datalen);
    	return -1;
    }
    
    int ast_channel_sendurl(struct ast_channel *chan, char *url)
    {
    	if (chan->pvt->send_html)
    		return chan->pvt->send_html(chan, AST_HTML_URL, url, strlen(url) + 1);
    	return -1;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_channel_make_compatible(struct ast_channel *chan, struct ast_channel *peer)
    {
    	int peerf;
    	int chanf;
    	int res;
    	peerf = peer->nativeformats;
    	chanf = chan->nativeformats;
    	res = ast_translator_best_choice(&peerf, &chanf);
    	if (res < 0) {
    		ast_log(LOG_WARNING, "No path to translate from %s(%d) to %s(%d)\n", chan, chan->nativeformats, peer, peer->nativeformats);
    		return -1;
    	}
    	/* Set read format on channel */
    	res = ast_set_read_format(chan, peerf);
    	if (res < 0) {
    		ast_log(LOG_WARNING, "Unable to set read format on channel %s to %d\n", chan, chanf);
    		return -1;
    	}
    	/* Set write format on peer channel */
    	res = ast_set_write_format(peer, peerf);
    	if (res < 0) {
    		ast_log(LOG_WARNING, "Unable to set write format on channel %s to %d\n", peer, peerf);
    		return -1;
    	}
    	/* Now we go the other way */
    	peerf = peer->nativeformats;
    	chanf = chan->nativeformats;
    	res = ast_translator_best_choice(&chanf, &peerf);
    	if (res < 0) {
    		ast_log(LOG_WARNING, "No path to translate from %s(%d) to %s(%d)\n", peer, peer->nativeformats, chan, chan->nativeformats);
    		return -1;
    	}