/* * Copyright (C) 2021 Aon plc * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #include #include #include #include #include #include #include #include #include pa_simple *pa_s1; pa_simple *pa_s2; const pa_sample_spec pa_ss = { .format = PA_SAMPLE_S16LE, .rate = 9600, .channels = 1 }; int wave_recv; int wave_transmit; #define SIGNATURE PJMEDIA_SIG_CLASS_PORT_AUD('D','M') struct dmodem { pjmedia_port base; pj_timestamp timestamp; pj_sock_t sock; }; static struct dmodem port; static bool destroying = false; static pj_pool_t *pool; static void error_exit(const char *title, pj_status_t status) { pjsua_perror(__FILE__, title, status); if (!destroying) { destroying = true; pjsua_destroy(); exit(1); } } static pj_status_t dmodem_put_frame(pjmedia_port *this_port, pjmedia_frame *frame) { struct dmodem *sm = (struct dmodem *)this_port; int len; if (frame->type == PJMEDIA_FRAME_TYPE_AUDIO) { pa_simple_write(pa_s1, frame->buf, frame->size, NULL); write(wave_recv, frame->buf, frame->size); if ((len=write(sm->sock, frame->buf, frame->size)) != frame->size) { error_exit("error writing frame",0); } } return PJ_SUCCESS; } static pj_status_t dmodem_get_frame(pjmedia_port *this_port, pjmedia_frame *frame) { struct dmodem *sm = (struct dmodem *)this_port; frame->size = PJMEDIA_PIA_AVG_FSZ(&this_port->info); // MAX? what is int len; if ((len=read(sm->sock, frame->buf, frame->size)) != frame->size) { error_exit("error reading frame",0); } frame->timestamp.u64 = sm->timestamp.u64; frame->type = PJMEDIA_FRAME_TYPE_AUDIO; sm->timestamp.u64 += PJMEDIA_PIA_SPF(&this_port->info); pa_simple_write(pa_s2, frame->buf, frame->size, NULL); write(wave_transmit, frame->buf, frame->size); return PJ_SUCCESS; } static pj_status_t dmodem_on_destroy(pjmedia_port *this_port) { printf("destroy\n"); exit(-1); } /* Callback called by the library when call's state has changed */ static void on_call_state(pjsua_call_id call_id, pjsip_event *e) { pjsua_call_info ci; PJ_UNUSED_ARG(e); pjsua_call_get_info(call_id, &ci); PJ_LOG(3,(__FILE__, "Call %d state=%.*s", call_id, (int)ci.state_text.slen, ci.state_text.ptr)); if (ci.state == PJSIP_INV_STATE_DISCONNECTED) { close(port.sock); if (!destroying) { destroying = true; pjsua_destroy(); exit(0); } } } /* Callback called by the library when call's media state has changed */ static void on_call_media_state(pjsua_call_id call_id) { pjsua_call_info ci; pjsua_conf_port_id port_id; static int done=0; pjsua_call_get_info(call_id, &ci); // printf("media_status %d media_cnt %d ci.conf_slot %d aud.conf_slot %d\n",ci.media_status,ci.media_cnt,ci.conf_slot,ci.media[0].stream.aud.conf_slot); if (ci.media_status == PJSUA_CALL_MEDIA_ACTIVE) { if (!done) { pjsua_conf_add_port(pool, &port.base, &port_id); pjsua_conf_connect(ci.conf_slot, port_id); pjsua_conf_connect(port_id, ci.conf_slot); done = 1; } } else { done = 0; } } void start_pa() { wave_recv = open("/tmp/dmodem.recv.s16le.9600hz.pcm", O_WRONLY | O_CREAT, 00644); wave_transmit = open("/tmp/dmodem.transmit.s16le.9600hz.pcm", O_WRONLY | O_CREAT, 00644); if (wave_recv <= 0 || wave_transmit <= 0) error_exit("open wave files", 1); pa_s1 = pa_simple_new(NULL, // Use the default server. "dmodem", // Our application's name. PA_STREAM_PLAYBACK, NULL, // Use the default device. "recv", // Description of our stream. &pa_ss, // Our sample format. NULL, // Use default channel map NULL, // Use default buffering attributes. NULL // Ignore error code. ); pa_s2 = pa_simple_new(NULL, // Use the default server. "dmodem", // Our application's name. PA_STREAM_PLAYBACK, NULL, // Use the default device. "transmit", // Description of our stream. &pa_ss, // Our sample format. NULL, // Use default channel map NULL, // Use default buffering attributes. NULL // Ignore error code. ); if (!pa_s1 || !pa_s2) error_exit("pulseaudio", 1); } void stop_pa() { pa_simple_free(pa_s1); pa_simple_free(pa_s2); close(wave_recv); close(wave_transmit); } int main(int argc, char *argv[]) { pjsua_acc_id acc_id; pj_status_t status; if (argc != 3) { return -1; } signal(SIGPIPE,SIG_IGN); char *dialstr = argv[1]; int has_sip_user = 1; char *sip_user = getenv("SIP_LOGIN"); if (!sip_user) { has_sip_user = 0; printf("[!] SIP_LOGIN is empty, no registration will be attempted\n"); char sip_user_buf[40]; strcpy(sip_user_buf, "placeholder:placeholder@placeholder"); sip_user = sip_user_buf; } if (!sip_user) { return -1; } char *sip_domain = strchr(sip_user,'@'); if (!sip_domain) { return -1; } *sip_domain++ = '\0'; char *sip_pass = strchr(sip_user,':'); if (!sip_pass) { return -1; } *sip_pass++ = '\0'; status = pjsua_create(); if (status != PJ_SUCCESS) error_exit("Error in pjsua_create()", status); /* Init pjsua */ { pjsua_config cfg; pjsua_logging_config log_cfg; pjsua_media_config med_cfg; pjsua_config_default(&cfg); cfg.cb.on_call_media_state = &on_call_media_state; cfg.cb.on_call_state = &on_call_state; pjsua_logging_config_default(&log_cfg); log_cfg.console_level = 4; pjsua_media_config_default(&med_cfg); med_cfg.no_vad = true; med_cfg.ec_tail_len = 0; med_cfg.jb_max = 2000; // med_cfg.jb_init = 200; /* // med_cfg.jb_init = 200 */ med_cfg.jb_init = 100; /* med_cfg.audio_frame_ptime = 5; */ med_cfg.audio_frame_ptime = 10; status = pjsua_init(&cfg, &log_cfg, &med_cfg); if (status != PJ_SUCCESS) error_exit("Error in pjsua_init()", status); } pjsua_set_ec(0,0); // maybe? pjsua_set_null_snd_dev(); /* g711 only */ pjsua_codec_info codecs[32]; unsigned count = sizeof(codecs)/sizeof(*codecs); pjsua_enum_codecs(codecs,&count); for (int i=0; itpdata[transport_id].has_bound_addr = PJ_TRUE; if (getenv("PJSIP_IPV6")) pjsua_get_var()->acc[acc_id].cfg.ipv6_media_use = PJSUA_IPV6_ENABLED; if (has_sip_user) snprintf(buf,sizeof(buf),"sip:%s@%s",dialstr,sip_domain); else snprintf(buf,sizeof(buf),"sip:%s",dialstr); printf("calling %s\n",buf); pj_str_t uri = pj_str(buf); start_pa(); pjsua_call_id callid; status = pjsua_call_make_call(acc_id, &uri, 0, NULL, NULL, &callid); if (status != PJ_SUCCESS) error_exit("Error making call", status); struct timespec ts = {100, 0}; while(1) { nanosleep(&ts,NULL); } stop_pa(); return 0; }