svn merge https://svn.blender.org/svnroot/bf-blender/trunk/blender -r22704:22717
[blender.git] / intern / audaspace / SRC / AUD_SRCResampleReader.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_SRCResampleReader.h"
27 #include "AUD_Buffer.h"
28
29 #include <math.h>
30 #include <cstring>
31 #include <stdio.h>
32
33 static long src_callback(void *cb_data, float **data)
34 {
35         return ((AUD_SRCResampleReader*)cb_data)->doCallback(data);
36 }
37
38 AUD_SRCResampleReader::AUD_SRCResampleReader(AUD_IReader* reader,
39                                                                                          AUD_Specs specs) :
40                 AUD_EffectReader(reader)
41 {
42         m_sspecs = reader->getSpecs();
43
44         if(m_sspecs.format != AUD_FORMAT_FLOAT32)
45         {
46                 delete m_reader; AUD_DELETE("reader")
47                 AUD_THROW(AUD_ERROR_READER);
48         }
49
50         m_tspecs = specs;
51         m_tspecs.channels = m_sspecs.channels;
52         m_tspecs.format = m_sspecs.format;
53         m_factor = (double)m_tspecs.rate / (double)m_sspecs.rate;
54
55         int error;
56         m_src = src_callback_new(src_callback,
57                                                          SRC_SINC_MEDIUM_QUALITY,
58                                                          m_sspecs.channels,
59                                                          &error,
60                                                          this);
61
62         if(!m_src)
63         {
64                 // XXX printf("%s\n", src_strerror(error));
65                 delete m_reader; AUD_DELETE("reader")
66                 AUD_THROW(AUD_ERROR_READER);
67         }
68
69         m_buffer = new AUD_Buffer(); AUD_NEW("buffer")
70 }
71
72 AUD_SRCResampleReader::~AUD_SRCResampleReader()
73 {
74         delete m_buffer; AUD_DELETE("buffer")
75
76         src_delete(m_src);
77 }
78
79 long AUD_SRCResampleReader::doCallback(float** data)
80 {
81         int length = m_buffer->getSize() / 4 / m_tspecs.channels;
82         sample_t* buffer;
83
84         m_reader->read(length, buffer);
85
86         *data = (float*)buffer;
87         return length;
88 }
89
90 void AUD_SRCResampleReader::seek(int position)
91 {
92         m_reader->seek(position / m_factor);
93         src_reset(m_src);
94 }
95
96 int AUD_SRCResampleReader::getLength()
97 {
98         return m_reader->getLength() * m_factor;
99 }
100
101 int AUD_SRCResampleReader::getPosition()
102 {
103         return m_reader->getPosition() * m_factor;
104 }
105
106 AUD_Specs AUD_SRCResampleReader::getSpecs()
107 {
108         return m_tspecs;
109 }
110
111 void AUD_SRCResampleReader::read(int & length, sample_t* & buffer)
112 {
113         if(m_buffer->getSize() < length * m_tspecs.channels * 4)
114                 m_buffer->resize(length * m_tspecs.channels * 4);
115
116         buffer = m_buffer->getBuffer();
117
118         length = src_callback_read(m_src, m_factor, length, (float*)buffer);
119 }