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