5d9ebbd618708fb9d133930f418bce75f148db92
[blender.git] / intern / audaspace / FX / AUD_DoubleReader.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_DoubleReader.cpp
28  *  \ingroup audfx
29  */
30
31
32 #include "AUD_DoubleReader.h"
33
34 #include <cstring>
35
36 static const char* specs_error = "AUD_DoubleReader: Both readers have to have "
37                                                                  "the same specs.";
38
39 AUD_DoubleReader::AUD_DoubleReader(AUD_Reference<AUD_IReader> reader1,
40                                                                    AUD_Reference<AUD_IReader> reader2) :
41                 m_reader1(reader1), m_reader2(reader2), m_finished1(false)
42 {
43         AUD_Specs s1, s2;
44         s1 = reader1->getSpecs();
45         s2 = reader2->getSpecs();
46         if(memcmp(&s1, &s2, sizeof(AUD_Specs)) != 0)
47         {
48                 AUD_THROW(AUD_ERROR_SPECS, specs_error);
49         }
50 }
51
52 AUD_DoubleReader::~AUD_DoubleReader()
53 {
54 }
55
56 bool AUD_DoubleReader::isSeekable() const
57 {
58         return m_reader1->isSeekable() && m_reader2->isSeekable();
59 }
60
61 void AUD_DoubleReader::seek(int position)
62 {
63         m_reader1->seek(position);
64
65         int pos1 = m_reader1->getPosition();
66
67         if((m_finished1 = (pos1 < position)))
68                 m_reader2->seek(position - pos1);
69         else
70                 m_reader2->seek(0);
71 }
72
73 int AUD_DoubleReader::getLength() const
74 {
75         int len1 = m_reader1->getLength();
76         int len2 = m_reader2->getLength();
77         if(len1 < 0 || len2 < 0)
78                 return -1;
79         return len1 + len2;
80 }
81
82 int AUD_DoubleReader::getPosition() const
83 {
84         return m_reader1->getPosition() + m_reader2->getPosition();
85 }
86
87 AUD_Specs AUD_DoubleReader::getSpecs() const
88 {
89         return m_reader1->getSpecs();
90 }
91
92 void AUD_DoubleReader::read(int& length, bool& eos, sample_t* buffer)
93 {
94         if(!m_finished1)
95         {
96                 int len = length;
97                 m_reader1->read(len, m_finished1, buffer);
98
99                 if(m_finished1)
100                 {
101                         const AUD_Specs specs = m_reader1->getSpecs();
102
103                         len = length - len;
104                         length -= len;
105
106                         m_reader2->read(len, eos, buffer + length * specs.channels);
107
108                         length += len;
109                 }
110         }
111         else
112         {
113                 m_reader2->read(length, eos, buffer);
114         }
115 }