Files
asterisk/apps/app_mp3.c
T

263 lines
6.1 KiB
C
Raw Normal View History

1999-11-15 05:04:48 +00:00
/*
* Asterisk -- An open source telephony toolkit.
1999-11-15 05:04:48 +00:00
*
2005-01-21 07:06:25 +00:00
* Copyright (C) 1999 - 2005, Digium, Inc.
1999-11-15 05:04:48 +00:00
*
* Mark Spencer <markster@digium.com>
1999-11-15 05:04:48 +00:00
*
* See http://www.asterisk.org for more information about
* the Asterisk project. Please do not directly contact
* any of the maintainers of this project for assistance;
* the project provides a web site, mailing lists and IRC
* channels for your use.
*
1999-11-15 05:04:48 +00:00
* This program is free software, distributed under the terms of
* the GNU General Public License Version 2. See the LICENSE file
* at the top of the source tree.
*/
/*! \file
*
* \brief Silly application to play an MP3 file -- uses mpg123
2005-12-30 21:18:06 +00:00
*
* \author Mark Spencer <markster@digium.com>
*
2005-11-06 15:09:47 +00:00
* \ingroup applications
1999-11-15 05:04:48 +00:00
*/
2005-06-06 22:39:32 +00:00
#include <string.h>
#include <stdio.h>
#include <signal.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/time.h>
#include "asterisk.h"
ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
#include "asterisk/lock.h"
#include "asterisk/file.h"
#include "asterisk/logger.h"
#include "asterisk/channel.h"
#include "asterisk/frame.h"
#include "asterisk/pbx.h"
#include "asterisk/module.h"
#include "asterisk/translate.h"
1999-11-15 05:04:48 +00:00
2003-09-27 00:30:07 +00:00
#define LOCAL_MPG_123 "/usr/local/bin/mpg123"
1999-11-15 05:04:48 +00:00
#define MPG_123 "/usr/bin/mpg123"
static char *tdesc = "Silly MP3 Application";
static char *app = "MP3Player";
2001-04-10 20:52:03 +00:00
static char *synopsis = "Play an MP3 file or stream";
static char *descrip =
2005-11-07 22:01:22 +00:00
" MP3Player(location) Executes mpg123 to play the given location,\n"
"which typically would be a filename or a URL. User can exit by pressing\n"
"any key on the dialpad, or by hanging up.";
2001-04-10 20:52:03 +00:00
1999-11-15 05:04:48 +00:00
STANDARD_LOCAL_USER;
LOCAL_USER_DECL;
static int mp3play(char *filename, int fd)
{
int res;
2001-07-17 16:57:20 +00:00
int x;
1999-11-15 05:04:48 +00:00
res = fork();
if (res < 0)
ast_log(LOG_WARNING, "Fork failed\n");
if (res)
return res;
dup2(fd, STDOUT_FILENO);
2001-07-17 16:57:20 +00:00
for (x=0;x<256;x++) {
if (x != STDOUT_FILENO)
close(x);
}
1999-11-15 05:04:48 +00:00
/* Execute mpg123, but buffer if it's a net connection */
if (!strncasecmp(filename, "http://", 7)) {
2003-09-27 00:30:07 +00:00
/* Most commonly installed in /usr/local/bin */
2004-05-03 03:32:04 +00:00
execl(LOCAL_MPG_123, "mpg123", "-q", "-s", "-b", "1024", "-f", "8192", "--mono", "-r", "8000", filename, (char *)NULL);
2003-09-27 00:30:07 +00:00
/* But many places has it in /usr/bin */
2004-05-03 03:32:04 +00:00
execl(MPG_123, "mpg123", "-q", "-s", "-b", "1024","-f", "8192", "--mono", "-r", "8000", filename, (char *)NULL);
2003-09-27 00:30:07 +00:00
/* As a last-ditch effort, try to use PATH */
2004-05-03 03:32:04 +00:00
execlp("mpg123", "mpg123", "-q", "-s", "-b", "1024", "-f", "8192", "--mono", "-r", "8000", filename, (char *)NULL);
2003-09-27 00:30:07 +00:00
}
else {
/* Most commonly installed in /usr/local/bin */
2004-05-03 03:32:04 +00:00
execl(MPG_123, "mpg123", "-q", "-s", "-f", "8192", "--mono", "-r", "8000", filename, (char *)NULL);
2003-09-27 00:30:07 +00:00
/* But many places has it in /usr/bin */
2004-05-03 03:32:04 +00:00
execl(LOCAL_MPG_123, "mpg123", "-q", "-s", "-f", "8192", "--mono", "-r", "8000", filename, (char *)NULL);
2003-09-27 00:30:07 +00:00
/* As a last-ditch effort, try to use PATH */
2004-05-03 03:32:04 +00:00
execlp("mpg123", "mpg123", "-q", "-s", "-f", "8192", "--mono", "-r", "8000", filename, (char *)NULL);
2003-09-27 00:30:07 +00:00
}
1999-11-15 05:04:48 +00:00
ast_log(LOG_WARNING, "Execute of mpg123 failed\n");
return -1;
}
static int timed_read(int fd, void *data, int datalen, int timeout)
{
int res;
struct pollfd fds[1];
fds[0].fd = fd;
fds[0].events = POLLIN;
res = poll(fds, 1, timeout);
if (res < 1) {
ast_log(LOG_NOTICE, "Poll timed out/errored out with %d\n", res);
return -1;
}
return read(fd, data, datalen);
}
1999-11-15 05:04:48 +00:00
static int mp3_exec(struct ast_channel *chan, void *data)
{
int res=0;
struct localuser *u;
int fds[2];
int ms = -1;
2001-07-17 16:57:20 +00:00
int pid = -1;
2001-03-10 19:12:11 +00:00
int owriteformat;
int timeout = 2000;
struct timeval next;
1999-11-15 05:04:48 +00:00
struct ast_frame *f;
struct myframe {
struct ast_frame f;
char offset[AST_FRIENDLY_OFFSET];
2003-03-27 15:11:03 +00:00
short frdata[160];
1999-11-15 05:04:48 +00:00
} myf;
if (ast_strlen_zero(data)) {
1999-11-15 05:04:48 +00:00
ast_log(LOG_WARNING, "MP3 Playback requires an argument (filename)\n");
return -1;
}
LOCAL_USER_ADD(u);
1999-11-15 05:04:48 +00:00
if (pipe(fds)) {
ast_log(LOG_WARNING, "Unable to create pipe\n");
LOCAL_USER_REMOVE(u);
1999-11-15 05:04:48 +00:00
return -1;
}
1999-11-15 05:04:48 +00:00
ast_stopstream(chan);
2001-03-10 19:12:11 +00:00
owriteformat = chan->writeformat;
res = ast_set_write_format(chan, AST_FORMAT_SLINEAR);
2001-03-10 19:12:11 +00:00
if (res < 0) {
ast_log(LOG_WARNING, "Unable to set write format to signed linear\n");
LOCAL_USER_REMOVE(u);
2001-03-10 19:12:11 +00:00
return -1;
}
res = mp3play((char *)data, fds[1]);
if (!strncasecmp((char *)data, "http://", 7)) {
timeout = 10000;
}
2003-03-27 15:11:03 +00:00
/* Wait 1000 ms first */
next = ast_tvnow();
2004-05-01 14:03:37 +00:00
next.tv_sec += 1;
2001-03-10 19:12:11 +00:00
if (res >= 0) {
pid = res;
/* Order is important -- there's almost always going to be mp3... we want to prioritize the
user */
for (;;) {
ms = ast_tvdiff_ms(next, ast_tvnow());
2004-05-01 14:03:37 +00:00
if (ms <= 0) {
res = timed_read(fds[0], myf.frdata, sizeof(myf.frdata), timeout);
2001-03-10 19:12:11 +00:00
if (res > 0) {
myf.f.frametype = AST_FRAME_VOICE;
myf.f.subclass = AST_FORMAT_SLINEAR;
myf.f.datalen = res;
2002-11-29 02:14:13 +00:00
myf.f.samples = res / 2;
2001-03-10 19:12:11 +00:00
myf.f.mallocd = 0;
myf.f.offset = AST_FRIENDLY_OFFSET;
myf.f.src = __PRETTY_FUNCTION__;
myf.f.delivery.tv_sec = 0;
myf.f.delivery.tv_usec = 0;
2001-03-10 19:12:11 +00:00
myf.f.data = myf.frdata;
if (ast_write(chan, &myf.f) < 0) {
1999-11-15 05:04:48 +00:00
res = -1;
break;
}
} else {
2001-03-10 19:12:11 +00:00
ast_log(LOG_DEBUG, "No more mp3\n");
res = 0;
2003-03-27 15:11:03 +00:00
break;
1999-11-15 05:04:48 +00:00
}
next = ast_tvadd(next, ast_samp2tv(myf.f.samples, 8000));
2004-05-01 14:03:37 +00:00
} else {
ms = ast_waitfor(chan, ms);
if (ms < 0) {
ast_log(LOG_DEBUG, "Hangup detected\n");
res = -1;
break;
}
if (ms) {
f = ast_read(chan);
if (!f) {
ast_log(LOG_DEBUG, "Null frame == hangup() detected\n");
res = -1;
break;
}
if (f->frametype == AST_FRAME_DTMF) {
ast_log(LOG_DEBUG, "User pressed a key\n");
ast_frfree(f);
res = 0;
break;
}
ast_frfree(f);
}
1999-11-15 05:04:48 +00:00
}
}
2001-03-10 19:12:11 +00:00
}
1999-11-15 05:04:48 +00:00
close(fds[0]);
close(fds[1]);
2001-07-17 16:57:20 +00:00
if (pid > -1)
kill(pid, SIGKILL);
if (!res && owriteformat)
ast_set_write_format(chan, owriteformat);
LOCAL_USER_REMOVE(u);
1999-11-15 05:04:48 +00:00
return res;
}
int unload_module(void)
{
int res;
res = ast_unregister_application(app);
1999-11-15 05:04:48 +00:00
STANDARD_HANGUP_LOCALUSERS;
return res;
1999-11-15 05:04:48 +00:00
}
int load_module(void)
{
2001-04-10 20:52:03 +00:00
return ast_register_application(app, mp3_exec, synopsis, descrip);
1999-11-15 05:04:48 +00:00
}
char *description(void)
{
return tdesc;
}
int usecount(void)
{
int res;
STANDARD_USECOUNT(res);
return res;
}
2001-03-10 19:12:11 +00:00
char *key()
{
return ASTERISK_GPL_KEY;
}