mirror of
https://github.com/signalwire/freeswitch.git
synced 2025-02-08 08:51:50 +00:00
FS-4969 --resolve
This commit is contained in:
parent
4baf0276d6
commit
bd2862080c
@ -1022,6 +1022,11 @@ struct record_helper {
|
|||||||
int native;
|
int native;
|
||||||
uint32_t packet_len;
|
uint32_t packet_len;
|
||||||
int min_sec;
|
int min_sec;
|
||||||
|
int final_timeout_ms;
|
||||||
|
int initial_timeout_ms;
|
||||||
|
int silence_threshold;
|
||||||
|
int silence_timeout_ms;
|
||||||
|
switch_time_t silence_time;
|
||||||
int rready;
|
int rready;
|
||||||
int wready;
|
int wready;
|
||||||
switch_time_t last_read_time;
|
switch_time_t last_read_time;
|
||||||
@ -1030,6 +1035,34 @@ struct record_helper {
|
|||||||
switch_codec_implementation_t read_impl;
|
switch_codec_implementation_t read_impl;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
static switch_bool_t is_silence_frame(switch_frame_t *frame, int silence_threshold, switch_codec_implementation_t *codec_impl)
|
||||||
|
{
|
||||||
|
int16_t *fdata = (int16_t *) frame->data;
|
||||||
|
uint32_t samples = frame->datalen / sizeof(*fdata);
|
||||||
|
switch_bool_t is_silence = SWITCH_TRUE;
|
||||||
|
uint32_t channel_num = 0;
|
||||||
|
|
||||||
|
int divisor = 0;
|
||||||
|
if (!(divisor = codec_impl->samples_per_second / 8000)) {
|
||||||
|
divisor = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* is silence only if every channel is silent */
|
||||||
|
for (channel_num = 0; channel_num < codec_impl->number_of_channels && is_silence; channel_num++) {
|
||||||
|
uint32_t count = 0, j = channel_num;
|
||||||
|
double energy = 0;
|
||||||
|
for (count = 0; count < samples; count++) {
|
||||||
|
energy += abs(fdata[j]);
|
||||||
|
j += codec_impl->number_of_channels;
|
||||||
|
}
|
||||||
|
is_silence &= (uint32_t) (energy / (samples / divisor)) < silence_threshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
return is_silence;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static switch_bool_t record_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
static switch_bool_t record_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
||||||
{
|
{
|
||||||
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
||||||
@ -1048,6 +1081,8 @@ static switch_bool_t record_callback(switch_media_bug_t *bug, void *user_data, s
|
|||||||
switch_channel_event_set_data(channel, event);
|
switch_channel_event_set_data(channel, event);
|
||||||
switch_event_fire(&event);
|
switch_event_fire(&event);
|
||||||
}
|
}
|
||||||
|
rh->silence_time = switch_micro_time_now();
|
||||||
|
rh->silence_timeout_ms = rh->initial_timeout_ms;
|
||||||
|
|
||||||
switch_core_session_get_read_impl(session, &rh->read_impl);
|
switch_core_session_get_read_impl(session, &rh->read_impl);
|
||||||
|
|
||||||
@ -1204,6 +1239,7 @@ static switch_bool_t record_callback(switch_media_bug_t *bug, void *user_data, s
|
|||||||
status = switch_core_media_bug_read(bug, &frame, SWITCH_FALSE);
|
status = switch_core_media_bug_read(bug, &frame, SWITCH_FALSE);
|
||||||
|
|
||||||
if (status == SWITCH_STATUS_SUCCESS || status == SWITCH_STATUS_BREAK) {
|
if (status == SWITCH_STATUS_SUCCESS || status == SWITCH_STATUS_BREAK) {
|
||||||
|
|
||||||
len = (switch_size_t) frame.datalen / 2;
|
len = (switch_size_t) frame.datalen / 2;
|
||||||
|
|
||||||
if (len && switch_core_file_write(rh->fh, mask ? null_data : data, &len) != SWITCH_STATUS_SUCCESS && rh->hangup_on_error) {
|
if (len && switch_core_file_write(rh->fh, mask ? null_data : data, &len) != SWITCH_STATUS_SUCCESS && rh->hangup_on_error) {
|
||||||
@ -1212,6 +1248,32 @@ static switch_bool_t record_callback(switch_media_bug_t *bug, void *user_data, s
|
|||||||
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
||||||
return SWITCH_FALSE;
|
return SWITCH_FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* check for silence timeout */
|
||||||
|
if (rh->silence_threshold) {
|
||||||
|
switch_codec_implementation_t read_impl = { 0 };
|
||||||
|
switch_core_session_get_read_impl(session, &read_impl);
|
||||||
|
if (is_silence_frame(&frame, rh->silence_threshold, &read_impl)) {
|
||||||
|
if (!rh->silence_time) {
|
||||||
|
/* start of silence */
|
||||||
|
rh->silence_time = switch_micro_time_now();
|
||||||
|
} else {
|
||||||
|
/* continuing silence */
|
||||||
|
int duration_ms = (switch_micro_time_now() - rh->silence_time) / 1000;
|
||||||
|
if (rh->silence_timeout_ms > 0 && duration_ms >= rh->silence_timeout_ms) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Recording file %s timeout: %i >= %i\n", rh->file, duration_ms, rh->silence_timeout_ms);
|
||||||
|
switch_core_media_bug_set_flag(bug, SMBF_PRUNE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else { /* not silence */
|
||||||
|
if (rh->silence_time) {
|
||||||
|
/* end of silence */
|
||||||
|
rh->silence_time = 0;
|
||||||
|
/* switch from initial timeout to final timeout */
|
||||||
|
rh->silence_timeout_ms = rh->final_timeout_ms;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -2080,6 +2142,29 @@ SWITCH_DECLARE(switch_status_t) switch_ivr_record_session(switch_core_session_t
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((p = switch_channel_get_variable(channel, "RECORD_INITIAL_TIMEOUT_MS"))) {
|
||||||
|
int tmp = atoi(p);
|
||||||
|
if (tmp >= 0) {
|
||||||
|
rh->initial_timeout_ms = tmp;
|
||||||
|
rh->silence_threshold = 200;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((p = switch_channel_get_variable(channel, "RECORD_FINAL_TIMEOUT_MS"))) {
|
||||||
|
int tmp = atoi(p);
|
||||||
|
if (tmp >= 0) {
|
||||||
|
rh->final_timeout_ms = tmp;
|
||||||
|
rh->silence_threshold = 200;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((p = switch_channel_get_variable(channel, "RECORD_SILENCE_THRESHOLD"))) {
|
||||||
|
int tmp = atoi(p);
|
||||||
|
if (tmp >= 0) {
|
||||||
|
rh->silence_threshold = tmp;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
rh->hangup_on_error = hangup_on_error;
|
rh->hangup_on_error = hangup_on_error;
|
||||||
|
|
||||||
if ((status = switch_core_media_bug_add(session, "session_record", file,
|
if ((status = switch_core_media_bug_add(session, "session_record", file,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user