1 /*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 //#define LOG_NDEBUG 0
18 #define LOG_TAG "JetPlayer-C"
19
20 #include <utils/Log.h>
21 #include <media/JetPlayer.h>
22
23
24 namespace android
25 {
26
27 static const int MIX_NUM_BUFFERS = 4;
28 static const S_EAS_LIB_CONFIG* pLibConfig = NULL;
29
30 //-------------------------------------------------------------------------------------------------
JetPlayer(void * javaJetPlayer,int maxTracks,int trackBufferSize)31 JetPlayer::JetPlayer(void *javaJetPlayer, int maxTracks, int trackBufferSize) :
32 mEventCallback(NULL),
33 mJavaJetPlayerRef(javaJetPlayer),
34 mTid(-1),
35 mRender(false),
36 mPaused(false),
37 mMaxTracks(maxTracks),
38 mEasData(NULL),
39 mEasJetFileLoc(NULL),
40 mTrackBufferSize(trackBufferSize)
41 {
42 ALOGV("JetPlayer constructor");
43 mPreviousJetStatus.currentUserID = -1;
44 mPreviousJetStatus.segmentRepeatCount = -1;
45 mPreviousJetStatus.numQueuedSegments = -1;
46 mPreviousJetStatus.paused = true;
47 }
48
49 //-------------------------------------------------------------------------------------------------
~JetPlayer()50 JetPlayer::~JetPlayer()
51 {
52 ALOGV("~JetPlayer");
53 release();
54
55 }
56
57 //-------------------------------------------------------------------------------------------------
init()58 int JetPlayer::init()
59 {
60 //Mutex::Autolock lock(&mMutex);
61
62 EAS_RESULT result;
63
64 // retrieve the EAS library settings
65 if (pLibConfig == NULL)
66 pLibConfig = EAS_Config();
67 if (pLibConfig == NULL) {
68 ALOGE("JetPlayer::init(): EAS library configuration could not be retrieved, aborting.");
69 return EAS_FAILURE;
70 }
71
72 // init the EAS library
73 result = EAS_Init(&mEasData);
74 if (result != EAS_SUCCESS) {
75 ALOGE("JetPlayer::init(): Error initializing Sonivox EAS library, aborting.");
76 mState = EAS_STATE_ERROR;
77 return result;
78 }
79 // init the JET library with the default app event controller range
80 result = JET_Init(mEasData, NULL, sizeof(S_JET_CONFIG));
81 if (result != EAS_SUCCESS) {
82 ALOGE("JetPlayer::init(): Error initializing JET library, aborting.");
83 mState = EAS_STATE_ERROR;
84 return result;
85 }
86
87 // create the output AudioTrack
88 mAudioTrack = new AudioTrack();
89 mAudioTrack->set(AUDIO_STREAM_MUSIC, //TODO parameterize this
90 pLibConfig->sampleRate,
91 AUDIO_FORMAT_PCM_16_BIT,
92 audio_channel_out_mask_from_count(pLibConfig->numChannels),
93 (size_t) mTrackBufferSize,
94 AUDIO_OUTPUT_FLAG_NONE);
95
96 // create render and playback thread
97 {
98 Mutex::Autolock l(mMutex);
99 ALOGV("JetPlayer::init(): trying to start render thread");
100 mThread = new JetPlayerThread(this);
101 mThread->run("jetRenderThread", ANDROID_PRIORITY_AUDIO);
102 mCondition.wait(mMutex);
103 }
104 if (mTid > 0) {
105 // render thread started, we're ready
106 ALOGV("JetPlayer::init(): render thread(%d) successfully started.", mTid);
107 mState = EAS_STATE_READY;
108 } else {
109 ALOGE("JetPlayer::init(): failed to start render thread.");
110 mState = EAS_STATE_ERROR;
111 return EAS_FAILURE;
112 }
113
114 return EAS_SUCCESS;
115 }
116
setEventCallback(jetevent_callback eventCallback)117 void JetPlayer::setEventCallback(jetevent_callback eventCallback)
118 {
119 Mutex::Autolock l(mMutex);
120 mEventCallback = eventCallback;
121 }
122
123 //-------------------------------------------------------------------------------------------------
release()124 int JetPlayer::release()
125 {
126 ALOGV("JetPlayer::release()");
127 Mutex::Autolock lock(mMutex);
128 mPaused = true;
129 mRender = false;
130 if (mEasData) {
131 JET_Pause(mEasData);
132 JET_CloseFile(mEasData);
133 JET_Shutdown(mEasData);
134 EAS_Shutdown(mEasData);
135 }
136 if (mEasJetFileLoc) {
137 free(mEasJetFileLoc);
138 mEasJetFileLoc = NULL;
139 }
140 if (mAudioTrack != 0) {
141 mAudioTrack->stop();
142 mAudioTrack->flush();
143 mAudioTrack.clear();
144 }
145 if (mAudioBuffer) {
146 delete mAudioBuffer;
147 mAudioBuffer = NULL;
148 }
149 mEasData = NULL;
150
151 return EAS_SUCCESS;
152 }
153
154
155 //-------------------------------------------------------------------------------------------------
render()156 int JetPlayer::render() {
157 EAS_RESULT result = EAS_FAILURE;
158 EAS_I32 count;
159 int temp;
160 bool audioStarted = false;
161
162 ALOGV("JetPlayer::render(): entering");
163
164 // allocate render buffer
165 mAudioBuffer =
166 new EAS_PCM[pLibConfig->mixBufferSize * pLibConfig->numChannels * MIX_NUM_BUFFERS];
167
168 // signal main thread that we started
169 {
170 Mutex::Autolock l(mMutex);
171 mTid = gettid();
172 ALOGV("JetPlayer::render(): render thread(%d) signal", mTid);
173 mCondition.signal();
174 }
175
176 while (1) {
177
178 mMutex.lock(); // [[[[[[[[ LOCK ---------------------------------------
179
180 if (mEasData == NULL) {
181 mMutex.unlock();
182 ALOGV("JetPlayer::render(): NULL EAS data, exiting render.");
183 goto threadExit;
184 }
185
186 // nothing to render, wait for client thread to wake us up
187 while (!mRender)
188 {
189 ALOGV("JetPlayer::render(): signal wait");
190 if (audioStarted) {
191 mAudioTrack->pause();
192 // we have to restart the playback once we start rendering again
193 audioStarted = false;
194 }
195 mCondition.wait(mMutex);
196 ALOGV("JetPlayer::render(): signal rx'd");
197 }
198
199 // render midi data into the input buffer
200 int num_output = 0;
201 EAS_PCM* p = mAudioBuffer;
202 for (int i = 0; i < MIX_NUM_BUFFERS; i++) {
203 result = EAS_Render(mEasData, p, pLibConfig->mixBufferSize, &count);
204 if (result != EAS_SUCCESS) {
205 ALOGE("JetPlayer::render(): EAS_Render returned error %ld", result);
206 }
207 p += count * pLibConfig->numChannels;
208 num_output += count * pLibConfig->numChannels * sizeof(EAS_PCM);
209
210 // send events that were generated (if any) to the event callback
211 fireEventsFromJetQueue();
212 }
213
214 // update playback state
215 //ALOGV("JetPlayer::render(): updating state");
216 JET_Status(mEasData, &mJetStatus);
217 fireUpdateOnStatusChange();
218 mPaused = mJetStatus.paused;
219
220 mMutex.unlock(); // UNLOCK ]]]]]]]] -----------------------------------
221
222 // check audio output track
223 if (mAudioTrack == NULL) {
224 ALOGE("JetPlayer::render(): output AudioTrack was not created");
225 goto threadExit;
226 }
227
228 // Write data to the audio hardware
229 //ALOGV("JetPlayer::render(): writing to audio output");
230 if ((temp = mAudioTrack->write(mAudioBuffer, num_output)) < 0) {
231 ALOGE("JetPlayer::render(): Error in writing:%d",temp);
232 return temp;
233 }
234
235 // start audio output if necessary
236 if (!audioStarted) {
237 ALOGV("JetPlayer::render(): starting audio playback");
238 mAudioTrack->start();
239 audioStarted = true;
240 }
241
242 }//while (1)
243
244 threadExit:
245 if (mAudioTrack != NULL) {
246 mAudioTrack->stop();
247 mAudioTrack->flush();
248 }
249 delete [] mAudioBuffer;
250 mAudioBuffer = NULL;
251 mMutex.lock();
252 mTid = -1;
253 mCondition.signal();
254 mMutex.unlock();
255 return result;
256 }
257
258
259 //-------------------------------------------------------------------------------------------------
260 // fire up an update if any of the status fields has changed
261 // precondition: mMutex locked
fireUpdateOnStatusChange()262 void JetPlayer::fireUpdateOnStatusChange()
263 {
264 if ( (mJetStatus.currentUserID != mPreviousJetStatus.currentUserID)
265 ||(mJetStatus.segmentRepeatCount != mPreviousJetStatus.segmentRepeatCount) ) {
266 if (mEventCallback) {
267 mEventCallback(
268 JetPlayer::JET_USERID_UPDATE,
269 mJetStatus.currentUserID,
270 mJetStatus.segmentRepeatCount,
271 mJavaJetPlayerRef);
272 }
273 mPreviousJetStatus.currentUserID = mJetStatus.currentUserID;
274 mPreviousJetStatus.segmentRepeatCount = mJetStatus.segmentRepeatCount;
275 }
276
277 if (mJetStatus.numQueuedSegments != mPreviousJetStatus.numQueuedSegments) {
278 if (mEventCallback) {
279 mEventCallback(
280 JetPlayer::JET_NUMQUEUEDSEGMENT_UPDATE,
281 mJetStatus.numQueuedSegments,
282 -1,
283 mJavaJetPlayerRef);
284 }
285 mPreviousJetStatus.numQueuedSegments = mJetStatus.numQueuedSegments;
286 }
287
288 if (mJetStatus.paused != mPreviousJetStatus.paused) {
289 if (mEventCallback) {
290 mEventCallback(JetPlayer::JET_PAUSE_UPDATE,
291 mJetStatus.paused,
292 -1,
293 mJavaJetPlayerRef);
294 }
295 mPreviousJetStatus.paused = mJetStatus.paused;
296 }
297
298 }
299
300
301 //-------------------------------------------------------------------------------------------------
302 // fire up all the JET events in the JET engine queue (until the queue is empty)
303 // precondition: mMutex locked
fireEventsFromJetQueue()304 void JetPlayer::fireEventsFromJetQueue()
305 {
306 if (!mEventCallback) {
307 // no callback, just empty the event queue
308 while (JET_GetEvent(mEasData, NULL, NULL)) { }
309 return;
310 }
311
312 EAS_U32 rawEvent;
313 while (JET_GetEvent(mEasData, &rawEvent, NULL)) {
314 mEventCallback(
315 JetPlayer::JET_EVENT,
316 rawEvent,
317 -1,
318 mJavaJetPlayerRef);
319 }
320 }
321
322
323 //-------------------------------------------------------------------------------------------------
loadFromFile(const char * path)324 int JetPlayer::loadFromFile(const char* path)
325 {
326 ALOGV("JetPlayer::loadFromFile(): path=%s", path);
327
328 Mutex::Autolock lock(mMutex);
329
330 mEasJetFileLoc = (EAS_FILE_LOCATOR) malloc(sizeof(EAS_FILE));
331 strncpy(mJetFilePath, path, sizeof(mJetFilePath));
332 mJetFilePath[sizeof(mJetFilePath) - 1] = '\0';
333 mEasJetFileLoc->path = mJetFilePath;
334
335 mEasJetFileLoc->fd = 0;
336 mEasJetFileLoc->length = 0;
337 mEasJetFileLoc->offset = 0;
338
339 EAS_RESULT result = JET_OpenFile(mEasData, mEasJetFileLoc);
340 if (result != EAS_SUCCESS)
341 mState = EAS_STATE_ERROR;
342 else
343 mState = EAS_STATE_OPEN;
344 return( result );
345 }
346
347
348 //-------------------------------------------------------------------------------------------------
loadFromFD(const int fd,const long long offset,const long long length)349 int JetPlayer::loadFromFD(const int fd, const long long offset, const long long length)
350 {
351 ALOGV("JetPlayer::loadFromFD(): fd=%d offset=%lld length=%lld", fd, offset, length);
352
353 Mutex::Autolock lock(mMutex);
354
355 mEasJetFileLoc = (EAS_FILE_LOCATOR) malloc(sizeof(EAS_FILE));
356 mEasJetFileLoc->fd = fd;
357 mEasJetFileLoc->offset = offset;
358 mEasJetFileLoc->length = length;
359 mEasJetFileLoc->path = NULL;
360
361 EAS_RESULT result = JET_OpenFile(mEasData, mEasJetFileLoc);
362 if (result != EAS_SUCCESS)
363 mState = EAS_STATE_ERROR;
364 else
365 mState = EAS_STATE_OPEN;
366 return( result );
367 }
368
369
370 //-------------------------------------------------------------------------------------------------
closeFile()371 int JetPlayer::closeFile()
372 {
373 Mutex::Autolock lock(mMutex);
374 return JET_CloseFile(mEasData);
375 }
376
377
378 //-------------------------------------------------------------------------------------------------
play()379 int JetPlayer::play()
380 {
381 ALOGV("JetPlayer::play(): entering");
382 Mutex::Autolock lock(mMutex);
383
384 EAS_RESULT result = JET_Play(mEasData);
385
386 mPaused = false;
387 mRender = true;
388
389 JET_Status(mEasData, &mJetStatus);
390 this->dumpJetStatus(&mJetStatus);
391
392 fireUpdateOnStatusChange();
393
394 // wake up render thread
395 ALOGV("JetPlayer::play(): wakeup render thread");
396 mCondition.signal();
397
398 return result;
399 }
400
401 //-------------------------------------------------------------------------------------------------
pause()402 int JetPlayer::pause()
403 {
404 Mutex::Autolock lock(mMutex);
405 mPaused = true;
406 EAS_RESULT result = JET_Pause(mEasData);
407
408 mRender = false;
409
410 JET_Status(mEasData, &mJetStatus);
411 this->dumpJetStatus(&mJetStatus);
412 fireUpdateOnStatusChange();
413
414
415 return result;
416 }
417
418
419 //-------------------------------------------------------------------------------------------------
queueSegment(int segmentNum,int libNum,int repeatCount,int transpose,EAS_U32 muteFlags,EAS_U8 userID)420 int JetPlayer::queueSegment(int segmentNum, int libNum, int repeatCount, int transpose,
421 EAS_U32 muteFlags, EAS_U8 userID)
422 {
423 ALOGV("JetPlayer::queueSegment segmentNum=%d, libNum=%d, repeatCount=%d, transpose=%d",
424 segmentNum, libNum, repeatCount, transpose);
425 Mutex::Autolock lock(mMutex);
426 return JET_QueueSegment(mEasData, segmentNum, libNum, repeatCount, transpose, muteFlags, userID);
427 }
428
429 //-------------------------------------------------------------------------------------------------
setMuteFlags(EAS_U32 muteFlags,bool sync)430 int JetPlayer::setMuteFlags(EAS_U32 muteFlags, bool sync)
431 {
432 Mutex::Autolock lock(mMutex);
433 return JET_SetMuteFlags(mEasData, muteFlags, sync);
434 }
435
436 //-------------------------------------------------------------------------------------------------
setMuteFlag(int trackNum,bool muteFlag,bool sync)437 int JetPlayer::setMuteFlag(int trackNum, bool muteFlag, bool sync)
438 {
439 Mutex::Autolock lock(mMutex);
440 return JET_SetMuteFlag(mEasData, trackNum, muteFlag, sync);
441 }
442
443 //-------------------------------------------------------------------------------------------------
triggerClip(int clipId)444 int JetPlayer::triggerClip(int clipId)
445 {
446 ALOGV("JetPlayer::triggerClip clipId=%d", clipId);
447 Mutex::Autolock lock(mMutex);
448 return JET_TriggerClip(mEasData, clipId);
449 }
450
451 //-------------------------------------------------------------------------------------------------
clearQueue()452 int JetPlayer::clearQueue()
453 {
454 ALOGV("JetPlayer::clearQueue");
455 Mutex::Autolock lock(mMutex);
456 return JET_Clear_Queue(mEasData);
457 }
458
459 //-------------------------------------------------------------------------------------------------
dump()460 void JetPlayer::dump()
461 {
462 ALOGE("JetPlayer dump: JET file=%s", mEasJetFileLoc->path);
463 }
464
dumpJetStatus(S_JET_STATUS * pJetStatus)465 void JetPlayer::dumpJetStatus(S_JET_STATUS* pJetStatus)
466 {
467 if (pJetStatus!=NULL)
468 ALOGV(">> current JET player status: userID=%d segmentRepeatCount=%d numQueuedSegments=%d paused=%d",
469 pJetStatus->currentUserID, pJetStatus->segmentRepeatCount,
470 pJetStatus->numQueuedSegments, pJetStatus->paused);
471 else
472 ALOGE(">> JET player status is NULL");
473 }
474
475
476 } // end namespace android
477