1 /* 2 * Copyright (C) 2012 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_TAG "LibsndfileSource" 18 //#define LOG_NDEBUG 0 19 20 #include <utils/Errors.h> 21 #include <utils/Log.h> 22 #include <media/nbaio/LibsndfileSource.h> 23 24 namespace android { 25 26 LibsndfileSource::LibsndfileSource(SNDFILE *sndfile, const SF_INFO &sfinfo, bool loop) : 27 NBAIO_Source(Format_from_SR_C(sfinfo.samplerate, sfinfo.channels)), 28 mSndfile(sndfile), 29 mEstimatedFramesUntilEOF(sfinfo.frames), 30 mLooping(loop && sfinfo.seekable), 31 mReadAnyFramesThisLoopCycle(false) 32 { 33 } 34 35 LibsndfileSource::~LibsndfileSource() 36 { 37 // do not close mSndfile; we don't own it 38 } 39 40 ssize_t LibsndfileSource::availableToRead() 41 { 42 // after we reach the presumed EOF, report infinity just in case there's actually more 43 return !mLooping && mEstimatedFramesUntilEOF > 0 ? mEstimatedFramesUntilEOF : SSIZE_MAX; 44 } 45 46 ssize_t LibsndfileSource::read(void *buffer, size_t count) 47 { 48 if (!mNegotiated) { 49 return (ssize_t) NEGOTIATE; 50 } 51 if (mSndfile == NULL) { 52 return (ssize_t) NO_INIT; 53 } 54 sf_count_t actual = sf_readf_short(mSndfile, (short *) buffer, (sf_count_t) count); 55 // Detect EOF by zero frames read, not by mFramesUntilEOF as it could be inaccurate 56 if (actual == 0) { 57 if (mLooping) { 58 if (mReadAnyFramesThisLoopCycle) { 59 (void) sf_seek(mSndfile, (sf_count_t) 0, SEEK_SET); 60 mReadAnyFramesThisLoopCycle = false; 61 } else { 62 // We didn't read any frames during the current loop cycle, so disable 63 // further looping to prevent the caller from busy waiting at read(). 64 // This is especially important when looping an empty file. 65 mLooping = false; 66 } 67 } 68 } else { 69 mFramesRead += actual; 70 if (actual >= mEstimatedFramesUntilEOF) { 71 mEstimatedFramesUntilEOF = 0; 72 } else { 73 mEstimatedFramesUntilEOF -= actual; 74 } 75 mReadAnyFramesThisLoopCycle = true; 76 } 77 return actual; 78 } 79 80 } // namespace android 81