/* This file is part of MAUS: http://micewww.pp.rl.ac.uk:8080/projects/maus * * MAUS is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * MAUS is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with MAUS. If not, see . * */ #include #include #include "src/common_cpp/Utils/JsonWrapper.hh" #include "src/common_cpp/Utils/CppErrorHandler.hh" #include "Interface/Squeak.hh" #include "Utils/Exception.hh" #include "Interface/dataCards.hh" #include "API/PyWrapMapBase.hh" #include "src/map/MapCppTOFSlabHits/MapCppTOFSlabHits.hh" namespace MAUS { PyMODINIT_FUNC init_MapCppTOFSlabHits(void) { PyWrapMapBase::PyWrapMapBaseModInit ("MapCppTOFSlabHits", "", "", "", ""); } MapCppTOFSlabHits::MapCppTOFSlabHits() : MapBase("MapCppTOFSlabHits") { } void MapCppTOFSlabHits::_birth(const std::string& argJsonConfigDocument) { // Check if the JSON document can be parsed, else return error only _stationKeys.push_back("tof0"); _stationKeys.push_back("tof1"); _stationKeys.push_back("tof2"); // Check if the JSON document can be parsed, else return error only // JsonCpp setup Json::Value configJSON; configJSON = JsonWrapper::StringToJson(argJsonConfigDocument); // this will contain the configuration _tdcV1290_conversion_factor = JsonWrapper::GetProperty(configJSON, "TOFtdcConversionFactor", JsonWrapper::realValue).asDouble(); } void MapCppTOFSlabHits::_death() {} void MapCppTOFSlabHits::_process(Json::Value* document) const { // JsonCpp setup Json::Value& root = *document; Json::Value xEventType = JsonWrapper::GetProperty(root, "daq_event_type", JsonWrapper::stringValue); if (xEventType== "physics_event" || xEventType == "calibration_event") { Json::Value events = JsonWrapper::GetProperty(root, "recon_events", JsonWrapper::arrayValue); // Loop over each station. for (unsigned int n_event = 0; n_event < events.size(); n_event++) { Json::Value xDocTofEvent = JsonWrapper::GetItem(events, n_event, JsonWrapper::objectValue); xDocTofEvent = JsonWrapper::GetProperty(xDocTofEvent, "tof_event", JsonWrapper::objectValue); if (root["recon_events"][n_event]["tof_event"].isMember("tof_digits")) { root["recon_events"][n_event]["tof_event"]["tof_slab_hits"] = Json::Value(Json::objectValue); for (unsigned int n_station = 0; n_station < _stationKeys.size(); n_station++) { Json::Value xDocPartEvent = JsonWrapper::GetProperty(xDocTofEvent, "tof_digits", JsonWrapper::objectValue); // Ack! sometimes tofn is a nullValue xDocPartEvent = JsonWrapper::GetProperty(xDocPartEvent, _stationKeys[n_station], JsonWrapper::anyValue); Json::Value xDocSlabHits = makeSlabHits(xDocPartEvent); root["recon_events"][n_event]["tof_event"]["tof_slab_hits"] [_stationKeys[n_station]] = xDocSlabHits; } } } } } Json::Value MapCppTOFSlabHits::makeSlabHits(Json::Value xDocPartEvent) const { Json::Value xDocSlabHits; if (xDocPartEvent.isArray()) { int n_digits = xDocPartEvent.size(); // std::cout << xDocPartEvent << std::endl; // Create a map of all hited PMTs. std::map xDigitPos; std::map::iterator it; // Loop ovew the digits. for (int Digit = 0; Digit < n_digits; Digit++) { // Get the digit. Json::Value xThisDigit = JsonWrapper::GetItem(xDocPartEvent, Digit, JsonWrapper::objectValue); std::string xKeyStr = JsonWrapper::GetProperty(xThisDigit, "tof_key", JsonWrapper::stringValue).asString(); TOFChannelKey xKey(xKeyStr); // Check if we already have this key in the map. This will mean that there are two // digits in the same PMT. it = xDigitPos.find(xKeyStr); if (it != xDigitPos.end()) { // We have two digits coming from one PMT. // Get the other digit and check which comes first in time. Json::Value xOtherDigit = JsonWrapper::GetItem(xDocPartEvent, xDigitPos[xKeyStr], JsonWrapper::objectValue); int time_thisDigit = JsonWrapper::GetProperty(xThisDigit, "leading_time", JsonWrapper::intValue).asInt(); int time_otherDigit = JsonWrapper::GetProperty(xOtherDigit, "leading_time", JsonWrapper::intValue).asInt(); // Only the digit that comes first in time will be used to create a slab hit. // Most probably the second one is created due to afterpulsing of the PMT. if (time_thisDigit < time_otherDigit) // Else do nothing. xDigitPos[xKeyStr] = Digit; } else { // Add this PMT to the map. xDigitPos[xKeyStr] = Digit; } } // Now loop over the map of hited PMTs and create Slab hits. while ( xDigitPos.size() > 1 ) { // Get the first element of the map and check if we have a hit // at the opposite side of the slab. it = xDigitPos.begin(); TOFChannelKey xKey(it->first); // Get the digit. Json::Value xThisDigit = JsonWrapper::GetItem(xDocPartEvent, it->second, JsonWrapper::objectValue); // Get the opposite PMT coded as string. std::string xOppositPmtKey_str = xKey.GetOppositeSidePMTStr(); if (xDigitPos.find(xOppositPmtKey_str) != xDigitPos.end()) { Json::Value xDocTheSlabHit; Json::Value xOtherDigit = JsonWrapper::GetItem(xDocPartEvent, xDigitPos[xOppositPmtKey_str], JsonWrapper::objectValue); // Create Slab hit. if (xKey.pmt() == 0) { xDocTheSlabHit = fillSlabHit(xThisDigit, xOtherDigit); } if (xKey.pmt() == 1) { xDocTheSlabHit = fillSlabHit(xOtherDigit, xThisDigit); } xDocSlabHits.append(xDocTheSlabHit); // Erase both used hits from the map. xDigitPos.erase(it); xDigitPos.erase(xOppositPmtKey_str); } else { // Erese this hit from the map. xDigitPos.erase(it); } } } // std::cout << xDocSlabHits <(xTimeDigit0 - xTriggerReqDigit0)); double time_digit1 = _tdcV1290_conversion_factor*(static_cast(xTimeDigit1 - xTriggerReqDigit1)); xDocPMT0["raw_time"] = time_digit0; xDocPMT1["raw_time"] = time_digit1; xDocSlabHit["pmt0"] = xDocPMT0; xDocSlabHit["pmt1"] = xDocPMT1; double xRawTime = (time_digit0 + time_digit1)/2.; xDocSlabHit["raw_time"] = xRawTime; return xDocSlabHit; } }