Fixed compilation of coreaudiosource.cc
[ardour.git] / libs / ardour / coreaudiosource.cc
1 /*
2     Copyright (C) 2006 Paul Davis 
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include <pbd/error.h>
21 #include <ardour/coreaudiosource.h>
22
23 #include <appleutility/CAAudioFile.h>
24 #include <appleutility/CAStreamBasicDescription.h>
25
26 #include "i18n.h"
27
28 #include <AudioToolbox/AudioFormat.h>
29
30 using namespace ARDOUR;
31 using namespace PBD;
32
33 CoreAudioSource::CoreAudioSource (Session& s, const XMLNode& node)
34         : AudioFileSource (s, node)
35 {
36         init (_name);
37 }
38
39 CoreAudioSource::CoreAudioSource (Session& s, const string& idstr, Flag flags)
40         : AudioFileSource(s, idstr, flags)
41 {
42         init (idstr);
43 }
44
45 void 
46 CoreAudioSource::init (const string& idstr)
47 {
48         string::size_type pos;
49
50         tmpbuf = 0;
51         tmpbufsize = 0;
52
53         _name = idstr;
54
55         if ((pos = idstr.find_last_of (':')) == string::npos) {
56                 channel = 0;
57                 _path = idstr;
58         } else {
59                 channel = atoi (idstr.substr (pos+1).c_str());
60                 _path = idstr.substr (0, pos);
61         }
62
63         cerr << "CoreAudioSource::init() " << name() << endl;
64         
65         /* note that we temporarily truncated _id at the colon */
66         try {
67                 af.Open(_path.c_str());
68
69                 CAStreamBasicDescription file_asbd (af.GetFileDataFormat());
70                 n_channels = file_asbd.NumberChannels();
71                 cerr << "number of channels: " << n_channels << endl;
72                 
73                 if (channel >= n_channels) {
74                         error << string_compose("CoreAudioSource: file only contains %1 channels; %2 is invalid as a channel number (%3)", n_channels, channel, name()) << endmsg;
75                         throw failed_constructor();
76                 }
77
78                 _length = af.GetNumberFrames();
79
80                 CAStreamBasicDescription client_asbd(file_asbd);
81                 client_asbd.SetCanonical(client_asbd.NumberChannels(), false);
82                 af.SetClientFormat (client_asbd);
83         } catch (CAXException& cax) {
84                 error << string_compose ("CoreAudioSource: %1 (%2)", cax.mOperation, name()) << endmsg;
85                 throw failed_constructor ();
86         }
87         
88         if (_build_peakfiles) {
89                 if (initialize_peakfile (false, _path)) {
90                         error << string_compose("CoreAudioSource: initialize peakfile failed (%1)", name()) << endmsg;
91                         throw failed_constructor ();
92                 }
93         }
94 }
95
96 CoreAudioSource::~CoreAudioSource ()
97 {
98         cerr << "CoreAudioSource::~CoreAudioSource() " << name() << endl;
99         GoingAway (); /* EMIT SIGNAL */
100
101         if (tmpbuf) {
102                 delete [] tmpbuf;
103         }
104         
105         cerr << "deletion done" << endl;
106 }
107
108 jack_nframes_t
109 CoreAudioSource::read_unlocked (Sample *dst, jack_nframes_t start, jack_nframes_t cnt) const
110 {
111         try {
112                 af.Seek (start);
113         } catch (CAXException& cax) {
114                 error << string_compose("CoreAudioSource: %1 to %2 (%3)", cax.mOperation, start, _name.substr (1)) << endmsg;
115                 return 0;
116         }
117
118         AudioBufferList abl;
119         abl.mNumberBuffers = 1;
120         abl.mBuffers[0].mNumberChannels = n_channels;
121
122         UInt32 new_cnt = cnt;
123         if (n_channels == 1) {
124                 abl.mBuffers[0].mDataByteSize = cnt * sizeof(Sample);
125                 abl.mBuffers[0].mData = dst;
126                 try {
127                         af.Read (new_cnt, &abl);
128                 } catch (CAXException& cax) {
129                         error << string_compose("CoreAudioSource: %1 (%2)", cax.mOperation, _name);
130                 }
131                 _read_data_count = new_cnt * sizeof(float);
132                 return new_cnt;
133         }
134
135         UInt32 real_cnt = cnt * n_channels;
136
137         {
138                 Glib::Mutex::Lock lm (_tmpbuf_lock);
139                 
140                 if (tmpbufsize < real_cnt) {
141                         
142                         if (tmpbuf) {
143                                 delete [] tmpbuf;
144                         }
145                         tmpbufsize = real_cnt;
146                         tmpbuf = new float[tmpbufsize];
147                 }
148
149                 abl.mBuffers[0].mDataByteSize = tmpbufsize * sizeof(Sample);
150                 abl.mBuffers[0].mData = tmpbuf;
151
152                 cerr << "channel: " << channel << endl;
153                 
154                 try {
155                         af.Read (real_cnt, &abl);
156                 } catch (CAXException& cax) {
157                         error << string_compose("CoreAudioSource: %1 (%2)", cax.mOperation, _name);
158                 }
159                 float *ptr = tmpbuf + channel;
160                 real_cnt /= n_channels;
161                 
162                 /* stride through the interleaved data */
163                 
164                 for (uint32_t n = 0; n < real_cnt; ++n) {
165                         dst[n] = *ptr;
166                         ptr += n_channels;
167                 }
168         }
169
170         _read_data_count = cnt * sizeof(float);
171                 
172         return real_cnt;
173 }
174
175 float
176 CoreAudioSource::sample_rate() const
177 {
178         CAStreamBasicDescription client_asbd;
179
180         try {
181                 client_asbd = af.GetClientDataFormat ();
182         } catch (CAXException& cax) {
183                 error << string_compose("CoreAudioSource: %1 (%2)", cax.mOperation, _name);
184                 return 0.0;
185         }
186
187         return client_asbd.mSampleRate;
188 }
189
190 int
191 CoreAudioSource::update_header (jack_nframes_t when, struct tm&, time_t)
192 {
193         return 0;
194 }