// App.cpp : Implementation of CApp #include "stdafx.h" #include #include #include "activex-pjsua.h" #include "app.h" ///////////////////////////////////////////////////////////////////////////// // CApp // {9CE3052A-7A32-4229-B31C-5E02E0667A77} static const GUID IID_Pjsip_Cred_Info = { 0x9ce3052a, 0x7a32, 0x4229, { 0xb3, 0x1c, 0x5e, 0x2, 0xe0, 0x66, 0x7a, 0x77 } }; // {7F6CFF0F-C5B3-41e8-B278-61CD584C1F34} static const GUID IID_Pjsip_Sip_Uri = { 0x7f6cff0f, 0xc5b3, 0x41e8, { 0xb2, 0x78, 0x61, 0xcd, 0x58, 0x4c, 0x1f, 0x34 } }; // {3B12B04F-6E48-46a7-B9E0-6C4BF1594A96} static const GUID IID_Pjsua_Acc_Config = { 0x3b12b04f, 0x6e48, 0x46a7, { 0xb9, 0xe0, 0x6c, 0x4b, 0xf1, 0x59, 0x4a, 0x96 } }; // {E4B6573D-CF5E-484d-863F-ADAD5947FBE4} static const GUID IID_Pjsua_Config = { 0xe4b6573d, 0xcf5e, 0x484d, { 0x86, 0x3f, 0xad, 0xad, 0x59, 0x47, 0xfb, 0xe4 } }; // {5043AC9E-A417-4f03-927E-D7AE52DDD064} static const GUID IID_Pjsua_Call_Info = { 0x5043ac9e, 0xa417, 0x4f03, { 0x92, 0x7e, 0xd7, 0xae, 0x52, 0xdd, 0xd0, 0x64 } }; // {2729F0BC-8A5E-4f3f-BC29-C1740A86393A} static const GUID IID_Pjsua_Buddy_Info = { 0x2729f0bc, 0x8a5e, 0x4f3f, { 0xbc, 0x29, 0xc1, 0x74, 0xa, 0x86, 0x39, 0x3a } }; // {8D345956-10B7-4450-8A06-A80D2F319EFD} static const GUID IID_Pjsua_Acc_Info = { 0x8d345956, 0x10b7, 0x4450, { 0x8a, 0x6, 0xa8, 0xd, 0x2f, 0x31, 0x9e, 0xfd } }; // {0D05907A-3E1F-4c92-9FD0-26CB6E1CC56A} static const GUID IID_Pjsua_Conf_Port_Info = { 0xd05907a, 0x3e1f, 0x4c92, { 0x9f, 0xd0, 0x26, 0xcb, 0x6e, 0x1c, 0xc5, 0x6a } }; #define SA_SIZE(lbound,ubound) (ubound-lbound) class Temp_Pool { public: Temp_Pool() { pool_ = pjsip_endpt_create_pool( pjsua_get_pjsip_endpt(), "ActivePJSUA", 4000, 4000); } ~Temp_Pool() { pj_pool_release(pool_); } pj_pool_t *get_pool() { return pool_; } private: pj_pool_t *pool_; }; static pj_str_t Pj_str(pj_pool_t *pool, Pj_String s) { pj_str_t ret; unsigned len; len = wcslen(s); if (len) { ret.ptr = (char*)pj_pool_alloc(pool, len+1); ret.slen = len; pj_unicode_to_ansi(s, len, ret.ptr, len+1); ret.ptr[ret.slen] = '\0'; } else { ret.ptr = NULL; ret.slen = 0; } return ret; } BSTR str2bstr(const char *str, unsigned len) { if (len == 0) { return SysAllocString(L""); } else { OLECHAR *tmp; BSTR result; tmp = (OLECHAR*) malloc((len+1) * sizeof(OLECHAR)); pj_ansi_to_unicode(str, len, tmp, len+1); result = SysAllocString(tmp); free(tmp); return result; } } #define Cp(d,s) Cp2(&d,s) static void Cp2(BSTR *dst, const pj_str_t *src) { *dst = str2bstr(src->ptr, src->slen); } static void SafeStringArray2pjstrarray(pj_pool_t *pool, SAFEARRAY *sa, unsigned *count, pj_str_t a[]) { if (!sa) *count = 0; else { HRESULT hr; long lbound; unsigned i; hr = SafeArrayGetLBound(sa, 1, &lbound); if (FAILED(hr)) *count = 0; else { *count = 0; for (i=0; icbElements; ++i) { BSTR str = NULL; long rg = lbound + i; hr = SafeArrayGetElement(sa, &rg, &str); if (FAILED(hr)) break; a[*count] = Pj_str(pool, str); *count = *count + 1; } } } } static void pjstrarray2SafeStringArray(unsigned count, const pj_str_t a[], SAFEARRAY **psa) { unsigned i; SAFEARRAY *sa; sa = SafeArrayCreateVector( VT_BSTR, 0, count); for (i=0; iid = Pj_str(pool, c1->acc_uri); c2->reg_uri = Pj_str(pool, c1->reg_uri); c2->contact = Pj_str(pool, c1->contact_uri); c2->proxy = Pj_str(pool, c1->proxy_uri); c2->reg_timeout = c1->reg_timeout; if (c1->cred_info == NULL) { c2->cred_count = 0; } else { unsigned i; long lbound; HRESULT hr; hr = SafeArrayGetLBound(c1->cred_info, 1, &lbound); if (FAILED(hr)) { c2->cred_count = 0; } else { c2->cred_count = 0; for (i=0; icred_info->cbElements; ++i) { Pjsip_Cred_Info cred_info; long rg = lbound + i; pj_memset(&cred_info, 0, sizeof(cred_info)); hr = SafeArrayGetElement(c1->cred_info, &rg, &cred_info); if (FAILED(hr)) break; c2->cred_info[i].realm = Pj_str(pool, cred_info.realm); c2->cred_info[i].scheme = Pj_str(pool, cred_info.scheme); c2->cred_info[i].username = Pj_str(pool, cred_info.username); c2->cred_info[i].data_type = cred_info.hashed; c2->cred_info[i].data = Pj_str(pool, cred_info.data); } c2->cred_count = i; } } } static HRESULT accconfig2AccConfig(pjsua_acc_config *c1, Pjsua_Acc_Config *c2) { unsigned i; //pj_memset(c2, 0, sizeof(Pjsua_Acc_Config)); Cp(c2->acc_uri, &c1->id); Cp(c2->reg_uri, &c1->reg_uri); Cp(c2->contact_uri, &c1->contact); Cp(c2->proxy_uri, &c1->proxy); c2->reg_timeout = c1->reg_timeout; IRecordInfo *pUdtRecordInfo = NULL; HRESULT hr = GetRecordInfoFromGuids( LIBID_ACTIVEPJSUALib, 1, 0, 0, IID_Pjsip_Cred_Info, &pUdtRecordInfo ); if( FAILED( hr ) ) { return( hr ); //Return original HRESULT hr2 is for debug only } SAFEARRAYBOUND rgsabound[1]; rgsabound[0].lLbound = 0; rgsabound[0].cElements = c1->cred_count; c2->cred_info = ::SafeArrayCreateEx( VT_RECORD, 1, rgsabound, pUdtRecordInfo ); pUdtRecordInfo->Release(); //do not forget to release the interface for (i=0; icred_count; ++i) { Pjsip_Cred_Info cred_info; Cp(cred_info.realm, &c1->cred_info[i].realm); Cp(cred_info.scheme, &c1->cred_info[i].scheme); Cp(cred_info.username, &c1->cred_info[i].username); cred_info.hashed = (c1->cred_info[i].data_type != 0); Cp(cred_info.data, &c1->cred_info[i].data); long rg = i; SafeArrayPutElement(c2->cred_info, &rg, &cred_info); } return S_OK; } static HRESULT Config2config(pj_pool_t *pool, Pjsua_Config *c1, pjsua_config *c2) { pj_memset(c2, 0, sizeof(pjsua_config)); c2->udp_port = c1->udp_port; c2->sip_host = Pj_str(pool, c1->sip_host); c2->sip_port = c1->sip_port; c2->start_rtp_port = c1->rtp_port; c2->msg_logging = c1->msg_logging; c2->max_calls = c1->max_calls; c2->conf_ports = c1->conf_ports; c2->thread_cnt = c1->thread_cnt; c2->stun_srv1 = Pj_str(pool, c1->stun_srv1); c2->stun_port1 = c1->stun_port1; c2->stun_srv2 = Pj_str(pool, c1->stun_srv2); c2->stun_port2 = c1->stun_port2; c2->snd_player_id = c1->snd_player_id; c2->snd_capture_id = c1->snd_capture_id; c2->clock_rate = c1->clock_rate; c2->null_audio = c1->null_audio; c2->quality = c1->quality; c2->complexity = c1->complexity; SafeStringArray2pjstrarray(pool, c1->codec_arg, &c2->codec_cnt, c2->codec_arg); c2->auto_answer = c1->auto_answer; c2->uas_refresh = c1->uas_refresh; c2->outbound_proxy = Pj_str(pool, c1->outbound_proxy); if (!c1->acc_config) c2->acc_cnt = 0; else { HRESULT hr; long lbound; unsigned i; hr = SafeArrayGetLBound(c1->acc_config, 1, &lbound); if (FAILED(hr)) c2->acc_cnt = 0; else { c2->acc_cnt = 0; for (i=0; iacc_config->cbElements; ++i) { Pjsua_Acc_Config acc_config; long rg = lbound + i; pj_memset(&acc_config, 0, sizeof(acc_config)); hr = SafeArrayGetElement(c1->acc_config, &rg, &acc_config); if (FAILED(hr)) break; AccConfig2accconfig(pool, &acc_config, &c2->acc_config[i]); } c2->acc_cnt = i; } } c2->log_level = c1->log_level; c2->app_log_level = c1->app_log_level; c2->log_decor = c1->log_decor; c2->log_filename = Pj_str(pool, c1->log_filename); SafeStringArray2pjstrarray(pool, c1->buddy_uri, &c2->buddy_cnt, c2->buddy_uri); return S_OK; } static HRESULT config2Config(pjsua_config *c1, Pjsua_Config *c2) { unsigned i; HRESULT hr; //pj_memset(c2, 0, sizeof(Pjsua_Config)); c2->udp_port = c1->udp_port; Cp(c2->sip_host, &c1->sip_host); c2->sip_port = c1->sip_port; c2->rtp_port = c1->start_rtp_port; c2->msg_logging = c1->msg_logging; c2->max_calls = c1->max_calls; c2->conf_ports = c1->conf_ports; c2->thread_cnt = c1->thread_cnt; Cp(c2->stun_srv1, &c1->stun_srv1); c2->stun_port1 = c1->stun_port1; Cp(c2->stun_srv2, &c1->stun_srv2); c2->stun_port2 = c1->stun_port2; c2->snd_player_id = c1->snd_player_id; c2->snd_capture_id = c1->snd_capture_id; c2->clock_rate = c1->clock_rate; c2->null_audio = c1->null_audio; c2->quality = c1->quality; c2->complexity = c1->complexity; pjstrarray2SafeStringArray(c1->codec_cnt, c1->codec_arg, &c2->codec_arg); c2->auto_answer = c1->auto_answer; c2->uas_refresh = c1->uas_refresh; Cp(c2->outbound_proxy, &c1->outbound_proxy); IRecordInfo *pUdtRecordInfo = NULL; hr = GetRecordInfoFromGuids( LIBID_ACTIVEPJSUALib, 1, 0, 0, IID_Pjsua_Acc_Config, &pUdtRecordInfo ); if( FAILED( hr ) ) { return( hr ); //Return original HRESULT hr2 is for debug only } SAFEARRAYBOUND rgsabound[1]; rgsabound[0].lLbound = 0; rgsabound[0].cElements = c1->acc_cnt; c2->acc_config = ::SafeArrayCreateEx( VT_RECORD, 1, rgsabound, pUdtRecordInfo ); pUdtRecordInfo->Release(); //do not forget to release the interface for (i=0; iacc_cnt; ++i) { Pjsua_Acc_Config acc_cfg; hr = accconfig2AccConfig(&c1->acc_config[i], &acc_cfg); if (FAILED(hr)) return hr; long rg = i; SafeArrayPutElement(c2->acc_config, &rg, &acc_cfg); } c2->log_level = c1->log_level; c2->app_log_level = c1->app_log_level; c2->log_decor = c1->log_decor; Cp(c2->log_filename, &c1->log_filename); pjstrarray2SafeStringArray(c1->buddy_cnt, c1->buddy_uri, &c2->buddy_uri); return S_OK; } static void callinfo2CallInfo(pjsua_call_info *c1, Pjsua_Call_Info *c2) { pj_memset(c2, 0, sizeof(Pjsua_Call_Info)); c2->index = c1->index; c2->active = c1->active; c2->is_uac = (c1->role == PJSIP_ROLE_UAC); Cp(c2->local_info, &c1->local_info); Cp(c2->remote_info, &c1->remote_info); c2->state = (Pjsua_Call_State)c1->state; Cp(c2->state_text, &c1->state_text); c2->connect_duration = c1->connect_duration.sec; c2->total_duration = c1->total_duration.sec; c2->last_status = c1->last_status; Cp(c2->last_status_text, &c1->last_status_text); c2->has_media = c1->has_media; c2->conf_slot = c1->conf_slot; } static void accinfo2AccInfo(pjsua_acc_info *info1, Pjsua_Acc_Info *info2) { pj_memset(info2, 0, sizeof(Pjsua_Acc_Info)); info2->index = info1->index; Cp(info2->acc_uri, &info1->acc_id); info2->has_registration = info1->has_registration; info2->expires = info1->expires; info2->status_code = info1->status; Cp(info2->status_text, &info1->status_text); info2->online_status = info1->online_status; } static void buddyinfo2BuddyInfo(pjsua_buddy_info *info1, Pjsua_Buddy_Info *info2) { pj_memset(info2, 0, sizeof(Pjsua_Buddy_Info)); info2->index = info1->index; info2->is_valid = info1->is_valid; Cp(info2->name, &info1->name); Cp(info2->display, &info1->display_name); Cp(info2->host, &info1->host); info2->port = info1->port; Cp(info2->uri, &info1->uri); info2->status = (Pjsua_Buddy_State)info1->status; Cp(info2->status_text, &info1->status_text); info2->monitor = info1->monitor; } static CApp *CApp_Instance; CApp::CApp() { CApp_Instance = this; } STDMETHODIMP CApp::app_create(Pj_Status *ret) { *ret = pjsua_create(); return S_OK; } STDMETHODIMP CApp::app_default_config(Pjsua_Config *pConfig) { pjsua_config cfg; pjsua_default_config(&cfg); return config2Config(&cfg, pConfig); } STDMETHODIMP CApp::app_test_config(Pjsua_Config *pConfig, BSTR *retmsg) { pjsua_config cfg; HRESULT hr; Temp_Pool tp; char errmsg[PJ_ERR_MSG_SIZE]; hr = Config2config(tp.get_pool(), pConfig, &cfg); if (FAILED(hr)) return hr; pjsua_test_config(&cfg, errmsg, sizeof(errmsg)); *retmsg = str2bstr(errmsg, strlen(errmsg)); return S_OK; } static void on_call_state(int call_index, pjsip_event *e) { pjsua_call_info call_info; Pjsua_Call_Info *Call_Info = new Pjsua_Call_Info; pjsua_call_get_info(call_index, &call_info); callinfo2CallInfo(&call_info, Call_Info); CApp_Instance->Fire_OnCallState(call_index, Call_Info); } static void on_incoming_call(int acc_id, int call_index, pjsip_rx_data *rdata) { CApp_Instance->Fire_OnIncomingCall(call_index); } static void on_reg_state(int acc_id) { CApp_Instance->Fire_OnRegState(acc_id); } static void on_buddy_state(int buddy_index) { CApp_Instance->Fire_OnBuddyState(buddy_index); } static void on_pager(int call_index, const pj_str_t *from, const pj_str_t *to, const pj_str_t *txt) { BSTR fromURI, toURI, imText; Cp2(&fromURI, from); Cp2(&toURI, to); Cp2(&imText, txt); CApp_Instance->Fire_OnIncomingPager(call_index, fromURI, toURI, imText); } static void on_typing(int call_index, const pj_str_t *from, const pj_str_t *to, pj_bool_t is_typing) { BSTR fromURI, toURI; Cp2(&fromURI, from); Cp2(&toURI, to); CApp_Instance->Fire_OnTypingIndication(call_index, fromURI, toURI, is_typing); } STDMETHODIMP CApp::app_init(Pjsua_Config *pConfig, Pj_Status *pStatus) { pjsua_config cfg; pjsua_callback cb; Temp_Pool tp; HRESULT hr; pj_memset(&cb, 0, sizeof(cb)); cb.on_call_state = &on_call_state; cb.on_incoming_call = &on_incoming_call; cb.on_reg_state = &on_reg_state; cb.on_buddy_state = &on_buddy_state; cb.on_pager = &on_pager; cb.on_typing = &on_typing; hr = Config2config(tp.get_pool(), pConfig, &cfg); if (FAILED(hr)) return hr; *pStatus = pjsua_init(&cfg, &cb); return S_OK; } STDMETHODIMP CApp::app_start(Pj_Status *retStatus) { *retStatus = pjsua_start(); return S_OK; } STDMETHODIMP CApp::app_destroy(Pj_Status *retStatus) { *retStatus = pjsua_destroy(); return S_OK; } STDMETHODIMP CApp::call_get_max_count(int *retCount) { *retCount = pjsua_call_get_max_count(); return S_OK; } STDMETHODIMP CApp::call_get_count(int *retCount) { *retCount = pjsua_call_get_count(); return S_OK; } STDMETHODIMP CApp::call_is_active(int call_index, Pj_Bool *retVal) { *retVal = pjsua_call_is_active(call_index); return S_OK; } STDMETHODIMP CApp::call_has_media(int call_index, Pj_Bool *pRet) { *pRet = pjsua_call_has_media(call_index); return S_OK; } STDMETHODIMP CApp::call_get_info(int call_index, Pjsua_Call_Info *pInfo, Pj_Status *pRet) { pjsua_call_info info; *pRet = pjsua_call_get_info(call_index, &info); callinfo2CallInfo(&info, pInfo); return S_OK; } STDMETHODIMP CApp::call_make_call(int acc_id, Pj_String dst_uri, int *call_index, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), dst_uri); *pRet = pjsua_call_make_call(acc_id, &tmp, call_index); return S_OK; } STDMETHODIMP CApp::call_answer(int call_index, int status_code, Pj_Status *pRet) { *pRet = pjsua_call_answer(call_index, status_code); return S_OK; } STDMETHODIMP CApp::call_hangup(int call_index, Pj_Status *pRet) { pjsua_call_hangup(call_index); *pRet = PJ_SUCCESS; return S_OK; } STDMETHODIMP CApp::call_set_hold(int call_index, Pj_Status *pRet) { pjsua_call_set_hold(call_index); *pRet = PJ_SUCCESS; return S_OK; } STDMETHODIMP CApp::call_release_hold(int call_index, Pj_Status *pRet) { *pRet = pjsua_call_reinvite(call_index); return S_OK; } STDMETHODIMP CApp::call_xfer(int call_index, Pj_String dst_uri, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), dst_uri); *pRet = pjsua_call_xfer(call_index, &tmp); return S_OK; } STDMETHODIMP CApp::call_dial_dtmf(int call_index, Pj_String digits, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), digits); *pRet = pjsua_call_dial_dtmf(call_index, &tmp); return S_OK; } STDMETHODIMP CApp::call_send_im(int call_index, Pj_String text, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), text); *pRet = pjsua_call_send_im(call_index, &tmp); return S_OK; } STDMETHODIMP CApp::call_typing(int call_index, int is_typing, Pj_Status *pRet) { *pRet = pjsua_call_send_typing_ind(call_index, is_typing); return S_OK; } STDMETHODIMP CApp::call_hangup_all() { pjsua_call_hangup_all(); return S_OK; } STDMETHODIMP CApp::call_get_textstat(int call_index, BSTR *textstat) { char buf[1024]; pjsua_call_dump(call_index, 1, buf, sizeof(buf), ""); OLECHAR wbuf[1024]; pj_ansi_to_unicode(buf, strlen(buf), wbuf, PJ_ARRAY_SIZE(wbuf)); *textstat = SysAllocString(wbuf); return S_OK; } STDMETHODIMP CApp::acc_get_count(int *pCount) { *pCount = pjsua_get_acc_count(); return S_OK; } STDMETHODIMP CApp::acc_get_info(int acc_id, Pjsua_Acc_Info *pInfo, Pj_Status *pRet) { pjsua_acc_info info; *pRet = pjsua_acc_get_info(acc_id, &info); accinfo2AccInfo(&info, pInfo); return S_OK; } STDMETHODIMP CApp::acc_add(Pjsua_Acc_Config *pConfig, int *pAcc_Index, Pj_Status *pRet) { Temp_Pool tp; pjsua_acc_config config; AccConfig2accconfig(tp.get_pool(), pConfig, &config); *pRet = pjsua_acc_add(&config, pAcc_Index); return S_OK; } STDMETHODIMP CApp::acc_set_online_status(int acc_id, int is_online, Pj_Status *pRet) { *pRet = pjsua_acc_set_online_status(acc_id, is_online); return S_OK; } STDMETHODIMP CApp::acc_set_registration(int acc_id, int reg_active, Pj_Status *pRet) { *pRet = pjsua_acc_set_registration(acc_id, reg_active); return S_OK; } STDMETHODIMP CApp::buddy_get_count(int *pCount) { *pCount = pjsua_get_buddy_count(); return S_OK; } STDMETHODIMP CApp::buddy_get_info(int buddy_index, Pjsua_Buddy_Info *pInfo, Pj_Status *pRet) { pjsua_buddy_info info; *pRet = pjsua_buddy_get_info(buddy_index, &info); buddyinfo2BuddyInfo(&info, pInfo); return S_OK; } STDMETHODIMP CApp::buddy_add(Pj_String uri, int *pBuddy_Index, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), uri); *pRet = pjsua_buddy_add(&tmp, pBuddy_Index); return S_OK; } STDMETHODIMP CApp::buddy_subscribe_pres(int buddy_index, int subscribe, Pj_Status *pRet) { *pRet = pjsua_buddy_subscribe_pres(buddy_index, subscribe); return S_OK; } STDMETHODIMP CApp::im_send_text(int acc_id, Pj_String dst_uri, Pj_String text, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp_uri = Pj_str(tp.get_pool(), dst_uri); pj_str_t tmp_text = Pj_str(tp.get_pool(), text); *pRet = pjsua_im_send(acc_id, &tmp_uri, &tmp_text); return S_OK; } STDMETHODIMP CApp::im_typing(int acc_id, Pj_URI dst_uri, int is_typing, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp_uri = Pj_str(tp.get_pool(), dst_uri); *pRet = pjsua_im_typing(acc_id, &tmp_uri, is_typing); return S_OK; } STDMETHODIMP CApp::conf_connect(int src_port, int sink_port, Pj_Status *pRet) { *pRet = pjsua_conf_connect(src_port, sink_port); return S_OK; } STDMETHODIMP CApp::conf_disconnect(int src_port, int sink_port, Pj_Status *pRet) { *pRet = pjsua_conf_disconnect(src_port, sink_port); return S_OK; } STDMETHODIMP CApp::player_create(Pj_String filename, int *pPlayer_Id, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), filename); *pRet = pjsua_player_create(&tmp, pPlayer_Id); return S_OK; } STDMETHODIMP CApp::player_get_conf_port(int player_id, int *pPort) { *pPort = pjsua_player_get_conf_port(player_id); return S_OK; } STDMETHODIMP CApp::player_set_pos(int player_id, int pos, Pj_Status *pRet) { *pRet = pjsua_player_set_pos(player_id, pos); return S_OK; } STDMETHODIMP CApp::player_destroy(int player_id, Pj_Status *pRet) { *pRet = pjsua_player_destroy(player_id); return S_OK; } STDMETHODIMP CApp::recorder_create(Pj_String filename, int *pRecorder_Id, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), filename); *pRet = pjsua_recorder_create(&tmp, pRecorder_Id); return S_OK; } STDMETHODIMP CApp::recorder_get_conf_port(int recorder_id, int *pPort) { *pPort = pjsua_recorder_get_conf_port(recorder_id); return S_OK; } STDMETHODIMP CApp::recorder_destroy(int recorder_id, Pj_Status *pRet) { *pRet = pjsua_recorder_destroy(recorder_id); return S_OK; } STDMETHODIMP CApp::app_load_config(Pj_String filename, Pjsua_Config *pConfig, Pj_Status *pRet) { pjsua_config config; Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), filename); pjsua_default_config(&config); *pRet = pjsua_load_settings(tmp.ptr, &config, NULL); if (*pRet == PJ_SUCCESS) *pRet = config2Config(&config, pConfig); return S_OK; } STDMETHODIMP CApp::app_save_config(Pj_String filename, Pjsua_Config *pConfig, Pj_Status *pRet) { Temp_Pool tp; pjsua_config config; pj_str_t tmp = Pj_str(tp.get_pool(), filename); HRESULT hr; hr = Config2config(tp.get_pool(), pConfig, &config); if (FAILED(hr)) return hr; *pRet = pjsua_save_settings(tmp.ptr, &config); return S_OK; } STDMETHODIMP CApp::app_get_current_config(Pjsua_Config *pConfig) { Temp_Pool tp; pjsua_config config; pjsua_get_config(tp.get_pool(), &config); return config2Config(&config, pConfig); } STDMETHODIMP CApp::app_get_error_msg(Pj_Status status, BSTR * pRet) { char errmsg[PJ_ERR_MSG_SIZE]; OLECHAR werrmsg[PJ_ERR_MSG_SIZE]; pj_strerror(status, errmsg, sizeof(errmsg)); pj_ansi_to_unicode(errmsg, strlen(errmsg), werrmsg, PJ_ARRAY_SIZE(werrmsg)); *pRet = SysAllocString(werrmsg); return S_OK; } STDMETHODIMP CApp::app_verify_sip_url(Pj_String uri, Pj_Status *pRet) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), uri); *pRet = pjsua_verify_sip_url(tmp.ptr); return S_OK; } STDMETHODIMP CApp::app_handle_events(int msec_timeout, int *pEvCount) { if (msec_timeout < 0) msec_timeout = 0; *pEvCount = pjsua_handle_events(msec_timeout); return S_OK; } STDMETHODIMP CApp::app_parse_uri(BSTR uriString, Pjsip_Sip_Uri *pSipUri, Pj_Status *pStatus) { Temp_Pool tp; pj_str_t tmp = Pj_str(tp.get_pool(), uriString); pjsip_name_addr *addr; char buf[1024]; pj_str_t s; addr = (pjsip_name_addr*) pjsip_parse_uri(tp.get_pool(), tmp.ptr, tmp.slen, PJSIP_PARSE_URI_AS_NAMEADDR); if (addr == NULL) { *pStatus = PJSIP_EINVALIDURI; return S_OK; } if (!PJSIP_URI_SCHEME_IS_SIP(addr) && !PJSIP_URI_SCHEME_IS_SIPS(addr)) { *pStatus = PJSIP_EINVALIDSCHEME; return S_OK; } pjsip_sip_uri *sip = (pjsip_sip_uri*)addr->uri; Cp2(&pSipUri->display, &addr->display); Cp2(&pSipUri->user, &sip->user); Cp2(&pSipUri->passwd, &sip->passwd); Cp2(&pSipUri->host, &sip->host); pSipUri->port = sip->port; Cp2(&pSipUri->param_user, &sip->user_param); Cp2(&pSipUri->param_method, &sip->method_param); Cp2(&pSipUri->param_transport, &sip->transport_param); pSipUri->param_ttl = sip->ttl_param; pSipUri->param_lr = sip->lr_param; Cp2(&pSipUri->param_maddr, &sip->maddr_param); s.ptr = buf; s.slen = pjsip_param_print_on(&sip->other_param, buf, sizeof(buf), &pjsip_PARAM_CHAR_SPEC, &pjsip_PARAM_CHAR_SPEC, ';'); Cp2(&pSipUri->param_other, &s); s.slen = pjsip_param_print_on(&sip->header_param, buf, sizeof(buf), &pjsip_HDR_CHAR_SPEC, &pjsip_HDR_CHAR_SPEC, '?'); Cp2(&pSipUri->param_header, &s); *pStatus = PJ_SUCCESS; return S_OK; } STDMETHODIMP CApp::app_print_uri(Pjsip_Sip_Uri *pSipURI, Pjsip_Uri_Context context, BSTR *uriText) { Temp_Pool tp; pjsip_name_addr *addr; pj_str_t tmp; char buf[1024]; pjsip_sip_uri *sip; addr = pjsip_name_addr_create(tp.get_pool()); sip = pjsip_sip_uri_create(tp.get_pool(), PJ_FALSE); addr->uri = (pjsip_uri*)sip; addr->display = Pj_str(tp.get_pool(), pSipURI->display); sip->user = Pj_str(tp.get_pool(), pSipURI->user); sip->passwd = Pj_str(tp.get_pool(), pSipURI->passwd); sip->host = Pj_str(tp.get_pool(), pSipURI->host); sip->port = pSipURI->port; sip->user_param = Pj_str(tp.get_pool(), pSipURI->param_user); sip->method_param = Pj_str(tp.get_pool(), pSipURI->param_method); sip->transport_param = Pj_str(tp.get_pool(), pSipURI->param_transport); sip->ttl_param = pSipURI->param_ttl; sip->lr_param = pSipURI->param_lr; sip->maddr_param = Pj_str(tp.get_pool(), pSipURI->param_maddr); /* Unfortunately can't transport params yet (no parsing function) */ tmp.ptr = buf; tmp.slen = pjsip_uri_print((pjsip_uri_context_e)context, addr, buf, sizeof(buf)); Cp2(uriText, &tmp); return S_OK; } STDMETHODIMP CApp::app_compare_uri_string(Pjsip_Uri_Context context, BSTR uri1, BSTR uri2, Pj_Status *pStatus) { Temp_Pool tp; pj_str_t tmp_uri1 = Pj_str(tp.get_pool(), uri1); pj_str_t tmp_uri2 = Pj_str(tp.get_pool(), uri2); pjsip_uri *u1, *u2; u1 = pjsip_parse_uri(tp.get_pool(), tmp_uri1.ptr, tmp_uri1.slen, PJSIP_PARSE_URI_AS_NAMEADDR); if (u1 == NULL) { *pStatus = PJSIP_EINVALIDURI; return S_OK; } u2 = pjsip_parse_uri(tp.get_pool(), tmp_uri2.ptr, tmp_uri2.slen, PJSIP_PARSE_URI_AS_NAMEADDR); if (u2 == NULL) { *pStatus = PJSIP_EINVALIDURI; return S_OK; } *pStatus = pjsip_uri_cmp((pjsip_uri_context_e)context, u1, u2); return S_OK; } STDMETHODIMP CApp::buddy_del(int buddy_index, Pj_Status *pRet) { *pRet = pjsua_buddy_del(buddy_index); return S_OK; } STDMETHODIMP CApp::acc_del(int acc_id, Pj_Status *pRet) { *pRet = pjsua_acc_del(acc_id); return S_OK; } STDMETHODIMP CApp::acc_find_for_outgoing(BSTR url, int *acc_id) { Temp_Pool tp; pj_str_t tmp_uri = Pj_str(tp.get_pool(), url); *acc_id = pjsua_acc_find_for_outgoing(&tmp_uri); return S_OK; } STDMETHODIMP CApp::acc_enum_id(SAFEARRAY **accIdArray) { int id[32]; unsigned count = PJ_ARRAY_SIZE(id); unsigned i; pjsua_acc_enum_id(id, &count); *accIdArray = SafeArrayCreateVector(VT_INT, 0, count); for (i=0; islot_id = info.slot_id; Cp2(&pInfo->name, &info.name); pInfo->clock_rate = info.clock_rate; pInfo->channel_count = info.channel_count; pInfo->samples_per_frame = info.samples_per_frame; pInfo->bits_per_sample = info.bits_per_sample; pInfo->listeners = SafeArrayCreateVector(VT_INT, 0, info.listener_cnt); for (i=0; ilisteners, &rg, &info.listeners[i]); } return S_OK; }