midway snapshot of work done on managing Region & Source lifetimes correctly. may...
[ardour.git] / libs / ardour / session_playlists.cc
1 /*
2     Copyright (C) 2009 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 #include <vector>
20
21 #include "pbd/xml++.h"
22 #include "pbd/compose.h"
23 #include "ardour/debug.h"
24 #include "ardour/session_playlists.h"
25 #include "ardour/playlist.h"
26 #include "ardour/region.h"
27 #include "ardour/playlist_factory.h"
28 #include "ardour/session.h"
29 #include "i18n.h"
30
31 using namespace std;
32 using namespace PBD;
33 using namespace ARDOUR;
34
35 SessionPlaylists::~SessionPlaylists ()
36 {
37         DEBUG_TRACE (DEBUG::Destruction, "delete playlists\n");
38         
39         for (List::iterator i = playlists.begin(); i != playlists.end(); ) {
40                 SessionPlaylists::List::iterator tmp;
41
42                 tmp = i;
43                 ++tmp;
44
45                 DEBUG_TRACE(DEBUG::Destruction, string_compose ("Dropping for used playlist %1 ; pre-ref = %2\n", (*i)->name(), (*i).use_count()));
46                 boost::shared_ptr<Playlist> keeper (*i);
47                 (*i)->drop_references ();
48
49                 i = tmp;
50         }
51
52         DEBUG_TRACE (DEBUG::Destruction, "delete unused playlists\n");
53         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ) {
54                 List::iterator tmp;
55
56                 tmp = i;
57                 ++tmp;
58
59                 DEBUG_TRACE(DEBUG::Destruction, string_compose ("Dropping for unused playlist %1 ; pre-ref = %2\n", (*i)->name(), (*i).use_count()));
60                 boost::shared_ptr<Playlist> keeper (*i);
61                 (*i)->drop_references ();
62
63                 i = tmp;
64         }
65
66         playlists.clear ();
67         unused_playlists.clear ();
68 }
69
70 bool
71 SessionPlaylists::add (boost::shared_ptr<Playlist> playlist)
72 {
73         Glib::Mutex::Lock lm (lock);
74
75         bool const existing = find (playlists.begin(), playlists.end(), playlist) != playlists.end();
76
77         if (!existing) {
78                 playlists.insert (playlists.begin(), playlist);
79                 playlist->InUse.connect_same_thread (*this, boost::bind (&SessionPlaylists::track, this, _1, boost::weak_ptr<Playlist>(playlist)));
80         }
81
82         return existing;
83 }
84
85 void
86 SessionPlaylists::remove (boost::shared_ptr<Playlist> playlist)
87 {
88         Glib::Mutex::Lock lm (lock);
89
90         List::iterator i;
91
92         i = find (playlists.begin(), playlists.end(), playlist);
93         if (i != playlists.end()) {
94                 playlists.erase (i);
95         }
96
97         i = find (unused_playlists.begin(), unused_playlists.end(), playlist);
98         if (i != unused_playlists.end()) {
99                 unused_playlists.erase (i);
100         }
101 }
102         
103
104 void
105 SessionPlaylists::track (bool inuse, boost::weak_ptr<Playlist> wpl)
106 {
107         boost::shared_ptr<Playlist> pl(wpl.lock());
108
109         if (!pl) {
110                 return;
111         }
112
113         List::iterator x;
114
115         if (pl->hidden()) {
116                 /* its not supposed to be visible */
117                 return;
118         }
119
120         {
121                 Glib::Mutex::Lock lm (lock);
122
123                 if (!inuse) {
124
125                         unused_playlists.insert (pl);
126
127                         if ((x = playlists.find (pl)) != playlists.end()) {
128                                 playlists.erase (x);
129                         }
130
131
132                 } else {
133
134                         playlists.insert (pl);
135
136                         if ((x = unused_playlists.find (pl)) != unused_playlists.end()) {
137                                 unused_playlists.erase (x);
138                         }
139                 }
140         }
141 }
142
143 uint32_t
144 SessionPlaylists::n_playlists () const
145 {
146         Glib::Mutex::Lock lm (lock);
147         return playlists.size();
148 }
149
150 boost::shared_ptr<Playlist>
151 SessionPlaylists::by_name (string name)
152 {
153         Glib::Mutex::Lock lm (lock);
154
155         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i) {
156                 if ((*i)->name() == name) {
157                         return* i;
158                 }
159         }
160         
161         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ++i) {
162                 if ((*i)->name() == name) {
163                         return* i;
164                 }
165         }
166
167         return boost::shared_ptr<Playlist>();
168 }
169
170 boost::shared_ptr<Playlist>
171 SessionPlaylists::by_id (const PBD::ID& id)
172 {
173         Glib::Mutex::Lock lm (lock);
174
175         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i) {
176                 if ((*i)->id() == id) {
177                         return* i;
178                 }
179         }
180         
181         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ++i) {
182                 if ((*i)->id() == id) {
183                         return* i;
184                 }
185         }
186
187         return boost::shared_ptr<Playlist>();
188 }
189
190 void
191 SessionPlaylists::unassigned (std::list<boost::shared_ptr<Playlist> > & list)
192 {
193         Glib::Mutex::Lock lm (lock);
194
195         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i) {
196                 if (!(*i)->get_orig_diskstream_id().to_s().compare ("0")) {
197                         list.push_back (*i);
198                 }
199         }
200         
201         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ++i) {
202                 if (!(*i)->get_orig_diskstream_id().to_s().compare ("0")) {
203                         list.push_back (*i);
204                 }
205         }
206 }
207
208 void
209 SessionPlaylists::get (vector<boost::shared_ptr<Playlist> >& s)
210 {
211         Glib::Mutex::Lock lm (lock);
212
213         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i) {
214                 s.push_back (*i);
215         }
216         
217         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ++i) {
218                 s.push_back (*i);
219         }
220 }
221
222 void
223 SessionPlaylists::destroy_region (boost::shared_ptr<Region> r)
224 {
225         Glib::Mutex::Lock lm (lock);
226
227         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i) {
228                 (*i)->destroy_region (r);
229         }
230         
231         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ++i) {
232                 (*i)->destroy_region (r);
233         }
234 }
235
236
237 void
238 SessionPlaylists::find_equivalent_playlist_regions (boost::shared_ptr<Region> region, vector<boost::shared_ptr<Region> >& result)
239 {
240         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i)
241                 (*i)->get_region_list_equivalent_regions (region, result);
242 }
243
244 /** Return the number of playlists (not regions) that contain @a src */
245 uint32_t
246 SessionPlaylists::source_use_count (boost::shared_ptr<const Source> src) const
247 {
248         uint32_t count = 0;
249         for (List::const_iterator p = playlists.begin(); p != playlists.end(); ++p) {
250                 for (Playlist::RegionList::const_iterator r = (*p)->region_list().begin();
251                                 r != (*p)->region_list().end(); ++r) {
252                         if ((*r)->uses_source(src)) {
253                                 ++count;
254                                 break;
255                         }
256                 }
257         }
258         return count;
259 }
260
261
262 void
263 SessionPlaylists::update_after_tempo_map_change ()
264 {
265         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i) {
266                 (*i)->update_after_tempo_map_change ();
267         }
268
269         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ++i) {
270                 (*i)->update_after_tempo_map_change ();
271         }
272 }
273
274 void
275 SessionPlaylists::add_state (XMLNode* node, bool full_state)
276 {
277         XMLNode* child = node->add_child ("Playlists");
278         for (List::iterator i = playlists.begin(); i != playlists.end(); ++i) {
279                 if (!(*i)->hidden()) {
280                         if (full_state) {
281                                 child->add_child_nocopy ((*i)->get_state());
282                         } else {
283                                 child->add_child_nocopy ((*i)->get_template());
284                         }
285                 }
286         }
287
288         child = node->add_child ("UnusedPlaylists");
289         for (List::iterator i = unused_playlists.begin(); i != unused_playlists.end(); ++i) {
290                 if (!(*i)->hidden()) {
291                         if (!(*i)->empty()) {
292                                 if (full_state) {
293                                         child->add_child_nocopy ((*i)->get_state());
294                                 } else {
295                                         child->add_child_nocopy ((*i)->get_template());
296                                 }
297                         }
298                 }
299         }
300 }
301
302 /** @return true for `stop cleanup', otherwise false */
303 bool
304 SessionPlaylists::maybe_delete_unused (boost::function<int(boost::shared_ptr<Playlist>)> ask)
305 {
306         vector<boost::shared_ptr<Playlist> > playlists_tbd;
307
308         for (List::iterator x = unused_playlists.begin(); x != unused_playlists.end(); ++x) {
309
310                 int status = ask (*x);
311
312                 switch (status) {
313                 case -1:
314                         return true;
315
316                 case 0:
317                         playlists_tbd.push_back (*x);
318                         break;
319
320                 default:
321                         /* leave it alone */
322                         break;
323                 }
324         }
325
326         /* now delete any that were marked for deletion */
327
328         for (vector<boost::shared_ptr<Playlist> >::iterator x = playlists_tbd.begin(); x != playlists_tbd.end(); ++x) {
329                 boost::shared_ptr<Playlist> keeper (*x);
330                 (*x)->drop_references ();
331         }
332
333         playlists_tbd.clear ();
334
335         return false;
336 }
337
338 int
339 SessionPlaylists::load (Session& session, const XMLNode& node)
340 {
341         XMLNodeList nlist;
342         XMLNodeConstIterator niter;
343         boost::shared_ptr<Playlist> playlist;
344
345         nlist = node.children();
346
347         for (niter = nlist.begin(); niter != nlist.end(); ++niter) {
348
349                 if ((playlist = XMLPlaylistFactory (session, **niter)) == 0) {
350                         error << _("Session: cannot create Playlist from XML description.") << endmsg;
351                 }
352         }
353
354         return 0;
355 }
356
357 int
358 SessionPlaylists::load_unused (Session& session, const XMLNode& node)
359 {
360         XMLNodeList nlist;
361         XMLNodeConstIterator niter;
362         boost::shared_ptr<Playlist> playlist;
363
364         nlist = node.children();
365
366         for (niter = nlist.begin(); niter != nlist.end(); ++niter) {
367
368                 if ((playlist = XMLPlaylistFactory (session, **niter)) == 0) {
369                         error << _("Session: cannot create Playlist from XML description.") << endmsg;
370                         continue;
371                 }
372
373                 // now manually untrack it
374
375                 track (false, boost::weak_ptr<Playlist> (playlist));
376         }
377
378         return 0;
379 }
380
381 boost::shared_ptr<Playlist>
382 SessionPlaylists::XMLPlaylistFactory (Session& session, const XMLNode& node)
383 {
384         try {
385                 return PlaylistFactory::create (session, node);
386         }
387
388         catch (failed_constructor& err) {
389                 return boost::shared_ptr<Playlist>();
390         }
391 }
392