Files
Jungfraujoch/common/AzimuthalIntegrationProfile.cpp
2025-05-28 18:49:27 +02:00

104 lines
3.5 KiB
C++

// SPDX-FileCopyrightText: 2024 Filip Leonarski, Paul Scherrer Institute <filip.leonarski@psi.ch>
// SPDX-License-Identifier: GPL-3.0-only
#include "AzimuthalIntegrationProfile.h"
#include "JFJochException.h"
inline float sum_to_count(float sum, uint64_t count) {
if (count == 0)
return 0.0f;
else
return sum / (static_cast<float>(count));
}
AzimuthalIntegrationProfile::AzimuthalIntegrationProfile(const AzimuthalIntegration &mapping)
: bin_to_q(mapping.GetBinToQ()),
sum(mapping.GetBinNumber(), 0),
count(mapping.GetBinNumber(), 0){}
void AzimuthalIntegrationProfile::Clear(const AzimuthalIntegration &mapping) {
std::unique_lock ul(m);
bin_to_q = mapping.GetBinToQ();
sum = std::vector<float>(mapping.GetBinNumber(), 0);
count = std::vector<uint64_t>(mapping.GetBinNumber(), 0);
}
void AzimuthalIntegrationProfile::Add(const std::vector<float> &in_sum, const std::vector<uint32_t> &in_count) {
std::unique_lock ul(m);
if ((in_sum.size() == sum.size()) && (in_count.size() == count.size())) {
for (int i = 0; i < sum.size(); i++) {
sum[i] += in_sum[i];
count[i] += in_count[i];
}
} else if (!in_sum.empty() && !in_count.empty())
throw JFJochException(JFJochExceptionCategory::InputParameterInvalid, "Mismatch in size of sum/count datasets");
}
std::vector<float> AzimuthalIntegrationProfile::GetResult() const {
std::unique_lock ul(m);
std::vector<float> rad_int_profile(sum.size(), 0);
for (int i = 0; i < sum.size(); i++)
rad_int_profile[i] = sum_to_count(sum[i], count[i]);
return rad_int_profile;
}
void AzimuthalIntegrationProfile::SetTitle(const std::string &input) {
title = input;
}
MultiLinePlot AzimuthalIntegrationProfile::GetPlot() const {
MultiLinePlot ret;
ret.AddPlot(MultiLinePlotStruct{.title = title, .x = bin_to_q, .y = GetResult()});
return ret;
}
float AzimuthalIntegrationProfile::GetMeanValueOfBins(uint16_t min_bin, uint16_t max_bin) const {
std::unique_lock ul(m);
float ret_sum = 0;
uint64_t ret_count = 0;
for (int i = std::min((uint16_t)sum.size(),min_bin);
i <= std::min((uint16_t)(sum.size()-1),max_bin);
i++) {
ret_sum += sum[i];
ret_count += count[i];
}
return sum_to_count(ret_sum, ret_count);
}
float AzimuthalIntegrationProfile::GetBkgEstimate(const AzimuthalIntegrationSettings &settings) const {
auto min_bin = settings.QToBin(settings.GetBkgEstimateLowQ_recipA());
auto max_bin = settings.QToBin(settings.GetBkgEstimateHighQ_recipA());
return GetMeanValueOfBins(min_bin, max_bin);
}
AzimuthalIntegrationProfile &AzimuthalIntegrationProfile::operator+=(const AzimuthalIntegrationProfile &other) {
if ((other.bin_to_q != bin_to_q) || (sum.size() != other.sum.size())) {
throw JFJochException(JFJochExceptionCategory::InputParameterInvalid,
"Error combining two radial integration profiles");
}
for (int i = 0; i < sum.size(); i++) {
sum[i] += other.sum[i];
count[i] += other.count[i];
}
return *this;
}
void AzimuthalIntegrationProfile::Add(const DeviceOutput &result) {
std::unique_lock ul(m);
if (sum.size() > FPGA_INTEGRATION_BIN_COUNT )
throw JFJochException(JFJochExceptionCategory::InputParameterInvalid,
"Error in getting result from FPGA");
for (int i = 0; i < sum.size(); i++) {
sum[i] += result.integration_result[i].sum;
count[i] += result.integration_result[i].count;
}
}