| 
							- /*
 -  * Carla Native Plugins
 -  * Copyright (C) 2013 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 doc/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];
 -     uint64_t startFrame;
 -     uint32_t size;
 - 
 -     AudioFilePool()
 -         : startFrame(0),
 -           size(0)
 -     {
 -         buffer[0] = buffer[1] = nullptr;
 -     }
 - 
 -     ~AudioFilePool()
 -     {
 -         CARLA_ASSERT(buffer[0] == nullptr);
 -         CARLA_ASSERT(buffer[1] == nullptr);
 -         CARLA_ASSERT(startFrame == 0);
 -         CARLA_ASSERT(size == 0);
 -     }
 - 
 -     void create(const uint32_t sampleRate)
 -     {
 -         CARLA_ASSERT(buffer[0] == nullptr);
 -         CARLA_ASSERT(buffer[1] == nullptr);
 -         CARLA_ASSERT(startFrame == 0);
 -         CARLA_ASSERT(size == 0);
 - 
 -         size = sampleRate * 2;
 - 
 -         buffer[0] = new float[size];
 -         buffer[1] = new float[size];
 - 
 -         reset();
 -     }
 - 
 -     void destroy()
 -     {
 -         CARLA_ASSERT(buffer[0] != nullptr);
 -         CARLA_ASSERT(buffer[1] != nullptr);
 -         CARLA_ASSERT(size != 0);
 - 
 -         if (buffer[0] != nullptr)
 -         {
 -             delete[] buffer[0];
 -             buffer[0] = nullptr;
 -         }
 - 
 -         if (buffer[1] != nullptr)
 -         {
 -             delete[] buffer[1];
 -             buffer[1] = nullptr;
 -         }
 - 
 -         startFrame = 0;
 -         size = 0;
 -     }
 - 
 -     void reset()
 -     {
 -         CARLA_ASSERT(size != 0);
 - 
 -         startFrame = 0;
 - 
 -         FLOAT_CLEAR(buffer[0], size);
 -         FLOAT_CLEAR(buffer[1], size);
 -     }
 - };
 - 
 - class AbstractAudioPlayer
 - {
 - public:
 -     virtual ~AbstractAudioPlayer() {}
 -     virtual uint64_t getLastFrame() const = 0;
 - };
 - 
 - class AudioFileThread : public CarlaThread
 - {
 - public:
 -     AudioFileThread(AbstractAudioPlayer* const player, const double sampleRate)
 -         : CarlaThread("AudioFileThread"),
 -           kPlayer(player),
 -           fNeedsRead(false),
 -           fFilePtr(nullptr)
 -     {
 -         CARLA_ASSERT(kPlayer != nullptr);
 - 
 -         static bool adInitiated = false;
 - 
 -         if (! adInitiated)
 -         {
 -             ad_init();
 -             adInitiated = true;
 -         }
 - 
 -         ad_clear_nfo(&fFileNfo);
 - 
 -         fPool.create((uint32_t)sampleRate);
 -     }
 - 
 -     ~AudioFileThread() override
 -     {
 -         CARLA_ASSERT(! isThreadRunning());
 - 
 -         if (fFilePtr != nullptr)
 -             ad_close(fFilePtr);
 - 
 -         fPool.destroy();
 -     }
 - 
 -     void startNow()
 -     {
 -         fNeedsRead = true;
 -         startThread();
 -     }
 - 
 -     void stopNow()
 -     {
 -         fNeedsRead = false;
 -         stopThread(1000);
 - 
 -         const CarlaMutexLocker sl(fMutex);
 -         fPool.reset();
 -     }
 - 
 -     uint32_t getMaxFrame() const
 -     {
 -         return fFileNfo.frames > 0 ? (uint32_t)fFileNfo.frames : 0;
 -     }
 - 
 -     void setNeedsRead()
 -     {
 -         fNeedsRead = true;
 -     }
 - 
 -     bool loadFilename(const char* const filename)
 -     {
 -         CARLA_ASSERT(! isThreadRunning());
 -         CARLA_ASSERT(filename != nullptr);
 - 
 -         fPool.startFrame = 0;
 - 
 -         // clear old data
 -         if (fFilePtr != nullptr)
 -         {
 -             ad_close(fFilePtr);
 -             fFilePtr = nullptr;
 -         }
 - 
 -         ad_clear_nfo(&fFileNfo);
 - 
 -         // open new
 -         fFilePtr = ad_open(filename, &fFileNfo);
 - 
 -         if (fFilePtr == nullptr)
 -             return false;
 - 
 -         ad_dump_nfo(99, &fFileNfo);
 - 
 -         if (fFileNfo.frames == 0)
 -             carla_stderr("L: filename \"%s\" has 0 frames", filename);
 - 
 -         if ((fFileNfo.channels == 1 || fFileNfo.channels == 2) && fFileNfo.frames > 0)
 -         {
 -             // valid
 -             readPoll();
 -             return true;
 -         }
 -         else
 -         {
 -             // invalid
 -             ad_clear_nfo(&fFileNfo);
 -             ad_close(fFilePtr);
 -             fFilePtr = nullptr;
 -             return false;
 -         }
 -     }
 - 
 -     void tryPutData(AudioFilePool& pool)
 -     {
 -         CARLA_ASSERT(pool.size == fPool.size);
 - 
 -         if (pool.size != fPool.size)
 -             return;
 -         if (! fMutex.tryLock())
 -             return;
 - 
 -         //if (pool.startFrame != fPool.startFrame || pool.buffer[0] != fPool.buffer[0] || pool.buffer[1] != fPool.buffer[1])
 -         {
 -             pool.startFrame = fPool.startFrame;
 - 
 -             FLOAT_COPY(pool.buffer[0], fPool.buffer[0], fPool.size);
 -             FLOAT_COPY(pool.buffer[1], fPool.buffer[1], fPool.size);
 -         }
 - 
 -         fMutex.unlock();
 -     }
 - 
 -     void readPoll()
 -     {
 -         if (fFileNfo.frames <= 0 || fFilePtr == nullptr)
 -         {
 -             carla_stderr("R: no song loaded");
 -             fNeedsRead = false;
 -             return;
 -         }
 - 
 -         int64_t lastFrame = kPlayer->getLastFrame();
 -         int64_t readFrame = lastFrame;
 -         int64_t maxFrame  = fFileNfo.frames;
 - 
 -         if (lastFrame >= maxFrame)
 -         {
 - #if 0
 -             if (false)
 -             //if (handlePtr->loopMode)
 -             {
 -                 carla_stderr("R: DEBUG read loop, lastFrame:%i, maxFrame:%i", lastFrame, maxFrame);
 - 
 -                 if (maxFrame >= static_cast<int64_t>(fPool.size))
 -                 {
 -                     readFrame %= maxFrame;
 -                 }
 -                 else
 -                 {
 -                     readFrame  = 0;
 -                     lastFrame -= lastFrame % maxFrame;
 -                 }
 -             }
 -             else
 - #endif
 -             {
 -                 carla_stderr("R: transport out of bounds");
 -                 fNeedsRead = false;
 -                 return;
 -             }
 -         }
 - 
 -         // temp data buffer
 -         const size_t tmpSize = fPool.size * fFileNfo.channels;
 - 
 -         float tmpData[tmpSize];
 -         FLOAT_CLEAR(tmpData, int(tmpSize));
 - 
 -         {
 -             carla_stderr("R: poll data - reading at %li:%02li", readFrame/44100/60, (readFrame/44100) % 60);
 - 
 -             ad_seek(fFilePtr, readFrame);
 -             ssize_t i, j, rv = ad_read(fFilePtr, tmpData, tmpSize);
 -             i = j = 0;
 - 
 -             // lock, and put data asap
 -             const CarlaMutexLocker sl(fMutex);
 - 
 -             for (ssize_t size = (ssize_t)fPool.size; i < size && j < rv; ++j)
 -             {
 -                 if (fFileNfo.channels == 1)
 -                 {
 -                     fPool.buffer[0][i] = tmpData[j];
 -                     fPool.buffer[1][i] = tmpData[j];
 -                     i++;
 -                 }
 -                 else
 -                 {
 -                     if (j % 2 == 0)
 -                     {
 -                         fPool.buffer[0][i] = tmpData[j];
 -                     }
 -                     else
 -                     {
 -                         fPool.buffer[1][i] = tmpData[j];
 -                         i++;
 -                     }
 -                 }
 -             }
 - 
 - #if 0
 -             if (false)
 -             //if (handlePtr->loopMode && i < fPool.size)
 -             {
 -                 while (i < fPool.size)
 -                 {
 -                     for (j=0; i < fPool.size && j < rv; ++j)
 -                     {
 -                         if (fFileNfo.channels == 1)
 -                         {
 -                             fPool.buffer[0][i] = tmpData[j];
 -                             fPool.buffer[1][i] = tmpData[j];
 -                             i++;
 -                         }
 -                         else
 -                         {
 -                             if (j % 2 == 0)
 -                             {
 -                                 fPool.buffer[0][i] = tmpData[j];
 -                             }
 -                             else
 -                             {
 -                                 fPool.buffer[1][i] = tmpData[j];
 -                                 i++;
 -                             }
 -                         }
 -                     }
 -                 }
 -             }
 -             else
 - #endif
 -             {
 -                 for (ssize_t size = (ssize_t)fPool.size; i < size; ++i)
 -                 {
 -                     fPool.buffer[0][i] = 0.0f;
 -                     fPool.buffer[1][i] = 0.0f;
 -                 }
 -             }
 - 
 -             fPool.startFrame = lastFrame;
 -         }
 - 
 -         fNeedsRead = false;
 -     }
 - 
 - protected:
 -     void run() override
 -     {
 -         while (! shouldThreadExit())
 -         {
 -             const uint64_t lastFrame(kPlayer->getLastFrame());
 - 
 -             if (fNeedsRead || lastFrame < fPool.startFrame || (lastFrame - fPool.startFrame >= fPool.size*3/4 && lastFrame < (uint64_t)fFileNfo.frames))
 -                 readPoll();
 -             else
 -                 carla_msleep(50);
 -         }
 -     }
 - 
 - private:
 -     AbstractAudioPlayer* const kPlayer;
 - 
 -     volatile bool fNeedsRead;
 - 
 -     void*  fFilePtr;
 -     ADInfo fFileNfo;
 - 
 -     AudioFilePool fPool;
 -     CarlaMutex    fMutex;
 - };
 - 
 - #endif // AUDIO_BASE_HPP_INCLUDED
 
 
  |