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