]> sigrok.org Git - pulseview.git/blame_incremental - pv/session.hpp
Session: Enable logic data acquisition using gstreamer
[pulseview.git] / pv / session.hpp
... / ...
CommitLineData
1/*
2 * This file is part of the PulseView project.
3 *
4 * Copyright (C) 2012-14 Joel Holdsworth <joel@airwebreathe.org.uk>
5 *
6 * This program 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 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program 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.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, see <http://www.gnu.org/licenses/>.
18 */
19
20#ifndef PULSEVIEW_PV_SESSION_HPP
21#define PULSEVIEW_PV_SESSION_HPP
22
23#ifdef ENABLE_FLOW
24#include <atomic>
25#include <condition_variable>
26#endif
27
28#include <functional>
29#include <map>
30#include <memory>
31#include <mutex>
32#include <set>
33#include <string>
34#include <thread>
35#include <unordered_set>
36#include <vector>
37
38#include <QObject>
39#include <QSettings>
40#include <QString>
41
42#ifdef ENABLE_FLOW
43#include <gstreamermm.h>
44#include <libsigrokflow/libsigrokflow.hpp>
45#endif
46
47#include "util.hpp"
48#include "views/viewbase.hpp"
49
50
51using std::function;
52using std::list;
53using std::map;
54using std::mutex;
55using std::recursive_mutex;
56using std::shared_ptr;
57using std::string;
58using std::unordered_set;
59
60#ifdef ENABLE_FLOW
61using Glib::RefPtr;
62using Gst::AppSink;
63using Gst::Element;
64using Gst::Pipeline;
65#endif
66
67struct srd_decoder;
68struct srd_channel;
69
70namespace sigrok {
71class Analog;
72class Channel;
73class Device;
74class InputFormat;
75class Logic;
76class Meta;
77class Option;
78class OutputFormat;
79class Packet;
80class Session;
81} // namespace sigrok
82
83using sigrok::Option;
84
85namespace pv {
86
87class DeviceManager;
88
89namespace data {
90class Analog;
91class AnalogSegment;
92class DecodeSignal;
93class Logic;
94class LogicSegment;
95class SignalBase;
96class SignalData;
97}
98
99namespace devices {
100class Device;
101}
102
103namespace toolbars {
104class MainBar;
105}
106
107namespace views {
108class ViewBase;
109}
110
111class Session : public QObject
112{
113 Q_OBJECT
114
115public:
116 enum capture_state {
117 Stopped,
118 AwaitingTrigger,
119 Running
120 };
121
122 static shared_ptr<sigrok::Context> sr_context;
123
124public:
125 Session(DeviceManager &device_manager, QString name);
126
127 ~Session();
128
129 DeviceManager& device_manager();
130
131 const DeviceManager& device_manager() const;
132
133 shared_ptr<sigrok::Session> session() const;
134
135 shared_ptr<devices::Device> device() const;
136
137 QString name() const;
138
139 void set_name(QString name);
140
141 const list< shared_ptr<views::ViewBase> > views() const;
142
143 shared_ptr<views::ViewBase> main_view() const;
144
145 shared_ptr<pv::toolbars::MainBar> main_bar() const;
146
147 void set_main_bar(shared_ptr<pv::toolbars::MainBar> main_bar);
148
149 /**
150 * Indicates whether the captured data was saved to disk already or not
151 */
152 bool data_saved() const;
153
154 void save_settings(QSettings &settings) const;
155
156 void restore_settings(QSettings &settings);
157
158 /**
159 * Attempts to set device instance, may fall back to demo if needed
160 */
161 void select_device(shared_ptr<devices::Device> device);
162
163 /**
164 * Sets device instance that will be used in the next capture session.
165 */
166 void set_device(shared_ptr<devices::Device> device);
167
168 void set_default_device();
169
170 void load_init_file(const string &file_name, const string &format);
171
172 void load_file(QString file_name,
173 shared_ptr<sigrok::InputFormat> format = nullptr,
174 const map<string, Glib::VariantBase> &options =
175 map<string, Glib::VariantBase>());
176
177 capture_state get_capture_state() const;
178
179 void start_capture(function<void (const QString)> error_handler);
180
181 void stop_capture();
182
183 double get_samplerate() const;
184
185 uint32_t get_segment_count() const;
186
187 vector<util::Timestamp> get_triggers(uint32_t segment_id) const;
188
189 void register_view(shared_ptr<views::ViewBase> view);
190
191 void deregister_view(shared_ptr<views::ViewBase> view);
192
193 bool has_view(shared_ptr<views::ViewBase> view);
194
195 const unordered_set< shared_ptr<data::SignalBase> > signalbases() const;
196
197 bool all_segments_complete(uint32_t segment_id) const;
198
199#ifdef ENABLE_DECODE
200 shared_ptr<data::DecodeSignal> add_decode_signal();
201
202 void remove_decode_signal(shared_ptr<data::DecodeSignal> signal);
203#endif
204
205private:
206 void set_capture_state(capture_state state);
207
208 void update_signals();
209
210 shared_ptr<data::SignalBase> signalbase_from_channel(
211 shared_ptr<sigrok::Channel> channel) const;
212
213 static map<string, Glib::VariantBase> input_format_options(
214 vector<string> user_spec,
215 map<string, shared_ptr<Option>> fmt_opts);
216
217 void sample_thread_proc(function<void (const QString)> error_handler);
218
219 void free_unused_memory();
220
221 void signal_new_segment();
222 void signal_segment_completed();
223
224#ifdef ENABLE_FLOW
225 bool on_gst_bus_message(const Glib::RefPtr<Gst::Bus>& bus, const Glib::RefPtr<Gst::Message>& message);
226
227 Gst::FlowReturn on_gst_new_sample();
228#endif
229
230 void feed_in_header();
231
232 void feed_in_meta(shared_ptr<sigrok::Meta> meta);
233
234 void feed_in_trigger();
235
236 void feed_in_frame_begin();
237 void feed_in_frame_end();
238
239 void feed_in_logic(shared_ptr<sigrok::Logic> logic);
240
241 void feed_in_analog(shared_ptr<sigrok::Analog> analog);
242
243 void data_feed_in(shared_ptr<sigrok::Device> device,
244 shared_ptr<sigrok::Packet> packet);
245
246Q_SIGNALS:
247 void capture_state_changed(int state);
248 void device_changed();
249
250 void signals_changed();
251
252 void name_changed();
253
254 void trigger_event(int segment_id, util::Timestamp location);
255
256 void new_segment(int new_segment_id);
257 void segment_completed(int segment_id);
258
259 void data_received();
260
261 void add_view(const QString &title, views::ViewType type,
262 Session *session);
263
264public Q_SLOTS:
265 void on_data_saved();
266
267private:
268 DeviceManager &device_manager_;
269 shared_ptr<devices::Device> device_;
270 QString default_name_, name_;
271
272 list< shared_ptr<views::ViewBase> > views_;
273 shared_ptr<pv::views::ViewBase> main_view_;
274
275 shared_ptr<pv::toolbars::MainBar> main_bar_;
276
277 mutable mutex sampling_mutex_; //!< Protects access to capture_state_.
278 capture_state capture_state_;
279
280 unordered_set< shared_ptr<data::SignalBase> > signalbases_;
281 unordered_set< shared_ptr<data::SignalData> > all_signal_data_;
282
283 /// trigger_list_ contains pairs of <segment_id, timestamp> values.
284 vector< std::pair<uint32_t, util::Timestamp> > trigger_list_;
285
286 mutable recursive_mutex data_mutex_;
287 shared_ptr<data::Logic> logic_data_;
288 uint64_t cur_samplerate_;
289 shared_ptr<data::LogicSegment> cur_logic_segment_;
290 map< shared_ptr<sigrok::Channel>, shared_ptr<data::AnalogSegment> >
291 cur_analog_segments_;
292 int32_t highest_segment_id_;
293
294 std::thread sampling_thread_;
295
296 bool out_of_memory_;
297 bool data_saved_;
298 bool frame_began_;
299
300#ifdef ENABLE_FLOW
301 RefPtr<Pipeline> pipeline_;
302 RefPtr<Element> source_;
303 RefPtr<AppSink> sink_;
304
305 mutable mutex pipeline_done_mutex_;
306 mutable condition_variable pipeline_done_cond_;
307 atomic<bool> pipeline_done_interrupt_;
308#endif
309};
310
311} // namespace pv
312
313#endif // PULSEVIEW_PV_SESSION_HPP