merge and resolve conflict with master
[ardour.git] / libs / ardour / ardour / butler.h
1 /*
2     Copyright (C) 2000-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
20 #ifndef __ardour_butler_h__
21 #define __ardour_butler_h__
22
23 #include <pthread.h>
24
25 #include <glibmm/threads.h>
26
27 #ifdef PLATFORM_WINDOWS
28 #include "pbd/glib_semaphore.h"
29 #endif
30
31 #include "pbd/ringbuffer.h"
32 #include "pbd/pool.h"
33 #include "ardour/types.h"
34 #include "ardour/session_handle.h"
35
36
37
38 namespace ARDOUR {
39
40 /**
41  *  One of the Butler's functions is to clean up (ie delete) unused CrossThreadPools.
42  *  When a thread with a CrossThreadPool terminates, its CTP is added to pool_trash.
43  *  When the Butler thread wakes up, we check this trash buffer for CTPs, and if they
44  *  are empty they are deleted.
45  */
46
47 class Butler : public SessionHandleRef
48 {
49   public:
50         Butler (Session& session);
51         ~Butler();
52
53         int  start_thread();
54         void terminate_thread();
55         void schedule_transport_work();
56         void summon();
57         void stop();
58         void wait_until_finished();
59         bool transport_work_requested() const;
60         void drop_references ();
61
62         framecnt_t audio_diskstream_capture_buffer_size() const { return audio_dstream_capture_buffer_size; }
63         framecnt_t audio_diskstream_playback_buffer_size() const { return audio_dstream_playback_buffer_size; }
64         uint32_t midi_diskstream_buffer_size()  const { return midi_dstream_buffer_size; }
65
66         static void* _thread_work(void *arg);
67         void*         thread_work();
68
69         struct Request {
70                 enum Type {
71                         Run,
72                         Pause,
73                         Quit
74                 };
75         };
76
77         pthread_t    thread;
78         Glib::Threads::Mutex  request_lock;
79         Glib::Threads::Cond   paused;
80         bool         should_run;
81         mutable gint should_do_transport_work;
82         framecnt_t   audio_dstream_capture_buffer_size;
83         framecnt_t   audio_dstream_playback_buffer_size;
84         uint32_t     midi_dstream_buffer_size;
85         RingBuffer<CrossThreadPool*> pool_trash;
86
87 #ifdef PLATFORM_WINDOWS
88         PBD::atomic_counter m_request_state;
89         PBD::GlibSemaphore   m_request_sem;
90 #else
91         int          request_pipe[2];
92 #endif
93
94 private:
95         void empty_pool_trash ();
96         void config_changed (std::string);
97
98 #ifndef PLATFORM_WINDOWS
99         int setup_request_pipe ();
100 #endif
101
102         /**
103          * return true if there are requests to be processed
104          */
105         bool wait_for_requests ();
106
107         /**
108          * Remove request from butler request queue
109          *
110          * return true if there was another request and req is valid
111          */
112         bool dequeue_request (Request::Type& req);
113
114         /**
115          * Add request to butler thread request queue
116          */
117         void queue_request (Request::Type r);
118
119 };
120
121 } // namespace ARDOUR
122
123 #endif // __ardour_butler_h__