Prevent Echo() from relaying control, null, and modem frames
[asterisk/asterisk.git] / apps / app_echo.c
1 /*
2  * Asterisk -- An open source telephony toolkit.
3  *
4  * Copyright (C) 1999 - 2005, Digium, Inc.
5  *
6  * Mark Spencer <markster@digium.com>
7  *
8  * See http://www.asterisk.org for more information about
9  * the Asterisk project. Please do not directly contact
10  * any of the maintainers of this project for assistance;
11  * the project provides a web site, mailing lists and IRC
12  * channels for your use.
13  *
14  * This program is free software, distributed under the terms of
15  * the GNU General Public License Version 2. See the LICENSE file
16  * at the top of the source tree.
17  */
18
19 /*! \file
20  *
21  * \brief Echo application -- play back what you hear to evaluate latency
22  *
23  * \author Mark Spencer <markster@digium.com>
24  *
25  * \ingroup applications
26  */
27
28 /*** MODULEINFO
29         <support_level>core</support_level>
30  ***/
31
32 #include "asterisk.h"
33
34 ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
35
36 #include "asterisk/file.h"
37 #include "asterisk/module.h"
38 #include "asterisk/channel.h"
39
40 /*** DOCUMENTATION
41         <application name="Echo" language="en_US">
42                 <synopsis>
43                         Echo media, DTMF back to the calling party
44                 </synopsis>
45                 <syntax />
46                 <description>
47                         <para>Echos back any media or DTMF frames read from the calling 
48                         channel back to itself. This will not echo CONTROL, MODEM, or NULL
49                         frames. Note: If '#' detected application exits.</para>
50                         <para>This application does not automatically answer and should be
51                         preceeded by an application such as Answer() or Progress().</para>
52                 </description>
53         </application>
54  ***/
55
56 static const char app[] = "Echo";
57
58 static int echo_exec(struct ast_channel *chan, const char *data)
59 {
60         int res = -1;
61         struct ast_format format;
62
63         ast_best_codec(ast_channel_nativeformats(chan), &format);
64         ast_set_write_format(chan, &format);
65         ast_set_read_format(chan, &format);
66
67         while (ast_waitfor(chan, -1) > -1) {
68                 struct ast_frame *f = ast_read(chan);
69                 if (!f) {
70                         break;
71                 }
72                 f->delivery.tv_sec = 0;
73                 f->delivery.tv_usec = 0;
74                 if (f->frametype != AST_FRAME_CONTROL
75                         && f->frametype != AST_FRAME_MODEM
76                         && f->frametype != AST_FRAME_NULL
77                         && ast_write(chan, f)) {
78                         ast_frfree(f);
79                         goto end;
80                 }
81                 if ((f->frametype == AST_FRAME_DTMF) && (f->subclass.integer == '#')) {
82                         res = 0;
83                         ast_frfree(f);
84                         goto end;
85                 }
86                 ast_frfree(f);
87         }
88 end:
89         return res;
90 }
91
92 static int unload_module(void)
93 {
94         return ast_unregister_application(app);
95 }
96
97 static int load_module(void)
98 {
99         return ast_register_application_xml(app, echo_exec);
100 }
101
102 AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Simple Echo Application");