|  | /*
 * Carla Native Plugins
 * Copyright (C) 2013-2019 Filipe Coelho <falktx@falktx.com>
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation; either version 2 of
 * the License, or any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 *
 * For a full copy of the GNU General Public License see the GPL.txt file
 */
#ifndef AUDIO_BASE_HPP_INCLUDED
#define AUDIO_BASE_HPP_INCLUDED
#include "CarlaThread.hpp"
#include "CarlaMathUtils.hpp"
extern "C" {
#include "audio_decoder/ad.h"
}
typedef struct adinfo ADInfo;
struct AudioFilePool {
    float*   buffer[2];
    uint32_t numFrames;
    volatile uint64_t startFrame;
#ifdef CARLA_PROPER_CPP11_SUPPORT
    AudioFilePool() noexcept
        : buffer{nullptr},
          numFrames(0),
          startFrame(0) {}
#else
    AudioFilePool() noexcept
        : numFrames(0),
          startFrame(0)
    {
        buffer[0] = buffer[1] = nullptr;
    }
#endif
    ~AudioFilePool()
    {
        destroy();
    }
    void create(const uint32_t desiredNumFrames)
    {
        CARLA_ASSERT(buffer[0] == nullptr);
        CARLA_ASSERT(buffer[1] == nullptr);
        CARLA_ASSERT(startFrame == 0);
        CARLA_ASSERT(numFrames == 0);
        numFrames = desiredNumFrames;
        buffer[0] = new float[numFrames];
        buffer[1] = new float[numFrames];
        reset();
    }
    void destroy() noexcept
    {
        if (buffer[0] != nullptr)
        {
            delete[] buffer[0];
            buffer[0] = nullptr;
        }
        if (buffer[1] != nullptr)
        {
            delete[] buffer[1];
            buffer[1] = nullptr;
        }
        startFrame = 0;
        numFrames = 0;
    }
    void reset() noexcept
    {
        startFrame = 0;
        if (numFrames != 0)
        {
            carla_zeroFloats(buffer[0], numFrames);
            carla_zeroFloats(buffer[1], numFrames);
        }
    }
    CARLA_DECLARE_NON_COPY_STRUCT(AudioFilePool)
};
class AbstractAudioPlayer
{
public:
    virtual ~AbstractAudioPlayer() {}
    virtual uint64_t getLastFrame() const = 0;
};
class AudioFileThread : public CarlaThread
{
public:
    AudioFileThread(AbstractAudioPlayer* const player)
        : CarlaThread("AudioFileThread"),
          kPlayer(player),
          fEntireFileLoaded(false),
          fLoopingMode(true),
          fNeedsRead(false),
          fQuitNow(true),
          fFilePtr(nullptr),
          fFileNfo(),
          fNumFileFrames(0),
          fPollTempData(nullptr),
          fPollTempSize(0),
          fPool(),
          fMutex()
    {
        CARLA_ASSERT(kPlayer != nullptr);
        static bool adInitiated = false;
        if (! adInitiated)
        {
            ad_init();
            adInitiated = true;
        }
        ad_clear_nfo(&fFileNfo);
    }
    ~AudioFileThread() override
    {
        CARLA_ASSERT(fQuitNow);
        CARLA_ASSERT(! isThreadRunning());
        cleanup();
    }
    void cleanup()
    {
        fEntireFileLoaded = false;
        if (fFilePtr != nullptr)
        {
            ad_close(fFilePtr);
            fFilePtr = nullptr;
        }
        if (fPollTempData != nullptr)
        {
            delete[] fPollTempData;
            fPollTempData = nullptr;
            fPollTempSize = 0;
        }
        fPool.destroy();
    }
    void startNow()
    {
        if (fPollTempData == nullptr)
            return;
        fNeedsRead = true;
        fQuitNow = false;
        startThread();
    }
    void stopNow()
    {
        fNeedsRead = false;
        fQuitNow = true;
        stopThread(1000);
        const CarlaMutexLocker cml(fMutex);
        fPool.reset();
    }
    bool isEntireFileLoaded() const noexcept
    {
        return fEntireFileLoaded;
    }
    uint32_t getMaxFrame() const noexcept
    {
        return fNumFileFrames;
    }
    uint64_t getPoolStartFrame() const noexcept
    {
        return fPool.startFrame;
    }
    uint32_t getPoolNumFrames() const noexcept
    {
        return fPool.numFrames;
    }
    void setLoopingMode(const bool on) noexcept
    {
        fLoopingMode = on;
    }
    void setNeedsRead() noexcept
    {
        fNeedsRead = true;
    }
    bool loadFilename(const char* const filename, const uint32_t sampleRate)
    {
        CARLA_SAFE_ASSERT_RETURN(! isThreadRunning(), false);
        CARLA_SAFE_ASSERT_RETURN(filename != nullptr && *filename != '\0', false);
        cleanup();
        ad_clear_nfo(&fFileNfo);
        // open new
        fFilePtr = ad_open(filename, &fFileNfo);
        if (fFilePtr == nullptr)
            return false;
        ad_dump_nfo(99, &fFileNfo);
        // Fix for misinformation using libsndfile
        if (fFileNfo.frames % fFileNfo.channels)
            --fFileNfo.frames;
        if (fFileNfo.frames <= 0)
            carla_stderr("L: filename \"%s\" has 0 frames", filename);
        if ((fFileNfo.channels == 1 || fFileNfo.channels == 2) && fFileNfo.frames > 0)
        {
            // valid
            const uint32_t fileNumFrames = static_cast<uint32_t>(fFileNfo.frames);
            const uint32_t poolNumFrames = sampleRate * 5;
            if (fileNumFrames <= poolNumFrames)
            {
                // entire file fits in a small pool, lets read it now
                fPool.create(fileNumFrames);
                readEntireFileIntoPool();
                ad_close(fFilePtr);
                fFilePtr = nullptr;
            }
            else
            {
                // file is too big for our audio pool, we need an extra buffer
                fPool.create(poolNumFrames);
                const size_t pollTempSize = poolNumFrames * fFileNfo.channels;
                try {
                    fPollTempData = new float[pollTempSize];
                } catch (...) {
                    ad_close(fFilePtr);
                    fFilePtr = nullptr;
                    return false;
                }
                fPollTempSize = pollTempSize;
            }
            fNumFileFrames = fileNumFrames;
            readPoll();
            return true;
        }
        else
        {
            // invalid
            ad_clear_nfo(&fFileNfo);
            ad_close(fFilePtr);
            fFilePtr = nullptr;
            return false;
        }
    }
    void putAllData(AudioFilePool& pool)
    {
        CARLA_SAFE_ASSERT_RETURN(pool.numFrames == fPool.numFrames,);
        const CarlaMutexLocker cml(fMutex);
        pool.startFrame = fPool.startFrame;
        carla_copyFloats(pool.buffer[0], fPool.buffer[0], fPool.numFrames);
        carla_copyFloats(pool.buffer[1], fPool.buffer[1], fPool.numFrames);
    }
    bool tryPutData(AudioFilePool& pool, const uint64_t framePos, const uint32_t frames)
    {
        CARLA_SAFE_ASSERT_RETURN(pool.numFrames == fPool.numFrames, false);
        if (framePos >= fPool.numFrames)
            return false;
        const CarlaMutexLocker cml(fMutex);
        /*
        const CarlaMutexTryLocker cmtl(fMutex);
        if (! cmtl.wasLocked())
            return false;
        */
        pool.startFrame = fPool.startFrame;
        carla_copyFloats(pool.buffer[0] + framePos, fPool.buffer[0] + framePos, frames);
        carla_copyFloats(pool.buffer[1] + framePos, fPool.buffer[1] + framePos, frames);
        return true;
    }
    void readEntireFileIntoPool()
    {
        CARLA_SAFE_ASSERT_RETURN(fPool.numFrames > 0,);
        const uint numChannels = fFileNfo.channels;
        const size_t bufferSize = fPool.numFrames * numChannels;
        float* const buffer = (float*)std::malloc(bufferSize*sizeof(float));
        CARLA_SAFE_ASSERT_RETURN(buffer != nullptr,);
        carla_zeroFloats(buffer, bufferSize);
        ad_seek(fFilePtr, 0);
        ssize_t rv = ad_read(fFilePtr, buffer, bufferSize);
        CARLA_SAFE_ASSERT_INT2_RETURN(rv == static_cast<ssize_t>(bufferSize),
                                      static_cast<int>(rv),
                                      static_cast<int>(bufferSize),
                                      std::free(buffer));
        {
            // lock, and put data asap
            const CarlaMutexLocker cml(fMutex);
            for (ssize_t i=0, j=0; j < rv; ++j)
            {
                if (numChannels == 1)
                {
                    fPool.buffer[0][i] = buffer[j];
                    fPool.buffer[1][i] = buffer[j];
                    ++i;
                }
                else
                {
                    if (j % 2 == 0)
                    {
                        fPool.buffer[0][i] = buffer[j];
                    }
                    else
                    {
                        fPool.buffer[1][i] = buffer[j];
                        ++i;
                    }
                }
            }
        }
        std::free(buffer);
        fEntireFileLoaded = true;
    }
    void readPoll()
    {
        if (fNumFileFrames == 0 || fFileNfo.channels == 0 || fFilePtr == nullptr)
        {
            carla_debug("R: no song loaded");
            fNeedsRead = false;
            return;
        }
        if (fPollTempData == nullptr)
        {
            carla_debug("R: nothing to poll");
            fNeedsRead = false;
            return;
        }
        uint64_t lastFrame = kPlayer->getLastFrame();
        int64_t readFrameCheck;
        if (lastFrame >= fNumFileFrames)
        {
            if (fLoopingMode)
            {
                const uint64_t readFrameCheckLoop = lastFrame % fNumFileFrames;
                CARLA_SAFE_ASSERT_RETURN(readFrameCheckLoop < INT32_MAX,);
                carla_debug("R: transport out of bounds for loop");
                readFrameCheck = static_cast<int64_t>(readFrameCheckLoop);
            }
            else
            {
                carla_debug("R: transport out of bounds");
                fNeedsRead = false;
                return;
            }
        }
        else
        {
            CARLA_SAFE_ASSERT_RETURN(lastFrame < INT32_MAX,);
            readFrameCheck = static_cast<int64_t>(lastFrame);
        }
        const int64_t readFrame = readFrameCheck;
        // temp data buffer
        carla_zeroFloats(fPollTempData, fPollTempSize);
        {
#if 0
            const int32_t sampleRate = 44100;
            carla_debug("R: poll data - reading at frame %li, time %li:%02li, lastFrame %li",
                        readFrame, readFrame/sampleRate/60, (readFrame/sampleRate) % 60, lastFrame);
#endif
            ad_seek(fFilePtr, readFrame);
            size_t i = 0;
            ssize_t j = 0;
            ssize_t rv = ad_read(fFilePtr, fPollTempData, fPollTempSize);
            if (rv < 0)
            {
                carla_stderr("R: ad_read failed");
                fNeedsRead = false;
                return;
            }
            const size_t urv = static_cast<size_t>(rv);
            // see if we can read more
            if (readFrame + rv >= static_cast<ssize_t>(fFileNfo.frames) && urv < fPollTempSize)
            {
                carla_debug("R: from start");
                ad_seek(fFilePtr, 0);
                rv += ad_read(fFilePtr, fPollTempData+urv, fPollTempSize-urv);
            }
            // lock, and put data asap
            const CarlaMutexLocker cml(fMutex);
            do {
                for (; i < fPool.numFrames && j < rv; ++j)
                {
                    if (fFileNfo.channels == 1)
                    {
                        fPool.buffer[0][i] = fPollTempData[j];
                        fPool.buffer[1][i] = fPollTempData[j];
                        i++;
                    }
                    else
                    {
                        if (j % 2 == 0)
                        {
                            fPool.buffer[0][i] = fPollTempData[j];
                        }
                        else
                        {
                            fPool.buffer[1][i] = fPollTempData[j];
                            i++;
                        }
                    }
                }
                if (i >= fPool.numFrames)
                    break;
                if (rv == fFileNfo.frames)
                {
                    // full file read
                    j = 0;
                    carla_debug("R: full file was read, filling buffers again");
                }
                else
                {
                    carla_debug("read break, not enough space");
                    carla_zeroFloats(fPool.buffer[0] + i, fPool.numFrames - i);
                    carla_zeroFloats(fPool.buffer[1] + i, fPool.numFrames - i);
                    break;
                }
            } while (i < fPool.numFrames);
            fPool.startFrame = lastFrame;
        }
        fNeedsRead = false;
    }
protected:
    void run() override
    {
        const uint64_t numFramesNearEnd = fPool.numFrames*3/4;
        uint64_t lastFrame;
        while (! fQuitNow)
        {
            lastFrame = kPlayer->getLastFrame();
            if (fNeedsRead || lastFrame < fPool.startFrame || lastFrame - fPool.startFrame >= numFramesNearEnd)
                readPoll();
            carla_msleep(50);
        }
    }
private:
    AbstractAudioPlayer* const kPlayer;
    bool fEntireFileLoaded;
    bool fLoopingMode;
    volatile bool fNeedsRead;
    volatile bool fQuitNow;
    void*  fFilePtr;
    ADInfo fFileNfo;
    uint32_t fNumFileFrames;
    float* fPollTempData;
    size_t fPollTempSize;
    AudioFilePool fPool;
    CarlaMutex    fMutex;
    CARLA_DECLARE_NON_COPY_STRUCT(AudioFileThread)
};
#endif // AUDIO_BASE_HPP_INCLUDED
 |