1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
|
/**
* @file echo.c Echo module
*/
#include <re.h>
#include <baresip.h>
/**
*
* Multi Call Echo module
*
* REQUIRES: aubridge
* NOTE: This module is experimental.
*
*/
struct session {
struct le le;
struct call *call_in;
};
static struct list sessionl;
static void destructor(void *arg)
{
struct session *sess = arg;
debug("echo: session destroyed\n");
list_unlink(&sess->le);
}
static void call_event_handler(struct call *call, enum call_event ev,
const char *str, void *arg)
{
struct session *sess = arg;
(void)call;
switch (ev) {
case CALL_EVENT_CLOSED:
debug("echo: CALL_CLOSED: %s\n", str);
mem_deref(sess->call_in);
mem_deref(sess);
break;
default:
break;
}
}
static void call_dtmf_handler(struct call *call, char key, void *arg)
{
(void)arg;
debug("echo: relaying DTMF event: key = '%c'\n", key ? key : '.');
call_send_digit(call, key);
}
static int new_session(struct ua *ua, struct call *call)
{
struct session *sess;
char a[64];
int err = 0;
sess = mem_zalloc(sizeof(*sess), destructor);
if (!sess)
return ENOMEM;
sess->call_in = call;
re_snprintf(a, sizeof(a), "A-%x", sess);
audio_set_devicename(call_audio(sess->call_in), a, a);
video_set_devicename(call_video(sess->call_in), a, a);
call_set_handlers(sess->call_in, call_event_handler,
call_dtmf_handler, sess);
list_append(&sessionl, &sess->le, sess);
err = ua_answer(ua, call, VIDMODE_ON);
if (err)
mem_deref(sess);
return err;
}
static void ua_event_handler(struct ua *ua, enum ua_event ev,
struct call *call, const char *prm, void *arg)
{
int err;
(void)prm;
(void)arg;
switch (ev) {
case UA_EVENT_CALL_INCOMING:
info("echo: CALL_INCOMING: peer=%s --> local=%s\n",
call_peeruri(call),
call_localuri(call));
err = new_session(ua, call);
if (err) {
call_hangup(call, 500, "Server Error");
}
break;
default:
break;
}
}
static int module_init(void)
{
int err;
list_init(&sessionl);
err = uag_event_register(ua_event_handler, 0);
if (err)
return err;
debug("echo: module loaded\n");
return 0;
}
static int module_close(void)
{
debug("echo: module closing..\n");
if (!list_isempty(&sessionl)) {
info("echo: flushing %u sessions\n", list_count(&sessionl));
list_flush(&sessionl);
}
uag_event_unregister(ua_event_handler);
return 0;
}
const struct mod_export DECL_EXPORTS(echo) = {
"echo",
"application",
module_init,
module_close
};
|