| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- A telephony toolkit for Linux. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Channel Management | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Copyright (C) 1999, Mark Spencer | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Mark Spencer <markster@linux-support.net> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software, distributed under the terms of | 
					
						
							|  |  |  |  * the GNU General Public License | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <pthread.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <sys/time.h>
 | 
					
						
							|  |  |  | #include <sys/stat.h>
 | 
					
						
							|  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <asterisk/sched.h>
 | 
					
						
							|  |  |  | #include <asterisk/options.h>
 | 
					
						
							|  |  |  | #include <asterisk/channel.h>
 | 
					
						
							|  |  |  | #include <asterisk/channel_pvt.h>
 | 
					
						
							|  |  |  | #include <asterisk/logger.h>
 | 
					
						
							|  |  |  | #include <asterisk/file.h>
 | 
					
						
							|  |  |  | #include <asterisk/image.h>
 | 
					
						
							|  |  |  | #include <asterisk/translate.h>
 | 
					
						
							|  |  |  | #include <asterisk/cli.h>
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | #include "astconf.h"
 | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct ast_imager *list; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | static ast_mutex_t listlock = AST_MUTEX_INITIALIZER; | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int ast_image_register(struct ast_imager *img) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (option_verbose > 1) | 
					
						
							|  |  |  | 		ast_verbose(VERBOSE_PREFIX_2 "Registered format '%s' (%s)\n", img->name, img->desc); | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&listlock); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 	img->next = list; | 
					
						
							|  |  |  | 	list = img; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&listlock); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ast_image_unregister(struct ast_imager *img) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_imager *i, *prev = NULL; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&listlock); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 	i = list; | 
					
						
							|  |  |  | 	while(i) { | 
					
						
							|  |  |  | 		if (i == img) { | 
					
						
							|  |  |  | 			if (prev)  | 
					
						
							|  |  |  | 				prev->next = i->next; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				list = i->next; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		prev = i; | 
					
						
							|  |  |  | 		i = i->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&listlock); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 	if (i && (option_verbose > 1)) | 
					
						
							|  |  |  | 		ast_verbose(VERBOSE_PREFIX_2 "Registered format '%s' (%s)\n", img->name, img->desc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_supports_images(struct ast_channel *chan) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-01-23 20:56:52 +00:00
										 |  |  | 	if (!chan || !chan->pvt) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 	if (!chan->pvt->send_image) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int file_exists(char *filename) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	struct stat st; | 
					
						
							|  |  |  | 	res = stat(filename, &st); | 
					
						
							|  |  |  | 	if (!res) | 
					
						
							|  |  |  | 		return st.st_size; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void make_filename(char *buf, int len, char *filename, char *preflang, char *ext) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (filename[0] == '/') { | 
					
						
							|  |  |  | 		if (preflang && strlen(preflang)) | 
					
						
							|  |  |  | 			snprintf(buf, len, "%s-%s.%s", filename, preflang, ext); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			snprintf(buf, len, "%s.%s", filename, ext); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (preflang && strlen(preflang)) | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 			snprintf(buf, len, "%s/%s/%s-%s.%s", ast_config_AST_VAR_DIR, "images", filename, preflang, ext); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 			snprintf(buf, len, "%s/%s/%s.%s", ast_config_AST_VAR_DIR, "images", filename, ext); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_frame *ast_read_image(char *filename, char *preflang, int format) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_imager *i; | 
					
						
							|  |  |  | 	char buf[256]; | 
					
						
							|  |  |  | 	char tmp[80]; | 
					
						
							|  |  |  | 	char *e; | 
					
						
							|  |  |  | 	struct ast_imager *found = NULL; | 
					
						
							|  |  |  | 	int fd; | 
					
						
							|  |  |  | 	int len=0; | 
					
						
							|  |  |  | 	struct ast_frame *f = NULL; | 
					
						
							|  |  |  | #if 0 /* We need to have some sort of read-only lock */
 | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&listlock); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | #endif	
 | 
					
						
							|  |  |  | 	i = list; | 
					
						
							|  |  |  | 	while(!found && i) { | 
					
						
							|  |  |  | 		if (i->format & format) { | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 			char *stringp=NULL; | 
					
						
							| 
									
										
										
										
											2001-10-18 16:47:57 +00:00
										 |  |  | 			strncpy(tmp, i->exts, sizeof(tmp)-1); | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 			stringp=tmp; | 
					
						
							|  |  |  | 			e = strsep(&stringp, "|"); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 			while(e) { | 
					
						
							|  |  |  | 				make_filename(buf, sizeof(buf), filename, preflang, e); | 
					
						
							|  |  |  | 				if ((len = file_exists(buf))) { | 
					
						
							|  |  |  | 					found = i; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				make_filename(buf, sizeof(buf), filename, NULL, e); | 
					
						
							|  |  |  | 				if ((len = file_exists(buf))) { | 
					
						
							|  |  |  | 					found = i; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 				e = strsep(&stringp, "|"); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		i = i->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (found) { | 
					
						
							|  |  |  | 		fd = open(buf, O_RDONLY); | 
					
						
							|  |  |  | 		if (fd > -1) { | 
					
						
							|  |  |  | 			if (!found->identify || found->identify(fd)) { | 
					
						
							|  |  |  | 				/* Reset file pointer */ | 
					
						
							|  |  |  | 				lseek(fd, 0, SEEK_SET); | 
					
						
							|  |  |  | 				f = found->read_image(fd,len);  | 
					
						
							|  |  |  | 			} else | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "%s does not appear to be a %s file\n", buf, i->name); | 
					
						
							|  |  |  | 			close(fd); | 
					
						
							|  |  |  | 		} else | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Unable to open '%s': %s\n", buf, strerror(errno)); | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Image file '%s' not found\n", filename); | 
					
						
							|  |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&listlock); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | #endif	
 | 
					
						
							|  |  |  | 	return f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_send_image(struct ast_channel *chan, char *filename) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_frame *f; | 
					
						
							|  |  |  | 	int res = -1; | 
					
						
							|  |  |  | 	if (chan->pvt->send_image) { | 
					
						
							|  |  |  | 		f = ast_read_image(filename, chan->language, -1); | 
					
						
							|  |  |  | 		if (f) { | 
					
						
							|  |  |  | 			res = chan->pvt->send_image(chan, f); | 
					
						
							|  |  |  | 			ast_frfree(f); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int show_image_formats(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #define FORMAT "%10s %10s %50s %10s\n"
 | 
					
						
							| 
									
										
										
										
											2003-08-16 05:10:35 +00:00
										 |  |  | #define FORMAT2 "%10s %10s %50s %10s\n"
 | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 	struct ast_imager *i; | 
					
						
							|  |  |  | 	if (argc != 3) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	ast_cli(fd, FORMAT, "Name", "Extensions", "Description", "Format"); | 
					
						
							|  |  |  | 	i = list; | 
					
						
							|  |  |  | 	while(i) { | 
					
						
							| 
									
										
										
										
											2003-08-16 05:10:35 +00:00
										 |  |  | 		ast_cli(fd, FORMAT2, i->name, i->exts, i->desc, ast_getformatname(i->format)); | 
					
						
							| 
									
										
										
										
											2001-05-27 14:51:50 +00:00
										 |  |  | 		i = i->next; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_cli_entry show_images = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	{ "show", "image", "formats" }, | 
					
						
							|  |  |  | 	show_image_formats, | 
					
						
							|  |  |  | 	"Displays image formats", | 
					
						
							|  |  |  | "Usage: show image formats\n" | 
					
						
							|  |  |  | "       displays currently registered image formats (if any)\n" | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_image_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ast_cli_register(&show_images); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |