/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ //#define LOG_NDEBUG 0 #define LOG_TAG "MidiExtractor" #include <utils/Log.h> #include "MidiExtractor.h" #include <media/MidiIoWrapper.h> #include <media/stagefright/foundation/ADebug.h> #include <media/stagefright/MediaBufferGroup.h> #include <media/stagefright/MediaDefs.h> #include <media/stagefright/MetaData.h> #include <media/MediaTrack.h> #include <libsonivox/eas_reverb.h> namespace android { // how many Sonivox output buffers to aggregate into one MediaBufferBase static const int NUM_COMBINE_BUFFERS = 4; class MidiSource : public MediaTrack { public: MidiSource( MidiEngine &engine, MetaDataBase &trackMetadata); virtual status_t start(MetaDataBase *params); virtual status_t stop(); virtual status_t getFormat(MetaDataBase&); virtual status_t read( MediaBufferBase **buffer, const ReadOptions *options = NULL); protected: virtual ~MidiSource(); private: MidiEngine &mEngine; MetaDataBase &mTrackMetadata; bool mInitCheck; bool mStarted; status_t init(); // no copy constructor or assignment MidiSource(const MidiSource &); MidiSource &operator=(const MidiSource &); }; // Midisource MidiSource::MidiSource( MidiEngine &engine, MetaDataBase &trackMetadata) : mEngine(engine), mTrackMetadata(trackMetadata), mInitCheck(false), mStarted(false) { ALOGV("MidiSource ctor"); mInitCheck = init(); } MidiSource::~MidiSource() { ALOGV("MidiSource dtor"); if (mStarted) { stop(); } } status_t MidiSource::start(MetaDataBase * /* params */) { ALOGV("MidiSource::start"); CHECK(!mStarted); mStarted = true; mEngine.allocateBuffers(); return OK; } status_t MidiSource::stop() { ALOGV("MidiSource::stop"); CHECK(mStarted); mStarted = false; mEngine.releaseBuffers(); return OK; } status_t MidiSource::getFormat(MetaDataBase &meta) { meta = mTrackMetadata; return OK; } status_t MidiSource::read( MediaBufferBase **outBuffer, const ReadOptions *options) { ALOGV("MidiSource::read"); MediaBufferBase *buffer; // process an optional seek request int64_t seekTimeUs; ReadOptions::SeekMode mode; if ((NULL != options) && options->getSeekTo(&seekTimeUs, &mode)) { if (seekTimeUs <= 0LL) { seekTimeUs = 0LL; } mEngine.seekTo(seekTimeUs); } buffer = mEngine.readBuffer(); *outBuffer = buffer; ALOGV("MidiSource::read %p done", this); return buffer != NULL ? (status_t) OK : (status_t) ERROR_END_OF_STREAM; } status_t MidiSource::init() { ALOGV("MidiSource::init"); return OK; } // MidiEngine MidiEngine::MidiEngine(DataSourceBase *dataSource, MetaDataBase *fileMetadata, MetaDataBase *trackMetadata) : mGroup(NULL), mEasData(NULL), mEasHandle(NULL), mEasConfig(NULL), mIsInitialized(false) { mIoWrapper = new MidiIoWrapper(dataSource); // spin up a new EAS engine EAS_I32 temp; EAS_RESULT result = EAS_Init(&mEasData); if (result == EAS_SUCCESS) { result = EAS_OpenFile(mEasData, mIoWrapper->getLocator(), &mEasHandle); } if (result == EAS_SUCCESS) { result = EAS_Prepare(mEasData, mEasHandle); } if (result == EAS_SUCCESS) { result = EAS_ParseMetaData(mEasData, mEasHandle, &temp); } if (result != EAS_SUCCESS) { return; } if (fileMetadata != NULL) { fileMetadata->setCString(kKeyMIMEType, MEDIA_MIMETYPE_AUDIO_MIDI); } if (trackMetadata != NULL) { trackMetadata->setCString(kKeyMIMEType, MEDIA_MIMETYPE_AUDIO_RAW); trackMetadata->setInt64(kKeyDuration, 1000ll * temp); // milli->micro mEasConfig = EAS_Config(); trackMetadata->setInt32(kKeySampleRate, mEasConfig->sampleRate); trackMetadata->setInt32(kKeyChannelCount, mEasConfig->numChannels); trackMetadata->setInt32(kKeyPcmEncoding, kAudioEncodingPcm16bit); } mIsInitialized = true; } MidiEngine::~MidiEngine() { if (mEasHandle) { EAS_CloseFile(mEasData, mEasHandle); } if (mEasData) { EAS_Shutdown(mEasData); } delete mGroup; delete mIoWrapper; } status_t MidiEngine::initCheck() { return mIsInitialized ? OK : UNKNOWN_ERROR; } status_t MidiEngine::allocateBuffers() { // select reverb preset and enable EAS_SetParameter(mEasData, EAS_MODULE_REVERB, EAS_PARAM_REVERB_PRESET, EAS_PARAM_REVERB_CHAMBER); EAS_SetParameter(mEasData, EAS_MODULE_REVERB, EAS_PARAM_REVERB_BYPASS, EAS_FALSE); mGroup = new MediaBufferGroup; int bufsize = sizeof(EAS_PCM) * mEasConfig->mixBufferSize * mEasConfig->numChannels * NUM_COMBINE_BUFFERS; ALOGV("using %d byte buffer", bufsize); mGroup->add_buffer(MediaBufferBase::Create(bufsize)); return OK; } status_t MidiEngine::releaseBuffers() { delete mGroup; mGroup = NULL; return OK; } status_t MidiEngine::seekTo(int64_t positionUs) { ALOGV("seekTo %lld", (long long)positionUs); EAS_RESULT result = EAS_Locate(mEasData, mEasHandle, positionUs / 1000, false); return result == EAS_SUCCESS ? OK : UNKNOWN_ERROR; } MediaBufferBase* MidiEngine::readBuffer() { EAS_STATE state; EAS_State(mEasData, mEasHandle, &state); if ((state == EAS_STATE_STOPPED) || (state == EAS_STATE_ERROR)) { return NULL; } MediaBufferBase *buffer; status_t err = mGroup->acquire_buffer(&buffer); if (err != OK) { ALOGE("readBuffer: no buffer"); return NULL; } EAS_I32 timeMs; EAS_GetLocation(mEasData, mEasHandle, &timeMs); int64_t timeUs = 1000ll * timeMs; buffer->meta_data().setInt64(kKeyTime, timeUs); EAS_PCM* p = (EAS_PCM*) buffer->data(); int numBytesOutput = 0; for (int i = 0; i < NUM_COMBINE_BUFFERS; i++) { EAS_I32 numRendered; EAS_RESULT result = EAS_Render(mEasData, p, mEasConfig->mixBufferSize, &numRendered); if (result != EAS_SUCCESS) { ALOGE("EAS_Render() returned %ld, numBytesOutput = %d", result, numBytesOutput); buffer->release(); return NULL; // Stop processing to prevent infinite loops. } p += numRendered * mEasConfig->numChannels; numBytesOutput += numRendered * mEasConfig->numChannels * sizeof(EAS_PCM); } buffer->set_range(0, numBytesOutput); ALOGV("readBuffer: returning %zd in buffer %p", buffer->range_length(), buffer); return buffer; } // MidiExtractor MidiExtractor::MidiExtractor( DataSourceBase *dataSource) : mDataSource(dataSource), mInitCheck(false) { ALOGV("MidiExtractor ctor"); mEngine = new MidiEngine(mDataSource, &mFileMetadata, &mTrackMetadata); mInitCheck = mEngine->initCheck(); } MidiExtractor::~MidiExtractor() { ALOGV("MidiExtractor dtor"); } size_t MidiExtractor::countTracks() { return mInitCheck == OK ? 1 : 0; } MediaTrack *MidiExtractor::getTrack(size_t index) { if (mInitCheck != OK || index > 0) { return NULL; } return new MidiSource(*mEngine, mTrackMetadata); } status_t MidiExtractor::getTrackMetaData( MetaDataBase &meta, size_t index, uint32_t /* flags */) { ALOGV("MidiExtractor::getTrackMetaData"); if (mInitCheck != OK || index > 0) { return UNKNOWN_ERROR; } meta = mTrackMetadata; return OK; } status_t MidiExtractor::getMetaData(MetaDataBase &meta) { ALOGV("MidiExtractor::getMetaData"); meta = mFileMetadata; return OK; } // Sniffer bool SniffMidi(DataSourceBase *source, float *confidence) { MidiEngine p(source, NULL, NULL); if (p.initCheck() == OK) { *confidence = 0.8; ALOGV("SniffMidi: yes"); return true; } ALOGV("SniffMidi: no"); return false; } extern "C" { // This is the only symbol that needs to be exported __attribute__ ((visibility ("default"))) MediaExtractor::ExtractorDef GETEXTRACTORDEF() { return { MediaExtractor::EXTRACTORDEF_VERSION, UUID("ef6cca0a-f8a2-43e6-ba5f-dfcd7c9a7ef2"), 1, "MIDI Extractor", []( DataSourceBase *source, float *confidence, void **, MediaExtractor::FreeMetaFunc *) -> MediaExtractor::CreatorFunc { if (SniffMidi(source, confidence)) { return []( DataSourceBase *source, void *) -> MediaExtractor* { return new MidiExtractor(source);}; } return NULL; } }; } } // extern "C" } // namespace android