// SPDX-License-Identifier: LGPL-3.0-or-other // Copyright (C) 2021 Contributors to the SLS Detector Package #include "catch.hpp" #include "DetectorImpl.h" #include "Module.h" #include "sls/ClientSocket.h" #include "sls/logger.h" #include "sls/sls_detector_defs.h" #include "sls/Timer.h" #include "sls/sls_detector_funcs.h" #include #include #define VERBOSE // Header holding all configurations for different detectors #include "tests/config.h" #include "tests/globals.h" // using namespace test; // using dt = slsDetectorDefs::detectorType; // extern std::string hostname; // extern std::string detector_type; // extern dt type; namespace sls { TEST_CASE("Single detector no receiver", "[.integration][.single]") { auto t = Module::getTypeFromDetector(test::hostname); CHECK(t == test::type); Module d(t); CHECK(d.getDetectorTypeAsEnum() == t); CHECK(d.getDetectorTypeAsString() == test::detector_type); d.setHostname(test::hostname); CHECK(d.getHostname() == test::hostname); CHECK(d.setDetectorType() == test::type); d.freeSharedMemory(); } TEST_CASE("Set control port then create a new object with this control port", "[.integration][.single]") { /* TODO! Standard port but should not be hardcoded Is this the best way to initialize the detectors Using braces to make the object go out of scope */ int old_cport = DEFAULT_TCP_CNTRL_PORTNO; int old_sport = DEFAULT_TCP_STOP_PORTNO; int new_cport = 1993; int new_sport = 2000; { Module d(test::type); d.setHostname(test::hostname); CHECK(d.getControlPort() == old_cport); d.setControlPort(new_cport); CHECK(d.getStopPort() == old_sport); d.setStopPort(new_sport); d.freeSharedMemory(); } { Module d(test::type); d.setHostname(test::hostname); d.setControlPort(new_cport); d.setStopPort(new_sport); CHECK(d.getControlPort() == new_cport); CHECK(d.getStopPort() == new_sport); // Reset standard ports d.setControlPort(old_cport); d.setStopPort(old_sport); d.freeSharedMemory(); } Module d(test::type); d.setHostname(test::hostname); CHECK(d.getStopPort() == DEFAULT_TCP_STOP_PORTNO); d.freeSharedMemory(); } TEST_CASE("single EIGER detector no receiver basic set and get", "[.integration][eiger]") { // TODO! this test should take command line arguments for config SingleDetectorConfig c; // Read type by connecting to the detector auto type = Module::getTypeFromDetector(c.hostname); CHECK(type == c.type_enum); // Create Module of said type and set hostname and detector online Module d(type); CHECK(d.getDetectorTypeAsEnum() == type); CHECK(d.getDetectorTypeAsString() == c.type_string); d.setHostname(c.hostname); CHECK(d.getHostname() == c.hostname); CHECK(d.getUseReceiverFlag() == false); CHECK_NOTHROW(d.checkDetectorVersionCompatibility()); // Setting and reading exposure time auto t = 1000000000; d.setExptime(t); CHECK(d.getExptime() == t); // size of an eiger half module with and without gap pixels CHECK(d.getTotalNumberOfChannels() == 256 * 256 * 4); CHECK(d.getTotalNumberOfChannels(slsDetectorDefs::dimension::X) == 1024); CHECK(d.getTotalNumberOfChannels(slsDetectorDefs::dimension::Y) == 256); // CHECK(d.getTotalNumberOfChannels(slsDetectorDefs::dimension::Z) == 1); CHECK(d.getTotalNumberOfChannelsInclGapPixels( slsDetectorDefs::dimension::X) == 1024); CHECK(d.getTotalNumberOfChannelsInclGapPixels( slsDetectorDefs::dimension::Y) == 256); // CHECK(d.getTotalNumberOfChannelsInclGapPixels(slsDetectorDefs::dimension::Z) // == 1); CHECK(d.getNChans() == 256 * 256); CHECK(d.getNChans(slsDetectorDefs::dimension::X) == 256); CHECK(d.getNChans(slsDetectorDefs::dimension::Y) == 256); // CHECK(d.getNChans(slsDetectorDefs::dimension::Z) == 1); CHECK(d.getNChips() == 4); CHECK(d.getNChips(slsDetectorDefs::dimension::X) == 4); CHECK(d.getNChips(slsDetectorDefs::dimension::Y) == 1); // CHECK(d.getNChips(slsDetectorDefs::dimension::Z) == 1); d.freeSharedMemory(); } TEST_CASE("Locking mechanism and last ip", "[.integration][.single]") { Module d(test::type); d.setHostname(test::hostname); // Check that detector server is unlocked then lock CHECK(d.lockServer() == 0); d.lockServer(1); CHECK(d.lockServer() == 1); // Can we still access the detector while it's locked auto t = 1300000000; d.setExptime(t); CHECK(d.getExptime() == t); // unlock again and free d.lockServer(0); CHECK(d.lockServer() == 0); CHECK(d.getLastClientIP() == test::my_ip); d.freeSharedMemory(); } TEST_CASE("Set settings", "[.integration][.single]") { Module d(test::type); d.setHostname(test::hostname); CHECK(d.setSettings(defs::STANDARD) == defs::STANDARD); } TEST_CASE("Timer functions", "[.integration][cli]") { // FRAME_NUMBER, /**< number of real time frames: total number of // acquisitions is number or frames*number of triggers */ ACQUISITION_TIME, // /**< exposure time */ FRAME_PERIOD, /**< period between exposures */ // DELAY_AFTER_TRIGGER, /**< delay between trigger and start of exposure or // readout (in triggered mode) */ GATES_NUMBER, /**< number of gates per // frame (in gated mode) */ TRIGGER_NUMBER, /**< number of triggers: total // number of acquisitions is number or frames*number of triggers */ // ACTUAL_TIME, /**< Actual time of the detector's internal timer */ // MEASUREMENT_TIME, /**< Time of the measurement from the detector (fifo) // */ // PROGRESS, /**< fraction of measurement elapsed - only get! */ // MEASUREMENTS_NUMBER, // FRAMES_FROM_START, // FRAMES_FROM_START_PG, // SAMPLES, // SUBFRAME_ACQUISITION_TIME, /**< subframe exposure time */ // STORAGE_CELL_NUMBER, /** list = m.getReceiverDbitList(); list.clear(); for (int i = 0; i < 10; ++i) list.push_back(i); m.setReceiverDbitList(list); CHECK(m.getReceiverDbitList().size() == 10); list.push_back(64); CHECK_THROWS_AS(m.setReceiverDbitList(list), RuntimeError); CHECK_THROWS_WITH(m.setReceiverDbitList(list), Catch::Matchers::Contains("be between 0 and 63")); list.clear(); for (int i = 0; i < 65; ++i) list.push_back(i); CHECK(list.size() == 65); CHECK_THROWS_WITH(m.setReceiverDbitList(list), Catch::Matchers::Contains("be greater than 64")); list.clear(); m.setReceiverDbitList(list); CHECK(m.getReceiverDbitList().empty()); // adcinvert m.setADCInvert(0); CHECK(m.getADCInvert() == 0); m.setADCInvert(5); CHECK(m.getADCInvert() == 5); m.setADCInvert(-1); CHECK(m.getADCInvert() == -1); // ext sampling reg m.setExternalSamplingSource(0); CHECK(m.getExternalSamplingSource() == 0); m.setExternalSamplingSource(62); CHECK(m.getExternalSamplingSource() == 62); CHECK_THROWS_WITH(m.setExternalSamplingSource(64), Catch::Matchers::Contains("be 0-63")); CHECK(m.getExternalSamplingSource() == 62); m.setExternalSampling(1); CHECK(m.getExternalSampling() == 1); m.setExternalSampling(0); CHECK(m.getExternalSampling() == 0); m.setExternalSampling(1); CHECK(m.getExternalSampling() == 1); CHECK(m.readRegister(0x7b) == 0x1003E); } TEST_CASE("Eiger or Jungfrau nextframenumber", "[.eigerintegration][.jungfrauintegration][nextframenumber]") { SingleDetectorConfig c; // pick up multi detector from shm id 0 DetectorImpl m(0); // ensure ctb detector type, hostname and online REQUIRE( ((m.getDetectorTypeAsEnum() == slsDetectorDefs::detectorType::EIGER) || (m.getDetectorTypeAsEnum() == slsDetectorDefs::detectorType::JUNGFRAU))); REQUIRE(m.getHostname() == c.hostname); CHECK(m.setNumberOfFrames(1) == 1); // starting fnum uint64_t val = 8; m.setNextFrameNumber(val); CHECK(m.getNextFrameNumber() == val); CHECK(m.acquire() == slsDetectorDefs::OK); CHECK(m.getReceiverCurrentFrameIndex() == val); ++val; CHECK(m.acquire() == slsDetectorDefs::OK); CHECK(m.getReceiverCurrentFrameIndex() == val); CHECK_THROWS_AS(m.setNextFrameNumber(0), RuntimeError); if (m.getDetectorTypeAsString() == "Eiger") { val = 281474976710655; } else if (m.getDetectorTypeAsString() == "Jungfrau") { val = 18446744073709551615; } m.setNextFrameNumber(val); CHECK(m.getNextFrameNumber() == val); CHECK(m.acquire() == slsDetectorDefs::OK); CHECK(m.getReceiverCurrentFrameIndex() == val); CHECK(m.getNextFrameNumber() == (val + 1)); } TEST_CASE("Eiger partialread", "[.eigerintegration][partialread]") { SingleDetectorConfig c; // pick up multi detector from shm id 0 DetectorImpl m(0); // ensure detector type, hostname REQUIRE( (m.getDetectorTypeAsEnum() == slsDetectorDefs::detectorType::EIGER)); REQUIRE(m.getHostname() == c.hostname); m.setDynamicRange(16); m.enableTenGigabitEthernet(0); m.setPartialReadout(256); CHECK(m.getPartialReadout() == 256); m.setPartialReadout(1); CHECK(m.getPartialReadout() == 1); m.setDynamicRange(8); m.setPartialReadout(256); CHECK(m.getPartialReadout() == 256); CHECK_THROWS_AS(m.setPartialReadout(1), RuntimeError); CHECK(m.getPartialReadout() == 256); CHECK_THROWS_AS(m.setPartialReadout(0), RuntimeError); m.setPartialReadout(256); } } // namespace sls