Skip to content

Commit

Permalink
Refactor GP node handling
Browse files Browse the repository at this point in the history
Make sure that UI Blocks don't hold references to GR nodes after
creation (i.e. once scheduler started).

Exceptions are the drawable nodes (toolbar/plots), which are
currently both part of the graph and also accessed from the UI.
  • Loading branch information
frankosterfeld committed Apr 17, 2024
1 parent 7df1b3d commit b378339
Show file tree
Hide file tree
Showing 21 changed files with 273 additions and 320 deletions.
2 changes: 1 addition & 1 deletion cmake/Dependencies.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ FetchContent_Declare(
FetchContent_Declare(
graph-prototype
GIT_REPOSITORY https://github.com/fair-acc/graph-prototype.git
GIT_TAG 3cb5c2aff7046bc3760ccd8f8b701cbde361ecfd # main as of 2024-04-05
GIT_TAG frank/wip/od-refactoring
)

FetchContent_Declare(
Expand Down
16 changes: 6 additions & 10 deletions src/service/dashboard/defaultDashboard.flowgraph
Original file line number Diff line number Diff line change
@@ -1,33 +1,29 @@
blocks:
- name: FFT
id: FFT
id: gr::blocks::fft::FFT
- name: sum sigs
id: opendigitizer::Arithmetic
- name: sine source 1
id: opendigitizer::SineSource
parameters:
frequency: 0.100000
- name: source for sink 1
id: sink_source
- name: source for sink 2
id: sink_source
- name: remote source 1
id: opendigitizer::RemoteSource
parameters:
remote_uri: https://localhost:8080/GnuRadio/Acquisition?channelNameFilter=test
- name: sink 1
id: sink
id: opendigitizer::DataSink
- name: sink 2
id: sink
id: opendigitizer::DataSink
- name: sink 3
id: sink
id: opendigitizer::DataSink
- name: sink 4
id: sink
id: opendigitizer::DataSink
connections:
- [sine source 1, 0, FFT, 0]
- [FFT, 0, sink 1, 0]
- [sine source 1, 0, sum sigs, 0]
- [source for sink 1, 0, sum sigs, 1]
- [FFT, 0, sum sigs, 1]
- [sum sigs, 0, sink 2, 0]
- [sine source 1, 0, sink 3, 0]
- [remote source 1, 0, sink 4, 0]
1 change: 1 addition & 0 deletions src/ui/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -183,6 +183,7 @@ target_link_libraries(
gnuradio-algorithm
gr-basic
gr-fourier
gr-testing
fftw
vir
pmtv)
Expand Down
2 changes: 2 additions & 0 deletions src/ui/app.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -161,6 +161,8 @@ class App {
};

SchedWrapper _scheduler;
std::vector<gr::BlockModel *> _toolbarBlocks;
std::vector<gr::BlockModel *> _plotBlocks;

public:
App() noexcept { setStyle(Style::Light); }
Expand Down
18 changes: 5 additions & 13 deletions src/ui/assets/sampleDashboards/DemoDashboard.grc
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ blocks:
- name: sum sigs
id: opendigitizer::Arithmetic
- name: FFT
id: FFT
id: gr::blocks::fft::FFT
- name: sine source 1
id: opendigitizer::SineSource
parameters:
Expand All @@ -12,21 +12,13 @@ blocks:
parameters:
frequency: 1.300000
- name: sink 1
id: sink
- name: source for sink 1
id: sink_source
id: opendigitizer::DataSink
- name: sink 2
id: sink
- name: source for sink 2
id: sink_source
id: opendigitizer::DataSink
- name: sink 3
id: sink
- name: source for sink 3
id: sink_source
id: opendigitizer::DataSink
- name: sink 4
id: sink
- name: source for sink 4
id: sink_source
id: opendigitizer::DataSink
connections:
- [sine source 1, 0, sum sigs, 0]
- [sine source 3, 0, sum sigs, 1]
Expand Down
18 changes: 7 additions & 11 deletions src/ui/assets/sampleDashboards/ExtendedDemoDashboard.grc
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ blocks:
- name: sum sigs3
id: opendigitizer::Arithmetic
- name: FFT
id: FFT
id: gr::blocks::fft::FFT
- name: sine source 3
id: opendigitizer::SineSource
parameters:
Expand All @@ -23,22 +23,18 @@ blocks:
id: opendigitizer::SineSource
parameters:
frequency: 0.400000
- name: source for sink 1
id: sink_source
- name: source for sink 2
id: sink_source
- name: sink 1
id: sink
id: opendigitizer::DataSink
- name: sink 2
id: sink
id: opendigitizer::DataSink
- name: sink 3
id: sink
id: opendigitizer::DataSink
- name: sink 4
id: sink
id: opendigitizer::DataSink
- name: sink 5
id: sink
id: opendigitizer::DataSink
- name: sink 6
id: sink
id: opendigitizer::DataSink
connections:
- [sine source 3, 0, sum sigs1, 0]
- [sine source 4, 0, sum sigs1, 1]
Expand Down
2 changes: 2 additions & 0 deletions src/ui/blocks/Arithmetic.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -38,4 +38,6 @@ struct Arithmetic : public gr::Block<Arithmetic<T>> {

ENABLE_REFLECTION_FOR_TEMPLATE(opendigitizer::Arithmetic, in1, in2, out, operation)

auto registerArithmetic = gr::registerBlock<opendigitizer::Arithmetic, float, double>(gr::globalBlockRegistry());

#endif
12 changes: 9 additions & 3 deletions src/ui/blocks/RemoteSource.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,9 @@
namespace opendigitizer {

template<typename T>
requires std::is_same_v<T, float>
requires std::is_floating_point_v<T>
struct RemoteSource : public gr::Block<RemoteSource<T>> {
gr::PortOut<float> out;
gr::PortOut<T> out;
std::string remote_uri;
std::string signal_name;
std::string signal_unit;
Expand Down Expand Up @@ -53,7 +53,11 @@ struct RemoteSource : public gr::Block<RemoteSource<T>> {
auto in = std::span<const float>(d.acq.channelValue.begin(), d.acq.channelValue.end());
in = in.subspan(d.read, std::min(output.size() - written, in.size() - d.read));

std::copy(in.begin(), in.end(), output.begin() + written);
if constexpr (std::is_same_v<T, float>) {
std::ranges::copy(in, output.begin() + written);
} else {
std::ranges::transform(in, output.begin() + written, [](float v) { return static_cast<T>(v); });
}
written += in.size();
d.read += in.size();
if (d.read == d.acq.channelValue.size()) {
Expand Down Expand Up @@ -121,4 +125,6 @@ struct RemoteSource : public gr::Block<RemoteSource<T>> {

ENABLE_REFLECTION_FOR_TEMPLATE(opendigitizer::RemoteSource, out, remote_uri, signal_name, signal_unit, signal_min, signal_max)

auto registerRemoteSource = gr::registerBlock<opendigitizer::RemoteSource, float, double>(gr::globalBlockRegistry());

#endif
2 changes: 2 additions & 0 deletions src/ui/blocks/SineSource.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -73,4 +73,6 @@ struct SineSource : public gr::Block<SineSource<T>, gr::BlockingIO<true>> {

ENABLE_REFLECTION_FOR_TEMPLATE(opendigitizer::SineSource, out, freqIn, freqOut, frequency)

auto registerSineSource = gr::registerBlock<opendigitizer::SineSource, float, double>(gr::globalBlockRegistry());

#endif
10 changes: 2 additions & 8 deletions src/ui/cmake/Dependencies.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -21,12 +21,6 @@ FetchContent_Declare(
GIT_TAG v0.9.3 # latest as of 2023-12-19
)

FetchContent_Declare(
yaml-cpp
GIT_REPOSITORY https://github.com/jbeder/yaml-cpp.git
GIT_TAG yaml-cpp-0.7.0
)

FetchContent_Declare(
plf_colony
GIT_REPOSITORY https://github.com/mattreecebentley/plf_colony.git
Expand All @@ -49,10 +43,10 @@ FetchContent_Declare(
FetchContent_Declare(
graph-prototype
GIT_REPOSITORY https://github.com/fair-acc/graph-prototype.git
GIT_TAG 3cb5c2aff7046bc3760ccd8f8b701cbde361ecfd # main as of 2024-04-05
GIT_TAG frank/wip/od-refactoring
)

FetchContent_MakeAvailable(imgui implot imgui-node-editor yaml-cpp stb opencmw-cpp plf_colony graph-prototype)
FetchContent_MakeAvailable(imgui implot imgui-node-editor stb opencmw-cpp plf_colony graph-prototype)

if (NOT EMSCRIPTEN)
find_package(SDL2 REQUIRED)
Expand Down
59 changes: 20 additions & 39 deletions src/ui/dashboard.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -253,16 +253,6 @@ DataSink *Dashboard::createSink() {
return sinkptr;
}

DataSinkSource *Dashboard::createSource() {
int n = localFlowGraph.sourceBlocks().size() + 1;
auto name = fmt::format("source {}", n);
auto source = std::make_unique<DigitizerUi::DataSinkSource>(name);
auto sourceptr = source.get();
name = fmt::format("source for sink {}", n);
localFlowGraph.addSourceBlock(std::move(source));
return sourceptr;
}

void Dashboard::setNewDescription(const std::shared_ptr<DashboardDescription> &desc) {
m_desc = desc;
}
Expand All @@ -272,12 +262,19 @@ void Dashboard::load() {
fetch(
m_desc->source, m_desc->filename, { What::Flowgraph, What::Dashboard },
[_this = shared()](std::array<std::string, 2> &&data) {
_this->localFlowGraph.parse(std::move(data[0]));
// Load is called after parsing the flowgraph so that we already have the list of sources
_this->doLoad(data[1]);
try {
_this->localFlowGraph.parse(std::move(data[0]));
// Load is called after parsing the flowgraph so that we already have the list of sources
_this->doLoad(data[1]);
} catch (const std::exception &e) {
// TODO show error message
fmt::print(std::cerr, "Error: {}", e.what());
App::instance().closeDashboard();
}
},
[_this = shared()]() {
fmt::print("Invalid flowgraph for dashboard {}/{}\n", _this->m_desc->source->path, _this->m_desc->filename);
// TODO show error message
fmt::print(std::cerr, "Invalid flowgraph for dashboard {}/{}\n", _this->m_desc->source->path, _this->m_desc->filename);
App::instance().closeDashboard();
});
} else {
Expand All @@ -290,31 +287,17 @@ void Dashboard::doLoad(const std::string &desc) {

auto path = std::filesystem::path(m_desc->source->path) / m_desc->filename;

#ifdef NDEBUG
#define ERROR_RETURN(msg) \
{ \
fmt::print("Error parsing YAML {}: {}\n{}\n", path.native(), msg, desc); \
abort(); \
}
#else
#define ERROR_RETURN(msg) \
{ \
fmt::print("Error parsing YAML {}: {}\n{}\n", path.native(), msg, desc); \
return; \
}
#endif

auto sources = tree["sources"];
if (!sources || !sources.IsSequence()) ERROR_RETURN("sources entry invalid");
if (!sources || !sources.IsSequence()) throw std::runtime_error("sources entry invalid");

for (const auto &s : sources) {
if (!s.IsMap()) ERROR_RETURN("source is no map");
if (!s.IsMap()) throw std::runtime_error("source is no map");

auto block = s["block"];
auto port = s["port"];
auto name = s["name"];
auto color = s["color"];
if (!block || !block.IsScalar() || !port || !port.IsScalar() || !name || !name.IsScalar() || !color || !color.IsScalar()) ERROR_RETURN("invalid source color definition");
if (!block || !block.IsScalar() || !port || !port.IsScalar() || !name || !name.IsScalar() || !color || !color.IsScalar()) std::runtime_error("invalid source color definition");

auto blockStr = block.as<std::string>();
auto portNum = port.as<int>();
Expand All @@ -334,29 +317,29 @@ void Dashboard::doLoad(const std::string &desc) {
}

auto plots = tree["plots"];
if (!plots || !plots.IsSequence()) ERROR_RETURN("plots invalid");
if (!plots || !plots.IsSequence()) throw std::runtime_error("plots invalid");

for (const auto &p : plots) {
if (!p.IsMap()) ERROR_RETURN("plots is not map");
if (!p.IsMap()) std::runtime_error("plots is not map");

auto name = p["name"];
auto axes = p["axes"];
auto plotSources = p["sources"];
auto rect = p["rect"];
if (!name || !name.IsScalar() || !axes || !axes.IsSequence() || !plotSources || !plotSources.IsSequence() || !rect || !rect.IsSequence() || rect.size() != 4) ERROR_RETURN("invalid plot definition");
if (!name || !name.IsScalar() || !axes || !axes.IsSequence() || !plotSources || !plotSources.IsSequence() || !rect || !rect.IsSequence() || rect.size() != 4) throw std::runtime_error("invalid plot definition");

m_plots.emplace_back();
auto &plot = m_plots.back();
plot.name = name.as<std::string>();

for (const auto &a : axes) {
if (!a.IsMap()) ERROR_RETURN("axes is no map");
if (!a.IsMap()) throw std::runtime_error("axes is no map");

auto axis = a["axis"];
auto min = a["min"];
auto max = a["max"];

if (!axis || !axis.IsScalar() || !min || !min.IsScalar() || !max || !max.IsScalar()) ERROR_RETURN("invalid axis definition");
if (!axis || !axis.IsScalar() || !min || !min.IsScalar() || !max || !max.IsScalar()) throw std::runtime_error("invalid axis definition");

plot.axes.push_back({});
auto &ax = plot.axes.back();
Expand All @@ -377,7 +360,7 @@ void Dashboard::doLoad(const std::string &desc) {
}

for (const auto &s : plotSources) {
if (!s.IsScalar()) ERROR_RETURN("plot source is no scalar");
if (!s.IsScalar()) throw std::runtime_error("plot source is no scalar");

auto str = s.as<std::string>();
plot.sourceNames.push_back(str);
Expand All @@ -393,8 +376,6 @@ void Dashboard::doLoad(const std::string &desc) {
App::instance().fgItem.setSettings(&localFlowGraph, fgLayout && fgLayout.IsScalar() ? fgLayout.as<std::string>() : std::string{});

loadPlotSources();

#undef ERROR_RETURN
}

void Dashboard::save() {
Expand Down
3 changes: 0 additions & 3 deletions src/ui/dashboard.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -26,8 +26,6 @@ class Block;
class FlowGraph;
struct DashboardDescription;
class DataSink;
class DataSinkSource;

struct DashboardSource {
~DashboardSource() noexcept;

Expand Down Expand Up @@ -138,7 +136,6 @@ class Dashboard : public std::enable_shared_from_this<Dashboard> {
inline auto &remoteServices() { return m_services; }

DataSink *createSink();
DataSinkSource *createSource();

void loadPlotSources();

Expand Down
6 changes: 2 additions & 4 deletions src/ui/dashboardpage.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -300,8 +300,7 @@ void DashboardPage::drawPlot(DigitizerUi::Dashboard::Plot &plot) noexcept {
} else if (source->visible) {
switch (sink->dataType) {
case DigitizerUi::DataType::Float32: {
auto values = sink->data.asFloat32();
std::lock_guard lock(sink->m_mutex);
auto values = sink->data.asFloat32();
ImPlot::PlotLine(source->name.c_str(), values.data(), int(values.size()));
break;
}
Expand All @@ -310,8 +309,7 @@ void DashboardPage::drawPlot(DigitizerUi::Dashboard::Plot &plot) noexcept {
if (ds.extents.empty()) {
break;
}
auto &values = ds.signal_values;
std::lock_guard lock(sink->m_mutex);
auto &values = ds.signal_values;
for (int i = 0; i < ds.extents[0]; ++i) {
auto n = ds.extents[1];
ImPlot::PlotLine(ds.signal_names[i].c_str(), values.data() + n * i, n);
Expand Down
Loading

0 comments on commit b378339

Please sign in to comment.