2 * Copyright (c) 2011 Sveriges Television AB <info@casparcg.com>
4 * This file is part of CasparCG (www.casparcg.com).
6 * CasparCG is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
11 * CasparCG is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with CasparCG. If not, see <http://www.gnu.org/licenses/>.
19 * Author: Helge Norberg, helge.norberg@svt.se
24 #include "semaphore.h"
26 #include <boost/thread/mutex.hpp>
27 #include <boost/noncopyable.hpp>
32 * Adapts an unbounded non-blocking concurrent queue into a blocking bounded
35 * The queue Q to adapt must support the following use cases:
47 * It must also guarantee thread safety for those operations.
50 class blocking_bounded_queue_adapter : boost::noncopyable
53 typedef typename Q::value_type value_type;
54 typedef unsigned int size_type;
56 semaphore space_available_;
57 semaphore elements_available_;
59 mutable boost::mutex capacity_mutex_;
65 * @param capacity The capacity of the queue.
67 blocking_bounded_queue_adapter(size_type capacity)
68 : space_available_(capacity)
69 , elements_available_(0)
75 * Push an element to the queue, block until room is available.
77 * @param element The element to push.
79 void push(const value_type& element)
81 space_available_.acquire();
82 push_after_room_reserved(element);
86 * Try to push an element to the queue, returning immediately if room is not
89 * @param element The element to push.
91 * @return true if there was room for the element.
93 bool try_push(const value_type& element)
95 bool room_available = space_available_.try_acquire();
100 push_after_room_reserved(element);
106 * Pop an element from the queue, will block until an element is available.
108 * @param element The element to store the result in.
110 void pop(value_type& element)
112 elements_available_.acquire();
113 queue_.try_pop(element);
114 space_available_.release();
118 * Try to pop an element from the queue, returning immediately if no
119 * element is available.
121 * @param element The element to store the result in.
123 * @return true if an element was popped.
125 bool try_pop(value_type& element)
127 if (!elements_available_.try_acquire())
130 queue_.try_pop(element);
131 space_available_.release();
137 * Modify the capacity of the queue. May block if reducing the capacity.
139 * @param capacity The new capacity.
141 void set_capacity(size_type capacity)
143 boost::mutex::scoped_lock lock (capacity_mutex_);
145 if (capacity_ < capacity)
147 auto to_grow_with = capacity - capacity_;
149 space_available_.release(to_grow_with);
151 else if (capacity_ > capacity)
153 auto to_shrink_with = capacity_ - capacity;
155 // Will block until the desired capacity has been reached.
156 space_available_.acquire(to_shrink_with);
159 capacity_ = capacity;
163 * @return the current capacity of the queue.
165 size_type capacity() const
167 boost::mutex::scoped_lock lock (capacity_mutex_);
173 * @return the current size of the queue (may have changed at the time of
176 size_type size() const
178 return elements_available_.permits();
181 void push_after_room_reserved(const value_type& element)
185 queue_.push(element);
189 space_available_.release();
194 elements_available_.release();