Skip to content
Snippets Groups Projects
translate.c 14 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 - 2006, 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. */
       
    
    static AST_LIST_HEAD_STATIC(translators, ast_translator);
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    struct ast_translator_dir {
    
    	struct ast_translator *step;	/*!< Next step translator */
    
    	unsigned int cost;		/*!< Complete cost to destination */
    	unsigned int multistep;		/*!< Multiple conversions required for this translation */
    
    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);
    
    	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
    		}
    
    			tmp->next = ast_malloc(sizeof(*tmp));
    
    			tmp = tmp->next;
    		} else
    
    			tmp = ast_malloc(sizeof(*tmp));
    
    			
    		if (!tmp) {
    			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
    
      \note This function expects the list of translators to be locked
    */
    
    static void rebuild_matrix(int samples)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	struct ast_translator *t;
    	int changed;
    
    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));
    
    	AST_LIST_TRAVERSE(&translators, t, list) {
    
    			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) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    			tr_matrix[t->srcfmt][t->dstfmt].step = t;
    			tr_matrix[t->srcfmt][t->dstfmt].cost = t->cost;
    		}
    	}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	do {
    		changed = 0;
    
    Mark Spencer's avatar
    Mark Spencer committed
    		/* 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 */
    					continue;
    
    				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 */
    						continue;
    
    					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;
    						tr_matrix[x][z].multistep = 1;
    						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);
    						changed++;
    					}
    				}
    			}
    		}
    
    Mark Spencer's avatar
    Mark Spencer committed
    	} 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++;
    	}
    
    	AST_LIST_INSERT_HEAD(&translators, t, list);
    
    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)
    {
    
    	struct ast_translator *u;
    	AST_LIST_LOCK(&translators);
    	AST_LIST_TRAVERSE_SAFE_BEGIN(&translators, u, list) {
    
    Mark Spencer's avatar
    Mark Spencer committed
    		if (u == t) {
    
    			AST_LIST_REMOVE_CURRENT(&translators, list);
    
    			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
    	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 beststeps = INT_MAX;
    
    	int common;
    
    	if ((common = (*dst) & (*srcs))) {
    
    		for (y = 0; y < MAX_FORMAT; y++) {
    
    				/* This is a common format to both.  Pick it if we don't have one already */
    				bestdst = cur;
    				best = cur;
    
    Mark Spencer's avatar
    Mark Spencer committed
    			}
    
    			cur = cur << 1;
    		}
    	} else {
    		/* We will need to translate */
    
    		for (y = 0; y < MAX_FORMAT; y++) {
    
    			if (!(cur & *dst)) {
    				cur = cur << 1;
    
    
    			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 */
    					if (tr_matrix[x][y].cost > besttime)
    						continue;			/* It's more expensive, skip it */
    					
    					if (tr_matrix[x][y].cost == besttime &&
    					    tr_matrix[x][y].multistep >= beststeps)
    						continue; 			/* It requires the same (or more) steps,
    										   skip it */
    
    					/* It's better than what we have so far */
    					best = 1 << x;
    					bestdst = cur;
    					besttime = tr_matrix[x][y].cost;
    					beststeps = tr_matrix[x][y].multistep;
    
    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;
    }