/[svn]/linuxsampler/trunk/src/drivers/audio/AudioChannel.cpp
ViewVC logotype

Annotation of /linuxsampler/trunk/src/drivers/audio/AudioChannel.cpp

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1913 - (hide annotations) (download)
Sun Jun 7 16:24:55 2009 UTC (14 years, 10 months ago) by senoner
File size: 8298 byte(s)
* AudioChannel.cpp: added GCC vector extensions code for the functions
* copyTo() and MixTo() , gives 300% speedup
* should reduce CPU usage with large FX sends setups

1 schoenebeck 200 /***************************************************************************
2     * *
3     * LinuxSampler - modular, streaming capable sampler *
4     * *
5     * Copyright (C) 2003, 2004 by Benno Senoner and Christian Schoenebeck *
6 schoenebeck 1037 * Copyright (C) 2005 - 2007 Christian Schoenebeck *
7 schoenebeck 200 * *
8     * This program is free software; you can redistribute it and/or modify *
9     * it under the terms of the GNU General Public License as published by *
10     * the Free Software Foundation; either version 2 of the License, or *
11     * (at your option) any later version. *
12     * *
13     * This program is distributed in the hope that it will be useful, *
14     * but WITHOUT ANY WARRANTY; without even the implied warranty of *
15     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
16     * GNU General Public License for more details. *
17     * *
18     * You should have received a copy of the GNU General Public License *
19     * along with this program; if not, write to the Free Software *
20     * Foundation, Inc., 59 Temple Place, Suite 330, Boston, *
21     * MA 02111-1307 USA *
22     ***************************************************************************/
23    
24     #include "AudioChannel.h"
25    
26 schoenebeck 1424 #include "../../common/global_private.h"
27 senoner 1481 #include "../../common/Thread.h" // needed for allocAlignedMem() and freeAlignedMem()
28 schoenebeck 1424
29 schoenebeck 361 #if defined(__APPLE__)
30     # include <stdlib.h>
31     #else
32     # include <malloc.h>
33     #endif
34    
35    
36 schoenebeck 200 namespace LinuxSampler {
37    
38     /**
39     * Create real channel.
40     *
41 schoenebeck 226 * @param ChannelNr - channel number of this new channel
42 schoenebeck 200 * @param BufferSize - desired audio data buffer size
43     */
44 schoenebeck 226 AudioChannel::AudioChannel(uint ChannelNr, uint BufferSize) {
45     this->ChannelNr = ChannelNr;
46 senoner 1481 this->pBuffer = (float *) Thread::allocAlignedMem(16,BufferSize*sizeof(float));
47 schoenebeck 200 this->uiBufferSize = BufferSize;
48     this->pMixChannel = NULL;
49     this->UsesExternalBuffer = false;
50    
51 schoenebeck 226 Parameters["NAME"] = new ParameterName("Channel " + ToString(ChannelNr));
52     Parameters["IS_MIX_CHANNEL"] = new ParameterIsMixChannel(false);
53 schoenebeck 200
54     Clear();
55     }
56    
57     /**
58     * Create channel with external (already existing) audio buffer.
59     *
60 schoenebeck 226 * @param ChannelNr - channel number of this new channel
61 schoenebeck 200 * @param pBuffer - external audio buffer
62 schoenebeck 1424 * @param BufferSize - size of the external buffer
63 schoenebeck 200 */
64 schoenebeck 226 AudioChannel::AudioChannel(uint ChannelNr, float* pBuffer, uint BufferSize) {
65     this->ChannelNr = ChannelNr;
66 schoenebeck 200 this->pBuffer = pBuffer;
67     this->uiBufferSize = BufferSize;
68     this->pMixChannel = NULL;
69     this->UsesExternalBuffer = true;
70    
71 schoenebeck 226 Parameters["NAME"] = new ParameterName("Channel " + ToString(ChannelNr));
72     Parameters["IS_MIX_CHANNEL"] = new ParameterIsMixChannel(false);
73 schoenebeck 200
74     Clear();
75     }
76    
77     /**
78     * Create mix channel.
79     *
80 schoenebeck 226 * @param ChannelNr - channel number of this new channel
81     * @param pMixChannelDestination - a real audio channel this new mix
82     * channel refers to
83 schoenebeck 200 */
84 schoenebeck 226 AudioChannel::AudioChannel(uint ChannelNr, AudioChannel* pMixChannelDestination) {
85     this->ChannelNr = ChannelNr;
86 senkov 331 this->pBuffer = pMixChannelDestination->Buffer();
87     this->uiBufferSize = pMixChannelDestination->uiBufferSize;
88     this->pMixChannel = pMixChannelDestination;
89 schoenebeck 200 this->UsesExternalBuffer = true;
90    
91 schoenebeck 226 Parameters["NAME"] = new ParameterName("Channel " + ToString(ChannelNr));
92     Parameters["IS_MIX_CHANNEL"] = new ParameterIsMixChannel(true);
93     //TODO: Parameters["MIX_CHANNEL_DESTINATION"] = new ParameterMixChannelDestination(dest_chan);
94 schoenebeck 200
95     Clear();
96     }
97    
98     /**
99     * Destructor
100     */
101     AudioChannel::~AudioChannel() {
102 schoenebeck 226 std::map<String,DeviceRuntimeParameter*>::iterator iter = Parameters.begin();
103     while (iter != Parameters.end()) { delete iter->second; iter++; }
104 senoner 1481 if (!UsesExternalBuffer) Thread::freeAlignedMem(pBuffer);
105 schoenebeck 200 }
106    
107 schoenebeck 1001 /**
108     * Copies audio data (unmodified) from this AudioChannel to the given
109     * destination AudioChannel.
110     *
111 schoenebeck 1037 * @e Caution: This method will overwrite the content in the destination
112     * channel buffer.
113     *
114 schoenebeck 1001 * @param pDst - destination channel
115     * @param Samples - amount of sample points to be copied
116     */
117     void AudioChannel::CopyTo(AudioChannel* pDst, const uint Samples) {
118 schoenebeck 1004 memcpy(pDst->Buffer(), Buffer(), Samples * sizeof(float));
119 schoenebeck 1001 }
120    
121     /**
122     * Copies audio data from this AudioChannel to the given destination
123     * AudioChannel and applies the given volume coefficient to the
124     * destination audio signal.
125     *
126 schoenebeck 1037 * @e Caution: This method will overwrite the content in the destination
127     * channel buffer.
128     *
129 schoenebeck 1001 * @param pDst - destination channel
130     * @param Samples - amount of sample points to be copied
131     * @param fLevel - volume coefficient to be applied
132     */
133     void AudioChannel::CopyTo(AudioChannel* pDst, const uint Samples, const float fLevel) {
134     if (fLevel == 1.0f) CopyTo(pDst, Samples);
135     else {
136 senoner 1913 #ifndef HAVE_GCC_VECTOR_EXTENSIONS
137 schoenebeck 1001 float* pSrcBuf = Buffer();
138     float* pDstBuf = pDst->Buffer();
139     for (int i = 0; i < Samples; i++)
140     pDstBuf[i] = pSrcBuf[i] * fLevel;
141 senoner 1913 #else
142     const v4sf vcoeff = { fLevel, fLevel, fLevel, fLevel };
143     const v4sf* src = static_cast<v4sf*>((void*)Buffer());
144     v4sf* dst = static_cast<v4sf*>((void*)pDst->Buffer());
145     const int cells = Samples / 4;
146     for (int i = 0; i < cells; ++i)
147     dst[i] = src[i] * vcoeff;
148     #endif
149 schoenebeck 1001 }
150     }
151    
152 schoenebeck 1037 /**
153     * Copies audio data (unmodified) from this AudioChannel and mixes it to the
154     * given destination AudioChannel.
155     *
156     * @param pDst - destination channel
157     * @param Samples - amount of sample points to be mixed over
158     */
159     void AudioChannel::MixTo(AudioChannel* pDst, const uint Samples) {
160 senoner 1913 #ifndef HAVE_GCC_VECTOR_EXTENSIONS
161 schoenebeck 1037 float* pSrcBuf = Buffer();
162     float* pDstBuf = pDst->Buffer();
163     for (int i = 0; i < Samples; i++)
164     pDstBuf[i] += pSrcBuf[i];
165 senoner 1913 #else
166     const v4sf* src = static_cast<v4sf*>((void*)Buffer());
167     v4sf* dst = static_cast<v4sf*>((void*)pDst->Buffer());
168     const int cells = Samples / 4;
169     for (int i = 0; i < cells; ++i)
170     dst[i] += src[i];
171     #endif
172 schoenebeck 1037 }
173    
174     /**
175     * Copies audio data from this AudioChannel, applies the given volume
176     * coefficient to the audio signal and mixes it to the given destination
177     * channel.
178     *
179     * @param pDst - destination channel
180     * @param Samples - amount of sample points to be mixed over
181     * @param fLevel - volume coefficient to be applied
182     */
183     void AudioChannel::MixTo(AudioChannel* pDst, const uint Samples, const float fLevel) {
184     if (fLevel == 1.0f) MixTo(pDst, Samples);
185     else {
186 senoner 1913 #ifndef HAVE_GCC_VECTOR_EXTENSIONS
187 schoenebeck 1037 float* pSrcBuf = Buffer();
188     float* pDstBuf = pDst->Buffer();
189     for (int i = 0; i < Samples; i++)
190     pDstBuf[i] += pSrcBuf[i] * fLevel;
191 senoner 1913 #else
192     const v4sf vcoeff = { fLevel, fLevel, fLevel, fLevel };
193     const v4sf* src = static_cast<v4sf*>((void*)Buffer());
194     v4sf* dst = static_cast<v4sf*>((void*)pDst->Buffer());
195     const int cells = Samples / 4;
196     for (int i = 0; i < cells; ++i)
197     dst[i] += src[i] * vcoeff;
198     #endif
199 schoenebeck 1037 }
200     }
201    
202 schoenebeck 200 std::map<String,DeviceRuntimeParameter*> AudioChannel::ChannelParameters() {
203 schoenebeck 226 return Parameters;
204 schoenebeck 200 }
205     }

  ViewVC Help
Powered by ViewVC