TAMSVIZ
Visualization and annotation tool for ROS
renderthread.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "renderthread.h"
5 
6 #include "../annotations/scene.h"
7 #include "../core/bagplayer.h"
8 #include "../core/loader.h"
9 #include "../core/log.h"
10 #include "../core/profiler.h"
11 #include "../core/topic.h"
12 #include "../core/workspace.h"
13 #include "../render/renderer.h"
14 #include "../render/resource.h"
15 #include "../render/shader.h"
16 #include "renderwindow.h"
17 
18 RenderThread::RenderThread() {
19  _running = true;
20  _thread = std::thread([this]() { run(); });
21 }
22 
23 void RenderThread::invalidate() {
24  // LOG_DEBUG("render thread invalidate");
25  std::unique_lock<std::mutex> lock(_mutex);
26  _redraw_flag = true;
27  _condition.notify_all();
28 }
29 
30 void RenderThread::run() {
31  LOG_DEBUG("render thread started");
32 
33  QOffscreenSurface offscreen_surface;
34  offscreen_surface.create();
35  if (!offscreen_surface.isValid()) {
36  throw std::runtime_error("failed to create offscreen surface");
37  }
38 
39  QOpenGLContext offscreen_context;
40  offscreen_context.setShareContext(QOpenGLContext::globalShareContext());
41  if (!offscreen_context.create()) {
42  throw std::runtime_error("failed to create offscreen context");
43  }
44 
45  offscreen_context.makeCurrent(&offscreen_surface);
46 
47  std::vector<std::function<void()>> garbage_list;
48  std::mutex garbage_mutex;
49  ResourceBase::setCleanupFunction([&](const std::function<void()> &garbage) {
50  std::lock_guard<std::mutex> lock(garbage_mutex);
51  garbage_list.push_back(garbage);
52  });
53  std::vector<std::shared_ptr<RenderWindowBase>> render_window_list;
54  std::vector<std::shared_ptr<Display>> display_list;
55  std::vector<std::shared_ptr<SceneAnnotationBase>> scene_annotations;
56 
57  Renderer renderer;
58  RenderList render_list;
59  while (true) {
60  // std::this_thread::sleep_for(std::chrono::milliseconds(10));
61  {
62  std::unique_lock<std::mutex> lock(_mutex);
63  while (true) {
64  if (_stop_flag) {
65  break;
66  }
67  if (_redraw_flag) {
68  _redraw_flag = false;
69  break;
70  }
71  _condition.wait(lock);
72  }
73  if (_stop_flag) {
74  break;
75  }
76  }
77  // LOG_DEBUG("render");
78 
79  PROFILER();
80 
81  render_list.clear();
82  {
83  NoMessageScope m_scope;
84  LockScope ws;
85  ws->recurse([&render_window_list,
86  &display_list](const std::shared_ptr<Object> &object) {
87  if (object) {
88  if (auto render_window =
89  std::dynamic_pointer_cast<RenderWindowBase>(object)) {
90  render_window_list.push_back(render_window);
91  }
92  if (auto display = std::dynamic_pointer_cast<Display>(object)) {
93  display_list.push_back(display);
94  }
95  }
96  });
97  {
98  std::lock_guard<std::mutex> lock(garbage_mutex);
99  for (auto &f : garbage_list) {
100  f();
101  }
102  garbage_list.clear();
103  }
104  {
105  RenderSyncContext render_context;
106  render_context.render_list = &render_list;
107  ws->document()->display()->renderSyncRecursive(render_context);
108  }
109  {
110  scene_annotations.clear();
111  if (ws->player && ws->document()->timeline()) {
112  auto current_time = ws->player->time();
113  for (auto &track : ws->document()->timeline()->tracks()) {
114  if (auto annotation_track =
115  std::dynamic_pointer_cast<AnnotationTrack>(track)) {
116  if (auto branch = annotation_track->branch(ws(), false)) {
117  for (auto &span : branch->spans()) {
118  if (span->start() <= current_time &&
119  span->start() + span->duration() >= current_time) {
120  for (auto &annotation : span->annotations()) {
121  if (auto scene_annotation =
122  std::dynamic_pointer_cast<SceneAnnotationBase>(
123  annotation)) {
124  {
125  RenderSyncContext render_context;
126  render_context.render_list = &render_list;
127  scene_annotation->renderSync(render_context, track,
128  span);
129  }
130  scene_annotations.push_back(scene_annotation);
131  }
132  }
133  }
134  }
135  }
136  }
137  }
138  }
139  }
140  for (auto &render_window : render_window_list) {
141  RenderWindowSyncContext render_context;
142  render_window->renderWindowSync(render_context);
143  }
144  }
145  {
146  RenderAsyncContext render_context;
147  render_context.render_list = &render_list;
148  for (auto &display : display_list) {
149  display->renderAsync(render_context);
150  }
151  for (auto &scene_annotation : scene_annotations) {
152  scene_annotation->renderAsync(render_context);
153  }
154  }
155  // V_GL(glFlush());
156  // V_GL(glFinish());
157  for (auto &render_window : render_window_list) {
158  RenderWindowAsyncContext render_context;
159  render_context.render_list = &render_list;
160  render_context.renderer = &renderer;
161  render_window->renderWindowAsync(render_context);
162  }
163  for (auto &render_window : render_window_list) {
164  startOnMainThreadAsync([render_window]() { render_window->present(); });
165  }
166  {
167  LockScope ws;
168  {
169  for (auto &display : display_list) {
170  if (display.unique()) {
171  auto ref = std::make_shared<std::shared_ptr<Display>>(display);
172  auto lambda = [ref]() {
173  LockScope ws;
174  ref->reset();
175  };
176  ref.reset();
177  display.reset();
178  startOnMainThreadAsync(lambda);
179  }
180  }
181  display_list.clear();
182  }
183  {
184  for (auto &render_window : render_window_list) {
185  if (render_window.unique()) {
186  auto ref = std::make_shared<std::shared_ptr<RenderWindowBase>>(
187  render_window);
188  auto lambda = [ref]() {
189  LockScope ws;
190  ref->reset();
191  };
192  ref.reset();
193  render_window.reset();
194  startOnMainThreadAsync(lambda);
195  }
196  }
197  render_window_list.clear();
198  }
199  }
200  }
201  offscreen_context.doneCurrent();
202  ResourceBase::setCleanupFunction(nullptr);
203 }
204 
205 void RenderThread::stop() {
206  if (_running) {
207  _running = false;
208  LOG_DEBUG("stopping render thread");
209  {
210  std::unique_lock<std::mutex> lock(_mutex);
211  _stop_flag = true;
212  _condition.notify_all();
213  }
214  _thread.join();
215  LOG_DEBUG("render thread stopped");
216  }
217 }
218 
219 RenderThread::~RenderThread() { stop(); }
220 
221 RenderThread *RenderThread::instance() {
222  static std::shared_ptr<RenderThread> instance = []() {
223  auto instance = std::make_shared<RenderThread>();
224  RenderThread *ptr = instance.get();
225  {
226  LockScope ws;
227  ws->modified.connect(instance, [ptr]() { ptr->invalidate(); });
228  GlobalEvents::instance()->redraw.connect(instance,
229  [ptr]() { ptr->invalidate(); });
230  }
231  LoaderThread::instance()->started.connect(instance,
232  [ptr]() { ptr->invalidate(); });
233  LoaderThread::instance()->finished.connect(instance,
234  [ptr]() { ptr->invalidate(); });
235  TopicManager::instance()->received.connect(instance,
236  [ptr]() { ptr->invalidate(); });
237  return instance;
238  }();
239  return instance.get();
240 }