blob: 0c9d5e28a3139ca703d0dd2960db2125a216a6f6 [file] [log] [blame]
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +00001/* $Id$ */
2/*
3 * Copyright (C) 2008-2009 Teluu Inc. (http://www.teluu.com)
4 * Copyright (C) 2003-2008 Benny Prijono <benny@prijono.org>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 */
20#include <pjmedia-audiodev/audiodev_imp.h>
Nanang Izzuddinda37ea32009-03-06 16:24:43 +000021#include <pjmedia-audiodev/errno.h>
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +000022#include <pjmedia/alaw_ulaw.h>
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +000023#include <pj/assert.h>
24#include <pj/log.h>
25#include <pj/math.h>
26#include <pj/os.h>
27#include <pj/string.h>
28
Nanang Izzuddinda37ea32009-03-06 16:24:43 +000029#if PJMEDIA_AUDIO_DEV_HAS_SYMB_MDA
30
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +000031/*
32 * This file provides sound implementation for Symbian Audio Streaming
33 * device. Application using this sound abstraction must link with:
34 * - mediaclientaudiostream.lib, and
35 * - mediaclientaudioinputstream.lib
36 */
37#include <mda/common/audio.h>
38#include <mdaaudiooutputstream.h>
39#include <mdaaudioinputstream.h>
40
41
42#define THIS_FILE "symb_mda_dev.c"
43#define BITS_PER_SAMPLE 16
44#define BYTES_PER_SAMPLE (BITS_PER_SAMPLE/8)
45
46
47#if 1
48# define TRACE_(st) PJ_LOG(3, st)
49#else
50# define TRACE_(st)
51#endif
52
53
54/* MDA factory */
55struct mda_factory
56{
57 pjmedia_aud_dev_factory base;
58 pj_pool_t *pool;
59 pj_pool_factory *pf;
60 pjmedia_aud_dev_info dev_info;
61};
62
63/* Forward declaration of internal engine. */
64class CPjAudioInputEngine;
65class CPjAudioOutputEngine;
66
67/* MDA stream. */
68struct mda_stream
69{
70 // Base
71 pjmedia_aud_stream base; /**< Base class. */
72
73 // Pool
74 pj_pool_t *pool; /**< Memory pool. */
75
76 // Common settings.
Benny Prijono10454dc2009-02-21 14:21:59 +000077 pjmedia_aud_param param; /**< Stream param. */
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +000078
79 // Audio engine
80 CPjAudioInputEngine *in_engine; /**< Record engine. */
81 CPjAudioOutputEngine *out_engine; /**< Playback engine. */
82};
83
84
85/* Prototypes */
86static pj_status_t factory_init(pjmedia_aud_dev_factory *f);
87static pj_status_t factory_destroy(pjmedia_aud_dev_factory *f);
Sauw Ming98766c72011-03-11 06:57:24 +000088static pj_status_t factory_refresh(pjmedia_aud_dev_factory *f);
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +000089static unsigned factory_get_dev_count(pjmedia_aud_dev_factory *f);
90static pj_status_t factory_get_dev_info(pjmedia_aud_dev_factory *f,
91 unsigned index,
92 pjmedia_aud_dev_info *info);
93static pj_status_t factory_default_param(pjmedia_aud_dev_factory *f,
94 unsigned index,
Benny Prijono10454dc2009-02-21 14:21:59 +000095 pjmedia_aud_param *param);
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +000096static pj_status_t factory_create_stream(pjmedia_aud_dev_factory *f,
Benny Prijono10454dc2009-02-21 14:21:59 +000097 const pjmedia_aud_param *param,
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +000098 pjmedia_aud_rec_cb rec_cb,
99 pjmedia_aud_play_cb play_cb,
100 void *user_data,
101 pjmedia_aud_stream **p_aud_strm);
102
103static pj_status_t stream_get_param(pjmedia_aud_stream *strm,
Benny Prijono10454dc2009-02-21 14:21:59 +0000104 pjmedia_aud_param *param);
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000105static pj_status_t stream_get_cap(pjmedia_aud_stream *strm,
106 pjmedia_aud_dev_cap cap,
107 void *value);
108static pj_status_t stream_set_cap(pjmedia_aud_stream *strm,
109 pjmedia_aud_dev_cap cap,
110 const void *value);
111static pj_status_t stream_start(pjmedia_aud_stream *strm);
112static pj_status_t stream_stop(pjmedia_aud_stream *strm);
113static pj_status_t stream_destroy(pjmedia_aud_stream *strm);
114
115
116/* Operations */
117static pjmedia_aud_dev_factory_op factory_op =
118{
119 &factory_init,
120 &factory_destroy,
121 &factory_get_dev_count,
122 &factory_get_dev_info,
123 &factory_default_param,
Sauw Ming98766c72011-03-11 06:57:24 +0000124 &factory_create_stream,
125 &factory_refresh
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000126};
127
128static pjmedia_aud_stream_op stream_op =
129{
130 &stream_get_param,
131 &stream_get_cap,
132 &stream_set_cap,
133 &stream_start,
134 &stream_stop,
135 &stream_destroy
136};
137
138
139/*
140 * Convert clock rate to Symbian's TMdaAudioDataSettings capability.
141 */
142static TInt get_clock_rate_cap(unsigned clock_rate)
143{
144 switch (clock_rate) {
145 case 8000: return TMdaAudioDataSettings::ESampleRate8000Hz;
146 case 11025: return TMdaAudioDataSettings::ESampleRate11025Hz;
147 case 12000: return TMdaAudioDataSettings::ESampleRate12000Hz;
148 case 16000: return TMdaAudioDataSettings::ESampleRate16000Hz;
149 case 22050: return TMdaAudioDataSettings::ESampleRate22050Hz;
150 case 24000: return TMdaAudioDataSettings::ESampleRate24000Hz;
151 case 32000: return TMdaAudioDataSettings::ESampleRate32000Hz;
152 case 44100: return TMdaAudioDataSettings::ESampleRate44100Hz;
153 case 48000: return TMdaAudioDataSettings::ESampleRate48000Hz;
154 case 64000: return TMdaAudioDataSettings::ESampleRate64000Hz;
155 case 96000: return TMdaAudioDataSettings::ESampleRate96000Hz;
156 default:
157 return 0;
158 }
159}
160
161/*
162 * Convert number of channels into Symbian's TMdaAudioDataSettings capability.
163 */
164static TInt get_channel_cap(unsigned channel_count)
165{
166 switch (channel_count) {
167 case 1: return TMdaAudioDataSettings::EChannelsMono;
168 case 2: return TMdaAudioDataSettings::EChannelsStereo;
169 default:
170 return 0;
171 }
172}
173
174/*
175 * Utility: print sound device error
176 */
177static void snd_perror(const char *title, TInt rc)
178{
179 PJ_LOG(1,(THIS_FILE, "%s: error code %d", title, rc));
180}
181
182//////////////////////////////////////////////////////////////////////////////
183//
184
185/*
186 * Implementation: Symbian Input Stream.
187 */
188class CPjAudioInputEngine : public CBase, MMdaAudioInputStreamCallback
189{
190public:
191 enum State
192 {
193 STATE_INACTIVE,
194 STATE_ACTIVE,
195 };
196
197 ~CPjAudioInputEngine();
198
199 static CPjAudioInputEngine *NewL(struct mda_stream *parent_strm,
200 pjmedia_aud_rec_cb rec_cb,
201 void *user_data);
202
203 static CPjAudioInputEngine *NewLC(struct mda_stream *parent_strm,
204 pjmedia_aud_rec_cb rec_cb,
205 void *user_data);
206
207 pj_status_t StartRecord();
208 void Stop();
209
210 pj_status_t SetGain(TInt gain) {
211 if (iInputStream_) {
212 iInputStream_->SetGain(gain);
213 return PJ_SUCCESS;
214 } else
215 return PJ_EINVALIDOP;
216 }
217
218 TInt GetGain() {
219 if (iInputStream_) {
220 return iInputStream_->Gain();
221 } else
222 return PJ_EINVALIDOP;
223 }
224
225 TInt GetMaxGain() {
226 if (iInputStream_) {
227 return iInputStream_->MaxGain();
228 } else
229 return PJ_EINVALIDOP;
230 }
231
232private:
233 State state_;
234 struct mda_stream *parentStrm_;
235 pjmedia_aud_rec_cb recCb_;
236 void *userData_;
237 CMdaAudioInputStream *iInputStream_;
238 HBufC8 *iStreamBuffer_;
239 TPtr8 iFramePtr_;
240 TInt lastError_;
241 pj_uint32_t timeStamp_;
242
243 // cache variable
244 // to avoid calculating frame length repeatedly
245 TInt frameLen_;
246
247 // sometimes recorded size != requested framesize, so let's
248 // provide a buffer to make sure the rec callback returning
249 // framesize as requested.
250 TUint8 *frameRecBuf_;
251 TInt frameRecBufLen_;
252
253 CPjAudioInputEngine(struct mda_stream *parent_strm,
254 pjmedia_aud_rec_cb rec_cb,
255 void *user_data);
256 void ConstructL();
257 TPtr8 & GetFrame();
258
259public:
260 virtual void MaiscOpenComplete(TInt aError);
261 virtual void MaiscBufferCopied(TInt aError, const TDesC8 &aBuffer);
262 virtual void MaiscRecordComplete(TInt aError);
263
264};
265
266
267CPjAudioInputEngine::CPjAudioInputEngine(struct mda_stream *parent_strm,
268 pjmedia_aud_rec_cb rec_cb,
269 void *user_data)
270 : state_(STATE_INACTIVE), parentStrm_(parent_strm),
271 recCb_(rec_cb), userData_(user_data),
272 iInputStream_(NULL), iStreamBuffer_(NULL), iFramePtr_(0, 0),
273 lastError_(KErrNone), timeStamp_(0),
274 frameLen_(parent_strm->param.samples_per_frame *
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000275 BYTES_PER_SAMPLE),
276 frameRecBuf_(NULL), frameRecBufLen_(0)
277{
278}
279
280CPjAudioInputEngine::~CPjAudioInputEngine()
281{
282 Stop();
283
284 delete iStreamBuffer_;
285 iStreamBuffer_ = NULL;
286
287 delete [] frameRecBuf_;
288 frameRecBuf_ = NULL;
289 frameRecBufLen_ = 0;
290}
291
292void CPjAudioInputEngine::ConstructL()
293{
294 iStreamBuffer_ = HBufC8::NewL(frameLen_);
295 CleanupStack::PushL(iStreamBuffer_);
296
297 frameRecBuf_ = new TUint8[frameLen_*2];
298 CleanupStack::PushL(frameRecBuf_);
299}
300
301CPjAudioInputEngine *CPjAudioInputEngine::NewLC(struct mda_stream *parent,
302 pjmedia_aud_rec_cb rec_cb,
303 void *user_data)
304{
305 CPjAudioInputEngine* self = new (ELeave) CPjAudioInputEngine(parent,
306 rec_cb,
307 user_data);
308 CleanupStack::PushL(self);
309 self->ConstructL();
310 return self;
311}
312
313CPjAudioInputEngine *CPjAudioInputEngine::NewL(struct mda_stream *parent,
314 pjmedia_aud_rec_cb rec_cb,
315 void *user_data)
316{
317 CPjAudioInputEngine *self = NewLC(parent, rec_cb, user_data);
318 CleanupStack::Pop(self->frameRecBuf_);
319 CleanupStack::Pop(self->iStreamBuffer_);
320 CleanupStack::Pop(self);
321 return self;
322}
323
324
325pj_status_t CPjAudioInputEngine::StartRecord()
326{
327
328 // Ignore command if recording is in progress.
329 if (state_ == STATE_ACTIVE)
330 return PJ_SUCCESS;
331
332 // According to Nokia's AudioStream example, some 2nd Edition, FP2 devices
333 // (such as Nokia 6630) require the stream to be reconstructed each time
334 // before calling Open() - otherwise the callback never gets called.
335 // For uniform behavior, lets just delete/re-create the stream for all
336 // devices.
337
338 // Destroy existing stream.
339 if (iInputStream_) delete iInputStream_;
340 iInputStream_ = NULL;
341
342 // Create the stream.
343 TRAPD(err, iInputStream_ = CMdaAudioInputStream::NewL(*this));
344 if (err != KErrNone)
345 return PJ_RETURN_OS_ERROR(err);
346
347 // Initialize settings.
348 TMdaAudioDataSettings iStreamSettings;
349 iStreamSettings.iChannels =
350 get_channel_cap(parentStrm_->param.channel_count);
351 iStreamSettings.iSampleRate =
352 get_clock_rate_cap(parentStrm_->param.clock_rate);
353
354 pj_assert(iStreamSettings.iChannels != 0 &&
355 iStreamSettings.iSampleRate != 0);
356
357 PJ_LOG(4,(THIS_FILE, "Opening sound device for capture, "
358 "clock rate=%d, channel count=%d..",
359 parentStrm_->param.clock_rate,
360 parentStrm_->param.channel_count));
361
362 // Open stream.
363 lastError_ = KRequestPending;
364 iInputStream_->Open(&iStreamSettings);
365
366 // Success
367 PJ_LOG(4,(THIS_FILE, "Sound capture started."));
368 return PJ_SUCCESS;
369}
370
371
372void CPjAudioInputEngine::Stop()
373{
374 // If capture is in progress, stop it.
375 if (iInputStream_ && state_ == STATE_ACTIVE) {
376 lastError_ = KRequestPending;
377 iInputStream_->Stop();
378
379 // Wait until it's actually stopped
380 while (lastError_ == KRequestPending)
381 pj_symbianos_poll(-1, 100);
382 }
383
384 if (iInputStream_) {
385 delete iInputStream_;
386 iInputStream_ = NULL;
387 }
388
389 state_ = STATE_INACTIVE;
390}
391
392
393TPtr8 & CPjAudioInputEngine::GetFrame()
394{
395 //iStreamBuffer_->Des().FillZ(frameLen_);
396 iFramePtr_.Set((TUint8*)(iStreamBuffer_->Ptr()), frameLen_, frameLen_);
397 return iFramePtr_;
398}
399
400void CPjAudioInputEngine::MaiscOpenComplete(TInt aError)
401{
402 lastError_ = aError;
403 if (aError != KErrNone) {
404 snd_perror("Error in MaiscOpenComplete()", aError);
405 return;
406 }
407
408 // set stream priority to normal and time sensitive
409 iInputStream_->SetPriority(EPriorityNormal,
410 EMdaPriorityPreferenceTime);
411
412 // Read the first frame.
413 TPtr8 & frm = GetFrame();
414 TRAPD(err2, iInputStream_->ReadL(frm));
415 if (err2) {
416 PJ_LOG(4,(THIS_FILE, "Exception in iInputStream_->ReadL()"));
417 }
418}
419
420void CPjAudioInputEngine::MaiscBufferCopied(TInt aError,
421 const TDesC8 &aBuffer)
422{
423 lastError_ = aError;
424 if (aError != KErrNone) {
425 snd_perror("Error in MaiscBufferCopied()", aError);
426 return;
427 }
428
Nanang Izzuddin551ace82009-05-06 15:44:12 +0000429 if (frameRecBufLen_ || aBuffer.Length() < frameLen_) {
430 pj_memcpy(frameRecBuf_ + frameRecBufLen_, (void*) aBuffer.Ptr(), aBuffer.Length());
431 frameRecBufLen_ += aBuffer.Length();
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000432 }
433
434 if (frameRecBufLen_) {
435 while (frameRecBufLen_ >= frameLen_) {
436 pjmedia_frame f;
437
438 f.type = PJMEDIA_FRAME_TYPE_AUDIO;
439 f.buf = frameRecBuf_;
440 f.size = frameLen_;
441 f.timestamp.u32.lo = timeStamp_;
442 f.bit_info = 0;
443
444 // Call the callback.
445 recCb_(userData_, &f);
446 // Increment timestamp.
447 timeStamp_ += parentStrm_->param.samples_per_frame;
448
449 frameRecBufLen_ -= frameLen_;
450 pj_memmove(frameRecBuf_, frameRecBuf_+frameLen_, frameRecBufLen_);
451 }
452 } else {
453 pjmedia_frame f;
454
455 f.type = PJMEDIA_FRAME_TYPE_AUDIO;
456 f.buf = (void*)aBuffer.Ptr();
Nanang Izzuddin551ace82009-05-06 15:44:12 +0000457 f.size = aBuffer.Length();
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000458 f.timestamp.u32.lo = timeStamp_;
459 f.bit_info = 0;
460
461 // Call the callback.
462 recCb_(userData_, &f);
463
464 // Increment timestamp.
465 timeStamp_ += parentStrm_->param.samples_per_frame;
466 }
467
468 // Record next frame
469 TPtr8 & frm = GetFrame();
470 TRAPD(err2, iInputStream_->ReadL(frm));
471 if (err2) {
472 PJ_LOG(4,(THIS_FILE, "Exception in iInputStream_->ReadL()"));
473 }
474}
475
476
477void CPjAudioInputEngine::MaiscRecordComplete(TInt aError)
478{
479 lastError_ = aError;
480 state_ = STATE_INACTIVE;
Benny Prijonobb51e502009-06-19 09:15:59 +0000481 if (aError != KErrNone && aError != KErrCancel) {
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000482 snd_perror("Error in MaiscRecordComplete()", aError);
483 }
484}
485
486
487
488//////////////////////////////////////////////////////////////////////////////
489//
490
491/*
492 * Implementation: Symbian Output Stream.
493 */
494
495class CPjAudioOutputEngine : public CBase, MMdaAudioOutputStreamCallback
496{
497public:
498 enum State
499 {
500 STATE_INACTIVE,
501 STATE_ACTIVE,
502 };
503
504 ~CPjAudioOutputEngine();
505
506 static CPjAudioOutputEngine *NewL(struct mda_stream *parent_strm,
507 pjmedia_aud_play_cb play_cb,
508 void *user_data);
509
510 static CPjAudioOutputEngine *NewLC(struct mda_stream *parent_strm,
511 pjmedia_aud_play_cb rec_cb,
512 void *user_data);
513
514 pj_status_t StartPlay();
515 void Stop();
516
517 pj_status_t SetVolume(TInt vol) {
518 if (iOutputStream_) {
519 iOutputStream_->SetVolume(vol);
520 return PJ_SUCCESS;
521 } else
522 return PJ_EINVALIDOP;
523 }
524
525 TInt GetVolume() {
526 if (iOutputStream_) {
527 return iOutputStream_->Volume();
528 } else
529 return PJ_EINVALIDOP;
530 }
531
532 TInt GetMaxVolume() {
533 if (iOutputStream_) {
534 return iOutputStream_->MaxVolume();
535 } else
536 return PJ_EINVALIDOP;
537 }
538
539private:
540 State state_;
541 struct mda_stream *parentStrm_;
542 pjmedia_aud_play_cb playCb_;
543 void *userData_;
544 CMdaAudioOutputStream *iOutputStream_;
545 TUint8 *frameBuf_;
546 unsigned frameBufSize_;
547 TPtrC8 frame_;
548 TInt lastError_;
549 unsigned timestamp_;
550
551 CPjAudioOutputEngine(struct mda_stream *parent_strm,
552 pjmedia_aud_play_cb play_cb,
553 void *user_data);
554 void ConstructL();
555
556 virtual void MaoscOpenComplete(TInt aError);
557 virtual void MaoscBufferCopied(TInt aError, const TDesC8& aBuffer);
558 virtual void MaoscPlayComplete(TInt aError);
559};
560
561
562CPjAudioOutputEngine::CPjAudioOutputEngine(struct mda_stream *parent_strm,
563 pjmedia_aud_play_cb play_cb,
564 void *user_data)
565: state_(STATE_INACTIVE), parentStrm_(parent_strm), playCb_(play_cb),
566 userData_(user_data), iOutputStream_(NULL), frameBuf_(NULL),
567 lastError_(KErrNone), timestamp_(0)
568{
569}
570
571
572void CPjAudioOutputEngine::ConstructL()
573{
574 frameBufSize_ = parentStrm_->param.samples_per_frame *
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000575 BYTES_PER_SAMPLE;
576 frameBuf_ = new TUint8[frameBufSize_];
577}
578
579CPjAudioOutputEngine::~CPjAudioOutputEngine()
580{
581 Stop();
582 delete [] frameBuf_;
583}
584
585CPjAudioOutputEngine *
586CPjAudioOutputEngine::NewLC(struct mda_stream *parent_strm,
587 pjmedia_aud_play_cb play_cb,
588 void *user_data)
589{
590 CPjAudioOutputEngine* self = new (ELeave) CPjAudioOutputEngine(parent_strm,
591 play_cb,
592 user_data);
593 CleanupStack::PushL(self);
594 self->ConstructL();
595 return self;
596}
597
598CPjAudioOutputEngine *
599CPjAudioOutputEngine::NewL(struct mda_stream *parent_strm,
600 pjmedia_aud_play_cb play_cb,
601 void *user_data)
602{
603 CPjAudioOutputEngine *self = NewLC(parent_strm, play_cb, user_data);
604 CleanupStack::Pop(self);
605 return self;
606}
607
608pj_status_t CPjAudioOutputEngine::StartPlay()
609{
610 // Ignore command if playing is in progress.
611 if (state_ == STATE_ACTIVE)
612 return PJ_SUCCESS;
613
614 // Destroy existing stream.
615 if (iOutputStream_) delete iOutputStream_;
616 iOutputStream_ = NULL;
617
618 // Create the stream
619 TRAPD(err, iOutputStream_ = CMdaAudioOutputStream::NewL(*this));
620 if (err != KErrNone)
621 return PJ_RETURN_OS_ERROR(err);
622
623 // Initialize settings.
624 TMdaAudioDataSettings iStreamSettings;
625 iStreamSettings.iChannels =
626 get_channel_cap(parentStrm_->param.channel_count);
627 iStreamSettings.iSampleRate =
628 get_clock_rate_cap(parentStrm_->param.clock_rate);
629
630 pj_assert(iStreamSettings.iChannels != 0 &&
631 iStreamSettings.iSampleRate != 0);
632
633 PJ_LOG(4,(THIS_FILE, "Opening sound device for playback, "
634 "clock rate=%d, channel count=%d..",
635 parentStrm_->param.clock_rate,
636 parentStrm_->param.channel_count));
637
638 // Open stream.
639 lastError_ = KRequestPending;
640 iOutputStream_->Open(&iStreamSettings);
641
642 // Success
643 PJ_LOG(4,(THIS_FILE, "Sound playback started"));
644 return PJ_SUCCESS;
645
646}
647
648void CPjAudioOutputEngine::Stop()
649{
650 // Stop stream if it's playing
651 if (iOutputStream_ && state_ != STATE_INACTIVE) {
652 lastError_ = KRequestPending;
653 iOutputStream_->Stop();
654
655 // Wait until it's actually stopped
656 while (lastError_ == KRequestPending)
657 pj_symbianos_poll(-1, 100);
658 }
659
660 if (iOutputStream_) {
661 delete iOutputStream_;
662 iOutputStream_ = NULL;
663 }
664
665 state_ = STATE_INACTIVE;
666}
667
668void CPjAudioOutputEngine::MaoscOpenComplete(TInt aError)
669{
670 lastError_ = aError;
671
672 if (aError==KErrNone) {
673 // output stream opened succesfully, set status to Active
674 state_ = STATE_ACTIVE;
675
676 // set stream properties, 16bit 8KHz mono
677 TMdaAudioDataSettings iSettings;
678 iSettings.iChannels =
679 get_channel_cap(parentStrm_->param.channel_count);
680 iSettings.iSampleRate =
681 get_clock_rate_cap(parentStrm_->param.clock_rate);
682
683 iOutputStream_->SetAudioPropertiesL(iSettings.iSampleRate,
684 iSettings.iChannels);
685
686 // set volume to 1/2th of stream max volume
687 iOutputStream_->SetVolume(iOutputStream_->MaxVolume()/2);
688
689 // set stream priority to normal and time sensitive
690 iOutputStream_->SetPriority(EPriorityNormal,
691 EMdaPriorityPreferenceTime);
692
693 // Call callback to retrieve frame from upstream.
694 pjmedia_frame f;
695 pj_status_t status;
696
697 f.type = PJMEDIA_FRAME_TYPE_AUDIO;
698 f.buf = frameBuf_;
699 f.size = frameBufSize_;
700 f.timestamp.u32.lo = timestamp_;
701 f.bit_info = 0;
702
703 status = playCb_(this->userData_, &f);
704 if (status != PJ_SUCCESS) {
705 this->Stop();
706 return;
707 }
708
Benny Prijono3105be62009-04-30 18:49:13 +0000709 if (f.type != PJMEDIA_FRAME_TYPE_AUDIO)
710 pj_bzero(frameBuf_, frameBufSize_);
711
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000712 // Increment timestamp.
713 timestamp_ += (frameBufSize_ / BYTES_PER_SAMPLE);
714
715 // issue WriteL() to write the first audio data block,
716 // subsequent calls to WriteL() will be issued in
717 // MMdaAudioOutputStreamCallback::MaoscBufferCopied()
718 // until whole data buffer is written.
719 frame_.Set(frameBuf_, frameBufSize_);
720 iOutputStream_->WriteL(frame_);
721 } else {
722 snd_perror("Error in MaoscOpenComplete()", aError);
723 }
724}
725
726void CPjAudioOutputEngine::MaoscBufferCopied(TInt aError,
727 const TDesC8& aBuffer)
728{
729 PJ_UNUSED_ARG(aBuffer);
730
731 if (aError==KErrNone) {
732 // Buffer successfully written, feed another one.
733
734 // Call callback to retrieve frame from upstream.
735 pjmedia_frame f;
736 pj_status_t status;
737
738 f.type = PJMEDIA_FRAME_TYPE_AUDIO;
739 f.buf = frameBuf_;
740 f.size = frameBufSize_;
741 f.timestamp.u32.lo = timestamp_;
742 f.bit_info = 0;
743
744 status = playCb_(this->userData_, &f);
745 if (status != PJ_SUCCESS) {
746 this->Stop();
747 return;
748 }
749
Benny Prijono3105be62009-04-30 18:49:13 +0000750 if (f.type != PJMEDIA_FRAME_TYPE_AUDIO)
751 pj_bzero(frameBuf_, frameBufSize_);
752
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000753 // Increment timestamp.
754 timestamp_ += (frameBufSize_ / BYTES_PER_SAMPLE);
755
756 // Write to playback stream.
757 frame_.Set(frameBuf_, frameBufSize_);
758 iOutputStream_->WriteL(frame_);
759
760 } else if (aError==KErrAbort) {
761 // playing was aborted, due to call to CMdaAudioOutputStream::Stop()
762 state_ = STATE_INACTIVE;
763 } else {
764 // error writing data to output
765 lastError_ = aError;
766 state_ = STATE_INACTIVE;
767 snd_perror("Error in MaoscBufferCopied()", aError);
768 }
769}
770
771void CPjAudioOutputEngine::MaoscPlayComplete(TInt aError)
772{
773 lastError_ = aError;
774 state_ = STATE_INACTIVE;
Benny Prijonobb51e502009-06-19 09:15:59 +0000775 if (aError != KErrNone && aError != KErrCancel) {
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000776 snd_perror("Error in MaoscPlayComplete()", aError);
777 }
778}
779
780/****************************************************************************
781 * Factory operations
782 */
Nanang Izzuddin6d17d502009-02-26 21:12:44 +0000783
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000784/*
Benny Prijono689f8a82009-02-24 08:23:36 +0000785 * C compatible declaration of MDA factory.
786 */
787PJ_BEGIN_DECL
788PJ_DECL(pjmedia_aud_dev_factory*) pjmedia_symb_mda_factory(pj_pool_factory *pf);
789PJ_END_DECL
790
791/*
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000792 * Init Symbian audio driver.
793 */
794pjmedia_aud_dev_factory* pjmedia_symb_mda_factory(pj_pool_factory *pf)
795{
796 struct mda_factory *f;
797 pj_pool_t *pool;
798
799 pool = pj_pool_create(pf, "symb_aud", 1000, 1000, NULL);
800 f = PJ_POOL_ZALLOC_T(pool, struct mda_factory);
801 f->pf = pf;
802 f->pool = pool;
803 f->base.op = &factory_op;
804
805 return &f->base;
806}
807
808/* API: init factory */
809static pj_status_t factory_init(pjmedia_aud_dev_factory *f)
810{
811 struct mda_factory *af = (struct mda_factory*)f;
812
813 pj_ansi_strcpy(af->dev_info.name, "Symbian Audio");
814 af->dev_info.default_samples_per_sec = 8000;
815 af->dev_info.caps = PJMEDIA_AUD_DEV_CAP_INPUT_VOLUME_SETTING |
Nanang Izzuddina940b362009-02-23 13:53:30 +0000816 PJMEDIA_AUD_DEV_CAP_OUTPUT_VOLUME_SETTING;
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000817 af->dev_info.input_count = 1;
818 af->dev_info.output_count = 1;
819
820 PJ_LOG(4, (THIS_FILE, "Symb Mda initialized"));
821
822 return PJ_SUCCESS;
823}
824
825/* API: destroy factory */
826static pj_status_t factory_destroy(pjmedia_aud_dev_factory *f)
827{
828 struct mda_factory *af = (struct mda_factory*)f;
829 pj_pool_t *pool = af->pool;
830
831 af->pool = NULL;
832 pj_pool_release(pool);
833
834 PJ_LOG(4, (THIS_FILE, "Symbian Mda destroyed"));
835
836 return PJ_SUCCESS;
837}
838
Sauw Ming98766c72011-03-11 06:57:24 +0000839/* API: refresh the device list */
840static pj_status_t factory_refresh(pjmedia_aud_dev_factory *f)
841{
842 PJ_UNUSED_ARG(f);
843 return PJ_ENOTSUP;
844}
845
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000846/* API: get number of devices */
847static unsigned factory_get_dev_count(pjmedia_aud_dev_factory *f)
848{
849 PJ_UNUSED_ARG(f);
850 return 1;
851}
852
853/* API: get device info */
854static pj_status_t factory_get_dev_info(pjmedia_aud_dev_factory *f,
855 unsigned index,
856 pjmedia_aud_dev_info *info)
857{
858 struct mda_factory *af = (struct mda_factory*)f;
859
860 PJ_ASSERT_RETURN(index == 0, PJMEDIA_EAUD_INVDEV);
861
862 pj_memcpy(info, &af->dev_info, sizeof(*info));
863
864 return PJ_SUCCESS;
865}
866
867/* API: create default device parameter */
868static pj_status_t factory_default_param(pjmedia_aud_dev_factory *f,
869 unsigned index,
Benny Prijono10454dc2009-02-21 14:21:59 +0000870 pjmedia_aud_param *param)
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000871{
872 struct mda_factory *af = (struct mda_factory*)f;
873
874 PJ_ASSERT_RETURN(index == 0, PJMEDIA_EAUD_INVDEV);
875
876 pj_bzero(param, sizeof(*param));
877 param->dir = PJMEDIA_DIR_CAPTURE_PLAYBACK;
878 param->rec_id = index;
879 param->play_id = index;
880 param->clock_rate = af->dev_info.default_samples_per_sec;
881 param->channel_count = 1;
882 param->samples_per_frame = af->dev_info.default_samples_per_sec * 20 / 1000;
883 param->bits_per_sample = BITS_PER_SAMPLE;
884 param->flags = af->dev_info.caps;
885
886 return PJ_SUCCESS;
887}
888
889
890/* API: create stream */
891static pj_status_t factory_create_stream(pjmedia_aud_dev_factory *f,
Benny Prijono10454dc2009-02-21 14:21:59 +0000892 const pjmedia_aud_param *param,
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000893 pjmedia_aud_rec_cb rec_cb,
894 pjmedia_aud_play_cb play_cb,
895 void *user_data,
896 pjmedia_aud_stream **p_aud_strm)
897{
898 struct mda_factory *mf = (struct mda_factory*)f;
899 pj_pool_t *pool;
900 struct mda_stream *strm;
901
902 /* Can only support 16bits per sample raw PCM format. */
903 PJ_ASSERT_RETURN(param->bits_per_sample == BITS_PER_SAMPLE, PJ_EINVAL);
Nanang Izzuddin6d17d502009-02-26 21:12:44 +0000904 PJ_ASSERT_RETURN((param->flags & PJMEDIA_AUD_DEV_CAP_EXT_FORMAT)==0 ||
905 param->ext_fmt.id == PJMEDIA_FORMAT_L16,
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000906 PJ_ENOTSUP);
Nanang Izzuddin551ace82009-05-06 15:44:12 +0000907
908 /* It seems that MDA recorder only supports for mono channel. */
909 PJ_ASSERT_RETURN(param->channel_count == 1, PJ_EINVAL);
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000910
911 /* Create and Initialize stream descriptor */
912 pool = pj_pool_create(mf->pf, "symb_aud_dev", 1000, 1000, NULL);
913 PJ_ASSERT_RETURN(pool, PJ_ENOMEM);
914
915 strm = PJ_POOL_ZALLOC_T(pool, struct mda_stream);
916 strm->pool = pool;
917 strm->param = *param;
918
919 // Create the output stream.
920 if (strm->param.dir & PJMEDIA_DIR_PLAYBACK) {
921 TRAPD(err, strm->out_engine = CPjAudioOutputEngine::NewL(strm, play_cb,
922 user_data));
923 if (err != KErrNone) {
924 pj_pool_release(pool);
925 return PJ_RETURN_OS_ERROR(err);
926 }
927 }
928
929 // Create the input stream.
930 if (strm->param.dir & PJMEDIA_DIR_CAPTURE) {
931 TRAPD(err, strm->in_engine = CPjAudioInputEngine::NewL(strm, rec_cb,
932 user_data));
933 if (err != KErrNone) {
934 strm->in_engine = NULL;
935 delete strm->out_engine;
936 strm->out_engine = NULL;
937 pj_pool_release(pool);
938 return PJ_RETURN_OS_ERROR(err);
939 }
940 }
941
942 /* Done */
943 strm->base.op = &stream_op;
944 *p_aud_strm = &strm->base;
945
946 return PJ_SUCCESS;
947}
948
949/* API: Get stream info. */
950static pj_status_t stream_get_param(pjmedia_aud_stream *s,
Benny Prijono10454dc2009-02-21 14:21:59 +0000951 pjmedia_aud_param *pi)
Nanang Izzuddin5efbdf22009-02-20 14:59:55 +0000952{
953 struct mda_stream *strm = (struct mda_stream*)s;
954
955 PJ_ASSERT_RETURN(strm && pi, PJ_EINVAL);
956
957 pj_memcpy(pi, &strm->param, sizeof(*pi));
958
959 return PJ_SUCCESS;
960}
961
962/* API: get capability */
963static pj_status_t stream_get_cap(pjmedia_aud_stream *s,
964 pjmedia_aud_dev_cap cap,
965 void *pval)
966{
967 struct mda_stream *strm = (struct mda_stream*)s;
968 pj_status_t status = PJ_ENOTSUP;
969
970 PJ_ASSERT_RETURN(s && pval, PJ_EINVAL);
971
972 switch (cap) {
973 case PJMEDIA_AUD_DEV_CAP_INPUT_VOLUME_SETTING:
974 if (strm->param.dir & PJMEDIA_DIR_CAPTURE) {
975 PJ_ASSERT_RETURN(strm->in_engine, PJ_EINVAL);
976
977 TInt max_gain = strm->in_engine->GetMaxGain();
978 TInt gain = strm->in_engine->GetGain();
979
980 if (max_gain > 0 && gain >= 0) {
981 *(unsigned*)pval = gain * 100 / max_gain;
982 status = PJ_SUCCESS;
983 } else {
984 status = PJMEDIA_EAUD_NOTREADY;
985 }
986 }
987 break;
988 case PJMEDIA_AUD_DEV_CAP_OUTPUT_VOLUME_SETTING:
989 if (strm->param.dir & PJMEDIA_DIR_PLAYBACK) {
990 PJ_ASSERT_RETURN(strm->out_engine, PJ_EINVAL);
991
992 TInt max_vol = strm->out_engine->GetMaxVolume();
993 TInt vol = strm->out_engine->GetVolume();
994
995 if (max_vol > 0 && vol >= 0) {
996 *(unsigned*)pval = vol * 100 / max_vol;
997 status = PJ_SUCCESS;
998 } else {
999 status = PJMEDIA_EAUD_NOTREADY;
1000 }
1001 }
1002 break;
1003 default:
1004 break;
1005 }
1006
1007 return status;
1008}
1009
1010/* API: set capability */
1011static pj_status_t stream_set_cap(pjmedia_aud_stream *s,
1012 pjmedia_aud_dev_cap cap,
1013 const void *pval)
1014{
1015 struct mda_stream *strm = (struct mda_stream*)s;
1016 pj_status_t status = PJ_ENOTSUP;
1017
1018 PJ_ASSERT_RETURN(s && pval, PJ_EINVAL);
1019
1020 switch (cap) {
1021 case PJMEDIA_AUD_DEV_CAP_INPUT_VOLUME_SETTING:
1022 if (strm->param.dir & PJMEDIA_DIR_CAPTURE) {
1023 PJ_ASSERT_RETURN(strm->in_engine, PJ_EINVAL);
1024
1025 TInt max_gain = strm->in_engine->GetMaxGain();
1026 if (max_gain > 0) {
1027 TInt gain;
1028
1029 gain = *(unsigned*)pval * max_gain / 100;
1030 status = strm->in_engine->SetGain(gain);
1031 } else {
1032 status = PJMEDIA_EAUD_NOTREADY;
1033 }
1034 }
1035 break;
1036 case PJMEDIA_AUD_DEV_CAP_OUTPUT_VOLUME_SETTING:
1037 if (strm->param.dir & PJMEDIA_DIR_CAPTURE) {
1038 PJ_ASSERT_RETURN(strm->out_engine, PJ_EINVAL);
1039
1040 TInt max_vol = strm->out_engine->GetMaxVolume();
1041 if (max_vol > 0) {
1042 TInt vol;
1043
1044 vol = *(unsigned*)pval * max_vol / 100;
1045 status = strm->out_engine->SetVolume(vol);
1046 } else {
1047 status = PJMEDIA_EAUD_NOTREADY;
1048 }
1049 }
1050 break;
1051 default:
1052 break;
1053 }
1054
1055 return status;
1056}
1057
1058/* API: Start stream. */
1059static pj_status_t stream_start(pjmedia_aud_stream *strm)
1060{
1061 struct mda_stream *stream = (struct mda_stream*)strm;
1062
1063 PJ_ASSERT_RETURN(stream, PJ_EINVAL);
1064
1065 if (stream->out_engine) {
1066 pj_status_t status;
1067 status = stream->out_engine->StartPlay();
1068 if (status != PJ_SUCCESS)
1069 return status;
1070 }
1071
1072 if (stream->in_engine) {
1073 pj_status_t status;
1074 status = stream->in_engine->StartRecord();
1075 if (status != PJ_SUCCESS)
1076 return status;
1077 }
1078
1079 return PJ_SUCCESS;
1080}
1081
1082/* API: Stop stream. */
1083static pj_status_t stream_stop(pjmedia_aud_stream *strm)
1084{
1085 struct mda_stream *stream = (struct mda_stream*)strm;
1086
1087 PJ_ASSERT_RETURN(stream, PJ_EINVAL);
1088
1089 if (stream->in_engine) {
1090 stream->in_engine->Stop();
1091 }
1092
1093 if (stream->out_engine) {
1094 stream->out_engine->Stop();
1095 }
1096
1097 return PJ_SUCCESS;
1098}
1099
1100
1101/* API: Destroy stream. */
1102static pj_status_t stream_destroy(pjmedia_aud_stream *strm)
1103{
1104 struct mda_stream *stream = (struct mda_stream*)strm;
1105
1106 PJ_ASSERT_RETURN(stream, PJ_EINVAL);
1107
1108 stream_stop(strm);
1109
1110 delete stream->in_engine;
1111 stream->in_engine = NULL;
1112
1113 delete stream->out_engine;
1114 stream->out_engine = NULL;
1115
1116 pj_pool_t *pool;
1117 pool = stream->pool;
1118 if (pool) {
1119 stream->pool = NULL;
1120 pj_pool_release(pool);
1121 }
1122
1123 return PJ_SUCCESS;
1124}
1125
Nanang Izzuddinda37ea32009-03-06 16:24:43 +00001126#endif /* PJMEDIA_AUDIO_DEV_HAS_SYMB_MDA */