Merging revision 30126:30261 from trunk.
[blender.git] / intern / audaspace / FX / AUD_SuperposeReader.cpp
1 /*
2  * $Id$
3  *
4  * ***** BEGIN LGPL LICENSE BLOCK *****
5  *
6  * Copyright 2009 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 Lesser General Public License as published by
12  * the Free Software Foundation, either version 3 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 Lesser General Public License for more details.
19  *
20  * You should have received a copy of the GNU Lesser General Public License
21  * along with AudaSpace.  If not, see <http://www.gnu.org/licenses/>.
22  *
23  * ***** END LGPL LICENSE BLOCK *****
24  */
25
26 #include "AUD_SuperposeReader.h"
27 #include "AUD_Buffer.h"
28
29 #include <cstring>
30
31 AUD_SuperposeReader::AUD_SuperposeReader(AUD_IReader* reader1, AUD_IReader* reader2) :
32         m_reader1(reader1), m_reader2(reader2)
33 {
34         try
35         {
36                 if(!reader1)
37                         AUD_THROW(AUD_ERROR_READER);
38
39                 if(!reader2)
40                         AUD_THROW(AUD_ERROR_READER);
41
42                 AUD_Specs s1, s2;
43                 s1 = reader1->getSpecs();
44                 s2 = reader2->getSpecs();
45                 if(memcmp(&s1, &s2, sizeof(AUD_Specs)) != 0)
46                         AUD_THROW(AUD_ERROR_READER);
47         }
48
49         catch(AUD_Exception)
50         {
51                 if(reader1)
52                 {
53                         delete reader1; AUD_DELETE("reader")
54                 }
55                 if(reader2)
56                 {
57                         delete reader2; AUD_DELETE("reader")
58                 }
59
60                 throw;
61         }
62
63         m_buffer = new AUD_Buffer(); AUD_NEW("buffer")
64 }
65
66 AUD_SuperposeReader::~AUD_SuperposeReader()
67 {
68         delete m_reader1; AUD_DELETE("reader")
69         delete m_reader2; AUD_DELETE("reader")
70         delete m_buffer; AUD_DELETE("buffer")
71 }
72
73 bool AUD_SuperposeReader::isSeekable()
74 {
75         return m_reader1->isSeekable() && m_reader2->isSeekable();
76 }
77
78 void AUD_SuperposeReader::seek(int position)
79 {
80         m_reader1->seek(position);
81         m_reader2->seek(position);
82 }
83
84 int AUD_SuperposeReader::getLength()
85 {
86         int len1 = m_reader1->getLength();
87         int len2 = m_reader2->getLength();
88         if((len1 < 0) || (len2 < 0))
89                 return -1;
90         if(len1 < len2)
91                 return len2;
92         return len1;
93 }
94
95 int AUD_SuperposeReader::getPosition()
96 {
97         int pos1 = m_reader1->getPosition();
98         int pos2 = m_reader2->getPosition();
99         return AUD_MAX(pos1, pos2);
100 }
101
102 AUD_Specs AUD_SuperposeReader::getSpecs()
103 {
104         return m_reader1->getSpecs();
105 }
106
107 AUD_ReaderType AUD_SuperposeReader::getType()
108 {
109         if(m_reader1->getType() == AUD_TYPE_BUFFER &&
110            m_reader2->getType() == AUD_TYPE_BUFFER)
111                 return AUD_TYPE_BUFFER;
112         return AUD_TYPE_STREAM;
113 }
114
115 bool AUD_SuperposeReader::notify(AUD_Message &message)
116 {
117         return m_reader1->notify(message) | m_reader2->notify(message);
118 }
119
120 void AUD_SuperposeReader::read(int & length, sample_t* & buffer)
121 {
122         AUD_Specs specs = m_reader1->getSpecs();
123         int samplesize = AUD_SAMPLE_SIZE(specs);
124
125         if(m_buffer->getSize() < length * samplesize)
126                 m_buffer->resize(length * samplesize);
127         buffer = m_buffer->getBuffer();
128
129         int len1 = length;
130         sample_t* buf;
131         m_reader1->read(len1, buf);
132         memcpy(buffer, buf, len1 * samplesize);
133
134         if(len1 < length)
135                 memset(buffer + len1 * specs.channels, 0, (length - len1) * samplesize);
136
137         int len2 = length;
138         m_reader2->read(len2, buf);
139
140         for(int i = 0; i < len2 * specs.channels; i++)
141                 buffer[i] += buf[i];
142
143         length = AUD_MAX(len1, len2);
144 }