| 1 | /** @file | 
| 2 | * $Author: kusanagi $ | 
| 3 | * $Date: 2005/08/29 09:46:13 $ | 
| 4 | * $Revision: 5.0 $ | 
| 5 | * | 
| 6 | * Implementation of the functions of a sample Algorithm class. | 
| 7 | * This file can be used as a templace to develop your own algorithm. | 
| 8 | */ | 
| 9 |  | 
| 10 | #include <log4cxx/logger.h> | 
| 11 | #include <fstream> | 
| 12 | #include "EventReader.h" | 
| 13 | #include "ReaderAlgorithms.h" | 
| 14 |  | 
| 15 |  | 
| 16 | extern "C" { | 
| 17 | #include "CRC.h" | 
| 18 | } | 
| 19 |  | 
| 20 | using namespace pamela; | 
| 21 | using namespace pamela::techmodel; | 
| 22 |  | 
| 23 | static log4cxx::LoggerPtr logger = log4cxx::Logger::getLogger(_T("pamela.techmodel.EventReader")); | 
| 24 |  | 
| 25 | unsigned int EventReader::maxPackets     = 0; | 
| 26 | unsigned int EventReader::prevPckCounter = 0; | 
| 27 | unsigned int EventReader::prevPckOBT     = 0; | 
| 28 |  | 
| 29 |  | 
| 30 | /** | 
| 31 | * Constructor. | 
| 32 | */ | 
| 33 | EventReader::EventReader(int packetsLimit = -1): | 
| 34 | TechmodelAlgorithm(0, "TechmodelEventReader"){ | 
| 35 | EventReader::maxPackets = packetsLimit; | 
| 36 | logger->debug(_T("Constructor")); | 
| 37 | Header = new EventHeader(); | 
| 38 |  | 
| 39 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::PhysEndRun,      new PhysEndRunReader)); | 
| 40 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibCalPulse1,  new CalibCalPulse1Reader)); | 
| 41 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibCalPulse2,  new CalibCalPulse2Reader)); | 
| 42 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Physics,         new PhysicsReader)); | 
| 43 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibTrk1,       new CalibTrk1Reader)); | 
| 44 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibTrk2,       new CalibTrk2Reader)); | 
| 45 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibTof,        new CalibTofReader)); | 
| 46 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibS4,         new CalibS4Reader)); | 
| 47 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibCalPed,     new CalibCalPedReader)); | 
| 48 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Calib1_Ac1,      new Calib1_Ac1Reader)); | 
| 49 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Calib2_Ac1,      new Calib2_Ac1Reader)); | 
| 50 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Calib1_Ac2,      new Calib1_Ac2Reader)); | 
| 51 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Calib2_Ac2,      new Calib2_Ac2Reader)); | 
| 52 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::RunHeader,       new RunHeaderReader)); | 
| 53 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::RunTrailer,      new RunTrailerReader)); | 
| 54 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibHeader,     new CalibHeaderReader)); | 
| 55 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalibTrailer,    new CalibTrailerReader)); | 
| 56 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::InitHeader,      new InitHeaderReader)); | 
| 57 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::InitTrailer,     new InitTrailerReader)); | 
| 58 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::EventTrk,        new EventTrkReader)); | 
| 59 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TestTrk,         new TestTrkReader)); | 
| 60 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TestTof,         new TestTofReader)); | 
| 61 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Log,             new LogReader)); | 
| 62 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::VarDump,         new VarDumpReader)); | 
| 63 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::ArrDump,         new ArrDumpReader)); | 
| 64 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TabDump,         new TabDumpReader)); | 
| 65 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Tmtc,            new TmtcReader)); | 
| 66 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Mcmd,            new McmdReader)); | 
| 67 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::ForcedFECmd,     new ForcedFECmdReader)); | 
| 68 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Ac1Init,         new Ac1InitReader)); | 
| 69 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalInit,         new CalInitReader)); | 
| 70 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TrkInit,         new TrkInitReader)); | 
| 71 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TofInit,         new TofInitReader)); | 
| 72 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TrgInit,         new TrgInitReader)); | 
| 73 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::NdInit,          new NdInitReader)); | 
| 74 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::S4Init,          new S4InitReader)); | 
| 75 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::Ac2Init,         new Ac2InitReader)); | 
| 76 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::CalAlarm,        new CalAlarmReader)); | 
| 77 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::AcAlarm,         new AcAlarmReader)); | 
| 78 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TrkAlarm,        new TrkAlarmReader)); | 
| 79 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TrgAlarm,        new TrgAlarmReader)); | 
| 80 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TofAlarm,        new TofAlarmReader)); | 
| 81 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::S4Alarm,         new S4AlarmReader)); | 
| 82 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TsbT,            new TsbTReader)); | 
| 83 | TechmodelAlgorithmMap.insert(AlgorithmMap::value_type(PacketType::TsbB,            new TsbBReader)); | 
| 84 | } | 
| 85 |  | 
| 86 | /** | 
| 87 | * Get a string with the version info of the algorithm. | 
| 88 | */ | 
| 89 | std::string EventReader::GetVersionInfo(void) const { | 
| 90 | return | 
| 91 | "$Header: /home/cvsmanager/yoda/techmodel/EventReader.cpp,v 5.0 2005/08/29 09:46:13 kusanagi Exp $\n"; | 
| 92 | } | 
| 93 |  | 
| 94 | /** | 
| 95 | * Initialize the algorithm with a special run. This will initialize the | 
| 96 | * event reader routines for all packet types. | 
| 97 | */ | 
| 98 | void EventReader::Init(PamelaRun *run) { | 
| 99 | SetInputStream(run); | 
| 100 |  | 
| 101 | //Create the structure of directories and create xxx.Header.root files | 
| 102 | run->WriteHeaders(this, &Header); | 
| 103 |  | 
| 104 | //Create the xxx.root in it's specific directory | 
| 105 | for (AlgorithmMap::iterator i = TechmodelAlgorithmMap.begin(); | 
| 106 | i != TechmodelAlgorithmMap.end(); i++) { | 
| 107 | oss.str(""); | 
| 108 | oss << "Initializing algo " << i->second->GetAlgorithmName(); | 
| 109 | logger->debug(oss.str().c_str()); | 
| 110 | i->second->Init(run); | 
| 111 | } | 
| 112 | Run = dynamic_cast<TechmodelPamelaRun*>(run); | 
| 113 | } | 
| 114 |  | 
| 115 | static void SkipToNextHeader(ifstream *); | 
| 116 |  | 
| 117 | /** | 
| 118 | * Read the next event header, call the reader algorithms that | 
| 119 | * correspond to its packet type, and read the event trailer. | 
| 120 | */ | 
| 121 | void EventReader::RunEvent(int EventNumber) { | 
| 122 | stringstream oss; | 
| 123 | int step = 0; | 
| 124 | const PacketType* type; | 
| 125 | while (!InputFile->eof() && ((step++ < maxPackets) || (maxPackets == 0))){ | 
| 126 | try { | 
| 127 | if (FindStart()) { | 
| 128 | UnpackPscuHeader(); | 
| 129 | type = Header->GetPscuHeader()->GetPacketType(); | 
| 130 | AlgorithmMap::iterator i = TechmodelAlgorithmMap.find(type); | 
| 131 | if (i != TechmodelAlgorithmMap.end()) { | 
| 132 | TechmodelAlgorithm *EventAlgorithm(i->second); | 
| 133 | EventAlgorithm->RunEvent(EventNumber, Header->GetPscuHeader()->GetPacketLenght()); | 
| 134 | Run->FillTrees(type); | 
| 135 | Header->GetCounter()->Increment(type); | 
| 136 | logger->info(Header->GetPscuHeader()->Print()); | 
| 137 | } else { | 
| 138 | oss.str(""); | 
| 139 | oss << "\n No way to read events of type  " << type->GetName().c_str() << Header->GetPscuHeader()->Print(); | 
| 140 | throw NotExistingAlgorithmException(oss.str().c_str()); //to exctract to an higher level and delete the logger! | 
| 141 | } | 
| 142 | } | 
| 143 | // In case of exception have to save the packet in a specific root file?? | 
| 144 | } catch (NotExistingAlgorithmException exc) { | 
| 145 | oss.str(""); | 
| 146 | oss << exc.print() << " " << Header->GetPscuHeader()->Print(); | 
| 147 | logger->error(oss.str().c_str()); | 
| 148 | } catch (WrongCRCHeaderException exc) { | 
| 149 | oss.str(""); | 
| 150 | oss << exc.print(); | 
| 151 | logger->error(oss.str().c_str()); | 
| 152 | } catch (WrongCRCException exc) { | 
| 153 | oss.str(""); | 
| 154 | oss << exc.print() << " " << Header->GetPscuHeader()->Print(); | 
| 155 | logger->error(oss.str().c_str()); | 
| 156 | InputFile->seekg( (-1)*(Header->GetPscuHeader()->GetPacketLenght() + 15) , std::ios::cur); | 
| 157 | } catch (UnidentifiedPacketException exc) { | 
| 158 | oss.str(""); | 
| 159 | oss << exc.print() << " " << Header->GetPscuHeader()->Print(); | 
| 160 | logger->error(oss.str().c_str()); | 
| 161 | } catch (NotExistingCounterException exc) { | 
| 162 | oss.str(""); | 
| 163 | oss << exc.print() << " " << Header->GetPscuHeader()->Print(); | 
| 164 | logger->error(oss.str().c_str()); | 
| 165 | } catch (LengthException exc) { | 
| 166 | oss.str(""); | 
| 167 | oss << exc.print() << " " << Header->GetPscuHeader()->Print(); | 
| 168 | logger->error(oss.str().c_str()); | 
| 169 | } catch (BackwardCounterException exc) { | 
| 170 | oss.str(""); | 
| 171 | oss << exc.print() << " " << Header->GetPscuHeader()->Print(); | 
| 172 | logger->error(oss.str().c_str()); | 
| 173 | } catch (...) { | 
| 174 | logger->error("Couldn't read the event. Skipping to the next header. \n"); | 
| 175 | } | 
| 176 | if ((step%1000) == 0) std::cout << step/1000 << "K \n"; | 
| 177 | oss.str(""); | 
| 178 | oss << "----endPck " << Header->GetPscuHeader()->GetCounter() << "\n"; | 
| 179 | logger->info(oss.str().c_str()); | 
| 180 | } | 
| 181 | Header->GetCounter()->PrintCounters(); | 
| 182 | } | 
| 183 |  | 
| 184 | /** | 
| 185 | * Unpack the PSCU header from a file into the structure. | 
| 186 | */ | 
| 187 | void EventReader::UnpackPscuHeader(void) throw (WrongCRCHeaderException, LengthException, BackwardCounterException) { | 
| 188 | stringstream oss; | 
| 189 | int response = 0; | 
| 190 | char buff[16]; | 
| 191 | InputFile->read(buff, sizeof(buff)); | 
| 192 |  | 
| 193 |  | 
| 194 | unsigned char PacketId1    = buff[3]; | 
| 195 | unsigned char PacketId2    = buff[4]; | 
| 196 | unsigned int  Counter      = (((UINT32)buff[5]<<16)&0x00FF0000) + (((UINT32)buff[6]<<8)&0x0000FF00) + (((UINT32)buff[7])&0x000000FF); | 
| 197 | unsigned int  OrbitalTime  = (((UINT32)buff[8]<<24)&0xFF000000) + (((UINT32)buff[9]<<16)&0x00FF0000) +  (((UINT32)buff[10]<<8)&0x0000FF00) + (((UINT32)buff[11])&0x000000FF); | 
| 198 | unsigned int  PacketLenght = (((UINT32)buff[12]<<16)&0x00FF0000) +  (((UINT32)buff[13]<<8)&0x0000FF00) + (((UINT32)buff[14])&0x000000FF); | 
| 199 | unsigned char CRC          = buff[15]; | 
| 200 | unsigned char FileOffset   = buff[15]; | 
| 201 |  | 
| 202 |  | 
| 203 | if (Counter < prevPckCounter){ | 
| 204 | response = prevPckCounter - Counter; | 
| 205 | //oss.str(""); | 
| 206 | //oss << "Packet counter is less than before of " << (prevPckCounter - Counter); | 
| 207 | //throw BackwardCounterException(oss.str().c_str()); | 
| 208 | //logger->error(oss.str().c_str()); | 
| 209 | } | 
| 210 |  | 
| 211 | if (Counter > prevPckCounter + 1){ | 
| 212 | oss.str(""); | 
| 213 | oss << "Packet counter is greater than before of " << (Counter - prevPckCounter); | 
| 214 | logger->error(oss.str().c_str()); | 
| 215 | } | 
| 216 |  | 
| 217 | if ((OrbitalTime == prevPckOBT) & (PacketId1 == 0x10)){ | 
| 218 | oss.str(""); | 
| 219 | oss << "Onboard Time of this packet is equal to the previous packet OBT"; | 
| 220 | logger->error(oss.str().c_str()); | 
| 221 | logger->error(Header->GetPscuHeader()->Print()); | 
| 222 | } | 
| 223 |  | 
| 224 | if (OrbitalTime < prevPckOBT){ | 
| 225 | oss.str(""); | 
| 226 | oss << " Onboard Time is less than before of " << (prevPckOBT - OrbitalTime); | 
| 227 | logger->error(oss.str().c_str()); | 
| 228 | } | 
| 229 |  | 
| 230 | if (((BYTE)CM_Compute_CRC16(0, (BYTE*)&buff, 15) == (BYTE)buff[15]) && (PacketId1 == PacketId2)){ | 
| 231 | prevPckCounter = Counter; | 
| 232 | prevPckOBT     = OrbitalTime; | 
| 233 | long int initPos = InputFile->tellg(); | 
| 234 | long int finalPos; | 
| 235 | Header->GetPscuHeader()->SetPacketId(PacketId1, PacketId2); | 
| 236 | Header->GetPscuHeader()->SetCounter(Counter); | 
| 237 | Header->GetPscuHeader()->SetOrbitalTime(OrbitalTime); | 
| 238 | //PacketLength is the length of the whole DATApacket starting from the first byte after the header | 
| 239 | //plus the CRC legth (which varies for each type of packet) | 
| 240 | Header->GetPscuHeader()->SetPacketLenght(PacketLenght); | 
| 241 | Header->GetPscuHeader()->SetCRC(CRC); | 
| 242 | Header->GetPscuHeader()->SetFileOffset(((long int)(InputFile->tellg()) - 16)); | 
| 243 | } else { | 
| 244 | /*Here i should extract the block of Data for later analysis */ | 
| 245 | InputFile->seekg(-(13), std::ios::cur); | 
| 246 | oss.str(""); | 
| 247 | oss << "CRC Header Error on packet:" << PscuHeader::Print(buff); | 
| 248 | throw WrongCRCHeaderException(oss.str().c_str()); | 
| 249 | } | 
| 250 |  | 
| 251 | if (response > 0){ | 
| 252 | oss.str(""); | 
| 253 | oss << "Packet counter is less than before of " << response; | 
| 254 | throw BackwardCounterException(oss.str().c_str()); | 
| 255 | } | 
| 256 | } | 
| 257 |  | 
| 258 | /** | 
| 259 | * Unpack the trailer of a PSCU event into the structure. | 
| 260 | */ | 
| 261 | void EventReader::UnpackPscuTrailer(void) throw (std::exception) { | 
| 262 |  | 
| 263 | } | 
| 264 |  | 
| 265 | /** | 
| 266 | * Find the next starting poin for the PSCU event looking for a {0xFA, 0xFE, 0xDE} sequence | 
| 267 | */ | 
| 268 | bool EventReader::FindStart(void) throw (std::exception) { | 
| 269 | //search an hexadecimal sequence in a file | 
| 270 | //subSign    ------> pointer to the sequence buffer | 
| 271 | //subSignDim ------> dimension of the buffer | 
| 272 | // at exit | 
| 273 | // return true if founds a match (else false) | 
| 274 | // subF point rigth after the match, if found. Else EOF. | 
| 275 | //Maurizio 15/11/2002----------------------- | 
| 276 | const int subSignDim = 3; | 
| 277 | const unsigned char subSign[subSignDim]={0xFA, 0xFE, 0xDE}; | 
| 278 | //------------------------------------------ | 
| 279 | int subIndex = 0; | 
| 280 | char dataByte; | 
| 281 |  | 
| 282 | int  buffSize = 64; | 
| 283 | int  index = 0; | 
| 284 | int  loop = -1; | 
| 285 | char buffer[buffSize]; | 
| 286 | bool flagOverPad = false; | 
| 287 |  | 
| 288 | while (!InputFile->eof()) { | 
| 289 | InputFile->read(buffer, sizeof(buffer)); | 
| 290 | index = 0; | 
| 291 | loop++; | 
| 292 | while (index < buffSize){ | 
| 293 | dataByte = buffer[index++]; | 
| 294 | if (dataByte == (char)(*(subSign+subIndex))){ | 
| 295 | if (subIndex++ == (subSignDim-1)) { | 
| 296 | InputFile->seekg( (index - (subIndex + buffSize)), std::ios::cur); | 
| 297 | if (flagOverPad){ | 
| 298 | oss.str(""); | 
| 299 | oss << "\n This packet beginning is farther than 64 byte from the end of the previous." | 
| 300 | << "\n Below the is the last already unpacked packet"; | 
| 301 | logger->error(oss.str().c_str()); | 
| 302 | logger->error(Header->GetPscuHeader()->Print()); | 
| 303 | } | 
| 304 | return true; | 
| 305 | } | 
| 306 | } else { | 
| 307 | index = index - (subIndex); | 
| 308 | subIndex = 0; | 
| 309 | } | 
| 310 | } | 
| 311 | //Needs to guarantee the overap of the buffer(s) in several loop | 
| 312 | flagOverPad = true; | 
| 313 | InputFile->seekg( (-1)*(subSignDim + 1) , std::ios::cur); | 
| 314 | } | 
| 315 | return false; | 
| 316 | } | 
| 317 |  | 
| 318 | ClassImp(EventReader) |