/[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 2377 - (hide annotations) (download)
Thu Oct 4 18:16:26 2012 UTC (11 years, 6 months ago) by schoenebeck
File size: 9630 byte(s)
* Various "const" and "restrict" optimizations.

1 schoenebeck 200 /***************************************************************************
2     * *
3     * LinuxSampler - modular, streaming capable sampler *
4     * *
5     * Copyright (C) 2003, 2004 by Benno Senoner and Christian Schoenebeck *
6 schoenebeck 2377 * Copyright (C) 2005 - 2012 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 2377 memcpy((float* __restrict)pDst->Buffer(), (const float* __restrict)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 schoenebeck 2377 const float* __restrict pSrcBuf = Buffer();
137     float* __restrict pDstBuf = pDst->Buffer();
138 schoenebeck 1930 #if HAVE_GCC_VECTOR_EXTENSIONS
139     if ((size_t)pSrcBuf % 16 == 0 && (size_t)pDstBuf % 16 == 0) {
140     const v4sf vcoeff = { fLevel, fLevel, fLevel, fLevel };
141 schoenebeck 2377 const v4sf* __restrict src =
142     static_cast<const v4sf* __restrict>(
143     (const void* __restrict)pSrcBuf
144     );
145     v4sf* __restrict dst =
146     static_cast<v4sf* __restrict>(
147     (void* __restrict)pDstBuf
148     );
149 schoenebeck 1930 const int cells = Samples / 4;
150     for (int i = 0; i < cells; ++i)
151     dst[i] = src[i] * vcoeff;
152     } else {
153     #endif
154     for (int i = 0; i < Samples; i++)
155     pDstBuf[i] = pSrcBuf[i] * fLevel;
156     #if HAVE_GCC_VECTOR_EXTENSIONS
157     }
158     #endif
159 schoenebeck 1001 }
160     }
161    
162 schoenebeck 1037 /**
163     * Copies audio data (unmodified) from this AudioChannel and mixes it to the
164     * given destination AudioChannel.
165     *
166     * @param pDst - destination channel
167     * @param Samples - amount of sample points to be mixed over
168     */
169     void AudioChannel::MixTo(AudioChannel* pDst, const uint Samples) {
170 schoenebeck 2377 const float* __restrict pSrcBuf = Buffer();
171     float* __restrict pDstBuf = pDst->Buffer();
172 schoenebeck 1930 #if HAVE_GCC_VECTOR_EXTENSIONS
173     if ((size_t)pSrcBuf % 16 == 0 && (size_t)pDstBuf % 16 == 0) {
174 schoenebeck 2377 const v4sf* __restrict src =
175     static_cast<const v4sf* __restrict>(
176     (const void* __restrict)pSrcBuf
177     );
178     v4sf* __restrict dst =
179     static_cast<v4sf* __restrict>(
180     (void* __restrict)pDstBuf
181     );
182 schoenebeck 1930 const int cells = Samples / 4;
183     for (int i = 0; i < cells; ++i)
184     dst[i] += src[i];
185     } else {
186     #endif
187     for (int i = 0; i < Samples; i++)
188     pDstBuf[i] += pSrcBuf[i];
189     #if HAVE_GCC_VECTOR_EXTENSIONS
190     }
191     #endif
192 schoenebeck 1037 }
193    
194     /**
195     * Copies audio data from this AudioChannel, applies the given volume
196     * coefficient to the audio signal and mixes it to the given destination
197     * channel.
198     *
199     * @param pDst - destination channel
200     * @param Samples - amount of sample points to be mixed over
201     * @param fLevel - volume coefficient to be applied
202     */
203     void AudioChannel::MixTo(AudioChannel* pDst, const uint Samples, const float fLevel) {
204     if (fLevel == 1.0f) MixTo(pDst, Samples);
205     else {
206 schoenebeck 2377 const float* __restrict pSrcBuf = Buffer();
207     float* __restrict pDstBuf = pDst->Buffer();
208 schoenebeck 1930 #if HAVE_GCC_VECTOR_EXTENSIONS
209     if ((size_t)pSrcBuf % 16 == 0 && (size_t)pDstBuf % 16 == 0) {
210     const v4sf vcoeff = { fLevel, fLevel, fLevel, fLevel };
211 schoenebeck 2377 const v4sf* __restrict src =
212     static_cast<const v4sf* __restrict>(
213     (const void* __restrict)pSrcBuf
214     );
215     v4sf* __restrict dst =
216     static_cast<v4sf* __restrict>(
217     (void* __restrict)pDstBuf
218     );
219 schoenebeck 1930 const int cells = Samples / 4;
220     for (int i = 0; i < cells; ++i)
221     dst[i] += src[i] * vcoeff;
222     } else {
223     #endif
224     for (int i = 0; i < Samples; i++)
225     pDstBuf[i] += pSrcBuf[i] * fLevel;
226     #if HAVE_GCC_VECTOR_EXTENSIONS
227     }
228     #endif
229 schoenebeck 1037 }
230     }
231    
232 schoenebeck 200 std::map<String,DeviceRuntimeParameter*> AudioChannel::ChannelParameters() {
233 schoenebeck 226 return Parameters;
234 schoenebeck 200 }
235     }

  ViewVC Help
Powered by ViewVC