86afe4cb8d8b4046291815476c0b5a344d753537
[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, string name, DataType type)
46         : _session (s)
47         , _type(type)
48 {
49         assert(_name.find("/") == string::npos);
50
51         _name = name;
52         _timestamp = 0;
53         _length = 0;
54         _in_use = 0;
55 }
56
57 Source::Source (Session& s, const XMLNode& node) 
58         : _session (s)
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         assert(_name.find("/") == string::npos);
69 }
70
71 Source::~Source ()
72 {
73         notify_callbacks ();
74 }
75
76 XMLNode&
77 Source::get_state ()
78 {
79         XMLNode *node = new XMLNode ("Source");
80         char buf[64];
81
82         node->add_property ("name", _name);
83         node->add_property ("type", _type.to_string());
84         _id.print (buf, sizeof (buf));
85         node->add_property ("id", buf);
86
87         if (_timestamp != 0) {
88                 snprintf (buf, sizeof (buf), "%ld", _timestamp);
89                 node->add_property ("timestamp", buf);
90         }
91
92         return *node;
93 }
94
95 int
96 Source::set_state (const XMLNode& node)
97 {
98         const XMLProperty* prop;
99
100         if ((prop = node.property ("name")) != 0) {
101                 _name = prop->value();
102         } else {
103                 return -1;
104         }
105         
106         if ((prop = node.property ("id")) != 0) {
107                 _id = prop->value ();
108         } else {
109                 return -1;
110         }
111
112         if ((prop = node.property ("type")) != 0) {
113                 _type = DataType(prop->value());
114         }
115
116         if ((prop = node.property ("timestamp")) != 0) {
117                 sscanf (prop->value().c_str(), "%ld", &_timestamp);
118         }
119         assert(_name.find("/") == string::npos);
120
121         return 0;
122 }
123
124 void
125 Source::update_length (jack_nframes_t pos, jack_nframes_t cnt)
126 {
127         if (pos + cnt > _length) {
128                 _length = pos+cnt;
129         }
130 }
131
132 void
133 Source::add_playlist (boost::shared_ptr<Playlist> pl)
134 {
135         std::pair<PlaylistMap::iterator,bool> res;
136         std::pair<boost::shared_ptr<Playlist>, uint32_t> newpair (pl, 1);
137         Glib::Mutex::Lock lm (playlist_lock);
138
139         res = _playlists.insert (newpair);
140
141         if (!res.second) {
142                 /* it already existed, bump count */
143                 res.first->second++;
144         }
145                 
146         pl->GoingAway.connect (bind (mem_fun (*this, &Source::remove_playlist), boost::weak_ptr<Playlist> (pl)));
147 }
148
149 void
150 Source::remove_playlist (boost::weak_ptr<Playlist> wpl)
151 {
152         boost::shared_ptr<Playlist> pl (wpl.lock());
153
154         if (!pl) {
155                 return;
156         }
157
158         PlaylistMap::iterator x;
159         Glib::Mutex::Lock lm (playlist_lock);
160
161         if ((x = _playlists.find (pl)) != _playlists.end()) {
162                 if (x->second > 1) {
163                         x->second--;
164                 } else {
165                         _playlists.erase (x);
166                 }
167         }
168 }
169
170 uint32_t
171 Source::used () const
172 {
173         return _playlists.size();
174 }