3D Audio GSoC:
[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         AUD_Specs s1, s2;
43         s1 = reader1->getSpecs();
44         s2 = reader2->getSpecs();
45         if(memcmp(&s1, &s2, sizeof(AUD_Specs)))
46                 AUD_THROW(AUD_ERROR_SPECS, specs_error);
47 }
48
49 AUD_SuperposeReader::~AUD_SuperposeReader()
50 {
51 }
52
53 bool AUD_SuperposeReader::isSeekable() const
54 {
55         return m_reader1->isSeekable() && m_reader2->isSeekable();
56 }
57
58 void AUD_SuperposeReader::seek(int position)
59 {
60         m_reader1->seek(position);
61         m_reader2->seek(position);
62 }
63
64 int AUD_SuperposeReader::getLength() const
65 {
66         int len1 = m_reader1->getLength();
67         int len2 = m_reader2->getLength();
68         if((len1 < 0) || (len2 < 0))
69                 return -1;
70         return AUD_MIN(len1, len2);
71 }
72
73 int AUD_SuperposeReader::getPosition() const
74 {
75         int pos1 = m_reader1->getPosition();
76         int pos2 = m_reader2->getPosition();
77         return AUD_MAX(pos1, pos2);
78 }
79
80 AUD_Specs AUD_SuperposeReader::getSpecs() const
81 {
82         return m_reader1->getSpecs();
83 }
84
85 void AUD_SuperposeReader::read(int & length, sample_t* & buffer)
86 {
87         AUD_Specs specs = m_reader1->getSpecs();
88         int samplesize = AUD_SAMPLE_SIZE(specs);
89
90         if(m_buffer.getSize() < length * samplesize)
91                 m_buffer.resize(length * samplesize);
92         buffer = m_buffer.getBuffer();
93
94         int len1 = length;
95         sample_t* buf;
96         m_reader1->read(len1, buf);
97         memcpy(buffer, buf, len1 * samplesize);
98
99         if(len1 < length)
100                 memset(buffer + len1 * specs.channels, 0, (length - len1) * samplesize);
101
102         int len2 = length;
103         m_reader2->read(len2, buf);
104
105         for(int i = 0; i < len2 * specs.channels; i++)
106                 buffer[i] += buf[i];
107
108         length = AUD_MAX(len1, len2);
109 }