Generic File Format Support. More...
#include "asterisk.h"#include <dirent.h>#include <sys/stat.h>#include <sys/wait.h>#include <math.h>#include "asterisk/_private.h"#include "asterisk/paths.h"#include "asterisk/mod_format.h"#include "asterisk/cli.h"#include "asterisk/channel.h"#include "asterisk/sched.h"#include "asterisk/translate.h"#include "asterisk/utils.h"#include "asterisk/lock.h"#include "asterisk/app.h"#include "asterisk/pbx.h"#include "asterisk/linkedlists.h"#include "asterisk/module.h"#include "asterisk/astobj2.h"#include "asterisk/test.h"
Go to the source code of this file.
Data Structures | |
| struct | formats |
Defines | |
| #define | FORMAT "%-10s %-10s %-20s\n" |
| #define | FORMAT2 "%-10s %-10s %-20s\n" |
Enumerations | |
| enum | file_action { ACTION_EXISTS = 1, ACTION_DELETE, ACTION_RENAME, ACTION_OPEN, ACTION_COPY } |
| enum | fsread_res { FSREAD_FAILURE, FSREAD_SUCCESS_SCHED, FSREAD_SUCCESS_NOSCHED } |
| enum | wrap_fn { WRAP_OPEN, WRAP_REWRITE } |
Functions | |
| int | __ast_format_def_register (const struct ast_format_def *f, struct ast_module *mod) |
| Register a new file format capability. Adds a format to Asterisk's format abilities. | |
| int | ast_applystream (struct ast_channel *chan, struct ast_filestream *s) |
| Applys a open stream to a channel. | |
| int | ast_closestream (struct ast_filestream *f) |
| Closes a stream. | |
| int | ast_file_init (void) |
| int | ast_filecopy (const char *filename, const char *filename2, const char *fmt) |
| Copies a file. | |
| int | ast_filedelete (const char *filename, const char *fmt) |
| Deletes a file. | |
| int | ast_fileexists (const char *filename, const char *fmt, const char *preflang) |
| Checks for the existence of a given file. | |
| int | ast_filerename (const char *filename, const char *filename2, const char *fmt) |
| Renames a file. | |
| int | ast_format_def_unregister (const char *name) |
| Unregisters a file format. | |
| char * | ast_format_str_reduce (char *fmts) |
| static int | ast_fsread_audio (const void *data) |
| static int | ast_fsread_video (const void *data) |
| struct ast_filestream * | ast_openstream (struct ast_channel *chan, const char *filename, const char *preflang) |
| Opens stream for use in seeking, playing. | |
| struct ast_filestream * | ast_openstream_full (struct ast_channel *chan, const char *filename, const char *preflang, int asis) |
| Opens stream for use in seeking, playing. | |
| struct ast_filestream * | ast_openvstream (struct ast_channel *chan, const char *filename, const char *preflang) |
| Opens stream for use in seeking, playing. | |
| int | ast_playstream (struct ast_filestream *s) |
| Play a open stream on a channel. | |
| static enum fsread_res | ast_readaudio_callback (struct ast_filestream *s) |
| struct ast_filestream * | ast_readfile (const char *filename, const char *type, const char *comment, int flags, int check, mode_t mode) |
| Starts reading from a file. | |
| struct ast_frame * | ast_readframe (struct ast_filestream *s) |
| Read a frame from a filestream. | |
| static enum fsread_res | ast_readvideo_callback (struct ast_filestream *s) |
| int | ast_seekstream (struct ast_filestream *fs, off_t sample_offset, int whence) |
| Seeks into stream. | |
| int | ast_stopstream (struct ast_channel *tmp) |
| Stops a stream. | |
| int | ast_stream_and_wait (struct ast_channel *chan, const char *file, const char *digits) |
| stream file until digit If the file name is non-empty, try to play it. | |
| int | ast_stream_fastforward (struct ast_filestream *fs, off_t ms) |
| Fast forward stream ms. | |
| int | ast_stream_rewind (struct ast_filestream *fs, off_t ms) |
| Rewind stream ms. | |
| int | ast_streamfile (struct ast_channel *chan, const char *filename, const char *preflang) |
| Streams a file. | |
| off_t | ast_tellstream (struct ast_filestream *fs) |
| Tell where we are in a stream. | |
| int | ast_truncstream (struct ast_filestream *fs) |
| Trunc stream at current location. | |
| int | ast_waitstream (struct ast_channel *c, const char *breakon) |
| Waits for a stream to stop or digit to be pressed. | |
| int | ast_waitstream_exten (struct ast_channel *c, const char *context) |
| Waits for a stream to stop or digit matching a valid one digit exten to be pressed. | |
| int | ast_waitstream_fr (struct ast_channel *c, const char *breakon, const char *forward, const char *reverse, int ms) |
| Same as waitstream but allows stream to be forwarded or rewound. | |
| int | ast_waitstream_fr_w_cb (struct ast_channel *c, const char *breakon, const char *forward, const char *reverse, int ms, ast_waitstream_fr_cb cb) |
| Same as waitstream_fr but allows a callback to be alerted when a user fastforwards or rewinds the file. | |
| int | ast_waitstream_full (struct ast_channel *c, const char *breakon, int audiofd, int cmdfd) |
| struct ast_filestream * | ast_writefile (const char *filename, const char *type, const char *comment, int flags, int check, mode_t mode) |
| Starts writing a file. | |
| int | ast_writestream (struct ast_filestream *fs, struct ast_frame *f) |
| Writes a frame to a stream. | |
| static char * | build_filename (const char *filename, const char *ext) |
| construct a filename. Absolute pathnames are preserved, relative names are prefixed by the sounds/ directory. The wav49 suffix is replaced by 'WAV'. Returns a malloc'ed string to be freed by the caller. | |
| static int | copy (const char *infile, const char *outfile) |
| static int | exts_compare (const char *exts, const char *type) |
| static void | file_shutdown (void) |
| static int | fileexists_core (const char *filename, const char *fmt, const char *preflang, char *buf, int buflen, struct ast_format_cap *result_cap) |
| helper routine to locate a file with a given format and language preference. | |
| static int | fileexists_test (const char *filename, const char *fmt, const char *lang, char *buf, int buflen, struct ast_format_cap *result_cap) |
| test if a file exists for a given format. | |
| static int | filehelper (const char *filename, const void *arg2, const char *fmt, const enum file_action action) |
| static void | filestream_close (struct ast_filestream *f) |
| static void | filestream_destructor (void *arg) |
| static int | fn_wrapper (struct ast_filestream *s, const char *comment, enum wrap_fn mode) |
| static struct ast_filestream * | get_filestream (struct ast_format_def *fmt, FILE *bfile) |
| static char * | handle_cli_core_show_file_formats (struct ast_cli_entry *e, int cmd, struct ast_cli_args *a) |
| static int | is_absolute_path (const char *filename) |
| static int | open_wrapper (struct ast_filestream *s) |
| static struct ast_frame * | read_frame (struct ast_filestream *s, int *whennext) |
| static int | rewrite_wrapper (struct ast_filestream *s, const char *comment) |
| static int | waitstream_core (struct ast_channel *c, const char *breakon, const char *forward, const char *reverse, int skip_ms, int audiofd, int cmdfd, const char *context, ast_waitstream_fr_cb cb) |
| the core of all waitstream() functions | |
Variables | |
| int | ast_language_is_prefix = 1 |
| The following variable controls the layout of localized sound files. If 0, use the historical layout with prefix just before the filename (i.e. digits/en/1.gsm , digits/it/1.gsm or default to digits/1.gsm), if 1 put the prefix at the beginning of the filename (i.e. en/digits/1.gsm, it/digits/1.gsm or default to digits/1.gsm). The latter permits a language to be entirely in one directory. | |
| static struct ast_cli_entry | cli_file [] |
| static struct formats | formats |
Generic File Format Support.
Definition in file file.c.
| #define FORMAT "%-10s %-10s %-20s\n" |
| #define FORMAT2 "%-10s %-10s %-20s\n" |
| enum file_action |
Definition at line 419 of file file.c.
{
ACTION_EXISTS = 1, /* return matching format if file exists, 0 otherwise */
ACTION_DELETE, /* delete file, return 0 on success, -1 on error */
ACTION_RENAME, /* rename file. return 0 on success, -1 on error */
ACTION_OPEN,
ACTION_COPY /* copy file. return 0 on success, -1 on error */
};
| enum fsread_res |
| enum wrap_fn |
Definition at line 389 of file file.c.
{ WRAP_OPEN, WRAP_REWRITE };
| int __ast_format_def_register | ( | const struct ast_format_def * | f, |
| struct ast_module * | mod | ||
| ) |
Register a new file format capability. Adds a format to Asterisk's format abilities.
| 0 | on success |
| -1 | on failure |
Definition at line 69 of file file.c.
References ast_calloc, ast_log(), AST_RWLIST_INSERT_HEAD, AST_RWLIST_TRAVERSE, AST_RWLIST_UNLOCK, AST_RWLIST_WRLOCK, ast_verb, ast_format_def::buf_size, ast_format_def::exts, f, ast_format_def::list, LOG_WARNING, ast_format_def::module, and ast_format_def::name.
{
struct ast_format_def *tmp;
AST_RWLIST_WRLOCK(&formats);
AST_RWLIST_TRAVERSE(&formats, tmp, list) {
if (!strcasecmp(f->name, tmp->name)) {
AST_RWLIST_UNLOCK(&formats);
ast_log(LOG_WARNING, "Tried to register '%s' format, already registered\n", f->name);
return -1;
}
}
if (!(tmp = ast_calloc(1, sizeof(*tmp)))) {
AST_RWLIST_UNLOCK(&formats);
return -1;
}
*tmp = *f;
tmp->module = mod;
if (tmp->buf_size) {
/*
* Align buf_size properly, rounding up to the machine-specific
* alignment for pointers.
*/
struct _test_align { void *a, *b; } p;
int align = (char *)&p.b - (char *)&p.a;
tmp->buf_size = ((f->buf_size + align - 1) / align) * align;
}
memset(&tmp->list, 0, sizeof(tmp->list));
AST_RWLIST_INSERT_HEAD(&formats, tmp, list);
AST_RWLIST_UNLOCK(&formats);
ast_verb(2, "Registered file format %s, extension(s) %s\n", f->name, f->exts);
return 0;
}
| int ast_applystream | ( | struct ast_channel * | chan, |
| struct ast_filestream * | s | ||
| ) |
Applys a open stream to a channel.
| chan | channel to work |
| s | ast_filestream to apply |
| 0 | on success. |
| -1 | on failure. |
Definition at line 921 of file file.c.
References ast_filestream::owner.
Referenced by ast_streamfile(), handle_getoption(), handle_recordfile(), handle_streamfile(), and speech_streamfile().
{
s->owner = chan;
return 0;
}
| int ast_closestream | ( | struct ast_filestream * | f | ) |
Closes a stream.
| f | filestream to close Close a playback or recording stream |
| 0 | on success. |
| -1 | on failure. |
Definition at line 964 of file file.c.
References ao2_ref, and filestream_close().
Referenced by __ast_play_and_record(), ast_hangup(), ast_moh_files_next(), ast_monitor_start(), ast_monitor_stop(), ast_readfile(), ast_stopstream(), ast_writefile(), dictate_exec(), filehelper(), filestream_destructor(), gen_closestream(), handle_cli_file_convert(), handle_recordfile(), local_ast_moh_stop(), mixmonitor_ds_close_fs(), moh_files_release(), record_exec(), and recordthread().
{
/* This used to destroy the filestream, but it now just decrements a refcount.
* We close the stream in order to quit queuing frames now, because we might
* change the writeformat, which could result in a subsequent write error, if
* the format is different. */
filestream_close(f);
ao2_ref(f, -1);
return 0;
}
| int ast_file_init | ( | void | ) |
Provided by file.c
Definition at line 1594 of file file.c.
References ARRAY_LEN, ast_cli_register_multiple(), ast_register_atexit(), and file_shutdown().
Referenced by main().
{
ast_cli_register_multiple(cli_file, ARRAY_LEN(cli_file));
ast_register_atexit(file_shutdown);
return 0;
}
| int ast_filecopy | ( | const char * | oldname, |
| const char * | newname, | ||
| const char * | fmt | ||
| ) |
Copies a file.
| oldname | name of the file you wish to copy (minus extension) |
| newname | name you wish the file to be copied to (minus extension) |
| fmt | the format of the file Copy a given file in a given format, or if fmt is NULL, then do so for all |
Definition at line 1001 of file file.c.
References ACTION_COPY, and filehelper().
Referenced by copy_plain_file(), msg_create_from_file(), and vm_forwardoptions().
{
return filehelper(filename, filename2, fmt, ACTION_COPY);
}
| int ast_filedelete | ( | const char * | filename, |
| const char * | fmt | ||
| ) |
Deletes a file.
| filename | name of the file you wish to delete (minus the extension) |
| fmt | of the file Delete a given file in a given format, or if fmt is NULL, then do so for all |
Definition at line 991 of file file.c.
References ACTION_DELETE, and filehelper().
Referenced by __ast_play_and_record(), announce_thread(), ast_monitor_start(), ast_monitor_stop(), conf_free(), conf_run(), confbridge_exec(), dial_exec_full(), handle_cli_file_convert(), leave_voicemail(), msg_create_from_file(), play_record_review(), record_exec(), setup_privacy_args(), and vm_delete().
{
return filehelper(filename, NULL, fmt, ACTION_DELETE);
}
| int ast_fileexists | ( | const char * | filename, |
| const char * | fmt, | ||
| const char * | preflang | ||
| ) |
Checks for the existence of a given file.
| filename | name of the file you wish to check, minus the extension |
| fmt | the format you wish to check (the extension) |
| preflang | (the preferred language you wisht to find the file in) See if a given file exists in a given format. If fmt is NULL, any format is accepted. |
| 0,false. | The file does not exist |
| 1,true. | The file does exist. |
Definition at line 979 of file file.c.
References ast_alloca, ast_filestream::buf, and fileexists_core().
Referenced by announce_thread(), app_exec(), ast_moh_files_next(), ast_monitor_start(), ast_monitor_stop(), common_exec(), conf_run(), dial_exec_full(), eivr_comm(), forward_message(), get_folder(), invent_message(), leave_voicemail(), meetme_menu_admin_extended(), minivm_delete_exec(), msg_create_from_file(), play_file(), play_message(), play_message_by_id_helper(), play_message_callerid(), readexten_exec(), record_exec(), retrydial_exec(), say_character_str_full(), say_digit_str_full(), say_phonetic_str_full(), sayname(), setup_privacy_args(), sound_file_exists(), vm_intro(), vm_msg_play(), vm_newuser(), vm_options(), and vm_tempgreeting().
{
char *buf;
int buflen;
if (preflang == NULL)
preflang = "";
buflen = strlen(preflang) + strlen(filename) + 4; /* room for everything */
buf = ast_alloca(buflen);
return fileexists_core(filename, fmt, preflang, buf, buflen, NULL) ? 1 : 0;
}
| int ast_filerename | ( | const char * | oldname, |
| const char * | newname, | ||
| const char * | fmt | ||
| ) |
Renames a file.
| oldname | the name of the file you wish to act upon (minus the extension) |
| newname | the name you wish to rename the file to (minus the extension) |
| fmt | the format of the file Rename a given file in a given format, or if fmt is NULL, then do so for all |
Definition at line 996 of file file.c.
References ACTION_RENAME, and filehelper().
Referenced by __ast_play_and_record(), ast_monitor_stop(), forward_message(), leave_voicemail(), msg_create_from_file(), play_record_review(), rename_file(), and vm_forwardoptions().
{
return filehelper(filename, filename2, fmt, ACTION_RENAME);
}
| int ast_format_def_unregister | ( | const char * | name | ) |
Unregisters a file format.
| name | the name of the format you wish to unregister Unregisters a format based on the name of the format. |
| 0 | on success |
| -1 | on failure to unregister |
Definition at line 106 of file file.c.
References ast_free, ast_log(), AST_RWLIST_REMOVE_CURRENT, AST_RWLIST_TRAVERSE_SAFE_BEGIN, AST_RWLIST_TRAVERSE_SAFE_END, AST_RWLIST_UNLOCK, AST_RWLIST_WRLOCK, ast_verb, ast_format_def::list, LOG_WARNING, and ast_format_def::name.
Referenced by unload_module().
{
struct ast_format_def *tmp;
int res = -1;
AST_RWLIST_WRLOCK(&formats);
AST_RWLIST_TRAVERSE_SAFE_BEGIN(&formats, tmp, list) {
if (!strcasecmp(name, tmp->name)) {
AST_RWLIST_REMOVE_CURRENT(list);
ast_free(tmp);
res = 0;
}
}
AST_RWLIST_TRAVERSE_SAFE_END;
AST_RWLIST_UNLOCK(&formats);
if (!res)
ast_verb(2, "Unregistered format %s\n", name);
else
ast_log(LOG_WARNING, "Tried to unregister format %s, already unregistered\n", name);
return res;
}
| char* ast_format_str_reduce | ( | char * | fmts | ) |
Remove duplicate formats from a format string.
| fmts | a format string, this string will be modified |
| NULL | error |
Definition at line 1473 of file file.c.
References ast_log(), AST_MAX_FORMATS, AST_RWLIST_RDLOCK, AST_RWLIST_TRAVERSE, AST_RWLIST_UNLOCK, ast_format_def::exts, exts_compare(), f, first, len(), ast_format_def::list, LOG_WARNING, and type.
Referenced by actual_load_config().
{
struct ast_format_def *f;
struct ast_format_def *fmts_ptr[AST_MAX_FORMATS];
char *fmts_str[AST_MAX_FORMATS];
char *stringp, *type;
char *orig = fmts;
int i, j, x, first, found = 0;
int len = strlen(fmts) + 1;
int res;
if (AST_RWLIST_RDLOCK(&formats)) {
ast_log(LOG_WARNING, "Unable to lock format list\n");
return NULL;
}
stringp = ast_strdupa(fmts);
for (x = 0; (type = strsep(&stringp, "|")) && x < AST_MAX_FORMATS; x++) {
AST_RWLIST_TRAVERSE(&formats, f, list) {
if (exts_compare(f->exts, type)) {
found = 1;
break;
}
}
fmts_str[x] = type;
if (found) {
fmts_ptr[x] = f;
} else {
fmts_ptr[x] = NULL;
}
}
AST_RWLIST_UNLOCK(&formats);
first = 1;
for (i = 0; i < x; i++) {
/* ignore invalid entries */
if (!fmts_ptr[i]) {
ast_log(LOG_WARNING, "ignoring unknown format '%s'\n", fmts_str[i]);
continue;
}
/* special handling for the first entry */
if (first) {
res = snprintf(fmts, len, "%s", fmts_str[i]);
fmts += res;
len -= res;
first = 0;
continue;
}
found = 0;
for (j = 0; j < i; j++) {
/* this is a duplicate */
if (fmts_ptr[j] == fmts_ptr[i]) {
found = 1;
break;
}
}
if (!found) {
res = snprintf(fmts, len, "|%s", fmts_str[i]);
fmts += res;
len -= res;
}
}
if (first) {
ast_log(LOG_WARNING, "no known formats found in format list (%s)\n", orig);
return NULL;
}
return orig;
}
| static int ast_fsread_audio | ( | const void * | data | ) | [static] |
Definition at line 863 of file file.c.
References ast_readaudio_callback(), and FSREAD_SUCCESS_SCHED.
Referenced by ast_readaudio_callback().
{
struct ast_filestream *fs = (struct ast_filestream *)data;
enum fsread_res res;
res = ast_readaudio_callback(fs);
if (res == FSREAD_SUCCESS_SCHED)
return 1;
return 0;
}
| static int ast_fsread_video | ( | const void * | data | ) | [static] |
Definition at line 908 of file file.c.
References ast_readvideo_callback(), and FSREAD_SUCCESS_SCHED.
Referenced by ast_readvideo_callback().
{
struct ast_filestream *fs = (struct ast_filestream *)data;
enum fsread_res res;
res = ast_readvideo_callback(fs);
if (res == FSREAD_SUCCESS_SCHED)
return 1;
return 0;
}
| struct ast_filestream* ast_openstream | ( | struct ast_channel * | chan, |
| const char * | filename, | ||
| const char * | preflang | ||
| ) | [read] |
Opens stream for use in seeking, playing.
| chan | channel to work with |
| filename | to use |
| preflang | prefered language to use |
| a | ast_filestream pointer if it opens the file. |
| NULL | on error. |
Definition at line 667 of file file.c.
References ast_openstream_full().
Referenced by ast_streamfile(), dictate_exec(), handle_getoption(), handle_streamfile(), and speech_streamfile().
{
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 | ||
| ) | [read] |
Opens stream for use in seeking, playing.
| chan | channel to work with |
| filename | to use |
| preflang | prefered language to use |
| asis | if set, don't clear generators |
| a | ast_filestream pointer if it opens the file. |
| NULL | on error. |
Definition at line 672 of file file.c.
References ACTION_OPEN, ast_alloca, ast_channel_generator(), ast_channel_oldwriteformat(), ast_channel_stream(), ast_channel_writeformat(), ast_deactivate_generator(), ast_format_cap_alloc_nolock(), ast_format_cap_destroy(), ast_format_cap_has_type(), ast_format_copy(), AST_FORMAT_TYPE_AUDIO, ast_log(), ast_set_write_format_from_cap(), ast_stopstream(), fileexists_core(), filehelper(), and LOG_WARNING.
Referenced by ast_moh_files_next(), ast_openstream(), and gen_nextfile().
{
/*
* Use fileexists_core() to find a file in a compatible
* language and format, set up a suitable translator,
* and open the stream.
*/
struct ast_format_cap *file_fmt_cap;
int res;
int buflen;
char *buf;
if (!asis) {
/* do this first, otherwise we detect the wrong writeformat */
ast_stopstream(chan);
if (ast_channel_generator(chan))
ast_deactivate_generator(chan);
}
if (preflang == NULL)
preflang = "";
buflen = strlen(preflang) + strlen(filename) + 4;
buf = ast_alloca(buflen);
if (!(file_fmt_cap = ast_format_cap_alloc_nolock())) {
return NULL;
}
if (!fileexists_core(filename, NULL, preflang, buf, buflen, file_fmt_cap) ||
!ast_format_cap_has_type(file_fmt_cap, AST_FORMAT_TYPE_AUDIO)) {
ast_log(LOG_WARNING, "File %s does not exist in any format\n", filename);
file_fmt_cap = ast_format_cap_destroy(file_fmt_cap);
return NULL;
}
/* Set the channel to a format we can work with and save off the previous format. */
ast_format_copy(ast_channel_oldwriteformat(chan), ast_channel_writeformat(chan));
/* Set the channel to the best format that exists for the file. */
res = ast_set_write_format_from_cap(chan, file_fmt_cap);
/* don't need this anymore now that the channel's write format is set. */
file_fmt_cap = ast_format_cap_destroy(file_fmt_cap);
if (res == -1) { /* No format available that works with this channel */
return NULL;
}
res = filehelper(buf, chan, NULL, ACTION_OPEN);
if (res >= 0)
return ast_channel_stream(chan);
return NULL;
}
| struct ast_filestream* ast_openvstream | ( | struct ast_channel * | chan, |
| const char * | filename, | ||
| const char * | preflang | ||
| ) | [read] |
Opens stream for use in seeking, playing.
| chan | channel to work with |
| filename | to use |
| preflang | prefered language to use |
| a | ast_filestream pointer if it opens the file. |
| NULL | on error. |
Definition at line 722 of file file.c.
References ACTION_OPEN, ast_alloca, ast_channel_nativeformats(), ast_channel_vstream(), ast_format_cap_alloc_nolock(), ast_format_cap_destroy(), ast_format_cap_has_type(), ast_format_cap_iscompatible(), ast_format_cap_iter_end(), ast_format_cap_iter_next(), ast_format_cap_iter_start(), AST_FORMAT_GET_TYPE, AST_FORMAT_TYPE_VIDEO, ast_getformatname(), ast_log(), fileexists_core(), filehelper(), ast_format::id, and LOG_WARNING.
Referenced by ast_streamfile(), handle_getoption(), and handle_streamfile().
{
/* As above, but for video. But here we don't have translators
* so we must enforce a format.
*/
struct ast_format tmp_fmt;
struct ast_format_cap *tmp_cap;
char *buf;
int buflen;
const char *fmt;
int fd;
if (preflang == NULL)
preflang = "";
buflen = strlen(preflang) + strlen(filename) + 4;
buf = ast_alloca(buflen);
/* is the channel capable of video without translation ?*/
if (!ast_format_cap_has_type(ast_channel_nativeformats(chan), AST_FORMAT_TYPE_VIDEO)) {
return NULL;
}
if (!(tmp_cap = ast_format_cap_alloc_nolock())) {
return NULL;
}
/* Video is supported, so see what video formats exist for this file */
if (!fileexists_core(filename, NULL, preflang, buf, buflen, tmp_cap)) {
tmp_cap = ast_format_cap_destroy(tmp_cap);
return NULL;
}
/* iterate over file formats and pick the first one compatible with the channel's native formats */
ast_format_cap_iter_start(tmp_cap);
while (!ast_format_cap_iter_next(tmp_cap, &tmp_fmt)) {
fmt = ast_getformatname(&tmp_fmt);
if ((AST_FORMAT_GET_TYPE(tmp_fmt.id) != AST_FORMAT_TYPE_VIDEO) ||
!ast_format_cap_iscompatible(ast_channel_nativeformats(chan), &tmp_fmt)) {
continue;
}
fd = filehelper(buf, chan, fmt, ACTION_OPEN);
if (fd >= 0) {
ast_format_cap_iter_end(tmp_cap);
tmp_cap = ast_format_cap_destroy(tmp_cap);
return ast_channel_vstream(chan);
}
ast_log(LOG_WARNING, "File %s has video but couldn't be opened\n", filename);
}
ast_format_cap_iter_end(tmp_cap);
tmp_cap = ast_format_cap_destroy(tmp_cap);
return NULL;
}
| int ast_playstream | ( | struct ast_filestream * | s | ) |
Play a open stream on a channel.
| s | filestream to play |
| 0 | on success. |
| -1 | on failure. |
Definition at line 927 of file file.c.
References AST_FORMAT_GET_TYPE, AST_FORMAT_TYPE_AUDIO, ast_readaudio_callback(), ast_readvideo_callback(), ast_filestream::fmt, ast_format_def::format, FSREAD_FAILURE, and ast_format::id.
Referenced by ast_streamfile(), handle_getoption(), handle_streamfile(), and speech_streamfile().
{
enum fsread_res res;
if (AST_FORMAT_GET_TYPE(s->fmt->format.id) == AST_FORMAT_TYPE_AUDIO)
res = ast_readaudio_callback(s);
else
res = ast_readvideo_callback(s);
return (res == FSREAD_FAILURE) ? -1 : 0;
}
| static enum fsread_res ast_readaudio_callback | ( | struct ast_filestream * | s | ) | [static] |
Definition at line 815 of file file.c.
References ast_channel_name(), ast_channel_sched(), ast_channel_streamid_set(), ast_channel_timingfd(), ast_format_rate(), ast_frfree, ast_fsread_audio(), ast_log(), ast_sched_add(), ast_settimeout(), ast_settimeout_full(), ast_write(), ast_filestream::fmt, ast_format_def::format, FSREAD_FAILURE, FSREAD_SUCCESS_NOSCHED, FSREAD_SUCCESS_SCHED, ast_filestream::lasttimeout, LOG_WARNING, ast_filestream::orig_chan_name, ast_filestream::owner, and read_frame().
Referenced by ast_fsread_audio(), and ast_playstream().
{
int whennext = 0;
while (!whennext) {
struct ast_frame *fr;
if (s->orig_chan_name && strcasecmp(ast_channel_name(s->owner), s->orig_chan_name)) {
goto return_failure;
}
fr = read_frame(s, &whennext);
if (!fr /* stream complete */ || ast_write(s->owner, fr) /* error writing */) {
if (fr) {
ast_log(LOG_WARNING, "Failed to write frame\n");
ast_frfree(fr);
}
goto return_failure;
}
if (fr) {
ast_frfree(fr);
}
}
if (whennext != s->lasttimeout) {
if (ast_channel_timingfd(s->owner) > -1) {
float samp_rate = (float) ast_format_rate(&s->fmt->format);
unsigned int rate;
rate = (unsigned int) roundf(samp_rate / ((float) whennext));
ast_settimeout_full(s->owner, rate, ast_fsread_audio, s, 1);
} else {
ast_channel_streamid_set(s->owner, ast_sched_add(ast_channel_sched(s->owner), whennext / (ast_format_rate(&s->fmt->format) / 1000), ast_fsread_audio, s));
}
s->lasttimeout = whennext;
return FSREAD_SUCCESS_NOSCHED;
}
return FSREAD_SUCCESS_SCHED;
return_failure:
ast_channel_streamid_set(s->owner, -1);
ast_settimeout(s->owner, 0, NULL, NULL);
return FSREAD_FAILURE;
}
| struct ast_filestream* ast_readfile | ( | const char * | filename, |
| const char * | type, | ||
| const char * | comment, | ||
| int | flags, | ||
| int | check, | ||
| mode_t | mode | ||
| ) | [read] |
Starts reading from a file.
| filename | the name of the file to read from |
| type | format of file you wish to read from |
| comment | comment to go with |
| flags | file flags |
| check | (unimplemented, hence negligible) |
| mode | Open mode Open an incoming file stream. flags are flags for the open() command, and if check is non-zero, then it will not read a file if there are any files that start with that name and have an extension Please note, this is a blocking function. Program execution will not return until ast_waitstream completes it's execution. |
| a | struct ast_filestream on success. |
| NULL | on failure. |
Definition at line 1056 of file file.c.
References ast_closestream(), ast_free, ast_log(), AST_RWLIST_RDLOCK, AST_RWLIST_TRAVERSE, AST_RWLIST_UNLOCK, ast_strdup, build_filename(), errno, ast_format_def::exts, exts_compare(), f, ast_filestream::filename, ast_filestream::flags, ast_filestream::fmt, get_filestream(), LOG_WARNING, ast_filestream::mode, open_wrapper(), ast_filestream::trans, and ast_filestream::vfs.
Referenced by __ast_play_and_record(), handle_cli_file_convert(), and msg_create_from_file().
{
FILE *bfile;
struct ast_format_def *f;
struct ast_filestream *fs = NULL;
char *fn;
int format_found = 0;
AST_RWLIST_RDLOCK(&formats);
AST_RWLIST_TRAVERSE(&formats, f, list) {
fs = NULL;
if (!exts_compare(f->exts, type))
continue;
else
format_found = 1;
fn = build_filename(filename, type);
if (!fn) {
continue;
}
errno = 0;
bfile = fopen(fn, "r");
if (!bfile || (fs = get_filestream(f, bfile)) == NULL || open_wrapper(fs) ) {
ast_log(LOG_WARNING, "Unable to open %s\n", fn);
if (fs) {
ast_closestream(fs);
}
fs = NULL;
bfile = NULL;
ast_free(fn);
break;
}
/* found it */
fs->trans = NULL;
fs->fmt = f;
fs->flags = flags;
fs->mode = mode;
fs->filename = ast_strdup(filename);
fs->vfs = NULL;
ast_free(fn);
break;
}
AST_RWLIST_UNLOCK(&formats);
if (!format_found)
ast_log(LOG_WARNING, "No such format '%s'\n", type);
return fs;
}
| struct ast_frame* ast_readframe | ( | struct ast_filestream * | s | ) | [read] |
Read a frame from a filestream.
| s | ast_filestream to act on |
| NULL | if read failed. |
Definition at line 800 of file file.c.
References read_frame().
Referenced by __ast_play_and_record(), dictate_exec(), gen_readframe(), handle_cli_file_convert(), and moh_files_readframe().
{
int whennext = 0;
return read_frame(s, &whennext);
}
| static enum fsread_res ast_readvideo_callback | ( | struct ast_filestream * | s | ) | [static] |
Definition at line 878 of file file.c.
References ast_channel_sched(), ast_channel_vstreamid_set(), ast_format_rate(), ast_frfree, ast_fsread_video(), ast_log(), ast_sched_add(), ast_write(), ast_filestream::fmt, ast_format_def::format, FSREAD_FAILURE, FSREAD_SUCCESS_NOSCHED, FSREAD_SUCCESS_SCHED, ast_filestream::lasttimeout, LOG_WARNING, ast_filestream::owner, and read_frame().
Referenced by ast_fsread_video(), and ast_playstream().
{
int whennext = 0;
while (!whennext) {
struct ast_frame *fr = read_frame(s, &whennext);
if (!fr /* stream complete */ || ast_write(s->owner, fr) /* error writing */) {
if (fr) {
ast_log(LOG_WARNING, "Failed to write frame\n");
ast_frfree(fr);
}
ast_channel_vstreamid_set(s->owner, -1);
return FSREAD_FAILURE;
}
if (fr) {
ast_frfree(fr);
}
}
if (whennext != s->lasttimeout) {
ast_channel_vstreamid_set(s->owner, ast_sched_add(ast_channel_sched(s->owner), whennext / (ast_format_rate(&s->fmt->format) / 1000), ast_fsread_video, s));
s->lasttimeout = whennext;
return FSREAD_SUCCESS_NOSCHED;
}
return FSREAD_SUCCESS_SCHED;
}
| int ast_seekstream | ( | struct ast_filestream * | fs, |
| off_t | sample_offset, | ||
| int | whence | ||
| ) |
Seeks into stream.
| fs | ast_filestream to perform seek on |
| sample_offset | numbers of samples to seek |
| whence | SEEK_SET, SEEK_CUR, SEEK_END |
| 0 | on success. |
| -1 | on failure. |
Definition at line 939 of file file.c.
References ast_filestream::fmt, and ast_format_def::seek.
Referenced by __ast_read(), ast_moh_files_next(), ast_stream_fastforward(), ast_stream_rewind(), ast_streamfile(), ast_write(), control_streamfile(), dictate_exec(), handle_getoption(), handle_recordfile(), handle_streamfile(), msg_create_from_file(), and speech_streamfile().
| int ast_stopstream | ( | struct ast_channel * | c | ) |
Stops a stream.
| c | The channel you wish to stop playback on |
Stop playback of a stream
| 0 | always |
Definition at line 130 of file file.c.
References ast_channel_lock, ast_channel_oldwriteformat(), ast_channel_stream(), ast_channel_stream_set(), ast_channel_unlock, ast_channel_vstream(), ast_channel_vstream_set(), ast_closestream(), ast_getformatname(), ast_log(), ast_set_write_format(), and LOG_WARNING.
Referenced by action_playback_and_continue(), adsi_transmit_message_full(), ast_openstream_full(), ast_play_and_wait(), ast_readstring_full(), ast_say_enumeration_full_da(), ast_say_enumeration_full_de(), ast_say_enumeration_full_en(), ast_say_enumeration_full_he(), ast_say_enumeration_full_vi(), ast_say_number_full_cs(), ast_say_number_full_da(), ast_say_number_full_de(), ast_say_number_full_en(), ast_say_number_full_en_GB(), ast_say_number_full_es(), ast_say_number_full_fr(), ast_say_number_full_gr(), ast_say_number_full_he(), ast_say_number_full_hu(), ast_say_number_full_it(), ast_say_number_full_ka(), ast_say_number_full_nl(), ast_say_number_full_no(), ast_say_number_full_pt(), ast_say_number_full_ru(), ast_say_number_full_se(), ast_say_number_full_th(), ast_say_number_full_ur(), ast_say_number_full_vi(), ast_say_number_full_zh(), background_detect_exec(), conf_exec(), conf_run(), control_streamfile(), dial_exec_full(), directory_exec(), grab_transfer(), handle_getoption(), handle_speechrecognize(), handle_streamfile(), ices_exec(), ivr_dispatch(), leave_voicemail(), meetme_menu_admin(), meetme_menu_admin_extended(), minivm_greet_exec(), mp3_exec(), NBScat_exec(), parkandannounce_exec(), pbx_builtin_background(), pl_odtworz_plik(), play_file(), play_files_helper(), play_mailbox_owner(), playback_exec(), queue_exec(), read_exec(), readexten_exec(), record_exec(), recordthread(), s_streamwait3(), say_character_str_full(), say_digit_str_full(), say_phonetic_str_full(), select_item_seq(), send_waveform_to_channel(), speech_background(), vm_authenticate(), vm_execmain(), wait_for_winner(), waitstream_core(), and zapateller_exec().
{
ast_channel_lock(tmp);
/* Stop a running stream if there is one */
if (ast_channel_stream(tmp)) {
ast_closestream(ast_channel_stream(tmp));
ast_channel_stream_set(tmp, NULL);
if (ast_channel_oldwriteformat(tmp)->id && ast_set_write_format(tmp, ast_channel_oldwriteformat(tmp)))
ast_log(LOG_WARNING, "Unable to restore format back to %s\n", ast_getformatname(ast_channel_oldwriteformat(tmp)));
}
/* Stop the video stream too */
if (ast_channel_vstream(tmp) != NULL) {
ast_closestream(ast_channel_vstream(tmp));
ast_channel_vstream_set(tmp, NULL);
}
ast_channel_unlock(tmp);
return 0;
}
| int ast_stream_and_wait | ( | struct ast_channel * | chan, |
| const char * | file, | ||
| const char * | digits | ||
| ) |
stream file until digit If the file name is non-empty, try to play it.
| -1 | if error. |
| digit | if interrupted by a digit. |
Definition at line 1460 of file file.c.
References ast_channel_language(), ast_channel_name(), ast_streamfile(), ast_strlen_zero(), ast_test_suite_event_notify, and ast_waitstream().
Referenced by __ast_play_and_record(), action_kick_last(), action_playback(), action_toggle_mute(), action_toggle_mute_participants(), announce_user_count(), app_exec(), ast_pickup_call(), ast_record_review(), bridge_playfile(), builtin_atxfer(), builtin_automixmonitor(), builtin_blindtransfer(), confbridge_exec(), directory_exec(), execute_menu_entry(), feature_attended_transfer(), feature_blind_transfer(), forward_message(), grab_transfer(), invent_message(), ivr_dispatch(), join_conference_bridge(), leave_voicemail(), masq_park_call(), park_call_exec(), parked_call_exec(), play_files_helper(), play_mailbox_owner(), play_message_callerid(), play_message_on_chan(), play_prompt_to_user(), play_record_review(), play_sound_helper(), sayname(), select_item_seq(), vm_forwardoptions(), vmsayname_exec(), wait_file2(), and xfer_park_call_helper().
{
int res = 0;
if (!ast_strlen_zero(file)) {
ast_test_suite_event_notify("PLAYBACK", "Message: %s\r\nChannel: %s", file, ast_channel_name(chan));
res = ast_streamfile(chan, file, ast_channel_language(chan));
if (!res) {
res = ast_waitstream(chan, digits);
}
}
return res;
}
| int ast_stream_fastforward | ( | struct ast_filestream * | fs, |
| off_t | ms | ||
| ) |
Fast forward stream ms.
| fs | filestream to act on |
| ms | milliseconds to move |
| 0 | on success. |
| -1 | on failure. |
Definition at line 954 of file file.c.
References ast_seekstream(), and DEFAULT_SAMPLES_PER_MS.
Referenced by waitstream_core().
{
return ast_seekstream(fs, ms * DEFAULT_SAMPLES_PER_MS, SEEK_CUR);
}
| int ast_stream_rewind | ( | struct ast_filestream * | fs, |
| off_t | ms | ||
| ) |
Rewind stream ms.
| fs | filestream to act on |
| ms | milliseconds to move |
| 0 | on success. |
| -1 | on failure. |
Definition at line 959 of file file.c.
References ast_seekstream(), and DEFAULT_SAMPLES_PER_MS.
Referenced by __ast_play_and_record(), handle_recordfile(), record_exec(), and waitstream_core().
{
return ast_seekstream(fs, -ms * DEFAULT_SAMPLES_PER_MS, SEEK_CUR);
}
| int ast_streamfile | ( | struct ast_channel * | c, |
| const char * | filename, | ||
| const char * | preflang | ||
| ) |
Streams a file.
| c | channel to stream the file to |
| filename | the name of the file you wish to stream, minus the extension |
| preflang | the preferred language you wish to have the file streamed to you in Prepares a channel for the streaming of a file. To start the stream, afterward do a ast_waitstream() on the channel Also, it will stop any existing streams on the channel. |
| 0 | on success. |
| -1 | on failure. |
Definition at line 1006 of file file.c.
References ast_applystream(), ast_channel_flags(), ast_channel_name(), ast_channel_nativeformats(), ast_channel_writeformat(), ast_debug, AST_FLAG_MASQ_NOSTREAM, ast_getformatname(), ast_getformatname_multiple(), ast_log(), ast_openstream(), ast_openvstream(), ast_playstream(), ast_seekstream(), ast_strdup, ast_test_flag, ast_verb, errno, ast_filestream::f, ast_filestream::fmt, ast_format_def::format, LOG_WARNING, ast_filestream::orig_chan_name, and ast_filestream::vfs.
Referenced by __analog_ss_thread(), action_bridge(), action_playback_and_continue(), agent_call(), analog_ss_thread(), announce_thread(), app_exec(), ast_app_getdata(), ast_app_getdata_full(), ast_play_and_wait(), ast_say_date_da(), ast_say_date_de(), ast_say_date_en(), ast_say_date_fr(), ast_say_date_gr(), ast_say_date_he(), ast_say_date_hu(), ast_say_date_ka(), ast_say_date_nl(), ast_say_date_th(), ast_say_date_with_format_gr(), ast_say_datetime_en(), ast_say_datetime_fr(), ast_say_datetime_from_now_en(), ast_say_datetime_from_now_fr(), ast_say_datetime_from_now_he(), ast_say_datetime_from_now_ka(), ast_say_datetime_gr(), ast_say_datetime_he(), ast_say_datetime_nl(), ast_say_datetime_pt(), ast_say_datetime_th(), ast_say_datetime_zh(), ast_say_enumeration_full_da(), ast_say_enumeration_full_de(), ast_say_enumeration_full_en(), ast_say_enumeration_full_he(), ast_say_enumeration_full_vi(), ast_say_number_full_cs(), ast_say_number_full_da(), ast_say_number_full_de(), ast_say_number_full_en(), ast_say_number_full_en_GB(), ast_say_number_full_es(), ast_say_number_full_fr(), ast_say_number_full_gr(), ast_say_number_full_he(), ast_say_number_full_hu(), ast_say_number_full_it(), ast_say_number_full_ka(), ast_say_number_full_nl(), ast_say_number_full_no(), ast_say_number_full_pt(), ast_say_number_full_ru(), ast_say_number_full_se(), ast_say_number_full_th(), ast_say_number_full_ur(), ast_say_number_full_vi(), ast_say_number_full_zh(), ast_say_time_de(), ast_say_time_en(), ast_say_time_fr(), ast_say_time_gr(), ast_say_time_hu(), ast_say_time_ka(), ast_say_time_nl(), ast_say_time_zh(), ast_stream_and_wait(), auth_exec(), background_detect_exec(), bridge_exec(), check_availability(), check_beep(), common_exec(), conf_exec(), conf_get_pin(), conf_run(), control_streamfile(), dial_exec_full(), do_directory(), find_conf_realtime(), forward_message(), gr_say_number_female(), handle_recordfile(), invent_message(), leave_voicemail(), local_attended_transfer(), login_exec(), meetme_menu_admin(), meetme_menu_admin_extended(), meetme_menu_normal(), minivm_greet_exec(), page_exec(), parkandannounce_exec(), pbx_builtin_background(), pl_odtworz_plik(), play_and_wait(), play_file(), play_record_review(), playback_exec(), privacy_exec(), readexten_exec(), record_exec(), retrydial_exec(), s_streamwait3(), say_character_str_full(), say_digit_str_full(), say_phonetic_str_full(), select_item_menu(), setup_privacy_args(), vm_authenticate(), wait_file(), and wait_for_winner().
{
struct ast_filestream *fs;
struct ast_filestream *vfs=NULL;
char fmt[256];
off_t pos;
int seekattempt;
int res;
fs = ast_openstream(chan, filename, preflang);
if (!fs) {
ast_log(LOG_WARNING, "Unable to open %s (format %s): %s\n", filename, ast_getformatname_multiple(fmt, sizeof(fmt), ast_channel_nativeformats(chan)), strerror(errno));
return -1;
}
/* check to see if there is any data present (not a zero length file),
* done this way because there is no where for ast_openstream_full to
* return the file had no data. */
pos = ftello(fs->f);
seekattempt = fseeko(fs->f, -1, SEEK_END);
if (seekattempt) {
if (errno == EINVAL) {
/* Zero-length file, as opposed to a pipe */
return 0;
} else {
ast_seekstream(fs, 0, SEEK_SET);
}
} else {
fseeko(fs->f, pos, SEEK_SET);
}
vfs = ast_openvstream(chan, filename, preflang);
if (vfs) {
ast_debug(1, "Ooh, found a video stream, too, format %s\n", ast_getformatname(&vfs->fmt->format));
}
if (ast_test_flag(ast_channel_flags(chan), AST_FLAG_MASQ_NOSTREAM))
fs->orig_chan_name = ast_strdup(ast_channel_name(chan));
if (ast_applystream(chan, fs))
return -1;
if (vfs && ast_applystream(chan, vfs))
return -1;
res = ast_playstream(fs);
if (!res && vfs)
res = ast_playstream(vfs);
ast_verb(3, "<%s> Playing '%s.%s' (language '%s')\n", ast_channel_name(chan), filename, ast_getformatname(ast_channel_writeformat(chan)), preflang ? preflang : "default");
return res;
}
| off_t ast_tellstream | ( | struct ast_filestream * | fs | ) |
Tell where we are in a stream.
| fs | fs to act on |
Definition at line 949 of file file.c.
References ast_filestream::fmt, and ast_format_def::tell.
Referenced by __ast_play_and_record(), ast_moh_files_next(), control_streamfile(), handle_getoption(), handle_recordfile(), handle_speechrecognize(), handle_streamfile(), msg_create_from_file(), and waitstream_core().
| int ast_truncstream | ( | struct ast_filestream * | fs | ) |
Trunc stream at current location.
| fs | filestream to act on |
| 0 | on success. |
| -1 | on failure. |
Definition at line 944 of file file.c.
References ast_filestream::fmt, and ast_format_def::trunc.
Referenced by __ast_play_and_record(), handle_recordfile(), and record_exec().
| int ast_waitstream | ( | struct ast_channel * | c, |
| const char * | breakon | ||
| ) |
Waits for a stream to stop or digit to be pressed.
| c | channel to waitstream on |
| breakon | string of DTMF digits to break upon Begins playback of a stream... Wait for a stream to stop or for any one of a given digit to arrive, |
| 0 | if the stream finishes |
| the | character if it was interrupted, |
| -1 | on error |
Definition at line 1433 of file file.c.
References waitstream_core().
Referenced by __analog_ss_thread(), action_bridge(), action_playback_and_continue(), agent_call(), analog_ss_thread(), announce_thread(), app_exec(), ast_play_and_wait(), ast_say_date_da(), ast_say_date_de(), ast_say_date_en(), ast_say_date_fr(), ast_say_date_gr(), ast_say_date_he(), ast_say_date_hu(), ast_say_date_ka(), ast_say_date_nl(), ast_say_date_th(), ast_say_date_with_format_gr(), ast_say_datetime_en(), ast_say_datetime_fr(), ast_say_datetime_from_now_en(), ast_say_datetime_from_now_fr(), ast_say_datetime_from_now_he(), ast_say_datetime_from_now_ka(), ast_say_datetime_gr(), ast_say_datetime_he(), ast_say_datetime_nl(), ast_say_datetime_pt(), ast_say_datetime_th(), ast_say_datetime_zh(), ast_say_enumeration_full_da(), ast_say_enumeration_full_de(), ast_say_enumeration_full_en(), ast_say_enumeration_full_he(), ast_say_enumeration_full_vi(), ast_say_number_full_cs(), ast_say_number_full_da(), ast_say_number_full_de(), ast_say_number_full_en(), ast_say_number_full_en_GB(), ast_say_number_full_es(), ast_say_number_full_fr(), ast_say_number_full_gr(), ast_say_number_full_he(), ast_say_number_full_hu(), ast_say_number_full_it(), ast_say_number_full_ka(), ast_say_number_full_nl(), ast_say_number_full_no(), ast_say_number_full_pt(), ast_say_number_full_ru(), ast_say_number_full_se(), ast_say_number_full_th(), ast_say_number_full_ur(), ast_say_number_full_vi(), ast_say_number_full_zh(), ast_say_time_de(), ast_say_time_en(), ast_say_time_gr(), ast_say_time_he(), ast_say_time_hu(), ast_say_time_ka(), ast_say_time_nl(), ast_say_time_zh(), ast_stream_and_wait(), auth_exec(), bridge_exec(), check_availability(), check_beep(), common_exec(), conf_exec(), conf_get_pin(), conf_run(), directory_exec(), find_conf_realtime(), gr_say_number_female(), handle_recordfile(), invent_message(), leave_voicemail(), local_attended_transfer(), login_exec(), meetme_menu_admin(), meetme_menu_admin_extended(), meetme_menu_normal(), minivm_greet_exec(), page_exec(), parkandannounce_exec(), pbx_builtin_background(), pl_odtworz_plik(), play_and_wait(), play_file(), play_record_review(), playback_exec(), privacy_exec(), record_exec(), retrydial_exec(), s_streamwait3(), say_character_str_full(), say_digit_str_full(), say_phonetic_str_full(), select_item_menu(), setup_privacy_args(), vm_authenticate(), and wait_file().
{
return waitstream_core(c, breakon, NULL, NULL, 0, -1, -1, NULL, NULL /* no callback */);
}
| int ast_waitstream_exten | ( | struct ast_channel * | c, |
| const char * | context | ||
| ) |
Waits for a stream to stop or digit matching a valid one digit exten to be pressed.
| c | channel to waitstream on |
| context | string of context to match digits to break upon Begins playback of a stream... Wait for a stream to stop or for any one of a valid extension digit to arrive, |
| 0 | if the stream finishes. |
| the | character if it was interrupted. |
| -1 | on error. |
Definition at line 1444 of file file.c.
References ast_channel_context(), and waitstream_core().
Referenced by pbx_builtin_background().
{
/* Waitstream, with return in the case of a valid 1 digit extension */
/* in the current or specified context being pressed */
if (!context)
context = ast_channel_context(c);
return waitstream_core(c, NULL, NULL, NULL, 0,
-1, -1, context, NULL /* no callback */);
}
| int ast_waitstream_fr | ( | struct ast_channel * | c, |
| const char * | breakon, | ||
| const char * | forward, | ||
| const char * | rewind, | ||
| int | ms | ||
| ) |
Same as waitstream but allows stream to be forwarded or rewound.
| c | channel to waitstream on |
| breakon | string of DTMF digits to break upon |
| forward | DTMF digit to fast forward upon |
| rewind | DTMF digit to rewind upon |
| ms | How many miliseconds to skip forward/back Begins playback of a stream... Wait for a stream to stop or for any one of a given digit to arrive, |
| 0 | if the stream finishes. |
| the | character if it was interrupted. |
| -1 | on error. |
Definition at line 1427 of file file.c.
References waitstream_core().
Referenced by control_streamfile().
{
return waitstream_core(c, breakon, forward, reverse, ms,
-1 /* no audiofd */, -1 /* no cmdfd */, NULL /* no context */, NULL /* no callback */);
}
| int ast_waitstream_fr_w_cb | ( | struct ast_channel * | c, |
| const char * | breakon, | ||
| const char * | forward, | ||
| const char * | rewind, | ||
| int | ms, | ||
| ast_waitstream_fr_cb | cb | ||
| ) |
Same as waitstream_fr but allows a callback to be alerted when a user fastforwards or rewinds the file.
| c | channel to waitstream on |
| breakon | string of DTMF digits to break upon |
| forward | DTMF digit to fast forward upon |
| rewind | DTMF digit to rewind upon |
| ms | How many milliseconds to skip forward/back |
| cb | to call when rewind or fastfoward occurs. Begins playback of a stream... Wait for a stream to stop or for any one of a given digit to arrive, |
| 0 | if the stream finishes. |
| the | character if it was interrupted. |
| -1 | on error. |
Definition at line 1416 of file file.c.
References waitstream_core().
Referenced by control_streamfile().
{
return waitstream_core(c, breakon, forward, reverse, ms,
-1 /* no audiofd */, -1 /* no cmdfd */, NULL /* no context */, cb);
}
| int ast_waitstream_full | ( | struct ast_channel * | c, |
| const char * | breakon, | ||
| int | audiofd, | ||
| int | monfd | ||
| ) |
Same as waitstream, but with audio output to fd and monitored fd checking.
Definition at line 1438 of file file.c.
References waitstream_core().
Referenced by ast_readstring_full(), ast_say_enumeration_full_da(), ast_say_enumeration_full_de(), ast_say_enumeration_full_en(), ast_say_enumeration_full_he(), ast_say_enumeration_full_vi(), ast_say_number_full_cs(), ast_say_number_full_da(), ast_say_number_full_de(), ast_say_number_full_en(), ast_say_number_full_en_GB(), ast_say_number_full_es(), ast_say_number_full_fr(), ast_say_number_full_gr(), ast_say_number_full_he(), ast_say_number_full_hu(), ast_say_number_full_it(), ast_say_number_full_ka(), ast_say_number_full_nl(), ast_say_number_full_no(), ast_say_number_full_pt(), ast_say_number_full_ru(), ast_say_number_full_se(), ast_say_number_full_th(), ast_say_number_full_ur(), ast_say_number_full_vi(), ast_say_number_full_zh(), handle_getoption(), handle_streamfile(), pl_odtworz_plik(), s_streamwait3(), say_character_str_full(), say_digit_str_full(), and say_phonetic_str_full().
{
return waitstream_core(c, breakon, NULL, NULL, 0,
audiofd, cmdfd, NULL /* no context */, NULL /* no callback */);
}
| struct ast_filestream* ast_writefile | ( | const char * | filename, |
| const char * | type, | ||
| const char * | comment, | ||
| int | flags, | ||
| int | check, | ||
| mode_t | mode | ||
| ) | [read] |
Starts writing a file.
| filename | the name of the file to write to |
| type | format of file you wish to write out to |
| comment | comment to go with |
| flags | output file flags |
| check | (unimplemented, hence negligible) |
| mode | Open mode Create an outgoing file stream. oflags are flags for the open() command, and if check is non-zero, then it will not write a file if there are any files that start with that name and have an extension Please note, this is a blocking function. Program execution will not return until ast_waitstream completes it's execution. |
| a | struct ast_filestream on success. |
| NULL | on failure. |
Definition at line 1108 of file file.c.
References ast_alloca, ast_closestream(), ast_free, ast_log(), ast_malloc, ast_opt_cache_record_files, AST_RWLIST_RDLOCK, AST_RWLIST_TRAVERSE, AST_RWLIST_UNLOCK, ast_strdup, ast_filestream::buf, build_filename(), errno, ast_format_def::exts, exts_compare(), ast_filestream::f, f, ast_filestream::filename, ast_filestream::flags, ast_filestream::fmt, get_filestream(), LOG_WARNING, ast_filestream::mode, ast_filestream::realfilename, record_cache_dir, rewrite_wrapper(), ast_format_def::seek, ast_filestream::trans, ast_filestream::vfs, and ast_filestream::write_buffer.
Referenced by __ast_play_and_record(), ast_monitor_start(), ast_writestream(), dictate_exec(), handle_cli_file_convert(), handle_recordfile(), mixmonitor_save_prep(), record_exec(), and recordthread().
{
int fd, myflags = 0;
/* compiler claims this variable can be used before initialization... */
FILE *bfile = NULL;
struct ast_format_def *f;
struct ast_filestream *fs = NULL;
char *buf = NULL;
size_t size = 0;
int format_found = 0;
AST_RWLIST_RDLOCK(&formats);
/* set the O_TRUNC flag if and only if there is no O_APPEND specified */
/* We really can't use O_APPEND as it will break WAV header updates */
if (flags & O_APPEND) {
flags &= ~O_APPEND;
} else {
myflags = O_TRUNC;
}
myflags |= O_WRONLY | O_CREAT;
/* XXX need to fix this - we should just do the fopen,
* not open followed by fdopen()
*/
AST_RWLIST_TRAVERSE(&formats, f, list) {
char *fn, *orig_fn = NULL;
if (fs)
break;
if (!exts_compare(f->exts, type))
continue;
else
format_found = 1;
fn = build_filename(filename, type);
if (!fn) {
continue;
}
fd = open(fn, flags | myflags, mode);
if (fd > -1) {
/* fdopen() the resulting file stream */
bfile = fdopen(fd, ((flags | myflags) & O_RDWR) ? "w+" : "w");
if (!bfile) {
ast_log(LOG_WARNING, "Whoa, fdopen failed: %s!\n", strerror(errno));
close(fd);
fd = -1;
}
}
if (ast_opt_cache_record_files && (fd > -1)) {
char *c;
fclose(bfile); /* this also closes 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);
for (c = fn; *c; c++)
if (*c == '/')
*c = '_';
size = strlen(fn) + strlen(record_cache_dir) + 2;
buf = ast_alloca(size);
strcpy(buf, record_cache_dir);
strcat(buf, "/");
strcat(buf, fn);
ast_free(fn);
fn = buf;
fd = open(fn, flags | myflags, mode);
if (fd > -1) {
/* fdopen() the resulting file stream */
bfile = fdopen(fd, ((flags | myflags) & O_RDWR) ? "w+" : "w");
if (!bfile) {
ast_log(LOG_WARNING, "Whoa, fdopen failed: %s!\n", strerror(errno));
close(fd);
fd = -1;
}
}
}
if (fd > -1) {
errno = 0;
fs = get_filestream(f, bfile);
if (fs) {
if ((fs->write_buffer = ast_malloc(32768))) {
setvbuf(fs->f, fs->write_buffer, _IOFBF, 32768);
}
}
if (!fs || rewrite_wrapper(fs, comment)) {
ast_log(LOG_WARNING, "Unable to rewrite %s\n", fn);
close(fd);
if (orig_fn) {
unlink(fn);
unlink(orig_fn);
}
if (fs) {
ast_closestream(fs);
fs = NULL;
}
if (!buf) {
ast_free(fn);
}
continue;
}
fs->trans = NULL;
fs->fmt = f;
fs->flags = flags;
fs->mode = mode;
if (orig_fn) {
fs->realfilename = ast_strdup(orig_fn);
fs->filename = ast_strdup(fn);
} else {
fs->realfilename = NULL;
fs->filename = ast_strdup(filename);
}
fs->vfs = NULL;
/* If truncated, we'll be at the beginning; if not truncated, then append */
f->seek(fs, 0, SEEK_END);
} else if (errno != EEXIST) {
ast_log(LOG_WARNING, "Unable to open file %s: %s\n", fn, strerror(errno));
if (orig_fn)
unlink(orig_fn);
}
/* if buf != NULL then fn is already free and pointing to it */
if (!buf)
ast_free(fn);
}
AST_RWLIST_UNLOCK(&formats);
if (!format_found)
ast_log(LOG_WARNING, "No such format '%s'\n", type);
return fs;
}
| int ast_writestream | ( | struct ast_filestream * | fs, |
| struct ast_frame * | f | ||
| ) |
Writes a frame to a stream.
| fs | filestream to write to |
| f | frame to write to the filestream Send a frame to a filestream -- note: does NOT free the frame, call ast_frfree manually |
| 0 | on success. |
| -1 | on failure. |
Definition at line 152 of file file.c.
References ast_debug, ast_format_cmp(), AST_FORMAT_CMP_EQUAL, AST_FORMAT_CMP_NOT_EQUAL, ast_format_copy(), AST_FORMAT_GET_TYPE, AST_FORMAT_TYPE_AUDIO, AST_FRAME_VIDEO, AST_FRAME_VOICE, ast_frfree, ast_getformatname(), AST_LIST_NEXT, ast_log(), ast_translate(), ast_translator_build_path(), ast_translator_free_path(), ast_writefile(), ast_writestream(), ast_filestream::filename, ast_filestream::flags, ast_filestream::fmt, ast_format_def::format, ast_frame_subclass::format, ast_frame::frametype, ast_format::id, ast_filestream::lastwriteformat, LOG_WARNING, ast_filestream::mode, ast_format_def::name, ast_frame::subclass, ast_filestream::trans, type, ast_filestream::vfs, and ast_format_def::write.
Referenced by __ast_play_and_record(), __ast_read(), ast_write(), ast_writestream(), dictate_exec(), handle_cli_file_convert(), handle_recordfile(), mixmonitor_thread(), record_exec(), and recordthread().
{
int res = -1;
if (f->frametype == AST_FRAME_VIDEO) {
if (AST_FORMAT_GET_TYPE(fs->fmt->format.id) == AST_FORMAT_TYPE_AUDIO) {
/* This is the audio portion. Call the video one... */
if (!fs->vfs && fs->filename) {
const char *type = ast_getformatname(&f->subclass.format);
fs->vfs = ast_writefile(fs->filename, type, NULL, fs->flags, 0, fs->mode);
ast_debug(1, "Opened video output file\n");
}
if (fs->vfs)
return ast_writestream(fs->vfs, f);
/* else ignore */
return 0;
}
} else if (f->frametype != AST_FRAME_VOICE) {
ast_log(LOG_WARNING, "Tried to write non-voice frame\n");
return -1;
}
if (ast_format_cmp(&f->subclass.format, &fs->fmt->format) != AST_FORMAT_CMP_NOT_EQUAL) {
res = fs->fmt->write(fs, f);
if (res < 0)
ast_log(LOG_WARNING, "Natural write failed\n");
else if (res > 0)
ast_log(LOG_WARNING, "Huh??\n");
} else {
/* 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 */
if (fs->trans && (ast_format_cmp(&f->subclass.format, &fs->lastwriteformat) != AST_FORMAT_CMP_EQUAL)) {
ast_translator_free_path(fs->trans);
fs->trans = NULL;
}
if (!fs->trans)
fs->trans = ast_translator_build_path(&fs->fmt->format, &f->subclass.format);
if (!fs->trans)
ast_log(LOG_WARNING, "Unable to translate to format %s, source format %s\n",
fs->fmt->name, ast_getformatname(&f->subclass.format));
else {
struct ast_frame *trf;
ast_format_copy(&fs->lastwriteformat, &f->subclass.format);
/* Get the translated frame but don't consume the original in case they're using it on another stream */
if ((trf = ast_translate(fs->trans, f, 0))) {
struct ast_frame *cur;
/* the translator may have returned multiple frames, so process them */
for (cur = trf; cur; cur = AST_LIST_NEXT(cur, frame_list)) {
if ((res = fs->fmt->write(fs, trf))) {
ast_log(LOG_WARNING, "Translated frame write failed\n");
break;
}
}
ast_frfree(trf);
} else {
res = 0;
}
}
}
return res;
}
| static char* build_filename | ( | const char * | filename, |
| const char * | ext | ||
| ) | [static] |
construct a filename. Absolute pathnames are preserved, relative names are prefixed by the sounds/ directory. The wav49 suffix is replaced by 'WAV'. Returns a malloc'ed string to be freed by the caller.
Definition at line 256 of file file.c.
References ast_asprintf, and ast_config_AST_DATA_DIR.
Referenced by ast_readfile(), ast_writefile(), and filehelper().
{
char *fn = NULL;
if (!strcmp(ext, "wav49"))
ext = "WAV";
if (filename[0] == '/') {
if (ast_asprintf(&fn, "%s.%s", filename, ext) < 0) {
fn = NULL;
}
} else {
if (ast_asprintf(&fn, "%s/sounds/%s.%s",
ast_config_AST_DATA_DIR, filename, ext) < 0) {
fn = NULL;
}
}
return fn;
}
| static int copy | ( | const char * | infile, |
| const char * | outfile | ||
| ) | [static] |
Definition at line 213 of file file.c.
References ast_log(), errno, len(), and LOG_WARNING.
Referenced by filehelper().
{
int ifd, ofd, len;
char buf[4096]; /* XXX make it lerger. */
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, AST_FILE_MODE)) < 0) {
ast_log(LOG_WARNING, "Unable to open %s in write-only mode\n", outfile);
close(ifd);
return -1;
}
while ( (len = read(ifd, buf, sizeof(buf)) ) ) {
int res;
if (len < 0) {
ast_log(LOG_WARNING, "Read failed on %s: %s\n", infile, strerror(errno));
break;
}
/* XXX handle partial writes */
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));
len = -1; /* error marker */
break;
}
}
close(ifd);
close(ofd);
if (len < 0) {
unlink(outfile);
return -1; /* error */
}
return 0; /* success */
}
| static int exts_compare | ( | const char * | exts, |
| const char * | type | ||
| ) | [static] |
Definition at line 278 of file file.c.
References ast_copy_string(), and ext.
Referenced by ast_format_str_reduce(), ast_readfile(), ast_writefile(), and filehelper().
{
char tmp[256];
char *stringp = tmp, *ext;
ast_copy_string(tmp, exts, sizeof(tmp));
while ((ext = strsep(&stringp, "|"))) {
if (!strcmp(ext, type))
return 1;
}
return 0;
}
| static void file_shutdown | ( | void | ) | [static] |
Definition at line 1589 of file file.c.
References ARRAY_LEN, and ast_cli_unregister_multiple().
Referenced by ast_file_init().
| static int fileexists_core | ( | const char * | filename, |
| const char * | fmt, | ||
| const char * | preflang, | ||
| char * | buf, | ||
| int | buflen, | ||
| struct ast_format_cap * | result_cap | ||
| ) | [static] |
helper routine to locate a file with a given format and language preference.
| filename,name | of the file. |
| fmt,format | to look for the file in. OPTIONAL |
| preflang,the | perfered language |
| buf,returns | the matching filename |
| buflen,size | of the buf |
| result_cap,OPTIONAL | format capabilities result structure returns what formats the file was found in. |
| 1,true. | file exists and result format is set |
| 0,false. | file does not exist. |
Definition at line 618 of file file.c.
References ast_strlen_zero(), DEFAULT_LANGUAGE, and fileexists_test().
Referenced by ast_fileexists(), ast_openstream_full(), and ast_openvstream().
{
char *lang;
if (buf == NULL) {
return 0;
}
/* We try languages in the following order:
* preflang (may include dialect and style codes)
* lang (preflang without dialect - if any)
* <none>
* default (unless the same as preflang or lang without dialect)
*/
lang = ast_strdupa(preflang);
/* Try preferred language, including removing any style or dialect codes */
while (!ast_strlen_zero(lang)) {
char *end;
if (fileexists_test(filename, fmt, lang, buf, buflen, result_cap)) {
return 1;
}
if ((end = strrchr(lang, '_')) != NULL) {
*end = '\0';
continue;
}
break;
}
/* Try without any language */
if (fileexists_test(filename, fmt, NULL, buf, buflen, result_cap)) {
return 1;
}
/* Finally try the default language unless it was already tried before */
if ((ast_strlen_zero(preflang) || strcmp(preflang, DEFAULT_LANGUAGE)) && (ast_strlen_zero(lang) || strcmp(lang, DEFAULT_LANGUAGE))) {
if ((fileexists_test(filename, fmt, DEFAULT_LANGUAGE, buf, buflen, result_cap)) > 0) {
return 1;
}
}
return 0;
}
| static int fileexists_test | ( | const char * | filename, |
| const char * | fmt, | ||
| const char * | lang, | ||
| char * | buf, | ||
| int | buflen, | ||
| struct ast_format_cap * | result_cap | ||
| ) | [static] |
test if a file exists for a given format.
| 1,true | and result_cap represents format capabilities file exists in. |
| 0,false |
Definition at line 572 of file file.c.
References ACTION_EXISTS, filehelper(), and is_absolute_path().
Referenced by fileexists_core().
{
if (buf == NULL) {
return 0;
}
if (ast_language_is_prefix && !is_absolute_path(filename)) { /* new layout */
if (lang) {
snprintf(buf, buflen, "%s/%s", lang, filename);
} else {
snprintf(buf, buflen, "%s", filename);
}
} else { /* old layout */
strcpy(buf, filename); /* first copy the full string */
if (lang) {
/* insert the language and suffix if needed */
const char *c = strrchr(filename, '/');
int offset = c ? c - filename + 1 : 0; /* points right after the last '/' */
snprintf(buf + offset, buflen - offset, "%s/%s", lang, filename + offset);
}
}
return filehelper(buf, result_cap, fmt, ACTION_EXISTS);
}
| static int filehelper | ( | const char * | filename, |
| const void * | arg2, | ||
| const char * | fmt, | ||
| const enum file_action | action | ||
| ) | [static] |
Definition at line 438 of file file.c.
References ACTION_COPY, ACTION_DELETE, ACTION_EXISTS, ACTION_OPEN, ACTION_RENAME, ast_channel_stream(), ast_channel_stream_set(), ast_channel_vstream(), ast_channel_vstream_set(), ast_channel_writeformat(), ast_closestream(), ast_format_cap_add(), ast_format_cmp(), AST_FORMAT_CMP_NOT_EQUAL, AST_FORMAT_GET_TYPE, AST_FORMAT_TYPE_AUDIO, AST_FORMAT_TYPE_VIDEO, ast_free, ast_log(), AST_RWLIST_RDLOCK, AST_RWLIST_TRAVERSE, AST_RWLIST_UNLOCK, build_filename(), copy(), errno, ext, ast_format_def::exts, exts_compare(), f, ast_filestream::filename, ast_filestream::fmt, ast_format_def::format, get_filestream(), ast_format::id, ast_filestream::lasttimeout, ast_format_def::list, LOG_WARNING, open_wrapper(), and ast_filestream::trans.
Referenced by ast_filecopy(), ast_filedelete(), ast_filerename(), ast_openstream_full(), ast_openvstream(), and fileexists_test().
{
struct ast_format_def *f;
int res = (action == ACTION_EXISTS) ? 0 : -1;
AST_RWLIST_RDLOCK(&formats);
/* Check for a specific format */
AST_RWLIST_TRAVERSE(&formats, f, list) {
char *stringp, *ext = NULL;
if (fmt && !exts_compare(f->exts, fmt))
continue;
/* Look for a file matching the supported extensions.
* The file must exist, and for OPEN, must match
* one of the formats supported by the channel.
*/
stringp = ast_strdupa(f->exts); /* this is in the stack so does not need to be freed */
while ( (ext = strsep(&stringp, "|")) ) {
struct stat st;
char *fn = build_filename(filename, ext);
if (fn == NULL)
continue;
if ( stat(fn, &st) ) { /* file not existent */
ast_free(fn);
continue;
}
/* for 'OPEN' we need to be sure that the format matches
* what the channel can process
*/
if (action == ACTION_OPEN) {
struct ast_channel *chan = (struct ast_channel *)arg2;
FILE *bfile;
struct ast_filestream *s;
if ((ast_format_cmp(ast_channel_writeformat(chan), &f->format) == AST_FORMAT_CMP_NOT_EQUAL) &&
!(((AST_FORMAT_GET_TYPE(f->format.id) == AST_FORMAT_TYPE_AUDIO) && fmt) ||
((AST_FORMAT_GET_TYPE(f->format.id) == AST_FORMAT_TYPE_VIDEO) && fmt))) {
ast_free(fn);
continue; /* not a supported format */
}
if ( (bfile = fopen(fn, "r")) == NULL) {
ast_free(fn);
continue; /* cannot open file */
}
s = get_filestream(f, bfile);
if (!s) {
fclose(bfile);
ast_free(fn); /* cannot allocate descriptor */
continue;
}
if (open_wrapper(s)) {
ast_free(fn);
ast_closestream(s);
continue; /* cannot run open on file */
}
if (st.st_size == 0) {
ast_log(LOG_WARNING, "File %s detected to have zero size.\n", fn);
}
/* ok this is good for OPEN */
res = 1; /* found */
s->lasttimeout = -1;
s->fmt = f;
s->trans = NULL;
s->filename = NULL;
if (AST_FORMAT_GET_TYPE(s->fmt->format.id) == AST_FORMAT_TYPE_AUDIO) {
if (ast_channel_stream(chan))
ast_closestream(ast_channel_stream(chan));
ast_channel_stream_set(chan, s);
} else {
if (ast_channel_vstream(chan))
ast_closestream(ast_channel_vstream(chan));
ast_channel_vstream_set(chan, s);
}
ast_free(fn);
break;
}
switch (action) {
case ACTION_OPEN:
break; /* will never get here */
case ACTION_EXISTS: /* return the matching format */
/* if arg2 is present, it is a format capabilities structure.
* Add this format to the set of formats this file can be played in */
if (arg2) {
ast_format_cap_add((struct ast_format_cap *) arg2, &f->format);
}
res = 1; /* file does exist and format it exists in is returned in arg2 */
break;
case ACTION_DELETE:
if ( (res = unlink(fn)) )
ast_log(LOG_WARNING, "unlink(%s) failed: %s\n", fn, strerror(errno));
break;
case ACTION_RENAME:
case ACTION_COPY: {
char *nfn = build_filename((const char *)arg2, ext);
if (!nfn)
ast_log(LOG_WARNING, "Out of memory\n");
else {
res = action == ACTION_COPY ? copy(fn, nfn) : rename(fn, nfn);
if (res)
ast_log(LOG_WARNING, "%s(%s,%s) failed: %s\n",
action == ACTION_COPY ? "copy" : "rename",
fn, nfn, strerror(errno));
ast_free(nfn);
}
}
break;
default:
ast_log(LOG_WARNING, "Unknown helper %d\n", action);
}
ast_free(fn);
}
}
AST_RWLIST_UNLOCK(&formats);
return res;
}
| static void filestream_close | ( | struct ast_filestream * | f | ) | [static] |
Definition at line 293 of file file.c.
References ast_channel_sched(), ast_channel_stream_set(), ast_channel_streamid(), ast_channel_streamid_set(), ast_channel_vstream_set(), ast_channel_vstreamid(), ast_channel_vstreamid_set(), AST_FORMAT_GET_TYPE, AST_FORMAT_TYPE_AUDIO, AST_FORMAT_TYPE_VIDEO, ast_log(), AST_LOG_WARNING, AST_SCHED_DEL_ACCESSOR, ast_settimeout(), ast_filestream::fmt, ast_format_def::format, ast_format::id, ast_format_def::name, and ast_filestream::owner.
Referenced by ast_closestream(), and filestream_destructor().
{
enum ast_format_type format_type = AST_FORMAT_GET_TYPE(f->fmt->format.id);
if (!f->owner) {
return;
}
/* Stop a running stream if there is one */
switch (format_type)
{
case AST_FORMAT_TYPE_AUDIO:
ast_channel_stream_set(f->owner, NULL);
AST_SCHED_DEL_ACCESSOR(ast_channel_sched(f->owner), f->owner, ast_channel_streamid, ast_channel_streamid_set);
ast_settimeout(f->owner, 0, NULL, NULL);
break;
case AST_FORMAT_TYPE_VIDEO:
ast_channel_vstream_set(f->owner, NULL);
AST_SCHED_DEL_ACCESSOR(ast_channel_sched(f->owner), f->owner, ast_channel_vstreamid, ast_channel_vstreamid_set);
break;
default:
ast_log(AST_LOG_WARNING, "Unable to schedule deletion of filestream with unsupported type %s\n", f->fmt->name);
break;
}
}
| static void filestream_destructor | ( | void * | arg | ) | [static] |
Definition at line 319 of file file.c.
References ast_closestream(), ast_free, ast_module_unref(), ast_safe_fork(), ast_translator_free_path(), ast_format_def::close, ast_filestream::f, f, ast_filestream::filename, filestream_close(), ast_filestream::fmt, free, ast_format_def::module, ast_filestream::orig_chan_name, ast_filestream::realfilename, SENTINEL, status, ast_filestream::trans, ast_filestream::vfs, and ast_filestream::write_buffer.
Referenced by get_filestream().
{
struct ast_filestream *f = arg;
int status;
int pid = -1;
/* Stop a running stream if there is one */
filestream_close(f);
/* destroy the translator on exit */
if (f->trans)
ast_translator_free_path(f->trans);
if (f->fmt->close) {
void (*closefn)(struct ast_filestream *) = f->fmt->close;
closefn(f);
}
if (f->f) {
fclose(f->f);
}
if (f->realfilename && f->filename) {
pid = ast_safe_fork(0);
if (!pid) {
execl("/bin/mv", "mv", "-f", f->filename, f->realfilename, SENTINEL);
_exit(1);
}
else if (pid > 0) {
/* Block the parent until the move is complete.*/
waitpid(pid, &status, 0);
}
}
if (f->filename)
free(f->filename);
if (f->realfilename)
free(f->realfilename);
if (f->vfs)
ast_closestream(f->vfs);
if (f->write_buffer) {
ast_free(f->write_buffer);
}
if (f->orig_chan_name)
free((void *) f->orig_chan_name);
ast_module_unref(f->fmt->module);
}
| static int fn_wrapper | ( | struct ast_filestream * | s, |
| const char * | comment, | ||
| enum wrap_fn | mode | ||
| ) | [static] |
Definition at line 391 of file file.c.
References ast_log(), ast_module_ref(), f, ast_filestream::fmt, LOG_WARNING, ast_format_def::module, ast_format_def::name, ast_format_def::open, ast_format_def::rewrite, WRAP_OPEN, and WRAP_REWRITE.
Referenced by open_wrapper(), and rewrite_wrapper().
{
struct ast_format_def *f = s->fmt;
int ret = -1;
int (*openfn)(struct ast_filestream *s);
if (mode == WRAP_OPEN && (openfn = f->open) && openfn(s))
ast_log(LOG_WARNING, "Unable to open format %s\n", f->name);
else if (mode == WRAP_REWRITE && f->rewrite && f->rewrite(s, comment))
ast_log(LOG_WARNING, "Unable to rewrite format %s\n", f->name);
else {
/* preliminary checks succeed. update usecount */
ast_module_ref(f->module);
ret = 0;
}
return ret;
}
| static struct ast_filestream* get_filestream | ( | struct ast_format_def * | fmt, |
| FILE * | bfile | ||
| ) | [static, read] |
Definition at line 367 of file file.c.
References ast_filestream::_private, ao2_alloc, ast_filestream::buf, ast_format_def::buf_size, ast_format_def::desc_size, ast_filestream::f, filestream_destructor(), ast_filestream::fmt, ast_filestream::fr, ast_format_def::name, and ast_frame::src.
Referenced by ast_readfile(), ast_writefile(), and filehelper().
{
struct ast_filestream *s;
int l = sizeof(*s) + fmt->buf_size + fmt->desc_size; /* total allocation size */
if ( (s = ao2_alloc(l, filestream_destructor)) == NULL)
return NULL;
s->fmt = fmt;
s->f = bfile;
if (fmt->desc_size)
s->_private = ((char *)(s + 1)) + fmt->buf_size;
if (fmt->buf_size)
s->buf = (char *)(s + 1);
s->fr.src = fmt->name;
return s;
}
| static char* handle_cli_core_show_file_formats | ( | struct ast_cli_entry * | e, |
| int | cmd, | ||
| struct ast_cli_args * | a | ||
| ) | [static] |
Definition at line 1549 of file file.c.
References ast_cli_args::argc, ast_cli(), ast_getformatname(), AST_RWLIST_RDLOCK, AST_RWLIST_TRAVERSE, AST_RWLIST_UNLOCK, CLI_GENERATE, CLI_INIT, CLI_SHOWUSAGE, CLI_SUCCESS, ast_cli_entry::command, ast_format_def::exts, f, ast_cli_args::fd, ast_format_def::format, FORMAT, FORMAT2, ast_format_def::list, ast_format_def::name, and ast_cli_entry::usage.
{
#define FORMAT "%-10s %-10s %-20s\n"
#define FORMAT2 "%-10s %-10s %-20s\n"
struct ast_format_def *f;
int count_fmt = 0;
switch (cmd) {
case CLI_INIT:
e->command = "core show file formats";
e->usage =
"Usage: core show file formats\n"
" Displays currently registered file formats (if any).\n";
return NULL;
case CLI_GENERATE:
return NULL;
}
if (a->argc != 4)
return CLI_SHOWUSAGE;
ast_cli(a->fd, FORMAT, "Format", "Name", "Extensions");
ast_cli(a->fd, FORMAT, "------", "----", "----------");
AST_RWLIST_RDLOCK(&formats);
AST_RWLIST_TRAVERSE(&formats, f, list) {
ast_cli(a->fd, FORMAT2, ast_getformatname(&f->format), f->name, f->exts);
count_fmt++;
}
AST_RWLIST_UNLOCK(&formats);
ast_cli(a->fd, "%d file formats registered.\n", count_fmt);
return CLI_SUCCESS;
#undef FORMAT
#undef FORMAT2
}
| static int is_absolute_path | ( | const char * | filename | ) | [static] |
Definition at line 561 of file file.c.
Referenced by fileexists_test().
{
return filename[0] == '/';
}
| static int open_wrapper | ( | struct ast_filestream * | s | ) | [static] |
Definition at line 414 of file file.c.
References fn_wrapper(), and WRAP_OPEN.
Referenced by ast_readfile(), and filehelper().
{
return fn_wrapper(s, NULL, WRAP_OPEN);
}
| static struct ast_frame* read_frame | ( | struct ast_filestream * | s, |
| int * | whennext | ||
| ) | [static, read] |
Definition at line 775 of file file.c.
References ast_frfree, ast_frisolate(), ast_filestream::fmt, and ast_format_def::read.
Referenced by ast_readaudio_callback(), ast_readframe(), ast_readvideo_callback(), and audiohook_read_frame_helper().
{
struct ast_frame *fr, *new_fr;
if (!s || !s->fmt) {
return NULL;
}
if (!(fr = s->fmt->read(s, whennext))) {
return NULL;
}
if (!(new_fr = ast_frisolate(fr))) {
ast_frfree(fr);
return NULL;
}
if (new_fr != fr) {
ast_frfree(fr);
fr = new_fr;
}
return fr;
}
| static int rewrite_wrapper | ( | struct ast_filestream * | s, |
| const char * | comment | ||
| ) | [static] |
Definition at line 409 of file file.c.
References fn_wrapper(), and WRAP_REWRITE.
Referenced by ast_writefile().
{
return fn_wrapper(s, comment, WRAP_REWRITE);
}
| static int waitstream_core | ( | struct ast_channel * | c, |
| const char * | breakon, | ||
| const char * | forward, | ||
| const char * | reverse, | ||
| int | skip_ms, | ||
| int | audiofd, | ||
| int | cmdfd, | ||
| const char * | context, | ||
| ast_waitstream_fr_cb | cb | ||
| ) | [static] |
the core of all waitstream() functions
Definition at line 1249 of file file.c.
References ast_channel_caller(), ast_channel_flags(), ast_channel_name(), ast_channel_sched(), ast_channel_softhangup_internal_flag(), ast_channel_stream(), ast_channel_timingfunc(), ast_clear_flag, AST_CONTROL_ANSWER, AST_CONTROL_AOC, AST_CONTROL_BUSY, AST_CONTROL_CONGESTION, AST_CONTROL_CONNECTED_LINE, AST_CONTROL_HANGUP, AST_CONTROL_HOLD, AST_CONTROL_PVT_CAUSE_CODE, AST_CONTROL_REDIRECTING, AST_CONTROL_RINGING, AST_CONTROL_SRCCHANGE, AST_CONTROL_SRCUPDATE, AST_CONTROL_UNHOLD, AST_CONTROL_UPDATE_RTP_PEER, AST_CONTROL_VIDUPDATE, ast_exists_extension(), AST_FLAG_END_DTMF_ONLY, AST_FLAG_MASQ_NOSTREAM, ast_format_rate(), AST_FRAME_CONTROL, AST_FRAME_DTMF_END, AST_FRAME_VOICE, ast_frfree, ast_log(), ast_read(), ast_sched_runq(), ast_sched_wait(), ast_set_flag, ast_stopstream(), ast_stream_fastforward(), ast_stream_rewind(), ast_tellstream(), ast_test_flag, ast_waitfor(), ast_waitfor_nandfds(), AST_WAITSTREAM_CB_FASTFORWARD, AST_WAITSTREAM_CB_REWIND, AST_WAITSTREAM_CB_START, ast_frame::data, ast_frame::datalen, errno, exten, f, ast_filestream::fmt, ast_format_def::format, ast_frame::frametype, ast_frame_subclass::integer, LOG_WARNING, ast_filestream::orig_chan_name, ast_frame::ptr, S_COR, and ast_frame::subclass.
Referenced by ast_waitstream(), ast_waitstream_exten(), ast_waitstream_fr(), ast_waitstream_fr_w_cb(), and ast_waitstream_full().
{
const char *orig_chan_name = NULL;
int err = 0;
if (!breakon)
breakon = "";
if (!forward)
forward = "";
if (!reverse)
reverse = "";
/* Switch the channel to end DTMF frame only. waitstream_core doesn't care about the start of DTMF. */
ast_set_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
if (ast_test_flag(ast_channel_flags(c), AST_FLAG_MASQ_NOSTREAM))
orig_chan_name = ast_strdupa(ast_channel_name(c));
if (ast_channel_stream(c) && cb) {
long ms_len = ast_tellstream(ast_channel_stream(c)) / (ast_format_rate(&ast_channel_stream(c)->fmt->format) / 1000);
cb(c, ms_len, AST_WAITSTREAM_CB_START);
}
while (ast_channel_stream(c)) {
int res;
int ms;
if (orig_chan_name && strcasecmp(orig_chan_name, ast_channel_name(c))) {
ast_stopstream(c);
err = 1;
break;
}
ms = ast_sched_wait(ast_channel_sched(c));
if (ms < 0 && !ast_channel_timingfunc(c)) {
ast_stopstream(c);
break;
}
if (ms < 0)
ms = 1000;
if (cmdfd < 0) {
res = ast_waitfor(c, ms);
if (res < 0) {
ast_log(LOG_WARNING, "Select failed (%s)\n", strerror(errno));
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return res;
}
} else {
int outfd;
struct ast_channel *rchan = ast_waitfor_nandfds(&c, 1, &cmdfd, (cmdfd > -1) ? 1 : 0, NULL, &outfd, &ms);
if (!rchan && (outfd < 0) && (ms)) {
/* Continue */
if (errno == EINTR)
continue;
ast_log(LOG_WARNING, "Wait failed (%s)\n", strerror(errno));
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return -1;
} else if (outfd > -1) { /* this requires cmdfd set */
/* The FD we were watching has something waiting */
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return 1;
}
/* if rchan is set, it is 'c' */
res = rchan ? 1 : 0; /* map into 'res' values */
}
if (res > 0) {
struct ast_frame *fr = ast_read(c);
if (!fr) {
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return -1;
}
switch (fr->frametype) {
case AST_FRAME_DTMF_END:
if (context) {
const char exten[2] = { fr->subclass.integer, '\0' };
if (ast_exists_extension(c, context, exten, 1,
S_COR(ast_channel_caller(c)->id.number.valid, ast_channel_caller(c)->id.number.str, NULL))) {
res = fr->subclass.integer;
ast_frfree(fr);
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return res;
}
} else {
enum ast_waitstream_fr_cb_values cb_val = 0;
res = fr->subclass.integer;
if (strchr(forward, res)) {
int eoftest;
ast_stream_fastforward(ast_channel_stream(c), skip_ms);
eoftest = fgetc(ast_channel_stream(c)->f);
if (feof(ast_channel_stream(c)->f)) {
ast_stream_rewind(ast_channel_stream(c), skip_ms);
} else {
ungetc(eoftest, ast_channel_stream(c)->f);
}
cb_val = AST_WAITSTREAM_CB_FASTFORWARD;
} else if (strchr(reverse, res)) {
ast_stream_rewind(ast_channel_stream(c), skip_ms);
cb_val = AST_WAITSTREAM_CB_REWIND;
} else if (strchr(breakon, res)) {
ast_frfree(fr);
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return res;
}
if (cb_val && cb) {
long ms_len = ast_tellstream(ast_channel_stream(c)) / (ast_format_rate(&ast_channel_stream(c)->fmt->format) / 1000);
cb(c, ms_len, cb_val);
}
}
break;
case AST_FRAME_CONTROL:
switch (fr->subclass.integer) {
case AST_CONTROL_HANGUP:
case AST_CONTROL_BUSY:
case AST_CONTROL_CONGESTION:
ast_frfree(fr);
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return -1;
case AST_CONTROL_RINGING:
case AST_CONTROL_ANSWER:
case AST_CONTROL_VIDUPDATE:
case AST_CONTROL_SRCUPDATE:
case AST_CONTROL_SRCCHANGE:
case AST_CONTROL_HOLD:
case AST_CONTROL_UNHOLD:
case AST_CONTROL_CONNECTED_LINE:
case AST_CONTROL_REDIRECTING:
case AST_CONTROL_AOC:
case AST_CONTROL_UPDATE_RTP_PEER:
case AST_CONTROL_PVT_CAUSE_CODE:
case -1:
/* Unimportant */
break;
default:
ast_log(LOG_WARNING, "Unexpected control subclass '%d'\n", fr->subclass.integer);
}
break;
case AST_FRAME_VOICE:
/* Write audio if appropriate */
if (audiofd > -1) {
if (write(audiofd, fr->data.ptr, fr->datalen) < 0) {
ast_log(LOG_WARNING, "write() failed: %s\n", strerror(errno));
}
}
default:
/* Ignore all others */
break;
}
ast_frfree(fr);
}
ast_sched_runq(ast_channel_sched(c));
}
ast_clear_flag(ast_channel_flags(c), AST_FLAG_END_DTMF_ONLY);
return (err || ast_channel_softhangup_internal_flag(c)) ? -1 : 0;
}
| int ast_language_is_prefix = 1 |
The following variable controls the layout of localized sound files. If 0, use the historical layout with prefix just before the filename (i.e. digits/en/1.gsm , digits/it/1.gsm or default to digits/1.gsm), if 1 put the prefix at the beginning of the filename (i.e. en/digits/1.gsm, it/digits/1.gsm or default to digits/1.gsm). The latter permits a language to be entirely in one directory.
This is settable in asterisk.conf.
Definition at line 65 of file file.c.
Referenced by ast_readconfig(), handle_show_settings(), and main().
struct ast_cli_entry cli_file[] [static] |
{
AST_CLI_DEFINE(handle_cli_core_show_file_formats, "Displays file formats")
}