= Using APS-Direct and VAS-Direct in PJMEDIA = '''Table of Contents''' [[PageOutline(1-3,,inline)]] The '''APS-Direct''' and '''VAS-Direct''' is our codenames for functionalities to use the hardware codecs that are supported by Nokia APS and/or VAS directly, bypassing media processing in PJMEDIA. These features will be introduced gradually beginning in PJSIP version 1.1. APS stands for '''Audio Proxy Server''', and it is available as plug-in for Nokia S60 3rd Edition up to Feature Pack 2 version. This has been deprecated for FP2 devices and above, and it is being replaced by '''VoIP Audio Services (VAS)''', which is available as plug-in for S60 FP1/FP2 devices and will be available built-in in later S60 versions. [[BR]] = Introduction = The Nokia APS and VAS support codecs such as G.711 (PCMA and PCMU), G.729, iLBC, and AMR-NB, though the availability of these codecs may vary according to the handset types. There are significant benefits of using these codecs instead of software codecs (in PJMEDIA-CODEC), with the main benefits are performance (hardware vs software codecs, latency) and the given codec licenses/royalties. Due to these benefits, the ability to use these codecs in PJSIP applications is very desirable. Note that non-APS codecs can still be used as usual, e.g: GSM, Speex/8000, and G.722. [[BR]] = Concepts = Before starting working with APS-Direct, please make sure you understand the concepts behind APS-Direct so that you can design the application appropriately. The whole point of APS-Direct is to enable end-to-end '''encoded audio format media flow''', that is from microphone device down to network/socket and from network/socket to the speaker device. This may sound obvious, but it has the following serious implications which will impact your application design. == What APS-Direct is really == To use APS-Direct means that you're opening the sound device in codec (i.e. non-PCM) mode. You '''still have the choice''', at run-time, to open the sound device in PCM mode, if you wish, for example to make use of the PCM features in PJMEDIA such as the tone generator, or if you want to use the software codecs such as Speex or G.722 in PJMEDIA. Note that if you use PJSUA-LIB then the management of closing/re-opening the sound device using the correct codec may be done by PJSUA-LIB automatically. To use APS-Direct also means that you are restricted to use the audio switchboard at compile time (audio switchboard will be explained later). This means that you loose the capability of mixing audio with PJSUA-LIB, as well as several other restrictions on your audio routing arrangements. Among other things, you can't have two calls active and connected to the audio device at the same time. You can have more than one calls, but one of them must be put on-hold. The sound device can only handle one format at a time, meaning that if it is currently opened with G.729 format (for one call for example), you can't feed it with PCM frames for example from the tone generator or PCM WAV files. All PJMEDIA features that work with PCM audio will no longer work if the audio device is currently opened in codec mode. This includes the tone generator (tonegen) and WAV files. If you wish to use any of the features above, you must close the sound device and re-open it in PCM mode. == When APS-Direct is activated == APS-Direct is used when passthrough codec is used (and vice versa): - when APS-Direct is used, the sound device emits and takes encoded audio frames (in contrast to PCM audio frames), and the stream needs special codec which handles this. This special codec is called passthrough codec, since it just do (de)packetization and not the actual encoding/decoding. - similarly when passthrough codec is selected in the stream, the stream will emit and take encoded audio frames (rather than PCM frames), hence it needs APS-Direct on the other side. One important thing to note: '''you may still use software codecs such as Speex and G.722 even when your application is compiled with APS-Direct support'''. When one of these software codecs are selected to be used by the stream, the stream will work as usual (i.e. emitting and taking PCM audio frames) and the audio device should be opened in normal/PCM mode (i.e. non-APS-Direct mode). [[BR]] = Changes = The use of APS-Direct and VAS-Direct is very different than traditional PJMEDIA media processing, with the main difference being the audio frames returned by/given to the sound device are now in encoded format rather than in raw PCM format. The following changes will be done in order to support this. == Support for non-PCM format == Media ports may now support non-PCM media, and this is signaled by adding a new "format" field in the {{{pjmedia_port_info}}}. {{{ typedef enum pjmedia_format { PJMEDIA_FORMAT_PCM, PJMEDIA_FORMAT_ALAW, PJMEDIA_FORMAT_ULAW, PJMEDIA_FORMAT_G729, PJMEDIA_FORMAT_AMR_NB, .. } pjmedia_format; }}} Support for new frame type (the {{{enum pjmedia_frame_type}}}): '''{{{PJMEDIA_FRAME_TYPE_EXTENDED}}}'''. When the frame's type is set to this type, the {{{pjmedia_frame}}} structure may be typecasted to '''{{{pjmedia_frame_ext}}}''' struct (new): {{{ typedef struct pjmedia_frame_ext { pjmedia_frame base; /**< Base frame info */ pj_uint16_t samples_cnt; /**< Number of samples in this frame */ pj_uint16_t subframe_cnt; /**< Number of (sub)frames in this frame */ /* Zero or more (sub)frames follows immediately after this, * each will be represented by pjmedia_frame_ext_subframe */ } pjmedia_frame_ext; typedef struct pjmedia_frame_ext_subframe { pj_uint16_t bitlen; /**< Number of bits in the data */ pj_uint8_t data[1]; /**< The encoded data */ } struct pjmedia_frame_ext_subframe; }}} The stream must also support non-PCM audio frames in its {{{get_frame()}}} and {{{put_frame()}}} port interface. == Passthrough Codecs == While the actual codec encoding/decoding will be done by APS/VAS, "dummy" codec instances still need to be created in PJMEDIA: - PJMEDIA needs the list of codecs supported to be offered/negotiated in SDP, - some codecs have special framing requirements which are not handled by the hardware codecs, for example the framing rules of AMR codecs ([http://tools.ietf.org/html/rfc3267 RFC 3267]). Passthrough codecs will be implemented for: PCMA, PCMU, iLBC, G.729, and AMR-NB. == (Symbian) Sound Device API == The APS/VAS based sound device backends will support additional APIs: - to query the list of supported codecs/formats (for APS, the list is currently hardcoded), - to set which format to use when opening the sound device, - audio routing to loudspeaker or earpiece (this API is already available) == New Audio Switchboard (the non-mixing conference bridge) == Since audio frames are forwarded back and forth in encoded format, obviously the traditional conference bridge would not be able to handle it. A new object will be added, we call this audio switchboard ({{{conf_switch.c}}}) and it's API will be compatible with the existing conference bridge API, so that it can replace the bridge in the application by compile time switch. Understandably some conference bridge features will not be available: - audio mixing feature (no conferencing feature), - audio level adjustment and query, - passive ports. Some of the features of the switchboard: - uses the conference bridge API to control it (so it is compile-time compatible), - one source may transmit to more than one destinations (though obviously one destination cannot take more than one sources since the switchboard cannot mix audio). This is useful for example to implement call recording feature in the application. - it is optimized for low latency to the sound device (no ''delaybuf'' buffering for the microphone), - much more lightweight (footprint and performance), - supports routing audio from ports with different ''ptime'' settings. [[BR]] = Using APS-Direct or VAS-Direct = Currently only APS-Direct is implemented, and here are the steps to build the application with APS- Direct feature. 1. Enable APS sound device implementation as described [http://trac.pjsip.org/repos/wiki/APS here]. 1. Enable audio switch board, i.e. in config_site.h: {{{ #define PJMEDIA_CONF_USE_SWITCH_BOARD 1 }}} 1. Enable passthrough codecs, i.e. in config_site.h: {{{ #define PJMEDIA_HAS_PASSTHROUGH_CODECS 1 }}} For building sample application {{{symbian_ua}}}, those steps are enough since it's already prepared to use APS- Direct. For general application, there are few more things to be handled: - Reopening sound device when it needs to change the active format/codec, e.g: when a call is confirmed and stream has been created, the sound device format should be matched to the SDP negotiation result. Here is the sample code for application that using pjsua-lib, reopening sound device is done in {{{on_stream_created()}}} pjsua callback, this will replace the precreated pjsua-lib sound device instance: {{{ /* Global sound port. */ static pjmedia_snd_port *g_snd_port; /* Reopen sound device on on_stream_created() pjsua callback. */ static void on_stream_created(pjsua_call_id, pjmedia_session *sess, unsigned stream_idx, pjmedia_port**) { pjmedia_port *conf; pjmedia_session_info sess_info; pjmedia_stream_info *strm_info; pjmedia_snd_setting setting; unsigned samples_per_frame; /* Get active format for this stream, based on SDP negotiation result. */ pjmedia_session_get_info(sess, &sess_info); strm_info = &sess_info.stream_info[stream_idx]; /* Init sound device setting based on stream info. */ pj_bzero(&setting, sizeof(setting)); setting.format = strm_info->param->info.format; setting.bitrate = strm_info->param->info.avg_bps; setting.cng = strm_info->param->setting.cng; setting.vad = strm_info->param->setting.vad; setting.plc = strm_info->param->setting.plc; /* Close sound device and get the conference port. */ conf = pjsua_set_no_snd_dev(); samples_per_frame = strm_info->param->info.clock_rate * strm_info->param->info.frm_ptime * strm_info->param->info.channel_cnt / 1000; /* Reset conference port attributes. */ conf->info.samples_per_frame = samples_per_frame; conf->info.clock_rate = strm_info->param->info.clock_rate; conf->info.channel_count = strm_info->param->info.channel_cnt; conf->info.bits_per_sample = 16; /* Reopen sound device. */ pjmedia_snd_port_create2(app_pool, PJMEDIA_DIR_CAPTURE_PLAYBACK, 0, 0, strm_info->param->info.clock_rate, strm_info->param->info.channel_cnt, samples_per_frame, 16, &setting, &g_snd_port); /* Connect sound to conference port. */ pjmedia_snd_port_connect(g_snd_port, conf); } }}} - Note that sound device instance is now owned and managed by application, so {{{pjsua_media_config.snd_auto_close_time}}} will not work. Here is the sample code to utilize {{{on_stream_destroyed()}}} pjsua callback as the trigger of closing the sound device: {{{ /* Close sound device on on_stream_destroyed() pjsua callback. */ static void on_stream_destroyed(pjsua_call_id, pjmedia_session*, unsigned) { if (g_snd_port) { pjmedia_snd_port_destroy(g_snd_port); g_snd_port = NULL; } } }}} [[BR]] = References = Internal documentations: 1. [wiki:APS Using APS in PJSIP] 1. [wiki:FAQ#symbian Known problems of PJSIP with APS and Symbian target in general] 1. [http://www.pjsip.org/pjmedia/docs/html/index.htm PJMEDIA and PJMEDIA-CODEC Documentation] External links: 1. [http://www.forum.nokia.com/info/sw.nokia.com/id/baae1f23-214a-42e0-96fc-aca65d86bcee/Audio_Proxy_Server_and_VoIP_Audio_Services_v1_0_en.pdf.html Audio Proxy Server and VoIP Audio Services] (PDF presentation) 1. [http://wiki.forum.nokia.com/index.php/Audio_Proxy_Server Audio Proxy Server] - Forum Nokia 1. [http://wiki.forum.nokia.com/index.php/VoIP_Audio_Service_API Nokia VoIP Audio Service API] - Forum Nokia