Files
Jungfraujoch/tools/jfjoch_lite_perf_test.cpp
2025-10-20 20:43:44 +02:00

203 lines
7.5 KiB
C++

// SPDX-FileCopyrightText: 2025 Filip Leonarski, Paul Scherrer Institute <filip.leonarski@psi.ch>
// SPDX-License-Identifier: GPL-3.0-only
#include "../common/print_license.h"
#include "../common/Logger.h"
#include "../common/DiffractionExperiment.h"
#include "../writer/HDF5Objects.h"
#include "../common/PixelMask.h"
#include "../image_pusher/HDF5FilePusher.h"
#include "../image_puller/TestImagePuller.h"
#include "../acquisition_device/AcquisitionDeviceGroup.h"
#include "../receiver/JFJochReceiverService.h"
#include "JFJochCompressor.h"
void print_usage(Logger &logger) {
logger.Info("Usage ./jfjoch_fpga_test {<options>} <path to repository>");
logger.Info("Options:");
logger.Info(" -i<num> Number of images");
logger.Info(" -N<num> Number of image processing threads (default: 8)");
logger.Info(" -P<txt> NUMA policy: none|n2g2|n8g4|n8g4_hbm (default: none)");
logger.Info(" -F{<txt>} Write file, optional parameter is name (default: lyso_lite_perf_test)");
logger.Info(" -X<txt> Indexing (none|fft|ffbidx), ffbidx is default");
logger.Info(" -t<num> Indexing thread pool size (default: 4)");
logger.Info(" -f<num> FFT indexing search vectors");
logger.Info(" -Q Quick integration");
logger.Info(" -G Geometry refinement");
}
int main(int argc, char **argv) {
print_license("jfjoch_fpga_test");
Logger logger("jfjoch_fpga_test");
logger.Verbose(false);
bool use_geom = false;
uint16_t nthreads = 8;
size_t nimages = 1000;
std::string numa_policy_name;
std::string filename = "";
std::string ml_model = "";
IndexingAlgorithmEnum indexing = IndexingAlgorithmEnum::FFBIDX;
std::optional<int64_t> fft_num_vectors;
uint16_t indexing_threads = 4;
bool quick_integrate = false;
RegisterHDF5Filter();
if (argc == 1) {
print_usage(logger);
exit(EXIT_FAILURE);
}
int opt;
while ((opt = getopt(argc, argv, "N:P:i:F::QGvX:t:f:")) != -1) {
switch (opt) {
case 'N':
nthreads = atol(optarg);
break;
case 'P':
numa_policy_name = std::string(optarg);
break;
case 'i':
nimages = atol(optarg);
break;
case 'X':
if (std::string(optarg) == "none")
indexing = IndexingAlgorithmEnum::None;
else if (std::string(optarg) == "fft" || std::string(optarg) == "FFT")
indexing = IndexingAlgorithmEnum::FFT;
else if (std::string(optarg) == "ffbidx" || std::string(optarg) == "FFBIDX")
indexing = IndexingAlgorithmEnum::FFBIDX;
break;
case 't':
indexing_threads = atol(optarg);
break;
case 'f':
fft_num_vectors = atol(optarg);
break;
case 'F':
if (optarg)
filename = std::string(optarg);
else
filename = "lyso_lite_perf_test";
break;
case 'Q':
quick_integrate = true;
break;
case 'G':
use_geom = true;
break;
case 'v':
logger.Verbose(true);
break;
default: /* '?' */
print_usage(logger);
exit(EXIT_FAILURE);
}
}
if (optind != argc - 1) {
print_usage(logger);
exit(EXIT_FAILURE);
}
std::string jfjoch_path = std::string(argv[optind]) + "/";
DiffractionExperiment experiment(DetDECTRIS(2068, 2164, "Test", {}));
experiment.ImagesPerTrigger(nimages).NumTriggers(1).UseInternalPacketGenerator(true).ImagesPerFile(1000)
.FilePrefix(filename).JungfrauConvPhotonCnt(false).SetFileWriterFormat(FileWriterFormat::NXmxVDS).OverwriteExistingFiles(true)
.DetectorDistance_mm(75).BeamY_pxl(1136).BeamX_pxl(1090).IncidentEnergy_keV(12.4)
.SetUnitCell(UnitCell{.a = 36.9, .b = 78.95, .c = 78.95, .alpha =90, .beta = 90, .gamma = 90}).PixelSigned(true);
PixelMask pixel_mask(experiment);
experiment.ROI().SetROI(ROIDefinition{.boxes = {ROIBox("ROI1", 123, 180, 500,800) }});
// Load example image
HDF5ReadOnlyFile data(jfjoch_path + "tests/test_data/compression_benchmark.h5");
HDF5DataSet dataset(data, "/entry/data/data");
HDF5DataSpace file_space(dataset);
std::vector<int16_t> image_conv(file_space.GetDimensions()[1] * file_space.GetDimensions()[2]);
std::vector<hsize_t> start = {4,0,0};
std::vector<hsize_t> file_size = {1, file_space.GetDimensions()[1], file_space.GetDimensions()[2]};
dataset.ReadVector(image_conv, start, file_size);
JFJochBitShuffleCompressor compressor(CompressionAlgorithm::BSHUF_LZ4);
auto image_compressed = compressor.Compress(image_conv);
HDF5FilePusher pusher;
auto puller = std::make_shared<TestImagePuller>(nimages + 5);
StartMessage start_msg;
experiment.FillMessage(start_msg);
puller->Put(ImagePullerOutput{
.cbor = std::make_shared<CBORStream2DeserializerOutput>(start_msg)
});
DataMessage data_msg;
data_msg.image = CompressedImage(image_compressed,
file_space.GetDimensions()[2],
file_space.GetDimensions()[1],
CompressedImageMode::Int16,
CompressionAlgorithm::BSHUF_LZ4);
for (int i = 0; i < experiment.GetImageNum(); i++) {
data_msg.number = i;
puller->Put(ImagePullerOutput{
.cbor = std::make_shared<CBORStream2DeserializerOutput>(data_msg)
});
}
EndMessage end_msg{};
puller->Put(ImagePullerOutput{
.cbor = std::make_shared<CBORStream2DeserializerOutput>(end_msg)
});
AcquisitionDeviceGroup group;
JFJochReceiverService service(group, logger, pusher);
service.NUMAPolicy(numa_policy_name);
service.NumThreads(nthreads);
IndexingSettings i_settings;
i_settings.Algorithm(indexing);
if (fft_num_vectors)
i_settings.FFT_NumVectors(fft_num_vectors.value());
i_settings.IndexingThreads(indexing_threads);
if (use_geom)
i_settings.GeomRefinementAlgorithm(GeomRefinementAlgorithmEnum::BeamCenter);
experiment.ImportIndexingSettings(i_settings);
service.Indexing(i_settings);
SpotFindingSettings settings = DiffractionExperiment::DefaultDataProcessingSettings();
settings.signal_to_noise_threshold = 2.5;
settings.photon_count_threshold = 5;
settings.min_pix_per_spot = 1;
settings.max_pix_per_spot = 200;
settings.high_resolution_limit = 2.0;
settings.low_resolution_limit = 50.0;
settings.quick_integration = quick_integrate;
service.SetSpotFindingSettings(settings);
auto start_time = std::chrono::system_clock::now();
service.Start(experiment, pixel_mask, nullptr, puller);
auto output = service.Stop();
auto end_time = std::chrono::system_clock::now();
double receiving_time_s = static_cast<double>(output.end_time_ms - output.start_time_ms) / 1000.0;
logger.Info("Throughput {:.1f} Hz", experiment.GetImageNum() / receiving_time_s);
if (output.status.indexing_rate)
logger.Info("Indexing rate {:.0f}%", output.status.indexing_rate.value() * 100.0);
if (output.status.max_receive_delay)
logger.Info("Max delay {}", output.status.max_receive_delay.value());
}