Skip to content
Snippets Groups Projects
translate.c 13.5 KiB
Newer Older
  • Learn to ignore specific revisions
  • Mark Spencer's avatar
    Mark Spencer committed
    /*
    
     * Asterisk -- An open source telephony toolkit.
    
    Mark Spencer's avatar
    Mark Spencer committed
     *
    
     * Copyright (C) 1999 - 2005, Digium, Inc.
    
    Mark Spencer's avatar
    Mark Spencer committed
     *
    
     * Mark Spencer <markster@digium.com>
    
    Mark Spencer's avatar
    Mark Spencer committed
     *
    
     * See http://www.asterisk.org for more information about
     * the Asterisk project. Please do not directly contact
     * any of the maintainers of this project for assistance;
     * the project provides a web site, mailing lists and IRC
     * channels for your use.
     *
    
    Mark Spencer's avatar
    Mark Spencer committed
     * This program is free software, distributed under the terms of
    
     * the GNU General Public License Version 2. See the LICENSE file
     * at the top of the source tree.
     */
    
    
     * \brief Translate via the use of pseudo channels
    
     *
     * \author Mark Spencer <markster@digium.com> 
    
    #include <sys/socket.h>
    #include <sys/time.h>
    #include <unistd.h>
    #include <stdlib.h>
    #include <string.h>
    #include <stdio.h>
    
    
    #include "asterisk.h"
    
    
    Kevin P. Fleming's avatar
    Kevin P. Fleming committed
    ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
    
    #include "asterisk/lock.h"
    #include "asterisk/channel.h"
    #include "asterisk/logger.h"
    #include "asterisk/translate.h"
    #include "asterisk/options.h"
    #include "asterisk/frame.h"
    #include "asterisk/sched.h"
    #include "asterisk/cli.h"
    #include "asterisk/term.h"
    
    #define MAX_RECALC 200 /* max sample recalc */
    
    
    /*! \note
       This could all be done more efficiently *IF* we chained packets together
    
    Mark Spencer's avatar
    Mark Spencer committed
       by default, but it would also complicate virtually every application. */
       
    
    AST_MUTEX_DEFINE_STATIC(list_lock);
    
    Mark Spencer's avatar
    Mark Spencer committed
    static struct ast_translator *list = NULL;
    
    struct ast_translator_dir {
    
    	struct ast_translator *step;	/*!< Next step translator */
    	int cost;			/*!< Complete cost to destination */
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_frame_delivery {
    	struct ast_frame *f;
    	struct ast_channel *chan;
    	int fd;
    	struct translator_pvt *owner;
    	struct ast_frame_delivery *prev;
    	struct ast_frame_delivery *next;
    };
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    static struct ast_translator_dir tr_matrix[MAX_FORMAT][MAX_FORMAT];
    
    struct ast_trans_pvt {
    	struct ast_translator *step;
    	struct ast_translator_pvt *state;
    	struct ast_trans_pvt *next;
    
    	struct timeval nextin;
    	struct timeval nextout;
    
    Mark Spencer's avatar
    Mark Spencer committed
    };
    
    
    static int powerof(int d)
    {
    	int x;
    	for (x = 0; x < 32; x++)
    		if ((1 << x) & d)
    			return x;
    	ast_log(LOG_WARNING, "Powerof %d: No power??\n", d);
    	return -1;
    }
    
    void ast_translator_free_path(struct ast_trans_pvt *p)
    {
    
    	struct ast_trans_pvt *pl, *pn;
    	pn = p;
    	while(pn) {
    		pl = pn;
    		pn = pn->next;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (pl->state && pl->step->destroy)
    			pl->step->destroy(pl->state);
    		free(pl);
    	}
    }
    
    
    /*! Build a set of translators based upon the given source and destination formats */
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_trans_pvt *ast_translator_build_path(int dest, int source)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	struct ast_trans_pvt *tmpr = NULL, *tmp = NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	source = powerof(source);
    	dest = powerof(dest);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	while(source != dest) {
    
    		if (!tr_matrix[source][dest].step) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    			/* We shouldn't have allocated any memory */
    
    			ast_log(LOG_WARNING, "No translator path from %s to %s\n", 
    				ast_getformatname(source), ast_getformatname(dest));
    
    Mark Spencer's avatar
    Mark Spencer committed
    			return NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		}
    
    
    		if (tmp) {
    			tmp->next = malloc(sizeof(*tmp));
    			tmp = tmp->next;
    		} else
    			tmp = malloc(sizeof(*tmp));
    			
    		if (!tmp) {
    			ast_log(LOG_WARNING, "Out of memory\n");
    			if (tmpr)
    				ast_translator_free_path(tmpr);	
    			return NULL;
    		}
    
    		/* Set the root, if it doesn't exist yet... */
    		if (!tmpr)
    			tmpr = tmp;
    
    		tmp->next = NULL;
    		tmp->nextin = tmp->nextout = ast_tv(0, 0);
    		tmp->step = tr_matrix[source][dest].step;
    		tmp->state = tmp->step->newpvt();
    		
    		if (!tmp->state) {
    			ast_log(LOG_WARNING, "Failed to build translator step from %d to %d\n", source, dest);
    			ast_translator_free_path(tmpr);	
    			return NULL;
    		}
    		
    		/* Keep going if this isn't the final destination */
    		source = tmp->step->dstfmt;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    	return tmpr;
    }
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    struct ast_frame *ast_translate(struct ast_trans_pvt *path, struct ast_frame *f, int consume)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	struct ast_trans_pvt *p;
    	struct ast_frame *out;
    
    	struct timeval delivery;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	p = path;
    	/* Feed the first frame into the first translator */
    	p->step->framein(p->state, f);
    
    	if (!ast_tvzero(f->delivery)) {
    		if (!ast_tvzero(path->nextin)) {
    
    			/* Make sure this is in line with what we were expecting */
    
    			if (!ast_tveq(path->nextin, f->delivery)) {
    
    				/* The time has changed between what we expected and this
    
    				   most recent time on the new packet.  If we have a
    				   valid prediction adjust our output time appropriately */
    				if (!ast_tvzero(path->nextout)) {
    					path->nextout = ast_tvadd(path->nextout,
    								  ast_tvsub(f->delivery, path->nextin));
    				}
    
    			}
    		} else {
    			/* This is our first pass.  Make sure the timing looks good */
    
    			path->nextin = f->delivery;
    			path->nextout = f->delivery;
    
    		}
    		/* Predict next incoming sample */
    
    		path->nextin = ast_tvadd(path->nextin, ast_samp2tv(f->samples, 8000));
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (consume)
    		ast_frfree(f);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	while(p) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		out = p->step->frameout(p->state);
    		/* If we get nothing out, return NULL */
    		if (!out)
    			return NULL;
    		/* If there is a next state, feed it in there.  If not,
    		   return this frame  */
    		if (p->next) 
    			p->next->step->framein(p->next->state, out);
    
    				/* Regenerate prediction after a discontinuity */
    				if (ast_tvzero(path->nextout))
    					path->nextout = ast_tvnow();
    
    
    				/* Use next predicted outgoing timestamp */
    
    				
    				/* Predict next outgoing timestamp from samples in this
    				   frame. */
    
    				path->nextout = ast_tvadd(path->nextout, ast_samp2tv( out->samples, 8000));
    
    			/* Invalidate prediction if we're entering a silence period */
    			if (out->frametype == AST_FRAME_CNG)
    				path->nextout = ast_tv(0, 0);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			return out;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		p = p->next;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	ast_log(LOG_WARNING, "I should never get here...\n");
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return NULL;
    }
    
    
    static void calc_cost(struct ast_translator *t, int samples)
    
    {
    	int sofar=0;
    	struct ast_translator_pvt *pvt;
    	struct ast_frame *f, *out;
    
    	
    	/* If they don't make samples, give them a terrible score */
    	if (!t->sample) {
    		ast_log(LOG_WARNING, "Translator '%s' does not produce sample frames.\n", t->name);
    		t->cost = 99999;
    		return;
    	}
    
    	if (!pvt) {
    		ast_log(LOG_WARNING, "Translator '%s' appears to be broken and will probably fail.\n", t->name);
    		t->cost = 99999;
    		return;
    	}
    
    	/* Call the encoder until we've processed one second of time */
    	while(sofar < samples * 8000) {
    		f = t->sample();
    		if (!f) {
    			ast_log(LOG_WARNING, "Translator '%s' failed to produce a sample frame.\n", t->name);
    			t->destroy(pvt);
    			t->cost = 99999;
    			return;
    		}
    		t->framein(pvt, f);
    		ast_frfree(f);
    		while((out = t->frameout(pvt))) {
    			sofar += out->samples;
    			ast_frfree(out);
    		}
    	}
    
    	t->destroy(pvt);
    	t->cost = cost / samples;
    	if (!t->cost)
    		t->cost = 1;
    }
    
    
    /*! \brief Use the list of translators to build a translation matrix */
    
    static void rebuild_matrix(int samples)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	struct ast_translator *t;
    	int changed;
    	int x,y,z;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (option_debug)
    
    		ast_log(LOG_DEBUG, "Resetting translation matrix\n");
    
    Mark Spencer's avatar
    Mark Spencer committed
    	bzero(tr_matrix, sizeof(tr_matrix));
    	t = list;
    	while(t) {
    
    		if(samples)
    			calc_cost(t, samples);
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (!tr_matrix[t->srcfmt][t->dstfmt].step ||
    		     tr_matrix[t->srcfmt][t->dstfmt].cost > t->cost) {
    			tr_matrix[t->srcfmt][t->dstfmt].step = t;
    			tr_matrix[t->srcfmt][t->dstfmt].cost = t->cost;
    		}
    		t = t->next;
    	}
    	do {
    		changed = 0;
    		/* Don't you just love O(N^3) operations? */
    		for (x=0; x< MAX_FORMAT; x++)				/* For each source format */
    			for (y=0; y < MAX_FORMAT; y++) 			/* And each destination format */
    
    				if (x != y)				/* Except ourselves, of course */
    
    Mark Spencer's avatar
    Mark Spencer committed
    					for (z=0; z < MAX_FORMAT; z++) 	/* And each format it might convert to */
    						if ((x!=z) && (y!=z)) 		/* Don't ever convert back to us */
    							if (tr_matrix[x][y].step && /* We can convert from x to y */
    								tr_matrix[y][z].step && /* And from y to z and... */
    								(!tr_matrix[x][z].step || 	/* Either there isn't an x->z conversion */
    								(tr_matrix[x][y].cost + 
    								 tr_matrix[y][z].cost <	/* Or we're cheaper than the existing */
    								 tr_matrix[x][z].cost)  /* solution */
    							     )) {
    								 			/* We can get from x to z via y with a cost that
    											   is the sum of the transition from x to y and
    											   from y to z */
    								 
    								 	tr_matrix[x][z].step = tr_matrix[x][y].step;
    									tr_matrix[x][z].cost = tr_matrix[x][y].cost + 
    														   tr_matrix[y][z].cost;
    									if (option_debug)
    
    										ast_log(LOG_DEBUG, "Discovered %d cost path from %s to %s, via %d\n", tr_matrix[x][z].cost, ast_getformatname(x), ast_getformatname(z), y);
    
    Mark Spencer's avatar
    Mark Spencer committed
    									changed++;
    								 }
    		
    	} while (changed);
    }
    
    
    /*! \brief CLI "show translation" command handler */
    
    Mark Spencer's avatar
    Mark Spencer committed
    static int show_translation(int fd, int argc, char *argv[])
    {
    
    James Golovich's avatar
    James Golovich committed
    	int x, y, z;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char line[80];
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return RESULT_SHOWUSAGE;
    
    James Golovich's avatar
    James Golovich committed
    	if (argv[2] && !strcasecmp(argv[2],"recalc")) {
    		z = argv[3] ? atoi(argv[3]) : 1;
    
    James Golovich's avatar
    James Golovich committed
    		if (z <= 0) {
    			ast_cli(fd,"         C'mon let's be serious here... defaulting to 1.\n");
    			z = 1;
    		}
    
    James Golovich's avatar
    James Golovich committed
    		if (z > MAX_RECALC) {
    			ast_cli(fd,"         Maximum limit of recalc exceeded by %d, truncating value to %d\n",z-MAX_RECALC,MAX_RECALC);
    			z = MAX_RECALC;
    		}
    		ast_cli(fd,"         Recalculating Codec Translation (number of sample seconds: %d)\n\n",z);
    		rebuild_matrix(z);
    
    	ast_cli(fd, "         Translation times between formats (in milliseconds)\n");
    	ast_cli(fd, "          Source Format (Rows) Destination Format(Columns)\n\n");
    
    	for (x = -1; x < SHOW_TRANS; x++) {
    
    		/* next 2 lines run faster than using strcpy() */
    		line[0] = ' ';
    		line[1] = '\0';
    
    		for (y=-1;y<SHOW_TRANS;y++) {
    
    			if (x >= 0 && y >= 0 && tr_matrix[x][y].step)
    
    James Golovich's avatar
    James Golovich committed
    				snprintf(line + strlen(line), sizeof(line) - strlen(line), " %5d", tr_matrix[x][y].cost >= 99999 ? tr_matrix[x][y].cost-99999 : tr_matrix[x][y].cost);
    
    Mark Spencer's avatar
    Mark Spencer committed
    			else
    
    				if (((x == -1 && y >= 0) || (y == -1 && x >= 0))) {
    
    					snprintf(line + strlen(line), sizeof(line) - strlen(line), 
    
    						" %5s", ast_getformatname(1<<(x+y+1)) );
    
    James Golovich's avatar
    James Golovich committed
    				} else if (x != -1 && y != -1) {
    
    					snprintf(line + strlen(line), sizeof(line) - strlen(line), "     -");
    
    James Golovich's avatar
    James Golovich committed
    				} else {
    
    					snprintf(line + strlen(line), sizeof(line) - strlen(line), "      ");
    
    Mark Spencer's avatar
    Mark Spencer committed
    		}
    		snprintf(line + strlen(line), sizeof(line) - strlen(line), "\n");
    		ast_cli(fd, line);			
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return RESULT_SUCCESS;
    }
    
    static int added_cli = 0;
    
    static char show_trans_usage[] =
    
    "Usage: show translation [recalc] [<recalc seconds>]\n"
    
    Mark Spencer's avatar
    Mark Spencer committed
    "       Displays known codec translators and the cost associated\n"
    
    "with each conversion.  If the argument 'recalc' is supplied along\n"
    
    "with optional number of seconds to test a new test will be performed\n"
    "as the chart is being displayed.\n";
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    static struct ast_cli_entry show_trans =
    { { "show", "translation", NULL }, show_translation, "Display translation matrix", show_trans_usage };
    
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_register_translator(struct ast_translator *t)
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	char tmp[80];
    
    Mark Spencer's avatar
    Mark Spencer committed
    	t->srcfmt = powerof(t->srcfmt);
    	t->dstfmt = powerof(t->dstfmt);
    
    	if (t->srcfmt >= MAX_FORMAT) {
    		ast_log(LOG_WARNING, "Source format %s is larger than MAX_FORMAT\n", ast_getformatname(t->srcfmt));
    		return -1;
    	}
    	if (t->dstfmt >= MAX_FORMAT) {
    		ast_log(LOG_WARNING, "Destination format %s is larger than MAX_FORMAT\n", ast_getformatname(t->dstfmt));
    
    Mark Spencer's avatar
    Mark Spencer committed
    		return -1;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (option_verbose > 1)
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ast_verbose(VERBOSE_PREFIX_2 "Registered translator '%s' from format %s to %s, cost %d\n", term_color(tmp, t->name, COLOR_MAGENTA, COLOR_BLACK, sizeof(tmp)), ast_getformatname(1 << t->srcfmt), ast_getformatname(1 << t->dstfmt), t->cost);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!added_cli) {
    		ast_cli_register(&show_trans);
    		added_cli++;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	t->next = list;
    	list = t;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return 0;
    }
    
    
    /*! \brief unregister codec translator */
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_unregister_translator(struct ast_translator *t)
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	struct ast_translator *u, *ul = NULL;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	u = list;
    	while(u) {
    		if (u == t) {
    			if (ul)
    				ul->next = u->next;
    			else
    				list = u->next;
    
    			if (option_verbose > 1)
    				ast_verbose(VERBOSE_PREFIX_2 "Unregistered translator '%s' from format %s to %s\n", term_color(tmp, t->name, COLOR_MAGENTA, COLOR_BLACK, sizeof(tmp)), ast_getformatname(1 << t->srcfmt), ast_getformatname(1 << t->dstfmt));
    
    Mark Spencer's avatar
    Mark Spencer committed
    			break;
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    		ul = u;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		u = u->next;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return (u ? 0 : -1);
    }
    
    
    /*! \brief Calculate our best translator source format, given costs, and a desired destination */
    
    Mark Spencer's avatar
    Mark Spencer committed
    int ast_translator_best_choice(int *dst, int *srcs)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int x,y;
    
    	int best = -1;
    	int bestdst = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    	int cur = 1;
    
    	int besttime = INT_MAX;
    	int common;
    
    	if ((common = (*dst) & (*srcs))) {
    
    		for (y=0; y < MAX_FORMAT; y++) {
    			if (cur & common) {
    
    				/* This is a common format to both.  Pick it if we don't have one already */
    
    Mark Spencer's avatar
    Mark Spencer committed
    			}
    
    			cur = cur << 1;
    		}
    	} else {
    		/* We will need to translate */
    		ast_mutex_lock(&list_lock);
    
    		for (y=0; y < MAX_FORMAT; y++) {
    
    				for (x=0; x < MAX_FORMAT; x++) {
    					if ((*srcs & (1 << x)) &&			/* x is a valid source format */
    					    tr_matrix[x][y].step &&			/* There's a step */
    					    (tr_matrix[x][y].cost < besttime)) {	/* It's better than what we have so far */
    						best = 1 << x;
    						bestdst = cur;
    						besttime = tr_matrix[x][y].cost;
    					}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (best > -1) {
    		*srcs = best;
    		*dst = bestdst;
    		best = 0;
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return best;
    }