Files
Jungfraujoch/common/Histogram.h
Filip Leonarski e2b240356c
All checks were successful
Build Packages / build:rpm (rocky8_nocuda) (push) Successful in 9m28s
Build Packages / build:rpm (ubuntu2404_nocuda) (push) Successful in 8m25s
Build Packages / build:rpm (ubuntu2204_nocuda) (push) Successful in 9m4s
Build Packages / build:rpm (rocky9_nocuda) (push) Successful in 10m27s
Build Packages / build:rpm (rocky8_sls9) (push) Successful in 9m36s
Build Packages / Generate python client (push) Successful in 32s
Build Packages / Build documentation (push) Successful in 45s
Build Packages / Create release (push) Has been skipped
Build Packages / build:rpm (rocky8) (push) Successful in 8m45s
Build Packages / build:rpm (ubuntu2404) (push) Successful in 7m51s
Build Packages / build:rpm (ubuntu2204) (push) Successful in 8m57s
Build Packages / build:rpm (rocky9) (push) Successful in 9m35s
Build Packages / Unit tests (push) Successful in 1h13m45s
v1.0.0-rc.121 (#28)
This is an UNSTABLE release.

* jfjoch_broker: Report changes in the image buffer, so viewer doesn't reload constantly
* jfjoch_viewer: Improve performance of loading images
* jfjoch_viewer: Auto-throttle image loading in HTTP-sync / movie modes
* jfjoch_viewer: Auto-foreground calculated with histogram
* jfjoch_viewer: Fix rare segmentation fault

Reviewed-on: #28
Co-authored-by: Filip Leonarski <filip.leonarski@psi.ch>
Co-committed-by: Filip Leonarski <filip.leonarski@psi.ch>
2025-12-12 21:24:20 +01:00

117 lines
3.1 KiB
C++

// SPDX-FileCopyrightText: 2024 Filip Leonarski, Paul Scherrer Institute <filip.leonarski@psi.ch>
// SPDX-License-Identifier: GPL-3.0-only
#ifndef JUNGFRAUJOCH_HISTOGRAM_H
#define JUNGFRAUJOCH_HISTOGRAM_H
#include <cstdint>
#include <cstddef>
#include <vector>
#include <mutex>
#include "MultiLinePlot.h"
#include "../common/JFJochException.h"
template<class T>
class SetAverage {
std::vector<T> sum;
std::vector<uint64_t> count;
mutable std::mutex m;
public:
explicit SetAverage(size_t bins) : sum(bins), count(bins) {
}
void Add(size_t bin, T val) {
std::unique_lock ul(m);
if (bin < sum.size()) {
sum[bin] += val;
count[bin] += 1;
}
}
MultiLinePlot GetPlot() const {
std::unique_lock ul(m);
MultiLinePlotStruct plot;
plot.x.resize(sum.size());
plot.y.resize(sum.size());
for (int i = 0; i < sum.size(); i++) {
plot.x[i] = static_cast<float>(i);
if (count[i] > 0)
plot.y[i] = static_cast<float>(sum[i]) / count[i];
else
plot.y[i] = 0;
}
MultiLinePlot ret;
ret.AddPlot(plot);
return ret;
}
};
class Histogram {
std::vector<uint64_t> count;
int32_t total_count = 0;
int32_t max_bin = 0;
public:
explicit Histogram(size_t bins) : count(bins) {}
void Add(int32_t val) {
if (val > 0 && val < count.size()) {
count[val] += 1;
if (val > max_bin)
max_bin = val;
++total_count;
}
}
void clear() {
for (auto &c: count) c = 0;
total_count = 0;
max_bin = 0;
}
[[nodiscard]] std::vector<float> GetCount() const {
std::vector<float> ret;
ret.reserve(max_bin + 1);
for (size_t i = 0; i < max_bin + 1; i++)
ret.emplace_back(count[i]);
return ret;
}
[[nodiscard]] uint64_t GetTotalCount() const {
return total_count;
}
// Returns the value x such that approximately `percent`% of samples are <= x.
// - percent must be in [0, 100]
// - returns std::nullopt if histogram is empty
[[nodiscard]] std::optional<int32_t> Percentile(float percent) const {
if (!std::isfinite(percent) || percent < 0.0f || percent > 100.0f) {
throw JFJochException(JFJochExceptionCategory::InputParameterBelowMin,
"FloatHistogram Percentile expects percent in [0, 100]");
}
if (total_count == 0)
return std::nullopt;
// Target rank in [0, total-1]
const double q = static_cast<double>(percent) / 100.0;
const auto target = static_cast<int64_t>(std::floor(q * static_cast<double>(total_count - 1)));
uint64_t cumulative = 0;
for (int64_t i = 0; i < max_bin + 1; i++) {
cumulative += count[i];
if (target < cumulative && count[i] > 0)
return i;
}
// If due to rounding we didn't return inside the loop, clamp to the last bin's upper edge.
return count.size() - 1;
}
};
#endif //JUNGFRAUJOCH_HISTOGRAM_H