00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034 #include "asterisk.h"
00035
00036 ASTERISK_FILE_VERSION(__FILE__, "$Revision: 361476 $")
00037
00038 #include <signal.h>
00039 #include <fcntl.h>
00040 #include <sys/time.h>
00041
00042 #include "asterisk/paths.h"
00043 #include "asterisk/lock.h"
00044 #include "asterisk/file.h"
00045 #include "asterisk/channel.h"
00046 #include "asterisk/frame.h"
00047 #include "asterisk/pbx.h"
00048 #include "asterisk/module.h"
00049 #include "asterisk/translate.h"
00050 #include "asterisk/app.h"
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068
00069
00070
00071 #define path_BIN "/usr/bin/"
00072 #define path_LOCAL "/usr/local/bin/"
00073
00074 static char *app = "ICES";
00075
00076 static int icesencode(char *filename, int fd)
00077 {
00078 int res;
00079
00080 res = ast_safe_fork(0);
00081 if (res < 0)
00082 ast_log(LOG_WARNING, "Fork failed\n");
00083 if (res) {
00084 return res;
00085 }
00086
00087 if (ast_opt_high_priority)
00088 ast_set_priority(0);
00089 dup2(fd, STDIN_FILENO);
00090 ast_close_fds_above_n(STDERR_FILENO);
00091
00092
00093
00094
00095
00096 execl(path_LOCAL "ices2", "ices", filename, SENTINEL);
00097 execl(path_BIN "ices2", "ices", filename, SENTINEL);
00098 execlp("ices2", "ices", filename, SENTINEL);
00099
00100 ast_debug(1, "Couldn't find ices version 2, attempting to use ices version 1.\n");
00101
00102 execl(path_LOCAL "ices", "ices", filename, SENTINEL);
00103 execl(path_BIN "ices", "ices", filename, SENTINEL);
00104 execlp("ices", "ices", filename, SENTINEL);
00105
00106 ast_log(LOG_WARNING, "Execute of ices failed, could not find command.\n");
00107 close(fd);
00108 _exit(0);
00109 }
00110
00111 static int ices_exec(struct ast_channel *chan, const char *data)
00112 {
00113 int res = 0;
00114 int fds[2];
00115 int ms = -1;
00116 int pid = -1;
00117 int flags;
00118 struct ast_format oreadformat;
00119 struct ast_frame *f;
00120 char filename[256]="";
00121 char *c;
00122
00123 ast_format_clear(&oreadformat);
00124 if (ast_strlen_zero(data)) {
00125 ast_log(LOG_WARNING, "ICES requires an argument (configfile.xml)\n");
00126 return -1;
00127 }
00128
00129 if (pipe(fds)) {
00130 ast_log(LOG_WARNING, "Unable to create pipe\n");
00131 return -1;
00132 }
00133 flags = fcntl(fds[1], F_GETFL);
00134 fcntl(fds[1], F_SETFL, flags | O_NONBLOCK);
00135
00136 ast_stopstream(chan);
00137
00138 if (ast_channel_state(chan) != AST_STATE_UP)
00139 res = ast_answer(chan);
00140
00141 if (res) {
00142 close(fds[0]);
00143 close(fds[1]);
00144 ast_log(LOG_WARNING, "Answer failed!\n");
00145 return -1;
00146 }
00147
00148 ast_format_copy(&oreadformat, ast_channel_readformat(chan));
00149 res = ast_set_read_format_by_id(chan, AST_FORMAT_SLINEAR);
00150 if (res < 0) {
00151 close(fds[0]);
00152 close(fds[1]);
00153 ast_log(LOG_WARNING, "Unable to set write format to signed linear\n");
00154 return -1;
00155 }
00156 if (((char *)data)[0] == '/')
00157 ast_copy_string(filename, (char *) data, sizeof(filename));
00158 else
00159 snprintf(filename, sizeof(filename), "%s/%s", ast_config_AST_CONFIG_DIR, (char *)data);
00160
00161 c = strchr(filename, '|');
00162 if (c)
00163 *c = '\0';
00164 res = icesencode(filename, fds[0]);
00165 if (res >= 0) {
00166 pid = res;
00167 for (;;) {
00168
00169 ms = ast_waitfor(chan, -1);
00170 if (ms < 0) {
00171 ast_debug(1, "Hangup detected\n");
00172 res = -1;
00173 break;
00174 }
00175 f = ast_read(chan);
00176 if (!f) {
00177 ast_debug(1, "Null frame == hangup() detected\n");
00178 res = -1;
00179 break;
00180 }
00181 if (f->frametype == AST_FRAME_VOICE) {
00182 res = write(fds[1], f->data.ptr, f->datalen);
00183 if (res < 0) {
00184 if (errno != EAGAIN) {
00185 ast_log(LOG_WARNING, "Write failed to pipe: %s\n", strerror(errno));
00186 res = -1;
00187 ast_frfree(f);
00188 break;
00189 }
00190 }
00191 }
00192 ast_frfree(f);
00193 }
00194 }
00195 close(fds[0]);
00196 close(fds[1]);
00197
00198 if (pid > -1)
00199 kill(pid, SIGKILL);
00200 if (!res && oreadformat.id)
00201 ast_set_read_format(chan, &oreadformat);
00202
00203 return res;
00204 }
00205
00206 static int unload_module(void)
00207 {
00208 return ast_unregister_application(app);
00209 }
00210
00211 static int load_module(void)
00212 {
00213 return ast_register_application_xml(app, ices_exec);
00214 }
00215
00216 AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Encode and Stream via icecast and ices");