| /* |
| * Copyright (C) 2012 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_TAG "LibsndfileSource" |
| //#define LOG_NDEBUG 0 |
| |
| #include <utils/Errors.h> |
| #include <utils/Log.h> |
| #include <media/nbaio/LibsndfileSource.h> |
| |
| namespace android { |
| |
| LibsndfileSource::LibsndfileSource(SNDFILE *sndfile, const SF_INFO &sfinfo, bool loop) : |
| NBAIO_Source(Format_from_SR_C(sfinfo.samplerate, sfinfo.channels)), |
| mSndfile(sndfile), |
| mEstimatedFramesUntilEOF(sfinfo.frames), |
| mLooping(loop && sfinfo.seekable), |
| mReadAnyFramesThisLoopCycle(false) |
| { |
| } |
| |
| LibsndfileSource::~LibsndfileSource() |
| { |
| // do not close mSndfile; we don't own it |
| } |
| |
| ssize_t LibsndfileSource::availableToRead() |
| { |
| // after we reach the presumed EOF, report infinity just in case there's actually more |
| return !mLooping && mEstimatedFramesUntilEOF > 0 ? mEstimatedFramesUntilEOF : SSIZE_MAX; |
| } |
| |
| ssize_t LibsndfileSource::read(void *buffer, size_t count) |
| { |
| if (!mNegotiated) { |
| return (ssize_t) NEGOTIATE; |
| } |
| if (mSndfile == NULL) { |
| return (ssize_t) NO_INIT; |
| } |
| sf_count_t actual = sf_readf_short(mSndfile, (short *) buffer, (sf_count_t) count); |
| // Detect EOF by zero frames read, not by mFramesUntilEOF as it could be inaccurate |
| if (actual == 0) { |
| if (mLooping) { |
| if (mReadAnyFramesThisLoopCycle) { |
| (void) sf_seek(mSndfile, (sf_count_t) 0, SEEK_SET); |
| mReadAnyFramesThisLoopCycle = false; |
| } else { |
| // We didn't read any frames during the current loop cycle, so disable |
| // further looping to prevent the caller from busy waiting at read(). |
| // This is especially important when looping an empty file. |
| mLooping = false; |
| } |
| } |
| } else { |
| mFramesRead += actual; |
| if (actual >= mEstimatedFramesUntilEOF) { |
| mEstimatedFramesUntilEOF = 0; |
| } else { |
| mEstimatedFramesUntilEOF -= actual; |
| } |
| mReadAnyFramesThisLoopCycle = true; |
| } |
| return actual; |
| } |
| |
| } // namespace android |