| 
									
										
										
										
											2003-06-28 18:35:58 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  |  * Asterisk -- A telephony toolkit for Linux. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Generic File Format Support. | 
					
						
							|  |  |  |  *  | 
					
						
							| 
									
										
										
										
											1999-12-11 20:09:45 +00:00
										 |  |  |  * Copyright (C) 1999, Mark Spencer | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Mark Spencer <markster@linux-support.net> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software, distributed under the terms of | 
					
						
							|  |  |  |  * the GNU General Public License | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-23 19:09:13 +00:00
										 |  |  | #include <sys/types.h>
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | #include <asterisk/frame.h>
 | 
					
						
							|  |  |  | #include <asterisk/file.h>
 | 
					
						
							| 
									
										
										
										
											2004-06-30 03:22:29 +00:00
										 |  |  | #include <asterisk/cli.h>
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | #include <asterisk/logger.h>
 | 
					
						
							|  |  |  | #include <asterisk/channel.h>
 | 
					
						
							|  |  |  | #include <asterisk/sched.h>
 | 
					
						
							|  |  |  | #include <asterisk/options.h>
 | 
					
						
							|  |  |  | #include <asterisk/translate.h>
 | 
					
						
							| 
									
										
										
										
											2004-05-06 21:17:06 +00:00
										 |  |  | #include <asterisk/utils.h>
 | 
					
						
							| 
									
										
										
										
											2004-06-22 18:49:00 +00:00
										 |  |  | #include <asterisk/lock.h>
 | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | #include <asterisk/app.h>
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <fcntl.h>
 | 
					
						
							|  |  |  | #include <dirent.h>
 | 
					
						
							| 
									
										
										
										
											2004-06-23 18:00:50 +00:00
										 |  |  | #include <sys/types.h>
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | #include <sys/stat.h>
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | #include "astconf.h"
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct ast_format { | 
					
						
							|  |  |  | 	/* Name of format */ | 
					
						
							|  |  |  | 	char name[80]; | 
					
						
							|  |  |  | 	/* Extensions (separated by | if more than one) 
 | 
					
						
							|  |  |  | 	   this format can read.  First is assumed for writing (e.g. .mp3) */ | 
					
						
							|  |  |  | 	char exts[80]; | 
					
						
							|  |  |  | 	/* Format of frames it uses/provides (one only) */ | 
					
						
							|  |  |  | 	int format; | 
					
						
							|  |  |  | 	/* Open an input stream, and start playback */ | 
					
						
							|  |  |  | 	struct ast_filestream * (*open)(int fd); | 
					
						
							|  |  |  | 	/* Open an output stream, of a given file descriptor and comment it appropriately if applicable */ | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | 	struct ast_filestream * (*rewrite)(int fd, const char *comment); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	/* Write a frame to a channel */ | 
					
						
							|  |  |  | 	int (*write)(struct ast_filestream *, struct ast_frame *); | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	/* seek num samples into file, whence(think normal seek) */ | 
					
						
							|  |  |  | 	int (*seek)(struct ast_filestream *, long offset, int whence); | 
					
						
							|  |  |  | 	/* trunc file to current position */ | 
					
						
							|  |  |  | 	int (*trunc)(struct ast_filestream *fs); | 
					
						
							|  |  |  | 	/* tell current position */ | 
					
						
							|  |  |  | 	long (*tell)(struct ast_filestream *fs); | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 	/* Read the next frame from the filestream (if available) and report when to get next one
 | 
					
						
							|  |  |  | 		(in samples) */ | 
					
						
							|  |  |  | 	struct ast_frame * (*read)(struct ast_filestream *, int *whennext); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	/* Close file, and destroy filestream structure */ | 
					
						
							|  |  |  | 	void (*close)(struct ast_filestream *); | 
					
						
							|  |  |  | 	/* Retrieve file comment */ | 
					
						
							|  |  |  | 	char * (*getcomment)(struct ast_filestream *); | 
					
						
							|  |  |  | 	/* Link */ | 
					
						
							|  |  |  | 	struct ast_format *next; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_filestream { | 
					
						
							|  |  |  | 	/* Everybody reserves a block of AST_RESERVED_POINTERS pointers for us */ | 
					
						
							|  |  |  | 	struct ast_format *fmt; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	int flags; | 
					
						
							|  |  |  | 	mode_t mode; | 
					
						
							|  |  |  | 	char *filename; | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 	char *realfilename; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	/* Video file stream */ | 
					
						
							|  |  |  | 	struct ast_filestream *vfs; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	/* Transparently translate from another format -- just once */ | 
					
						
							|  |  |  | 	struct ast_trans_pvt *trans; | 
					
						
							|  |  |  | 	struct ast_tranlator_pvt *tr; | 
					
						
							| 
									
										
										
										
											2003-05-28 19:45:07 +00:00
										 |  |  | 	int lastwriteformat; | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 	int lasttimeout; | 
					
						
							|  |  |  | 	struct ast_channel *owner; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-09 01:45:08 +00:00
										 |  |  | AST_MUTEX_DEFINE_STATIC(formatlock); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct ast_format *formats = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | int ast_format_register(const char *name, const char *exts, int format, | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 						struct ast_filestream * (*open)(int fd), | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | 						struct ast_filestream * (*rewrite)(int fd, const char *comment), | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 						int (*write)(struct ast_filestream *, struct ast_frame *), | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 						int (*seek)(struct ast_filestream *, long sample_offset, int whence), | 
					
						
							|  |  |  | 						int (*trunc)(struct ast_filestream *), | 
					
						
							|  |  |  | 						long (*tell)(struct ast_filestream *), | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 						struct ast_frame * (*read)(struct ast_filestream *, int *whennext), | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 						void (*close)(struct ast_filestream *), | 
					
						
							|  |  |  | 						char * (*getcomment)(struct ast_filestream *)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_format *tmp; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	if (ast_mutex_lock(&formatlock)) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Unable to lock format list\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	tmp = formats; | 
					
						
							|  |  |  | 	while(tmp) { | 
					
						
							|  |  |  | 		if (!strcasecmp(name, tmp->name)) { | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 			ast_mutex_unlock(&formatlock); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			ast_log(LOG_WARNING, "Tried to register '%s' format, already registered\n", name); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		tmp = tmp->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	tmp = malloc(sizeof(struct ast_format)); | 
					
						
							|  |  |  | 	if (!tmp) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Out of memory\n"); | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 		ast_mutex_unlock(&formatlock); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-10-18 16:47:57 +00:00
										 |  |  | 	strncpy(tmp->name, name, sizeof(tmp->name)-1); | 
					
						
							|  |  |  | 	strncpy(tmp->exts, exts, sizeof(tmp->exts)-1); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	tmp->open = open; | 
					
						
							|  |  |  | 	tmp->rewrite = rewrite; | 
					
						
							|  |  |  | 	tmp->read = read; | 
					
						
							|  |  |  | 	tmp->write = write; | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	tmp->seek = seek; | 
					
						
							|  |  |  | 	tmp->trunc = trunc; | 
					
						
							|  |  |  | 	tmp->tell = tell; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	tmp->close = close; | 
					
						
							|  |  |  | 	tmp->format = format; | 
					
						
							|  |  |  | 	tmp->getcomment = getcomment; | 
					
						
							|  |  |  | 	tmp->next = formats; | 
					
						
							|  |  |  | 	formats = tmp; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&formatlock); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	if (option_verbose > 1) | 
					
						
							|  |  |  | 		ast_verbose( VERBOSE_PREFIX_2 "Registered file format %s, extension(s) %s\n", name, exts); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | int ast_format_unregister(const char *name) | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_format *tmp, *tmpl = NULL; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	if (ast_mutex_lock(&formatlock)) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Unable to lock format list\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	tmp = formats; | 
					
						
							|  |  |  | 	while(tmp) { | 
					
						
							|  |  |  | 		if (!strcasecmp(name, tmp->name)) { | 
					
						
							|  |  |  | 			if (tmpl)  | 
					
						
							|  |  |  | 				tmpl->next = tmp->next; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				formats = tmp->next; | 
					
						
							|  |  |  | 			free(tmp); | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 			ast_mutex_unlock(&formatlock); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			if (option_verbose > 1) | 
					
						
							|  |  |  | 				ast_verbose( VERBOSE_PREFIX_2 "Unregistered format %s\n", name); | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-04-21 03:53:15 +00:00
										 |  |  | 		tmpl = tmp; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		tmp = tmp->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_log(LOG_WARNING, "Tried to unregister format %s, already unregistered\n", name); | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_stopstream(struct ast_channel *tmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Stop a running stream if there is one */ | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	if (tmp->vstream) | 
					
						
							|  |  |  | 		ast_closestream(tmp->vstream); | 
					
						
							|  |  |  | 	if (tmp->stream) { | 
					
						
							|  |  |  | 		ast_closestream(tmp->stream); | 
					
						
							| 
									
										
										
										
											2004-04-06 22:17:32 +00:00
										 |  |  | 		if (tmp->oldwriteformat && ast_set_write_format(tmp, tmp->oldwriteformat)) | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 			ast_log(LOG_WARNING, "Unable to restore format back to %d\n", tmp->oldwriteformat); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_writestream(struct ast_filestream *fs, struct ast_frame *f) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 	struct ast_frame *trf; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	int res = -1; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	int alt=0; | 
					
						
							|  |  |  | 	if (f->frametype == AST_FRAME_VIDEO) { | 
					
						
							|  |  |  | 		if (fs->fmt->format < AST_FORMAT_MAX_AUDIO) { | 
					
						
							|  |  |  | 			/* This is the audio portion.  Call the video one... */ | 
					
						
							|  |  |  | 			if (!fs->vfs && fs->filename) { | 
					
						
							|  |  |  | 				/* XXX Support other video formats XXX */ | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | 				const char *type = "h263"; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 				fs->vfs = ast_writefile(fs->filename, type, NULL, fs->flags, 0, fs->mode); | 
					
						
							|  |  |  | 				ast_log(LOG_DEBUG, "Opened video output file\n"); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (fs->vfs) | 
					
						
							|  |  |  | 				return ast_writestream(fs->vfs, f); | 
					
						
							|  |  |  | 			/* Ignore */ | 
					
						
							|  |  |  | 			return 0;				 | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* Might / might not have mark set */ | 
					
						
							|  |  |  | 			alt = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if (f->frametype != AST_FRAME_VOICE) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Tried to write non-voice frame\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	if (((fs->fmt->format | alt) & f->subclass) == f->subclass) { | 
					
						
							| 
									
										
										
										
											2000-01-02 23:51:30 +00:00
										 |  |  | 		res =  fs->fmt->write(fs, f); | 
					
						
							|  |  |  | 		if (res < 0)  | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Natural write failed\n"); | 
					
						
							|  |  |  | 		if (res > 0) | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Huh??\n"); | 
					
						
							|  |  |  | 		return res; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		/* XXX If they try to send us a type of frame that isn't the normal frame, and isn't
 | 
					
						
							|  |  |  | 		       the one we've setup a translator for, we do the "wrong thing" XXX */ | 
					
						
							| 
									
										
										
										
											2003-05-28 19:45:07 +00:00
										 |  |  | 		if (fs->trans && (f->subclass != fs->lastwriteformat)) { | 
					
						
							|  |  |  | 			ast_translator_free_path(fs->trans); | 
					
						
							|  |  |  | 			fs->trans = NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		if (!fs->trans)  | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 			fs->trans = ast_translator_build_path(fs->fmt->format, f->subclass); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		if (!fs->trans) | 
					
						
							| 
									
										
										
										
											2003-08-16 05:10:35 +00:00
										 |  |  | 			ast_log(LOG_WARNING, "Unable to translate to format %s, source format %s\n", fs->fmt->name, ast_getformatname(f->subclass)); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2003-05-28 19:45:07 +00:00
										 |  |  | 			fs->lastwriteformat = f->subclass; | 
					
						
							| 
									
										
										
										
											2000-01-02 23:51:30 +00:00
										 |  |  | 			res = 0; | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 			/* Get the translated frame but don't consume the original in case they're using it on another stream */ | 
					
						
							|  |  |  | 			trf = ast_translate(fs->trans, f, 0); | 
					
						
							| 
									
										
										
										
											2002-09-03 21:55:10 +00:00
										 |  |  | 			if (trf) { | 
					
						
							|  |  |  | 				res = fs->fmt->write(fs, trf); | 
					
						
							|  |  |  | 				if (res)  | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Translated frame write failed\n"); | 
					
						
							|  |  |  | 			} else | 
					
						
							|  |  |  | 				res = 0; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return res; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | static int copy(const char *infile, const char *outfile) | 
					
						
							| 
									
										
										
										
											2001-06-13 20:36:52 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int ifd; | 
					
						
							|  |  |  | 	int ofd; | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	int len; | 
					
						
							|  |  |  | 	char buf[4096]; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:00:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-13 20:36:52 +00:00
										 |  |  | 	if ((ifd = open(infile, O_RDONLY)) < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to open %s in read-only mode\n", infile); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if ((ofd = open(outfile, O_WRONLY | O_TRUNC | O_CREAT, 0600)) < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to open %s in write-only mode\n", outfile); | 
					
						
							|  |  |  | 		close(ifd); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		len = read(ifd, buf, sizeof(buf)); | 
					
						
							|  |  |  | 		if (len < 0) { | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Read failed on %s: %s\n", infile, strerror(errno)); | 
					
						
							|  |  |  | 			close(ifd); | 
					
						
							|  |  |  | 			close(ofd); | 
					
						
							|  |  |  | 			unlink(outfile); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (len) { | 
					
						
							|  |  |  | 			res = write(ofd, buf, len); | 
					
						
							|  |  |  | 			if (res != len) { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "Write failed on %s (%d of %d): %s\n", outfile, res, len, strerror(errno)); | 
					
						
							|  |  |  | 				close(ifd); | 
					
						
							|  |  |  | 				close(ofd); | 
					
						
							|  |  |  | 				unlink(outfile); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} while(len); | 
					
						
							|  |  |  | 	close(ifd); | 
					
						
							|  |  |  | 	close(ofd); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-14 22:38:34 +00:00
										 |  |  | static char *build_filename(const char *filename, const char *ext) | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	char *fn; | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	int fnsize = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-10 04:18:18 +00:00
										 |  |  | 	if (filename[0] == '/') { | 
					
						
							|  |  |  | 		fnsize = strlen(filename) + strlen(ext) + 2; | 
					
						
							|  |  |  | 		fn = malloc(fnsize); | 
					
						
							|  |  |  | 		if (fn) | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 			snprintf(fn, fnsize, "%s.%s", filename, ext); | 
					
						
							| 
									
										
										
										
											2005-01-10 04:18:18 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		char tmp[AST_CONFIG_MAX_PATH] = ""; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		snprintf(tmp, sizeof(tmp), "%s/%s", ast_config_AST_VAR_DIR, "sounds"); | 
					
						
							|  |  |  | 		fnsize = strlen(tmp) + strlen(filename) + strlen(ext) + 3; | 
					
						
							|  |  |  | 		fn = malloc(fnsize); | 
					
						
							|  |  |  | 		if (fn) | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 			snprintf(fn, fnsize, "%s/%s.%s", tmp, filename, ext); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-01-10 04:18:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	return fn; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-14 22:38:34 +00:00
										 |  |  | static int exts_compare(const char *exts, const char *type) | 
					
						
							| 
									
										
										
										
											2004-02-25 22:31:51 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	char *stringp = NULL, *ext; | 
					
						
							|  |  |  | 	char tmp[256]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	strncpy(tmp, exts, sizeof(tmp) - 1); | 
					
						
							|  |  |  | 	stringp = tmp; | 
					
						
							|  |  |  | 	while ((ext = strsep(&stringp, "|"))) { | 
					
						
							|  |  |  | 		if (!strcmp(ext, type)) { | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | #define ACTION_EXISTS 1
 | 
					
						
							|  |  |  | #define ACTION_DELETE 2
 | 
					
						
							|  |  |  | #define ACTION_RENAME 3
 | 
					
						
							|  |  |  | #define ACTION_OPEN   4
 | 
					
						
							| 
									
										
										
										
											2001-06-13 20:36:52 +00:00
										 |  |  | #define ACTION_COPY   5
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | static int ast_filehelper(const char *filename, const char *filename2, const char *fmt, int action) | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct stat st; | 
					
						
							|  |  |  | 	struct ast_format *f; | 
					
						
							|  |  |  | 	struct ast_filestream *s; | 
					
						
							|  |  |  | 	int res=0, ret = 0; | 
					
						
							|  |  |  | 	char *ext=NULL, *exts, *fn, *nfn; | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 	struct ast_channel *chan = (struct ast_channel *)filename2; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* Start with negative response */ | 
					
						
							|  |  |  | 	if (action == ACTION_EXISTS) | 
					
						
							|  |  |  | 		res = 0; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		res = -1; | 
					
						
							|  |  |  | 	if (action == ACTION_OPEN) | 
					
						
							|  |  |  | 		ret = -1; | 
					
						
							|  |  |  | 	/* Check for a specific format */ | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	if (ast_mutex_lock(&formatlock)) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Unable to lock format list\n"); | 
					
						
							|  |  |  | 		if (action == ACTION_EXISTS) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	f = formats; | 
					
						
							|  |  |  | 	while(f) { | 
					
						
							| 
									
										
										
										
											2004-02-25 22:31:51 +00:00
										 |  |  | 		if (!fmt || exts_compare(f->exts, fmt)) { | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 			char *stringp=NULL; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			exts = strdup(f->exts); | 
					
						
							|  |  |  | 			/* Try each kind of extension */ | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 			stringp=exts; | 
					
						
							|  |  |  | 			ext = strsep(&stringp, "|"); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:00:50 +00:00
										 |  |  | 			if (!strcmp(ext,"wav49")) { | 
					
						
							|  |  |  | 				ext = "WAV"; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			do { | 
					
						
							|  |  |  | 				fn = build_filename(filename, ext); | 
					
						
							|  |  |  | 				if (fn) { | 
					
						
							|  |  |  | 					res = stat(fn, &st); | 
					
						
							|  |  |  | 					if (!res) { | 
					
						
							|  |  |  | 						switch(action) { | 
					
						
							|  |  |  | 						case ACTION_EXISTS: | 
					
						
							|  |  |  | 							ret |= f->format; | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 						case ACTION_DELETE: | 
					
						
							|  |  |  | 							res = unlink(fn); | 
					
						
							|  |  |  | 							if (res) | 
					
						
							|  |  |  | 								ast_log(LOG_WARNING, "unlink(%s) failed: %s\n", fn, strerror(errno)); | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 						case ACTION_RENAME: | 
					
						
							|  |  |  | 							nfn = build_filename(filename2, ext); | 
					
						
							|  |  |  | 							if (nfn) { | 
					
						
							|  |  |  | 								res = rename(fn, nfn); | 
					
						
							|  |  |  | 								if (res) | 
					
						
							|  |  |  | 									ast_log(LOG_WARNING, "rename(%s,%s) failed: %s\n", fn, nfn, strerror(errno)); | 
					
						
							|  |  |  | 								free(nfn); | 
					
						
							|  |  |  | 							} else | 
					
						
							|  |  |  | 								ast_log(LOG_WARNING, "Out of memory\n"); | 
					
						
							|  |  |  | 							break; | 
					
						
							| 
									
										
										
										
											2001-06-13 20:36:52 +00:00
										 |  |  | 						case ACTION_COPY: | 
					
						
							|  |  |  | 							nfn = build_filename(filename2, ext); | 
					
						
							|  |  |  | 							if (nfn) { | 
					
						
							|  |  |  | 								res = copy(fn, nfn); | 
					
						
							|  |  |  | 								if (res) | 
					
						
							|  |  |  | 									ast_log(LOG_WARNING, "copy(%s,%s) failed: %s\n", fn, nfn, strerror(errno)); | 
					
						
							|  |  |  | 								free(nfn); | 
					
						
							|  |  |  | 							} else | 
					
						
							|  |  |  | 								ast_log(LOG_WARNING, "Out of memory\n"); | 
					
						
							|  |  |  | 							break; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 						case ACTION_OPEN: | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 							if ((ret < 0) && ((chan->writeformat & f->format) || | 
					
						
							|  |  |  | 										((f->format >= AST_FORMAT_MAX_AUDIO) && fmt))) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 								ret = open(fn, O_RDONLY); | 
					
						
							|  |  |  | 								if (ret >= 0) { | 
					
						
							|  |  |  | 									s = f->open(ret); | 
					
						
							|  |  |  | 									if (s) { | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 										s->lasttimeout = -1; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 										s->fmt = f; | 
					
						
							|  |  |  | 										s->trans = NULL; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 										s->filename = NULL; | 
					
						
							|  |  |  | 										if (s->fmt->format < AST_FORMAT_MAX_AUDIO) | 
					
						
							|  |  |  | 											chan->stream = s; | 
					
						
							|  |  |  | 										else | 
					
						
							|  |  |  | 											chan->vstream = s; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 									} else { | 
					
						
							|  |  |  | 										close(ret); | 
					
						
							| 
									
										
										
										
											2004-02-25 22:31:51 +00:00
										 |  |  | 										ast_log(LOG_WARNING, "Unable to open fd on %s\n", fn); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 									} | 
					
						
							|  |  |  | 								} else | 
					
						
							|  |  |  | 									ast_log(LOG_WARNING, "Couldn't open file %s\n", fn); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 						default: | 
					
						
							|  |  |  | 							ast_log(LOG_WARNING, "Unknown helper %d\n", action); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						/* Conveniently this logic is the same for all */ | 
					
						
							|  |  |  | 						if (res) | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					free(fn); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 				ext = strsep(&stringp, "|"); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			} while(ext); | 
					
						
							|  |  |  | 			free(exts); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		f = f->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&formatlock); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	if ((action == ACTION_EXISTS) || (action == ACTION_OPEN)) | 
					
						
							|  |  |  | 		res = ret ? ret : -1; | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | struct ast_filestream *ast_openstream(struct ast_channel *chan, const char *filename, const char *preflang) | 
					
						
							| 
									
										
										
										
											2004-12-24 01:40:07 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return ast_openstream_full(chan, filename, preflang, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_filestream *ast_openstream_full(struct ast_channel *chan, const char *filename, const char *preflang, int asis) | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* This is a fairly complex routine.  Essentially we should do 
 | 
					
						
							|  |  |  | 	   the following: | 
					
						
							|  |  |  | 	    | 
					
						
							|  |  |  | 	   1) Find which file handlers produce our type of format. | 
					
						
							|  |  |  | 	   2) Look for a filename which it can handle. | 
					
						
							|  |  |  | 	   3) If we find one, then great.   | 
					
						
							| 
									
										
										
										
											2000-01-02 23:51:30 +00:00
										 |  |  | 	   4) If not, see what files are there | 
					
						
							|  |  |  | 	   5) See what we can actually support | 
					
						
							|  |  |  | 	   6) Choose the one with the least costly translator path and | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	       set it up. | 
					
						
							|  |  |  | 		    | 
					
						
							|  |  |  | 	*/ | 
					
						
							|  |  |  | 	int fd = -1; | 
					
						
							| 
									
										
										
										
											2000-01-09 23:17:36 +00:00
										 |  |  | 	int fmts = -1; | 
					
						
							| 
									
										
										
										
											2003-10-06 00:36:17 +00:00
										 |  |  | 	char filename2[256]=""; | 
					
						
							|  |  |  | 	char filename3[256]=""; | 
					
						
							|  |  |  | 	char *endpart; | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2004-12-24 01:40:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!asis) { | 
					
						
							|  |  |  | 		/* do this first, otherwise we detect the wrong writeformat */ | 
					
						
							|  |  |  | 		ast_stopstream(chan); | 
					
						
							|  |  |  | 		if (chan->generator) | 
					
						
							|  |  |  | 			ast_deactivate_generator(chan); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-05-06 21:17:06 +00:00
										 |  |  | 	if (preflang && !ast_strlen_zero(preflang)) { | 
					
						
							| 
									
										
										
										
											2003-10-06 00:36:17 +00:00
										 |  |  | 		strncpy(filename3, filename, sizeof(filename3) - 1); | 
					
						
							|  |  |  | 		endpart = strrchr(filename3, '/'); | 
					
						
							|  |  |  | 		if (endpart) { | 
					
						
							|  |  |  | 			*endpart = '\0'; | 
					
						
							|  |  |  | 			endpart++; | 
					
						
							|  |  |  | 			snprintf(filename2, sizeof(filename2), "%s/%s/%s", filename3, preflang, endpart); | 
					
						
							|  |  |  | 		} else | 
					
						
							|  |  |  | 			snprintf(filename2, sizeof(filename2), "%s/%s", preflang, filename); | 
					
						
							| 
									
										
										
										
											2000-01-09 23:17:36 +00:00
										 |  |  | 		fmts = ast_fileexists(filename2, NULL, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (fmts < 1) { | 
					
						
							| 
									
										
										
										
											2001-10-18 16:47:57 +00:00
										 |  |  | 		strncpy(filename2, filename, sizeof(filename2)-1); | 
					
						
							| 
									
										
										
										
											2000-01-09 23:17:36 +00:00
										 |  |  | 		fmts = ast_fileexists(filename2, NULL, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	if (fmts < 1) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "File %s does not exist in any format\n", filename); | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 	chan->oldwriteformat = chan->writeformat; | 
					
						
							|  |  |  | 	/* Set the channel to a format we can work with */ | 
					
						
							| 
									
										
										
										
											2004-04-06 22:17:32 +00:00
										 |  |  | 	res = ast_set_write_format(chan, fmts); | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  |  	fd = ast_filehelper(filename2, (char *)chan, NULL, ACTION_OPEN); | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 	if (fd >= 0) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 		return chan->stream; | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | struct ast_filestream *ast_openvstream(struct ast_channel *chan, const char *filename, const char *preflang) | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* This is a fairly complex routine.  Essentially we should do 
 | 
					
						
							|  |  |  | 	   the following: | 
					
						
							|  |  |  | 	    | 
					
						
							|  |  |  | 	   1) Find which file handlers produce our type of format. | 
					
						
							|  |  |  | 	   2) Look for a filename which it can handle. | 
					
						
							|  |  |  | 	   3) If we find one, then great.   | 
					
						
							|  |  |  | 	   4) If not, see what files are there | 
					
						
							|  |  |  | 	   5) See what we can actually support | 
					
						
							|  |  |  | 	   6) Choose the one with the least costly translator path and | 
					
						
							|  |  |  | 	       set it up. | 
					
						
							|  |  |  | 		    | 
					
						
							|  |  |  | 	*/ | 
					
						
							|  |  |  | 	int fd = -1; | 
					
						
							|  |  |  | 	int fmts = -1; | 
					
						
							|  |  |  | 	char filename2[256]; | 
					
						
							|  |  |  | 	char lang2[MAX_LANGUAGE]; | 
					
						
							|  |  |  | 	/* XXX H.263 only XXX */ | 
					
						
							|  |  |  | 	char *fmt = "h263"; | 
					
						
							| 
									
										
										
										
											2004-05-06 21:17:06 +00:00
										 |  |  | 	if (preflang && !ast_strlen_zero(preflang)) { | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 		snprintf(filename2, sizeof(filename2), "%s/%s", preflang, filename); | 
					
						
							|  |  |  | 		fmts = ast_fileexists(filename2, fmt, NULL); | 
					
						
							|  |  |  | 		if (fmts < 1) { | 
					
						
							|  |  |  | 			strncpy(lang2, preflang, sizeof(lang2)-1); | 
					
						
							|  |  |  | 			snprintf(filename2, sizeof(filename2), "%s/%s", lang2, filename); | 
					
						
							|  |  |  | 			fmts = ast_fileexists(filename2, fmt, NULL); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (fmts < 1) { | 
					
						
							|  |  |  | 		strncpy(filename2, filename, sizeof(filename2)-1); | 
					
						
							|  |  |  | 		fmts = ast_fileexists(filename2, fmt, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (fmts < 1) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |  	fd = ast_filehelper(filename2, (char *)chan, fmt, ACTION_OPEN); | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 	if (fd >= 0) | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 		return chan->vstream; | 
					
						
							|  |  |  | 	ast_log(LOG_WARNING, "File %s has video but couldn't be opened\n", filename); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-12-19 18:06:29 +00:00
										 |  |  | struct ast_frame *ast_readframe(struct ast_filestream *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_frame *f = NULL; | 
					
						
							|  |  |  | 	int whennext = 0;	 | 
					
						
							|  |  |  | 	if (s && s->fmt) | 
					
						
							|  |  |  | 		f = s->fmt->read(s, &whennext); | 
					
						
							|  |  |  | 	return f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | static int ast_readaudio_callback(void *data) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 	struct ast_filestream *s = data; | 
					
						
							|  |  |  | 	struct ast_frame *fr; | 
					
						
							|  |  |  | 	int whennext = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(!whennext) { | 
					
						
							|  |  |  | 		fr = s->fmt->read(s, &whennext); | 
					
						
							|  |  |  | 		if (fr) { | 
					
						
							|  |  |  | 			if (ast_write(s->owner, fr)) { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "Failed to write frame\n"); | 
					
						
							|  |  |  | 				s->owner->streamid = -1; | 
					
						
							| 
									
										
										
										
											2003-06-29 20:32:26 +00:00
										 |  |  | #ifdef ZAPTEL_OPTIMIZATIONS
 | 
					
						
							|  |  |  | 				ast_settimeout(s->owner, 0, NULL, NULL); | 
					
						
							|  |  |  | #endif			
 | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 				return 0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* Stream has finished */ | 
					
						
							|  |  |  | 			s->owner->streamid = -1; | 
					
						
							| 
									
										
										
										
											2003-06-29 20:32:26 +00:00
										 |  |  | #ifdef ZAPTEL_OPTIMIZATIONS
 | 
					
						
							|  |  |  | 			ast_settimeout(s->owner, 0, NULL, NULL); | 
					
						
							|  |  |  | #endif			
 | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (whennext != s->lasttimeout) { | 
					
						
							| 
									
										
										
										
											2003-06-29 20:32:26 +00:00
										 |  |  | #ifdef ZAPTEL_OPTIMIZATIONS
 | 
					
						
							| 
									
										
										
										
											2003-07-01 23:13:05 +00:00
										 |  |  | 		if (s->owner->timingfd > -1) | 
					
						
							| 
									
										
										
										
											2003-07-01 23:09:18 +00:00
										 |  |  | 			ast_settimeout(s->owner, whennext, ast_readaudio_callback, s); | 
					
						
							|  |  |  | 		else | 
					
						
							| 
									
										
										
										
											2003-06-29 20:32:26 +00:00
										 |  |  | #endif		
 | 
					
						
							| 
									
										
										
										
											2003-07-01 23:17:10 +00:00
										 |  |  | 			s->owner->streamid = ast_sched_add(s->owner->sched, whennext/8, ast_readaudio_callback, s); | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 		s->lasttimeout = whennext; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | static int ast_readvideo_callback(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_filestream *s = data; | 
					
						
							|  |  |  | 	struct ast_frame *fr; | 
					
						
							|  |  |  | 	int whennext = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(!whennext) { | 
					
						
							|  |  |  | 		fr = s->fmt->read(s, &whennext); | 
					
						
							|  |  |  | 		if (fr) { | 
					
						
							|  |  |  | 			if (ast_write(s->owner, fr)) { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "Failed to write frame\n"); | 
					
						
							|  |  |  | 				s->owner->vstreamid = -1; | 
					
						
							|  |  |  | 				return 0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* Stream has finished */ | 
					
						
							|  |  |  | 			s->owner->vstreamid = -1; | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (whennext != s->lasttimeout) { | 
					
						
							|  |  |  | 		s->owner->vstreamid = ast_sched_add(s->owner->sched, whennext/8, ast_readvideo_callback, s); | 
					
						
							|  |  |  | 		s->lasttimeout = whennext; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | int ast_applystream(struct ast_channel *chan, struct ast_filestream *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	s->owner = chan; | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_playstream(struct ast_filestream *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	if (s->fmt->format < AST_FORMAT_MAX_AUDIO) | 
					
						
							|  |  |  | 		ast_readaudio_callback(s); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		ast_readvideo_callback(s); | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_seekstream(struct ast_filestream *fs, long sample_offset, int whence) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return fs->fmt->seek(fs, sample_offset, whence); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_truncstream(struct ast_filestream *fs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return fs->fmt->trunc(fs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | long ast_tellstream(struct ast_filestream *fs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return fs->fmt->tell(fs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_stream_fastforward(struct ast_filestream *fs, long ms) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* I think this is right, 8000 samples per second, 1000 ms a second so 8
 | 
					
						
							|  |  |  | 	 * samples per ms  */ | 
					
						
							|  |  |  | 	long samples = ms * 8; | 
					
						
							|  |  |  | 	return ast_seekstream(fs, samples, SEEK_CUR); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_stream_rewind(struct ast_filestream *fs, long ms) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	long samples = ms * 8; | 
					
						
							|  |  |  | 	samples = samples * -1; | 
					
						
							|  |  |  | 	return ast_seekstream(fs, samples, SEEK_CUR); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_closestream(struct ast_filestream *f) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 	char *cmd = NULL; | 
					
						
							|  |  |  | 	size_t size = 0; | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	/* Stop a running stream if there is one */ | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 	if (f->owner) { | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 		if (f->fmt->format < AST_FORMAT_MAX_AUDIO) { | 
					
						
							| 
									
										
										
										
											2003-06-29 04:50:45 +00:00
										 |  |  | 			f->owner->stream = NULL; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 			if (f->owner->streamid > -1) | 
					
						
							|  |  |  | 				ast_sched_del(f->owner->sched, f->owner->streamid); | 
					
						
							|  |  |  | 			f->owner->streamid = -1; | 
					
						
							| 
									
										
										
										
											2003-06-29 20:32:26 +00:00
										 |  |  | #ifdef ZAPTEL_OPTIMIZATIONS
 | 
					
						
							|  |  |  | 			ast_settimeout(f->owner, 0, NULL, NULL); | 
					
						
							|  |  |  | #endif			
 | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2003-06-29 04:50:45 +00:00
										 |  |  | 			f->owner->vstream = NULL; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 			if (f->owner->vstreamid > -1) | 
					
						
							|  |  |  | 				ast_sched_del(f->owner->sched, f->owner->vstreamid); | 
					
						
							|  |  |  | 			f->owner->vstreamid = -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-06-28 22:50:47 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-08-11 20:24:14 +00:00
										 |  |  | 	/* destroy the translator on exit */ | 
					
						
							|  |  |  | 	if (f->trans) { | 
					
						
							|  |  |  | 		ast_translator_free_path(f->trans); | 
					
						
							|  |  |  | 		f->trans = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (f->realfilename && f->filename) { | 
					
						
							|  |  |  | 			size = strlen(f->filename) + strlen(f->realfilename) + 15; | 
					
						
							|  |  |  | 			cmd = alloca(size); | 
					
						
							|  |  |  | 			memset(cmd,0,size); | 
					
						
							|  |  |  | 			snprintf(cmd,size,"/bin/mv -f %s %s",f->filename,f->realfilename); | 
					
						
							|  |  |  | 			ast_safe_system(cmd); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-09-20 23:15:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 	if (f->filename) { | 
					
						
							|  |  |  | 		free(f->filename); | 
					
						
							|  |  |  | 		f->filename = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (f->realfilename) { | 
					
						
							|  |  |  | 		free(f->realfilename); | 
					
						
							|  |  |  | 		f->realfilename = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-09-20 23:15:49 +00:00
										 |  |  | 	f->fmt->close(f); | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | int ast_fileexists(const char *filename, const char *fmt, const char *preflang) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	char filename2[256]; | 
					
						
							|  |  |  | 	char tmp[256]; | 
					
						
							|  |  |  | 	char *postfix; | 
					
						
							|  |  |  | 	char *prefix; | 
					
						
							|  |  |  | 	char *c; | 
					
						
							|  |  |  | 	char lang2[MAX_LANGUAGE]; | 
					
						
							|  |  |  | 	int res = -1; | 
					
						
							| 
									
										
										
										
											2004-05-06 21:17:06 +00:00
										 |  |  | 	if (preflang && !ast_strlen_zero(preflang)) { | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 		/* Insert the language between the last two parts of the path */ | 
					
						
							|  |  |  | 		strncpy(tmp, filename, sizeof(tmp) - 1); | 
					
						
							|  |  |  | 		c = strrchr(tmp, '/'); | 
					
						
							|  |  |  | 		if (c) { | 
					
						
							|  |  |  | 			*c = '\0'; | 
					
						
							|  |  |  | 			postfix = c+1; | 
					
						
							|  |  |  | 			prefix = tmp; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			postfix = tmp; | 
					
						
							|  |  |  | 			prefix=""; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		snprintf(filename2, sizeof(filename2), "%s/%s/%s", prefix, preflang, postfix); | 
					
						
							|  |  |  | 		res = ast_filehelper(filename2, NULL, fmt, ACTION_EXISTS); | 
					
						
							|  |  |  | 		if (res < 1) { | 
					
						
							|  |  |  | 			char *stringp=NULL; | 
					
						
							|  |  |  | 			strncpy(lang2, preflang, sizeof(lang2)-1); | 
					
						
							|  |  |  | 			stringp=lang2; | 
					
						
							|  |  |  | 			strsep(&stringp, "_"); | 
					
						
							|  |  |  | 			if (strcmp(lang2, preflang)) { | 
					
						
							|  |  |  | 				snprintf(filename2, sizeof(filename2), "%s/%s/%s", prefix, lang2, postfix); | 
					
						
							|  |  |  | 				res = ast_filehelper(filename2, NULL, fmt, ACTION_EXISTS); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (res < 1) { | 
					
						
							|  |  |  | 		res = ast_filehelper(filename, NULL, fmt, ACTION_EXISTS); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | int ast_filedelete(const char *filename, const char *fmt) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return ast_filehelper(filename, NULL, fmt, ACTION_DELETE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | int ast_filerename(const char *filename, const char *filename2, const char *fmt) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return ast_filehelper(filename, filename2, fmt, ACTION_RENAME); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | int ast_filecopy(const char *filename, const char *filename2, const char *fmt) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return ast_filehelper(filename, filename2, fmt, ACTION_COPY); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | int ast_streamfile(struct ast_channel *chan, const char *filename, const char *preflang) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_filestream *fs; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	struct ast_filestream *vfs; | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	fs = ast_openstream(chan, filename, preflang); | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 	vfs = ast_openvstream(chan, filename, preflang); | 
					
						
							|  |  |  | 	if (vfs) | 
					
						
							|  |  |  | 		ast_log(LOG_DEBUG, "Ooh, found a video stream, too\n"); | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 	if (fs){ | 
					
						
							|  |  |  | 		if (ast_applystream(chan, fs)) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 			return -1; | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 		if (vfs && ast_applystream(chan, vfs)) | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 			return -1; | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 		if (ast_playstream(fs)) | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 			return -1; | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 		if (vfs && ast_playstream(vfs)) | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 			return -1; | 
					
						
							| 
									
										
										
										
											2000-01-09 23:17:36 +00:00
										 |  |  | #if 1
 | 
					
						
							|  |  |  | 		if (option_verbose > 2) | 
					
						
							| 
									
										
										
										
											2003-10-25 18:15:04 +00:00
										 |  |  | 			ast_verbose(VERBOSE_PREFIX_3 "Playing '%s' (language '%s')\n", filename, preflang ? preflang : "default"); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-08-16 05:10:35 +00:00
										 |  |  | 	ast_log(LOG_WARNING, "Unable to open %s (format %s): %s\n", filename, ast_getformatname(chan->nativeformats), strerror(errno)); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | struct ast_filestream *ast_readfile(const char *filename, const char *type, const char *comment, int flags, int check, mode_t mode) | 
					
						
							| 
									
										
										
										
											2003-12-19 18:06:29 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int fd,myflags = 0; | 
					
						
							|  |  |  | 	struct ast_format *f; | 
					
						
							|  |  |  | 	struct ast_filestream *fs=NULL; | 
					
						
							|  |  |  | 	char *fn; | 
					
						
							|  |  |  | 	char *ext; | 
					
						
							|  |  |  | 	if (ast_mutex_lock(&formatlock)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to lock format list\n"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	f = formats; | 
					
						
							|  |  |  | 	while(f) { | 
					
						
							| 
									
										
										
										
											2004-02-25 22:31:51 +00:00
										 |  |  | 		if (exts_compare(f->exts, type)) { | 
					
						
							| 
									
										
										
										
											2003-12-19 18:06:29 +00:00
										 |  |  | 			char *stringp=NULL; | 
					
						
							|  |  |  | 			/* XXX Implement check XXX */ | 
					
						
							|  |  |  | 			ext = strdup(f->exts); | 
					
						
							|  |  |  | 			stringp=ext; | 
					
						
							|  |  |  | 			ext = strsep(&stringp, "|"); | 
					
						
							|  |  |  | 			fn = build_filename(filename, ext); | 
					
						
							|  |  |  | 			fd = open(fn, flags | myflags); | 
					
						
							|  |  |  | 			if (fd >= 0) { | 
					
						
							|  |  |  | 				errno = 0; | 
					
						
							|  |  |  | 				if ((fs = f->open(fd))) { | 
					
						
							|  |  |  | 					fs->trans = NULL; | 
					
						
							|  |  |  | 					fs->fmt = f; | 
					
						
							|  |  |  | 					fs->flags = flags; | 
					
						
							|  |  |  | 					fs->mode = mode; | 
					
						
							|  |  |  | 					fs->filename = strdup(filename); | 
					
						
							|  |  |  | 					fs->vfs = NULL; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Unable to open %s\n", fn); | 
					
						
							|  |  |  | 					close(fd); | 
					
						
							|  |  |  | 					unlink(fn); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else if (errno != EEXIST) | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "Unable to open file %s: %s\n", fn, strerror(errno)); | 
					
						
							|  |  |  | 			free(fn); | 
					
						
							|  |  |  | 			free(ext); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		f = f->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_mutex_unlock(&formatlock); | 
					
						
							|  |  |  | 	if (!f)  | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "No such format '%s'\n", type); | 
					
						
							|  |  |  | 	return fs; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | struct ast_filestream *ast_writefile(const char *filename, const char *type, const char *comment, int flags, int check, mode_t mode) | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-12-19 18:06:29 +00:00
										 |  |  | 	int fd,myflags = 0; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	struct ast_format *f; | 
					
						
							|  |  |  | 	struct ast_filestream *fs=NULL; | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 	char *fn,*orig_fn=NULL; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	char *ext; | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 	char *buf=NULL; | 
					
						
							|  |  |  | 	size_t size = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	if (ast_mutex_lock(&formatlock)) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Unable to lock format list\n"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 	/* set the O_TRUNC flag if and only if there is no O_APPEND specified */ | 
					
						
							| 
									
										
										
										
											2005-02-03 05:35:03 +00:00
										 |  |  | 	if (flags & O_APPEND) {  | 
					
						
							|  |  |  | 		/* We really can't use O_APPEND as it will break WAV header updates */ | 
					
						
							|  |  |  | 		flags &= ~O_APPEND; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2003-12-19 18:06:29 +00:00
										 |  |  | 		myflags = O_TRUNC; | 
					
						
							| 
									
										
										
										
											2005-02-03 05:35:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-12-19 18:06:29 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	myflags |= O_WRONLY | O_CREAT; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	f = formats; | 
					
						
							|  |  |  | 	while(f) { | 
					
						
							| 
									
										
										
										
											2004-02-25 22:31:51 +00:00
										 |  |  | 		if (exts_compare(f->exts, type)) { | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 			char *stringp=NULL; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			/* XXX Implement check XXX */ | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 			ext = ast_strdupa(f->exts); | 
					
						
							| 
									
										
										
										
											2003-02-06 22:11:43 +00:00
										 |  |  | 			stringp=ext; | 
					
						
							|  |  |  | 			ext = strsep(&stringp, "|"); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			fn = build_filename(filename, ext); | 
					
						
							| 
									
										
										
										
											2003-12-19 18:06:29 +00:00
										 |  |  | 			fd = open(fn, flags | myflags, mode); | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (option_cache_record_files && fd >= 0) { | 
					
						
							|  |  |  | 				close(fd); | 
					
						
							|  |  |  | 				/*
 | 
					
						
							|  |  |  | 				   We touch orig_fn just as a place-holder so other things (like vmail) see the file is there. | 
					
						
							|  |  |  | 				   What we are really doing is writing to record_cache_dir until we are done then we will mv the file into place. | 
					
						
							|  |  |  | 				*/ | 
					
						
							|  |  |  | 				orig_fn = ast_strdupa(fn);  | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 				for (size=0;size<strlen(fn);size++) { | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 					if (fn[size] == '/') | 
					
						
							|  |  |  | 						fn[size] = '_'; | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				size += (strlen(record_cache_dir) + 10); | 
					
						
							|  |  |  | 				buf = alloca(size); | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 				memset(buf, 0, size); | 
					
						
							|  |  |  | 				snprintf(buf, size, "%s/%s", record_cache_dir, fn); | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 				free(fn); | 
					
						
							|  |  |  | 				fn=buf; | 
					
						
							|  |  |  | 				fd = open(fn, flags | myflags, mode); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			if (fd >= 0) { | 
					
						
							|  |  |  | 				errno = 0; | 
					
						
							|  |  |  | 				if ((fs = f->rewrite(fd, comment))) { | 
					
						
							|  |  |  | 					fs->trans = NULL; | 
					
						
							|  |  |  | 					fs->fmt = f; | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 					fs->flags = flags; | 
					
						
							|  |  |  | 					fs->mode = mode; | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 					if (option_cache_record_files) { | 
					
						
							|  |  |  | 						fs->realfilename = build_filename(filename, ext); | 
					
						
							|  |  |  | 						fs->filename = strdup(fn); | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 					} else { | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 						fs->realfilename = NULL; | 
					
						
							|  |  |  | 						fs->filename = strdup(filename); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2003-06-29 03:24:39 +00:00
										 |  |  | 					fs->vfs = NULL; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 				} else { | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Unable to rewrite %s\n", fn); | 
					
						
							|  |  |  | 					close(fd); | 
					
						
							|  |  |  | 					unlink(fn); | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 					if (orig_fn) | 
					
						
							|  |  |  | 						unlink(orig_fn); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 			} else if (errno != EEXIST) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 				ast_log(LOG_WARNING, "Unable to open file %s: %s\n", fn, strerror(errno)); | 
					
						
							| 
									
										
										
										
											2004-09-07 15:01:48 +00:00
										 |  |  | 				if (orig_fn) | 
					
						
							| 
									
										
										
										
											2004-09-07 01:49:08 +00:00
										 |  |  | 					unlink(orig_fn); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (!buf) /* if buf != NULL then fn is already free and pointing to it */ | 
					
						
							|  |  |  | 				free(fn); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		f = f->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&formatlock); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	if (!f)  | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "No such format '%s'\n", type); | 
					
						
							|  |  |  | 	return fs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | char ast_waitstream(struct ast_channel *c, const char *breakon) | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-02-23 06:00:11 +00:00
										 |  |  | 	/* XXX Maybe I should just front-end ast_waitstream_full ? XXX */ | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 	struct ast_frame *fr; | 
					
						
							| 
									
										
										
										
											2004-07-15 14:37:09 +00:00
										 |  |  | 	if (!breakon) breakon = ""; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	while(c->stream) { | 
					
						
							|  |  |  | 		res = ast_sched_wait(c->sched); | 
					
						
							| 
									
										
										
										
											2003-06-29 20:32:26 +00:00
										 |  |  | 		if ((res < 0) && !c->timingfunc) { | 
					
						
							| 
									
										
										
										
											2003-07-01 16:16:28 +00:00
										 |  |  | 			ast_stopstream(c); | 
					
						
							| 
									
										
										
										
											2002-05-15 21:29:26 +00:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-06-29 20:32:26 +00:00
										 |  |  | 		if (res < 0) | 
					
						
							|  |  |  | 			res = 1000; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 		res = ast_waitfor(c, res); | 
					
						
							|  |  |  | 		if (res < 0) { | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Select failed (%s)\n", strerror(errno)); | 
					
						
							|  |  |  | 			return res; | 
					
						
							| 
									
										
										
										
											2003-05-12 04:23:55 +00:00
										 |  |  | 		} else if (res > 0) { | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 			fr = ast_read(c); | 
					
						
							|  |  |  | 			if (!fr) { | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 				ast_log(LOG_DEBUG, "Got hung up\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 				return -1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			switch(fr->frametype) { | 
					
						
							|  |  |  | 			case AST_FRAME_DTMF: | 
					
						
							|  |  |  | 				res = fr->subclass; | 
					
						
							| 
									
										
										
										
											2002-05-15 21:29:26 +00:00
										 |  |  | 				if (strchr(breakon, res)) { | 
					
						
							|  |  |  | 					ast_frfree(fr); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 					return res; | 
					
						
							| 
									
										
										
										
											2002-05-15 21:29:26 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			case AST_FRAME_CONTROL: | 
					
						
							|  |  |  | 				switch(fr->subclass) { | 
					
						
							|  |  |  | 				case AST_CONTROL_HANGUP: | 
					
						
							|  |  |  | 					ast_frfree(fr); | 
					
						
							|  |  |  | 					return -1; | 
					
						
							| 
									
										
										
										
											2001-05-03 23:57:34 +00:00
										 |  |  | 				case AST_CONTROL_RINGING: | 
					
						
							|  |  |  | 				case AST_CONTROL_ANSWER: | 
					
						
							|  |  |  | 					/* Unimportant */ | 
					
						
							|  |  |  | 					break; | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 				default: | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Unexpected control subclass '%d'\n", fr->subclass); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2002-05-15 21:29:26 +00:00
										 |  |  | 			/* Ignore */ | 
					
						
							|  |  |  | 			ast_frfree(fr); | 
					
						
							| 
									
										
										
										
											2003-05-12 04:23:55 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		ast_sched_runq(c->sched); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-09-03 21:55:10 +00:00
										 |  |  | 	return (c->_softhangup ? -1 : 0); | 
					
						
							| 
									
										
										
										
											1999-12-02 20:24:59 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | char ast_waitstream_fr(struct ast_channel *c, const char *breakon, const char *forward, const char *rewind, int ms) | 
					
						
							| 
									
										
										
										
											2003-04-03 07:37:30 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	struct ast_frame *fr; | 
					
						
							| 
									
										
										
										
											2005-02-06 06:43:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!breakon) | 
					
						
							|  |  |  | 			breakon = ""; | 
					
						
							|  |  |  | 	if (!forward) | 
					
						
							|  |  |  | 			forward = ""; | 
					
						
							|  |  |  | 	if (!rewind) | 
					
						
							|  |  |  | 			rewind = ""; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2003-04-03 07:37:30 +00:00
										 |  |  | 	while(c->stream) { | 
					
						
							|  |  |  | 		res = ast_sched_wait(c->sched); | 
					
						
							| 
									
										
										
										
											2003-06-29 21:54:58 +00:00
										 |  |  | 		if ((res < 0) && !c->timingfunc) { | 
					
						
							| 
									
										
										
										
											2003-07-01 16:16:28 +00:00
										 |  |  | 			ast_stopstream(c); | 
					
						
							| 
									
										
										
										
											2003-04-03 07:37:30 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-06-29 21:54:58 +00:00
										 |  |  | 		if (res < 0) | 
					
						
							|  |  |  | 			res = 1000; | 
					
						
							| 
									
										
										
										
											2003-04-03 07:37:30 +00:00
										 |  |  | 		res = ast_waitfor(c, res); | 
					
						
							|  |  |  | 		if (res < 0) { | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Select failed (%s)\n", strerror(errno)); | 
					
						
							|  |  |  | 			return res; | 
					
						
							|  |  |  | 		} else | 
					
						
							|  |  |  | 		if (res > 0) { | 
					
						
							|  |  |  | 			fr = ast_read(c); | 
					
						
							|  |  |  | 			if (!fr) { | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 				ast_log(LOG_DEBUG, "Got hung up\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 				return -1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			switch(fr->frametype) { | 
					
						
							|  |  |  | 			case AST_FRAME_DTMF: | 
					
						
							|  |  |  | 				res = fr->subclass; | 
					
						
							|  |  |  | 				if (strchr(forward,res)) { | 
					
						
							| 
									
										
										
										
											2003-04-09 02:10:29 +00:00
										 |  |  | 					ast_stream_fastforward(c->stream, ms); | 
					
						
							| 
									
										
										
										
											2003-04-03 07:37:30 +00:00
										 |  |  | 				} else if (strchr(rewind,res)) { | 
					
						
							| 
									
										
										
										
											2003-04-09 02:10:29 +00:00
										 |  |  | 					ast_stream_rewind(c->stream, ms); | 
					
						
							| 
									
										
										
										
											2003-04-03 07:37:30 +00:00
										 |  |  | 				} else if (strchr(breakon, res)) { | 
					
						
							|  |  |  | 					ast_frfree(fr); | 
					
						
							|  |  |  | 					return res; | 
					
						
							|  |  |  | 				}					 | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case AST_FRAME_CONTROL: | 
					
						
							|  |  |  | 				switch(fr->subclass) { | 
					
						
							|  |  |  | 				case AST_CONTROL_HANGUP: | 
					
						
							|  |  |  | 					ast_frfree(fr); | 
					
						
							|  |  |  | 					return -1; | 
					
						
							|  |  |  | 				case AST_CONTROL_RINGING: | 
					
						
							|  |  |  | 				case AST_CONTROL_ANSWER: | 
					
						
							|  |  |  | 					/* Unimportant */ | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				default: | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Unexpected control subclass '%d'\n", fr->subclass); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			/* Ignore */ | 
					
						
							|  |  |  | 			ast_frfree(fr); | 
					
						
							|  |  |  | 		} else | 
					
						
							|  |  |  | 			ast_sched_runq(c->sched); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return (c->_softhangup ? -1 : 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-13 16:13:07 +00:00
										 |  |  | char ast_waitstream_full(struct ast_channel *c, const char *breakon, int audiofd, int cmdfd) | 
					
						
							| 
									
										
										
										
											2003-02-23 06:00:11 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	int ms; | 
					
						
							|  |  |  | 	int outfd; | 
					
						
							|  |  |  | 	struct ast_frame *fr; | 
					
						
							|  |  |  | 	struct ast_channel *rchan; | 
					
						
							| 
									
										
										
										
											2005-02-06 06:43:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!breakon) | 
					
						
							|  |  |  | 		breakon = ""; | 
					
						
							| 
									
										
										
										
											2003-02-23 06:00:11 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	while(c->stream) { | 
					
						
							|  |  |  | 		ms = ast_sched_wait(c->sched); | 
					
						
							| 
									
										
										
										
											2003-06-29 21:54:58 +00:00
										 |  |  | 		if ((ms < 0) && !c->timingfunc) { | 
					
						
							| 
									
										
										
										
											2003-07-01 16:16:28 +00:00
										 |  |  | 			ast_stopstream(c); | 
					
						
							| 
									
										
										
										
											2003-02-23 06:00:11 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-06-29 21:54:58 +00:00
										 |  |  | 		if (ms < 0) | 
					
						
							|  |  |  | 			ms = 1000; | 
					
						
							| 
									
										
										
										
											2003-02-23 06:00:11 +00:00
										 |  |  | 		rchan = ast_waitfor_nandfds(&c, 1, &cmdfd, (cmdfd > -1) ? 1 : 0, NULL, &outfd, &ms); | 
					
						
							|  |  |  | 		if (!rchan && (outfd < 0) && (ms)) { | 
					
						
							| 
									
										
										
										
											2005-03-22 20:34:04 +00:00
										 |  |  | 			/* Continue */ | 
					
						
							|  |  |  | 			if (errno == EINTR) | 
					
						
							|  |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2003-02-23 06:00:11 +00:00
										 |  |  | 			ast_log(LOG_WARNING, "Wait failed (%s)\n", strerror(errno)); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} else if (outfd > -1) { | 
					
						
							|  |  |  | 			/* The FD we were watching has something waiting */ | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		} else if (rchan) { | 
					
						
							|  |  |  | 			fr = ast_read(c); | 
					
						
							|  |  |  | 			if (!fr) { | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 				ast_log(LOG_DEBUG, "Got hung up\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 				return -1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			switch(fr->frametype) { | 
					
						
							|  |  |  | 			case AST_FRAME_DTMF: | 
					
						
							|  |  |  | 				res = fr->subclass; | 
					
						
							|  |  |  | 				if (strchr(breakon, res)) { | 
					
						
							|  |  |  | 					ast_frfree(fr); | 
					
						
							|  |  |  | 					return res; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case AST_FRAME_CONTROL: | 
					
						
							|  |  |  | 				switch(fr->subclass) { | 
					
						
							|  |  |  | 				case AST_CONTROL_HANGUP: | 
					
						
							|  |  |  | 					ast_frfree(fr); | 
					
						
							|  |  |  | 					return -1; | 
					
						
							|  |  |  | 				case AST_CONTROL_RINGING: | 
					
						
							|  |  |  | 				case AST_CONTROL_ANSWER: | 
					
						
							|  |  |  | 					/* Unimportant */ | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				default: | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Unexpected control subclass '%d'\n", fr->subclass); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			case AST_FRAME_VOICE: | 
					
						
							|  |  |  | 				/* Write audio if appropriate */ | 
					
						
							|  |  |  | 				if (audiofd > -1) | 
					
						
							|  |  |  | 					write(audiofd, fr->data, fr->datalen); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			/* Ignore */ | 
					
						
							|  |  |  | 			ast_frfree(fr); | 
					
						
							| 
									
										
										
										
											2003-05-12 04:23:55 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		ast_sched_runq(c->sched); | 
					
						
							| 
									
										
										
										
											2003-02-23 06:00:11 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return (c->_softhangup ? -1 : 0); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2004-06-30 03:22:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int show_file_formats(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #define FORMAT "%-10s %-10s %-20s\n"
 | 
					
						
							|  |  |  | #define FORMAT2 "%-10s %-10s %-20s\n"
 | 
					
						
							|  |  |  | 	struct ast_format *f; | 
					
						
							|  |  |  | 	if (argc != 3) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	ast_cli(fd, FORMAT, "Format", "Name", "Extensions"); | 
					
						
							|  |  |  | 	         | 
					
						
							|  |  |  | 	if (ast_mutex_lock(&formatlock)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to lock format list\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2004-09-07 14:36:56 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-06-30 03:22:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	f = formats; | 
					
						
							|  |  |  | 	while(f) { | 
					
						
							|  |  |  | 		ast_cli(fd, FORMAT2, ast_getformatname(f->format), f->name, f->exts); | 
					
						
							|  |  |  | 		f = f->next; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	ast_mutex_unlock(&formatlock); | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_cli_entry show_file = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	{ "show", "file", "formats" }, | 
					
						
							|  |  |  | 	show_file_formats, | 
					
						
							|  |  |  | 	"Displays file formats", | 
					
						
							|  |  |  | 	"Usage: show file formats\n" | 
					
						
							|  |  |  | 	"       displays currently registered file formats (if any)\n" | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_file_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ast_cli_register(&show_file); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } |