mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-22 20:56:39 +00:00 
			
		
		
		
	We've been seeing crashes in libbfd when we attempt to generate a stack trace from multiple threads. It turns out that libbfd is NOT thread-safe. It can cache the bfd structure and give it to multiple threads without protecting itself. To get around this, we've added a global mutex around the bfd functions and also have refactored the use of those functions to be more efficient and to provide more information about inlined functions. Also added a few more tests to test_pbx.c. One just calls ast_assert() and the other calls ast_log_backtrace(). Neither are run by default. WARNING: This change necessitated changing the return value of ast_bt_get_symbols() from an array of strings to a VECTOR of strings. However, the use of this function outside Asterisk is not likely. ASTERISK-28140 Change-Id: I79d02862ddaa2423a0809caa4b3b85c128131621
		
			
				
	
	
		
			632 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			632 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * XXX this file probably need a fair amount of cleanup, at the very least:
 | |
|  *
 | |
|  * - documenting its purpose;
 | |
|  * - removing all unnecessary headers and other stuff from the sources
 | |
|  *   it was copied from;
 | |
|  * - fixing the formatting
 | |
|  */
 | |
| 
 | |
| /*** MODULEINFO
 | |
| 	<support_level>extended</support_level>
 | |
|  ***/
 | |
| 
 | |
| #define ASTMM_LIBC ASTMM_IGNORE
 | |
| #include "asterisk.h"
 | |
| 
 | |
| #include <locale.h>
 | |
| #include <ctype.h>
 | |
| #include <regex.h>
 | |
| #include <limits.h>
 | |
| 
 | |
| #include "asterisk/backtrace.h"
 | |
| #include "asterisk/channel.h"
 | |
| #include "asterisk/ast_expr.h"
 | |
| #include "asterisk/module.h"
 | |
| #include "asterisk/app.h"
 | |
| #include "asterisk/lock.h"
 | |
| #include "asterisk/hashtab.h"
 | |
| #include "asterisk/ael_structs.h"
 | |
| #include "asterisk/extconf.h"
 | |
| 
 | |
| int option_debug = 0;
 | |
| int option_verbose = 0;
 | |
| 
 | |
| /*** MODULEINFO
 | |
|   	<depend>res_ael_share</depend>
 | |
|  ***/
 | |
| 
 | |
| struct namelist
 | |
| {
 | |
| 	char name[100];
 | |
| 	char name2[100];
 | |
| 	struct namelist *next;
 | |
| };
 | |
| 
 | |
| struct ast_context
 | |
| {
 | |
| 	int extension_count;
 | |
| 	char name[100];
 | |
| 	char registrar[100];
 | |
| 	struct namelist *includes;
 | |
| 	struct namelist *ignorepats;
 | |
| 	struct namelist *switches;
 | |
| 	struct namelist *eswitches;
 | |
| 
 | |
| 	struct namelist *includes_last;
 | |
| 	struct namelist *ignorepats_last;
 | |
| 	struct namelist *switches_last;
 | |
| 	struct namelist *eswitches_last;
 | |
| 
 | |
| 	struct ast_context *next;
 | |
| };
 | |
| 
 | |
| #define ADD_LAST(headptr,memptr) if(!headptr){ headptr=(memptr); (headptr##_last)=(memptr);} else {(headptr##_last)->next = (memptr); (headptr##_last) = (memptr);}
 | |
| 
 | |
| void destroy_namelist(struct namelist *x);
 | |
| void destroy_namelist(struct namelist *x)
 | |
| {
 | |
| 	struct namelist *z,*z2;
 | |
| 	for(z=x; z; z = z2)
 | |
| 	{
 | |
| 		z2 = z->next;
 | |
| 		z->next = 0;
 | |
| 		free(z);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| struct namelist *create_name(const char *name);
 | |
| struct namelist *create_name(const char *name)
 | |
| {
 | |
| 	struct namelist *x = calloc(1, sizeof(*x));
 | |
| 	if (!x)
 | |
| 		return NULL;
 | |
| 	strncpy(x->name, name, sizeof(x->name) - 1);
 | |
| 	return x;
 | |
| }
 | |
| 
 | |
| struct ast_context *context_list;
 | |
| struct ast_context *last_context;
 | |
| struct namelist *globalvars;
 | |
| struct namelist *globalvars_last;
 | |
| 
 | |
| int conts=0, extens=0, priors=0;
 | |
| char last_exten[18000];
 | |
| 
 | |
| static char config_dir[PATH_MAX];
 | |
| static char var_dir[PATH_MAX];
 | |
| const char *ast_config_AST_CONFIG_DIR = config_dir;
 | |
| const char *ast_config_AST_VAR_DIR = var_dir;
 | |
| 
 | |
| void __ast_cli_register_multiple(void);
 | |
| int ast_add_extension2(struct ast_context *con,
 | |
| 					   int replace, const char *extension, int priority, const char *label, const char *callerid,
 | |
| 						const char *application, void *data, void (*datad)(void *),
 | |
| 					   const char *registrar);
 | |
| void pbx_builtin_setvar(void *chan, void *data);
 | |
| struct ast_context * ast_context_create(void **extcontexts, const char *name, const char *registrar);
 | |
| struct ast_context * ast_context_find_or_create(void **extcontexts, void *tab, const char *name, const char *registrar);
 | |
| void ast_context_add_ignorepat2(struct ast_context *con, const char *value, const char *registrar);
 | |
| void ast_context_add_include2(struct ast_context *con, const char *value, const char *registrar);
 | |
| void ast_context_add_switch2(struct ast_context *con, const char *value, const char *data, int eval, const char *registrar);
 | |
| void ast_merge_contexts_and_delete(void);
 | |
| void ast_context_verify_includes(void);
 | |
| struct ast_context * ast_walk_contexts(void);
 | |
| void ast_cli_unregister_multiple(void);
 | |
| void ast_context_destroy(void);
 | |
| void ast_log(int level, const char *file, int line, const char *function, const char *fmt, ...);
 | |
| char *ast_process_quotes_and_slashes(char *start, char find, char replace_with);
 | |
| void __ast_verbose(const char *file, int line, const char *func, int level, const char *fmt, ...);
 | |
| struct ast_app *pbx_findapp(const char *app);
 | |
| void filter_leading_space_from_exprs(char *str);
 | |
| void filter_newlines(char *str);
 | |
| static int quiet = 0;
 | |
| static int no_comp = 0;
 | |
| static int use_curr_dir = 0;
 | |
| static int dump_extensions = 0;
 | |
| static int FIRST_TIME = 0;
 | |
| static FILE *dumpfile;
 | |
| 
 | |
| void ast_log(int level, const char *file, int line, const char *function, const char *fmt, ...)
 | |
| {
 | |
| 	        va_list vars;
 | |
| 		        va_start(vars,fmt);
 | |
| 
 | |
| 			        printf("LOG: lev:%d file:%s  line:%d func: %s  ",
 | |
| 						                   level, file, line, function);
 | |
| 				        vprintf(fmt, vars);
 | |
| 					        fflush(stdout);
 | |
| 						        va_end(vars);
 | |
| }
 | |
| 
 | |
| struct ast_exten *pbx_find_extension(struct ast_channel *chan,
 | |
| 									 struct ast_context *bypass,
 | |
| 									 struct pbx_find_info *q,
 | |
| 									 const char *context,
 | |
| 									 const char *exten,
 | |
| 									 int priority,
 | |
| 									 const char *label,
 | |
| 									 const char *callerid,
 | |
| 									 enum ext_match_t action);
 | |
| 
 | |
| struct ast_exten *pbx_find_extension(struct ast_channel *chan,
 | |
| 									 struct ast_context *bypass,
 | |
| 									 struct pbx_find_info *q,
 | |
| 									 const char *context,
 | |
| 									 const char *exten,
 | |
| 									 int priority,
 | |
| 									 const char *label,
 | |
| 									 const char *callerid,
 | |
| 									 enum ext_match_t action)
 | |
| {
 | |
| 	return localized_find_extension(bypass, q, context, exten, priority, label, callerid, action);
 | |
| }
 | |
| 
 | |
| struct ast_app *pbx_findapp(const char *app)
 | |
| {
 | |
| 	return (struct ast_app*)1; /* so as not to trigger an error */
 | |
| }
 | |
| 
 | |
| struct ast_custom_function *ast_custom_function_find(const char *name);
 | |
| 
 | |
| 
 | |
| struct ast_custom_function *ast_custom_function_find(const char *name)
 | |
| {
 | |
| 	return 0; /* in "standalone" mode, functions are just not avail */
 | |
| }
 | |
| 
 | |
| int ast_add_profile(const char *x, uint64_t scale)
 | |
| {
 | |
| 	if (!no_comp)
 | |
| 		printf("Executed ast_add_profile();\n");
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int ast_loader_register(int (*updater)(void))
 | |
| {
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| int ast_loader_unregister(int (*updater)(void))
 | |
| {
 | |
| 	return 1;
 | |
| }
 | |
| void ast_module_register(const struct ast_module_info *x)
 | |
| {
 | |
| }
 | |
| 
 | |
| void ast_module_unregister(const struct ast_module_info *x)
 | |
| {
 | |
| }
 | |
| 
 | |
| 
 | |
| void __ast_cli_register_multiple(void)
 | |
| {
 | |
| 	if(!no_comp)
 | |
|         	printf("Executed ast_cli_register_multiple();\n");
 | |
| }
 | |
| 
 | |
| void pbx_substitute_variables_helper(struct ast_channel *c,const char *cp1,char *cp2,int count);
 | |
| void pbx_substitute_variables_helper(struct ast_channel *c,const char *cp1,char *cp2,int count)
 | |
| {
 | |
| 	if (cp1 && *cp1)
 | |
| 		strncpy(cp2,cp1,AST_MAX_EXTENSION); /* Right now, this routine is ONLY being called for
 | |
| 											   a possible var substitution on extension names,
 | |
| 											   so....! */
 | |
| 	else
 | |
| 		*cp2 = 0;
 | |
| }
 | |
| 
 | |
| int ast_add_extension2(struct ast_context *con,
 | |
| 			int replace, const char *extension, int priority, const char *label, const char *callerid,
 | |
| 			const char *application, void *data, void (*datad)(void *),
 | |
| 			const char *registrar)
 | |
| {
 | |
| 	priors++;
 | |
| 	con->extension_count++;
 | |
| 	if (strcmp(extension,last_exten) != 0) {
 | |
| 		extens++;
 | |
| 		strcpy(last_exten, extension);
 | |
| 	}
 | |
| 	if (!label) {
 | |
| 		label = "(null)";
 | |
| 	}
 | |
| 	if (!callerid) {
 | |
| 		callerid = "(null)";
 | |
| 	}
 | |
| 	if (!application) {
 | |
| 		application = "(null)";
 | |
| 	}
 | |
| 
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_add_extension2(context=%s, rep=%d, exten=%s, priority=%d, label=%s, callerid=%s, appl=%s, data=%s, FREE, registrar=%s);\n",
 | |
| 			   con->name, replace, extension, priority, label, callerid, application, (data?(char*)data:"(null)"), registrar);
 | |
| 
 | |
| 	if( dump_extensions && dumpfile ) {
 | |
| 		struct namelist *n;
 | |
| 
 | |
| 		if( FIRST_TIME ) {
 | |
| 			FIRST_TIME = 0;
 | |
| 
 | |
| 			if( globalvars )
 | |
| 				fprintf(dumpfile,"[globals]\n");
 | |
| 
 | |
| 			for(n=globalvars;n;n=n->next) {
 | |
| 				fprintf(dumpfile, "%s\n", n->name);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		/* print out each extension , possibly the context header also */
 | |
| 		if( con != last_context ) {
 | |
| 			fprintf(dumpfile,"\n\n[%s]\n", con->name);
 | |
| 			last_context = con;
 | |
| 			for(n=con->ignorepats;n;n=n->next) {
 | |
| 				fprintf(dumpfile, "ignorepat => %s\n", n->name);
 | |
| 			}
 | |
| 			for(n=con->includes;n;n=n->next) {
 | |
| 				fprintf(dumpfile, "include => %s\n", n->name);
 | |
| 			}
 | |
| 			for(n=con->switches;n;n=n->next) {
 | |
| 				fprintf(dumpfile, "switch => %s/%s\n", n->name, n->name2);
 | |
| 			}
 | |
| 			for(n=con->eswitches;n;n=n->next) {
 | |
| 				fprintf(dumpfile, "eswitch => %s/%s\n", n->name, n->name2);
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 		if( data ) {
 | |
| 			filter_newlines((char*)data);
 | |
| 			filter_leading_space_from_exprs((char*)data);
 | |
| 			/* in previous versions, commas were converted to '|' to separate
 | |
| 			   args in app calls, but now, commas are used. There used to be
 | |
| 			   code here to insert backslashes (escapes) before any commas
 | |
| 			   that may have been embedded in the app args. This code is no more. */
 | |
| 
 | |
| 			if( strcmp(label,"(null)") != 0  )
 | |
| 				fprintf(dumpfile,"exten => %s,%d(%s),%s(%s)\n", extension, priority, label, application, (char*)data);
 | |
| 			else
 | |
| 				fprintf(dumpfile,"exten => %s,%d,%s(%s)\n", extension, priority, application, (char*)data);
 | |
| 
 | |
| 		} else {
 | |
| 
 | |
| 			if( strcmp(label,"(null)") != 0  )
 | |
| 				fprintf(dumpfile,"exten => %s,%d(%s),%s\n", extension, priority, label, application);
 | |
| 			else
 | |
| 				fprintf(dumpfile,"exten => %s,%d,%s\n", extension, priority, application);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* since add_extension2 is responsible for the malloc'd data stuff */
 | |
| 	free(data);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void pbx_builtin_setvar(void *chan, void *data)
 | |
| {
 | |
| 	struct namelist *x = create_name(data);
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed pbx_builtin_setvar(chan, data=%s);\n", (char*)data);
 | |
| 
 | |
| 	if( dump_extensions ) {
 | |
| 		x = create_name(data);
 | |
| 		ADD_LAST(globalvars,x);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| struct ast_context * ast_context_create(void **extcontexts, const char *name, const char *registrar)
 | |
| {
 | |
| 	struct ast_context *x = calloc(1, sizeof(*x));
 | |
| 	if (!x)
 | |
| 		return NULL;
 | |
| 	x->next = context_list;
 | |
| 	context_list = x;
 | |
| 	if (!no_comp)
 | |
| 		printf("Executed ast_context_create(conts, name=%s, registrar=%s);\n", name, registrar);
 | |
| 	conts++;
 | |
| 	strncpy(x->name, name, sizeof(x->name) - 1);
 | |
| 	strncpy(x->registrar, registrar, sizeof(x->registrar) - 1);
 | |
| 	return x;
 | |
| }
 | |
| 
 | |
| struct ast_context * ast_context_find_or_create(void **extcontexts, void *tab, const char *name, const char *registrar)
 | |
| {
 | |
| 	struct ast_context *x = calloc(1, sizeof(*x));
 | |
| 	if (!x)
 | |
| 		return NULL;
 | |
| 	x->next = context_list;
 | |
| 	context_list = x;
 | |
| 	if (!no_comp)
 | |
| 		printf("Executed ast_context_find_or_create(conts, name=%s, registrar=%s);\n", name, registrar);
 | |
| 	conts++;
 | |
| 	strncpy(x->name, name, sizeof(x->name) - 1);
 | |
| 	strncpy(x->registrar, registrar, sizeof(x->registrar) - 1);
 | |
| 	return x;
 | |
| }
 | |
| 
 | |
| void ast_context_add_ignorepat2(struct ast_context *con, const char *value, const char *registrar)
 | |
| {
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_context_add_ignorepat2(con, value=%s, registrar=%s);\n", value, registrar);
 | |
| 	if( dump_extensions ) {
 | |
| 		struct namelist *x;
 | |
| 		x = create_name(value);
 | |
| 		ADD_LAST(con->ignorepats,x);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ast_context_add_include2(struct ast_context *con, const char *value, const char *registrar)
 | |
| {
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_context_add_include2(con, value=%s, registrar=%s);\n", value, registrar);
 | |
| 	if( dump_extensions ) {
 | |
| 		struct namelist *x;
 | |
| 		x = create_name((char*)value);
 | |
| 		ADD_LAST(con->includes,x);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ast_context_add_switch2(struct ast_context *con, const char *value, const char *data, int eval, const char *registrar)
 | |
| {
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_context_add_switch2(con, value=%s, data=%s, eval=%d, registrar=%s);\n", value, data, eval, registrar);
 | |
| 	if( dump_extensions ) {
 | |
| 		struct namelist *x;
 | |
| 		x = create_name((char*)value);
 | |
| 		strncpy(x->name2, data, 99);
 | |
| 		if( eval ) {
 | |
| 
 | |
| 			ADD_LAST(con->switches,x);
 | |
| 
 | |
| 		} else {
 | |
| 
 | |
| 			ADD_LAST(con->eswitches,x);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ast_merge_contexts_and_delete(void)
 | |
| {
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_merge_contexts_and_delete();\n");
 | |
| }
 | |
| 
 | |
| void ast_context_verify_includes(void)
 | |
| {
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_context_verify_includes();\n");
 | |
| }
 | |
| 
 | |
| struct ast_context * ast_walk_contexts(void)
 | |
| {
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_walk_contexts();\n");
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void ast_cli_unregister_multiple(void)
 | |
| {
 | |
| 	if(!no_comp)
 | |
| 		printf("Executed ast_cli_unregister_multiple();\n");
 | |
| }
 | |
| 
 | |
| void ast_context_destroy(void)
 | |
| {
 | |
| 	if( !no_comp)
 | |
| 		printf("Executed ast_context_destroy();\n");
 | |
| }
 | |
| 
 | |
| const char *ast_get_context_name(struct ast_context *con);
 | |
| const char *ast_get_context_name(struct ast_context *con)
 | |
| {
 | |
| 	return con ? con->name : NULL;
 | |
| }
 | |
| 
 | |
| struct ast_exten *ast_walk_context_extensions(struct ast_context *con, struct ast_exten *exten);
 | |
| struct ast_exten *ast_walk_context_extensions(struct ast_context *con, struct ast_exten *exten)
 | |
| {
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| struct ast_include *ast_walk_context_includes(struct ast_context *con, struct ast_include *inc);
 | |
| struct ast_include *ast_walk_context_includes(struct ast_context *con, struct ast_include *inc)
 | |
| {
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| struct ast_ignorepat *ast_walk_context_ignorepats(struct ast_context *con, struct ast_ignorepat *ip);
 | |
| struct ast_ignorepat *ast_walk_context_ignorepats(struct ast_context *con, struct ast_ignorepat *ip)
 | |
| {
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| struct ast_sw *ast_walk_context_switches(struct ast_context *con, struct ast_sw *sw);
 | |
| struct ast_sw *ast_walk_context_switches(struct ast_context *con, struct ast_sw *sw)
 | |
| {
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| void filter_leading_space_from_exprs(char *str)
 | |
| {
 | |
| 	/*  Mainly for aesthetics */
 | |
| 	char *t, *v, *u = str;
 | |
| 
 | |
| 	while ( u && *u ) {
 | |
| 
 | |
| 		if( *u == '$' && *(u+1) == '[' ) {
 | |
| 			t = u+2;
 | |
| 			while( *t == '\n' || *t == '\r' || *t == '\t' || *t == ' ' ) {
 | |
| 				v = t;
 | |
| 				while ( *v ) {
 | |
| 					*v = *(v+1);
 | |
| 					v++;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		u++;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void filter_newlines(char *str)
 | |
| {
 | |
| 	/* remove all newlines, returns  */
 | |
| 	char *t=str;
 | |
| 	while( t && *t ) {
 | |
| 		if( *t == '\n' || *t == '\r' ) {
 | |
| 			*t = ' '; /* just replace newlines and returns with spaces; they act as
 | |
| 						 token separators, and just blindly removing them could be
 | |
| 						 harmful. */
 | |
| 		}
 | |
| 		t++;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| extern struct module_symbols mod_data;
 | |
| int ael_external_load_module(void);
 | |
| 
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
| 	int i;
 | |
| 	struct namelist *n;
 | |
| 	struct ast_context *lp,*lp2;
 | |
| 
 | |
| 	for(i=1;i<argc;i++) {
 | |
| 		if( argv[i][0] == '-' && argv[i][1] == 'n' )
 | |
| 			no_comp =1;
 | |
| 		if( argv[i][0] == '-' && argv[i][1] == 'q' ) {
 | |
| 			quiet = 1;
 | |
| 			no_comp =1;
 | |
| 		}
 | |
| 		if( argv[i][0] == '-' && argv[i][1] == 'd' )
 | |
| 			use_curr_dir =1;
 | |
| 		if( argv[i][0] == '-' && argv[i][1] == 'w' )
 | |
| 			dump_extensions =1;
 | |
| 	}
 | |
| 
 | |
| 	if( !quiet ) {
 | |
| 		printf("\n(If you find progress and other non-error messages irritating, you can use -q to suppress them)\n");
 | |
| 		if( !no_comp )
 | |
| 			printf("\n(You can use the -n option if you aren't interested in seeing all the instructions generated by the compiler)\n\n");
 | |
| 		if( !use_curr_dir )
 | |
| 			printf("\n(You can use the -d option if you want to use the current working directory as the CONFIG_DIR. I will look in this dir for extensions.ael* and its included files)\n\n");
 | |
| 		if( !dump_extensions )
 | |
| 			printf("\n(You can use the -w option to dump extensions.conf format to extensions.conf.aeldump)\n");
 | |
| 	}
 | |
| 
 | |
| 	if( use_curr_dir ) {
 | |
| 		strcpy(config_dir, ".");
 | |
| 		localized_use_local_dir();
 | |
| 	}
 | |
| 	else {
 | |
| 		strcpy(config_dir, "/etc/asterisk");
 | |
| 		localized_use_conf_dir();
 | |
| 	}
 | |
| 	strcpy(var_dir, "/var/lib/asterisk");
 | |
| 
 | |
| 	if( dump_extensions ) {
 | |
| 		dumpfile = fopen("extensions.conf.aeldump","w");
 | |
| 		if( !dumpfile ) {
 | |
| 			printf("\n\nSorry, cannot open extensions.conf.aeldump for writing! Correct the situation and try again!\n\n");
 | |
| 			exit(10);
 | |
| 		}
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	FIRST_TIME = 1;
 | |
| 
 | |
| 	ael_external_load_module();
 | |
| 
 | |
| 	ast_log(4, "ael2_parse", __LINE__, "main", "%d contexts, %d extensions, %d priorities\n", conts, extens, priors);
 | |
| 
 | |
| 	if( dump_extensions && dumpfile ) {
 | |
| 
 | |
| 		for( lp = context_list; lp; lp = lp->next ) { /* print out any contexts that didn't have any
 | |
| 														 extensions in them */
 | |
| 			if( lp->extension_count == 0 ) {
 | |
| 
 | |
| 				fprintf(dumpfile,"\n\n[%s]\n", lp->name);
 | |
| 
 | |
| 				for(n=lp->ignorepats;n;n=n->next) {
 | |
| 					fprintf(dumpfile, "ignorepat => %s\n", n->name);
 | |
| 				}
 | |
| 				for(n=lp->includes;n;n=n->next) {
 | |
| 					fprintf(dumpfile, "include => %s\n", n->name);
 | |
| 				}
 | |
| 				for(n=lp->switches;n;n=n->next) {
 | |
| 					fprintf(dumpfile, "switch => %s/%s\n", n->name, n->name2);
 | |
| 				}
 | |
| 				for(n=lp->eswitches;n;n=n->next) {
 | |
| 					fprintf(dumpfile, "eswitch => %s/%s\n", n->name, n->name2);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if( dump_extensions && dumpfile )
 | |
| 		fclose(dumpfile);
 | |
| 
 | |
| 	for( lp = context_list; lp; lp = lp2 ) { /* free the ast_context structs */
 | |
| 		lp2 = lp->next;
 | |
| 		lp->next = 0;
 | |
| 
 | |
| 		destroy_namelist(lp->includes);
 | |
| 		destroy_namelist(lp->ignorepats);
 | |
| 		destroy_namelist(lp->switches);
 | |
| 		destroy_namelist(lp->eswitches);
 | |
| 
 | |
| 		free(lp);
 | |
| 	}
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| int ast_hashtab_compare_contexts(const void *ah_a, const void *ah_b);
 | |
| 
 | |
| int ast_hashtab_compare_contexts(const void *ah_a, const void *ah_b)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| unsigned int ast_hashtab_hash_contexts(const void *obj);
 | |
| 
 | |
| unsigned int ast_hashtab_hash_contexts(const void *obj)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| #ifdef DEBUG_THREADS
 | |
| void ast_mark_lock_acquired(void *lock_addr)
 | |
| {
 | |
| }
 | |
| void ast_remove_lock_info(void *lock_addr, struct ast_bt *bt)
 | |
| {
 | |
| }
 | |
| 
 | |
| void ast_store_lock_info(enum ast_lock_type type, const char *filename,
 | |
| 	int line_num, const char *func, const char *lock_name, void *lock_addr, struct ast_bt *bt)
 | |
| {
 | |
| }
 | |
| 
 | |
| #ifdef HAVE_BKTR
 | |
| int __ast_bt_get_addresses(struct ast_bt *bt)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| struct ast_vector_string *__ast_bt_get_symbols(void **addresses, size_t num_frames)
 | |
| {
 | |
| 	return NULL;
 | |
| }
 | |
| #endif /* HAVE_BKTR */
 | |
| void ast_suspend_lock_info(void *lock_addr)
 | |
| {
 | |
| }
 | |
| void ast_restore_lock_info(void *lock_addr)
 | |
| {
 | |
| }
 | |
| #endif /* DEBUG_THREADS */
 |