Audaspace:
[blender.git] / intern / audaspace / FX / AUD_LoopReader.cpp
1 /*
2  * ***** BEGIN GPL LICENSE BLOCK *****
3  *
4  * Copyright 2009-2011 Jörg Hermann Müller
5  *
6  * This file is part of AudaSpace.
7  *
8  * Audaspace is free software; you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation; either version 2 of the License, or
11  * (at your option) any later version.
12  *
13  * AudaSpace is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with Audaspace; if not, write to the Free Software Foundation,
20  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
21  *
22  * ***** END GPL LICENSE BLOCK *****
23  */
24
25 /** \file audaspace/FX/AUD_LoopReader.cpp
26  *  \ingroup audfx
27  */
28
29
30 #include "AUD_LoopReader.h"
31 #include "AUD_Buffer.h"
32
33 #include <cstring>
34
35 AUD_LoopReader::AUD_LoopReader(boost::shared_ptr<AUD_IReader> reader, int loop) :
36                 AUD_EffectReader(reader), m_count(loop), m_left(loop)
37 {
38 }
39
40 void AUD_LoopReader::seek(int position)
41 {
42         int len = m_reader->getLength();
43         if(len < 0)
44                 m_reader->seek(position);
45         else
46         {
47                 if(m_count >= 0)
48                 {
49                         m_left = m_count - (position / len);
50                         if(m_left < 0)
51                                 m_left = 0;
52                 }
53                 m_reader->seek(position % len);
54         }
55 }
56
57 int AUD_LoopReader::getLength() const
58 {
59         if(m_count < 0)
60                 return -1;
61         return m_reader->getLength() * m_count;
62 }
63
64 int AUD_LoopReader::getPosition() const
65 {
66         return m_reader->getPosition() * (m_count < 0 ? 1 : m_count);
67 }
68
69 void AUD_LoopReader::read(int& length, bool& eos, sample_t* buffer)
70 {
71         const AUD_Specs specs = m_reader->getSpecs();
72
73         int len = length;
74
75         m_reader->read(length, eos, buffer);
76
77         if(length < len && eos && m_left)
78         {
79                 int pos = length;
80                 length = len;
81
82                 while(pos < length && eos && m_left)
83                 {
84                         if(m_left > 0)
85                                 m_left--;
86
87                         m_reader->seek(0);
88
89                         len = length - pos;
90                         m_reader->read(len, eos, buffer + pos * specs.channels);
91
92                         // prevent endless loop
93                         if(!len)
94                                 break;
95
96                         pos += len;
97                 }
98
99                 length = pos;
100         }
101 }