Use ARDOUR::user_config_directory in ARDOUR::user_template_directory
[ardour.git] / libs / ardour / source.cc
1 /*
2     Copyright (C) 2000 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 <sys/stat.h>
21 #include <unistd.h>
22 #include <fcntl.h>
23 #include <poll.h>
24 #include <float.h>
25 #include <cerrno>
26 #include <ctime>
27 #include <cmath>
28 #include <iomanip>
29 #include <algorithm>
30
31 #include <glibmm/thread.h>
32 #include <pbd/xml++.h>
33 #include <pbd/pthread_utils.h>
34
35 #include <ardour/source.h>
36 #include <ardour/playlist.h>
37
38 #include "i18n.h"
39
40 using std::min;
41 using std::max;
42
43 using namespace ARDOUR;
44
45 Source::Source (Session& s, const string& name, DataType type)
46         : SessionObject(s, name)
47         , _type(type)
48 {
49         // not true.. is this supposed to be an assertion?
50         //assert(_name.find("/") == string::npos);
51
52         _timestamp = 0;
53         _length = 0;
54         _in_use = 0;
55 }
56
57 Source::Source (Session& s, const XMLNode& node) 
58         : SessionObject(s, "unnamed source")
59         , _type(DataType::AUDIO)
60 {
61         _timestamp = 0;
62         _length = 0;
63         _in_use = 0;
64
65         if (set_state (node) || _type == DataType::NIL) {
66                 throw failed_constructor();
67         }
68 }
69
70 Source::~Source ()
71 {
72         notify_callbacks ();
73 }
74
75 XMLNode&
76 Source::get_state ()
77 {
78         XMLNode *node = new XMLNode ("Source");
79         char buf[64];
80
81         node->add_property ("name", _name);
82         node->add_property ("type", _type.to_string());
83         _id.print (buf, sizeof (buf));
84         node->add_property ("id", buf);
85
86         if (_timestamp != 0) {
87                 snprintf (buf, sizeof (buf), "%ld", _timestamp);
88                 node->add_property ("timestamp", buf);
89         }
90
91         return *node;
92 }
93
94 int
95 Source::set_state (const XMLNode& node)
96 {
97         const XMLProperty* prop;
98
99         if ((prop = node.property ("name")) != 0) {
100                 _name = prop->value();
101         } else {
102                 return -1;
103         }
104         
105         if ((prop = node.property ("id")) != 0) {
106                 _id = prop->value ();
107         } else {
108                 return -1;
109         }
110
111         if ((prop = node.property ("type")) != 0) {
112                 _type = DataType(prop->value());
113         }
114
115         if ((prop = node.property ("timestamp")) != 0) {
116                 sscanf (prop->value().c_str(), "%ld", &_timestamp);
117         }
118         
119         // Don't think this is valid, absolute paths fail
120         //assert(_name.find("/") == string::npos);
121
122         return 0;
123 }
124
125 void
126 Source::update_length (nframes_t pos, nframes_t cnt)
127 {
128         if (pos + cnt > _length) {
129                 _length = pos+cnt;
130         }
131 }
132
133 void
134 Source::add_playlist (boost::shared_ptr<Playlist> pl)
135 {
136         std::pair<PlaylistMap::iterator,bool> res;
137         std::pair<boost::shared_ptr<Playlist>, uint32_t> newpair (pl, 1);
138         Glib::Mutex::Lock lm (playlist_lock);
139
140         res = _playlists.insert (newpair);
141
142         if (!res.second) {
143                 /* it already existed, bump count */
144                 res.first->second++;
145         }
146                 
147         pl->GoingAway.connect (bind (mem_fun (*this, &Source::remove_playlist), boost::weak_ptr<Playlist> (pl)));
148 }
149
150 void
151 Source::remove_playlist (boost::weak_ptr<Playlist> wpl)
152 {
153         boost::shared_ptr<Playlist> pl (wpl.lock());
154
155         if (!pl) {
156                 return;
157         }
158
159         PlaylistMap::iterator x;
160         Glib::Mutex::Lock lm (playlist_lock);
161
162         if ((x = _playlists.find (pl)) != _playlists.end()) {
163                 if (x->second > 1) {
164                         x->second--;
165                 } else {
166                         _playlists.erase (x);
167                 }
168         }
169 }
170
171 uint32_t
172 Source::used () const
173 {
174         return _playlists.size();
175 }