CoMPASS_BinReader/BinReader.h

381 lines
9.2 KiB
C
Raw Normal View History

2022-12-01 12:50:48 -05:00
#ifndef BINREADER_H
#define BINREADER_H
#include <stdio.h> /// for FILE
#include <iostream>
#include <fstream>
#include <stdlib.h>
#include <string.h>
#include "TString.h"
#include "TBenchmark.h"
#include "TMath.h"
struct Data{
2024-07-17 15:46:23 -04:00
uint16_t Header; /// only the last 4 bits
uint16_t BoardID;
uint16_t Channel;
2022-12-01 12:50:48 -05:00
2024-07-17 15:46:23 -04:00
uint64_t TimeStamp;
uint16_t Energy;
uint16_t Energy_short;
uint32_t Flags;
2022-12-01 12:50:48 -05:00
2024-07-17 15:46:23 -04:00
uint8_t WaveformCode;
uint32_t NSample;
std::vector<uint16_t> Trace;
2022-12-01 12:50:48 -05:00
void Clear(){
2024-07-17 15:46:23 -04:00
Header = 0xCAE0;
2022-12-01 12:50:48 -05:00
BoardID = 0;
Channel = 0;
TimeStamp = 0;
Energy = 0;
2024-07-17 15:46:23 -04:00
Energy_short = 0;
2022-12-01 12:50:48 -05:00
Flags = 0;
WaveformCode = 0;
NSample = 0;
2024-07-17 15:46:23 -04:00
Trace.clear();
2022-12-01 12:50:48 -05:00
};
void Print(){
printf("header : 0x%X \n", Header);
printf(" Board : %u , Channel : %u\n", BoardID, Channel);
2024-07-18 19:02:22 -04:00
if( Header & 4 ) {
2024-10-23 00:19:23 -04:00
printf("Energy : %5u, Energy2 : %5u, TimeStamp : %16lu ps\n", Energy, Energy_short, TimeStamp);
2024-07-18 19:02:22 -04:00
}else{
2024-10-23 00:19:23 -04:00
printf("Energy : %5u, TimeStamp : %16lu ps\n", Energy, TimeStamp);
2024-07-18 19:02:22 -04:00
}
2024-09-03 13:41:36 -04:00
printf(" Flag : 0x%08X\n", Flags);
2024-07-17 15:46:23 -04:00
if( (Header & 0x8 ) >= 1 ){ /// is waevform exist
2022-12-01 12:50:48 -05:00
printf(" Wave form code : %d , nSample : %d\n", WaveformCode, NSample);
for( unsigned int i = 0 ; i < NSample ; i++){
2022-12-01 12:50:48 -05:00
printf("%4d | %d \n", i, Trace[i]);
}
}
}
};
//^################################################
2022-12-01 12:50:48 -05:00
class BinReader{
public:
Data data;
private:
FILE * inFile;
long int inFileSize;
long int inFilePos;
bool endOfFile;
bool isOpened;
2024-07-18 19:02:22 -04:00
Long64_t hitID;
long int numHit;
int64_t timeOffset;
2022-12-01 12:50:48 -05:00
TBenchmark gClock;
long int inFilePosPrecent[10];
Long64_t blockIDPrecent[10];
bool isHeaderOK;
2024-09-03 13:41:36 -04:00
bool isOldFormat;
bool isNoHeaderFormat;
2022-12-01 12:50:48 -05:00
template <typename T> int FillData(T * dataItem); // if successful, return 1, else 0; cannot fill Trace
///============================================ Methods
public:
BinReader();
BinReader(TString inFileName, int64_t timeOffset = 0, bool noHeader = false);
2022-12-01 12:50:48 -05:00
~BinReader();
void OpenFile(TString inFileName, int64_t timeOffset = 0, bool noHeader = false);
void SetCustomHeader(uint16_t Header);
// void SetDataFormat(bool has );
2022-12-01 12:50:48 -05:00
void CloseFile();
void UpdateFileSize();
bool IsEndOfFile();
2024-07-18 19:02:22 -04:00
bool IsOpen() const {return isOpened;}
long int GetFilePos() const {return inFilePos;}
long int GetFileSize() const {return inFileSize;}
Long64_t GetHitID() const {return hitID;}
Long64_t GetNumHit() const {return numHit;}
2022-12-01 12:50:48 -05:00
2024-07-17 15:46:23 -04:00
int ReadBlock(int skipTrace = 0); /// 0 = default, fill waveform if any. slow
2022-12-01 12:50:48 -05:00
/// 1 = no fill waveform, fast
2024-07-18 19:02:22 -04:00
void ScanNumHit();
2022-12-01 12:50:48 -05:00
void JumptoPrecent(int precent); ///this is offset by 1 block
void PrintStatus(int mod);
void SetTimeOffset(int64_t timeOffset) { this->timeOffset = timeOffset; }
2022-12-01 12:50:48 -05:00
};
//========================== implementation
BinReader::BinReader(){
inFile = 0;
data.Clear();
inFileSize = 0;
inFilePos = 0;
2024-07-18 19:02:22 -04:00
numHit = 0;
hitID = -1;
2022-12-01 12:50:48 -05:00
endOfFile = false;
isOpened = false;
isHeaderOK = false;
2024-09-03 13:41:36 -04:00
isOldFormat = false;
isNoHeaderFormat = false;
2022-12-01 12:50:48 -05:00
}
BinReader::~BinReader(){
if( inFile ) fclose(inFile); /// fclose already delete inFile;
2022-12-01 12:50:48 -05:00
}
BinReader::BinReader(TString inFileName, int64_t timeOffset, bool noHeader){
2022-12-01 12:50:48 -05:00
inFile = 0;
data.Clear();
inFileSize = 0;
inFilePos = 0;
2024-07-18 19:02:22 -04:00
numHit = 0;
hitID = -1;
2022-12-01 12:50:48 -05:00
endOfFile = false;
isOpened = false;
OpenFile(inFileName, timeOffset, noHeader);
2022-12-01 12:50:48 -05:00
}
void BinReader::OpenFile(TString inFileName, int64_t timeOffset, bool noHeader){
2022-12-01 12:50:48 -05:00
inFile = fopen(inFileName, "r");
this->timeOffset = timeOffset;
2022-12-01 12:50:48 -05:00
if( inFile == NULL ){
printf("Cannot read file : %s \n", inFileName.Data());
}else{
fseek(inFile, 0L, SEEK_END);
inFileSize = ftell(inFile);
rewind(inFile); ///back to the File begining
data.Clear();
gClock.Reset();
gClock.Start("timer");
isOpened = true;
// normal format: isHeaderOK = true, isNoHeaderFormat= false, isOldFormat = false
// old format: isHeaderOK = true, isNoHeaderFormat= false, isOldFormat = true
//no header format: isHeaderOK = false, isNoHeaderFormat= true, isOldFormat = false
2022-12-01 12:50:48 -05:00
if( noHeader ){
data.Header = 0x0;
isOldFormat = false;
isHeaderOK = false;
isNoHeaderFormat = true;
return;
}
2022-12-01 12:50:48 -05:00
///============= Read the Header,
/// the header only at the beginning of a file
isHeaderOK = FillData(&data.Header);
if( isHeaderOK == false ){
printf(" header cannot read. \n");
return;
}
///printf("HEADER: 0x%X , 0x%X\n", data.Header, (data.Header >> 4));
2024-09-03 13:41:36 -04:00
if( (data.Header >> 4 ) == 0 ){
printf(" Old format of CoMPASS, 0x%04x\n", data.Header);
isHeaderOK = true;
isOldFormat = true;
isNoHeaderFormat = false;
2024-09-03 13:41:36 -04:00
data.Header = 0xCAE5; // only with energy and energy short
return;
}
2022-12-01 12:50:48 -05:00
if( (data.Header >> 4 ) != 0xCAE ) {
2024-09-03 13:41:36 -04:00
printf(" Header format not right. 0x%04x\n", data.Header);
isOldFormat = false;
2022-12-01 12:50:48 -05:00
isHeaderOK = false;
isNoHeaderFormat = false;
2022-12-01 12:50:48 -05:00
return ;
}
2024-09-25 19:13:49 -04:00
printf(" Header format. 0x%04x\n", data.Header);
2022-12-01 12:50:48 -05:00
isHeaderOK = true;
}
};
void BinReader::SetCustomHeader(uint16_t Header){
data.Header = Header;
isHeaderOK = true;
}
2022-12-01 12:50:48 -05:00
void BinReader::CloseFile(){
if(inFile) fclose(inFile);
inFile = nullptr;
2022-12-01 12:50:48 -05:00
isOpened = false;
data.Clear();
inFileSize = 0;
inFilePos = 0;
2024-07-18 19:02:22 -04:00
numHit = 0;
hitID = -1;
2022-12-01 12:50:48 -05:00
endOfFile = false;
};
void BinReader::UpdateFileSize(){
if( inFile == NULL ) return;
fseek(inFile, 0L, SEEK_END);
inFileSize = ftell(inFile);
fseek(inFile, inFilePos, SEEK_SET);
}
bool BinReader::IsEndOfFile() {
2024-07-18 19:02:22 -04:00
if( endOfFile ) return true;
if( inFile == nullptr ) return true;
2024-07-18 19:02:22 -04:00
return feof(inFile) > 0 ? true: false;
2022-12-01 12:50:48 -05:00
}
template <typename T> int BinReader::FillData(T * dataItem){
if ( fread(dataItem, sizeof(*dataItem), 1, inFile) != 1 ) {
2024-07-18 19:02:22 -04:00
printf("!!!!! problem for reading data\n");
2022-12-01 12:50:48 -05:00
endOfFile = IsEndOfFile();
return -1;
}
///printf("----- 0x%16llX \n", *dataItem);
return 1;
}
2024-07-17 15:46:23 -04:00
int BinReader::ReadBlock(int skipTrace){
if( inFile == nullptr ) return -1;
2022-12-01 12:50:48 -05:00
if( feof(inFile) ) return -1;
if( endOfFile ) return -1;
if( isNoHeaderFormat && data.Header == 0x0 ) {
FillData(&data.BoardID);
FillData(&data.Channel);
FillData(&data.TimeStamp);
if( timeOffset != 0 ) data.TimeStamp += timeOffset;
FillData(&data.Energy);
FillData(&data.Flags);
hitID ++;
inFilePos = ftell(inFile);
if( inFilePos >= inFileSize ) endOfFile = true;
return 1;
}
2022-12-01 12:50:48 -05:00
if( !isHeaderOK ) return -2;
/// see the CoMPASS manual v19, P.67
FillData(&data.BoardID);
FillData(&data.Channel);
FillData(&data.TimeStamp);
if( timeOffset != 0 ) data.TimeStamp += timeOffset;
2022-12-01 12:50:48 -05:00
2024-10-23 00:19:23 -04:00
if( (data.Header & 0x3) ) FillData(&data.Energy);
2022-12-01 12:50:48 -05:00
2024-10-23 00:19:23 -04:00
if( (data.Header & 0x2) ) {
2024-07-17 15:46:23 -04:00
uint64_t dummy = 0;
2022-12-01 12:50:48 -05:00
FillData(&dummy);
}
2024-07-17 15:46:23 -04:00
2024-10-23 00:19:23 -04:00
if( (data.Header & 0x4 ) ) FillData(&data.Energy_short);
2022-12-01 12:50:48 -05:00
FillData(&data.Flags);
2024-09-03 13:41:36 -04:00
if( isOldFormat ) {
uint32_t dummy = 0;
FillData(&dummy);
}
2022-12-01 12:50:48 -05:00
/// check is wave form exist
2024-10-23 00:19:23 -04:00
if( (data.Header & 0x8) ){
2022-12-01 12:50:48 -05:00
FillData(&data.WaveformCode);
FillData(&data.NSample);
2024-07-17 15:46:23 -04:00
if( skipTrace == 0 ){
// if ( fread(data.Trace, sizeof(data.Trace), 1, inFile) != 1 ) endOfFile = IsEndOfFile();
for( unsigned int i = 0; i < data.NSample; i++ ){
2024-07-17 15:46:23 -04:00
uint16_t haha;
2024-07-18 19:02:22 -04:00
size_t dummy = fread(&haha, 2, 1, inFile);
2024-07-17 15:46:23 -04:00
data.Trace.push_back(haha);
}
2022-12-01 12:50:48 -05:00
}else{ /// skip trace
fseek(inFile, inFilePos + data.NSample*2, SEEK_SET); /// 2 becasue each trace is 2 bytes
}
}
2024-07-18 19:02:22 -04:00
hitID ++;
2022-12-01 12:50:48 -05:00
inFilePos = ftell(inFile);
2024-07-17 15:46:23 -04:00
if( inFilePos >= inFileSize ) endOfFile = true;
2022-12-01 12:50:48 -05:00
return 1;
}
2024-07-18 19:02:22 -04:00
void BinReader::ScanNumHit(){
2022-12-01 12:50:48 -05:00
2024-07-18 19:02:22 -04:00
numHit = 0;
2022-12-01 12:50:48 -05:00
int count = 0;
while( ReadBlock(1) != -1 ){
2024-07-18 19:02:22 -04:00
numHit ++;
2022-12-01 12:50:48 -05:00
int haha = (inFilePos*10/inFileSize)%10;
if( haha == count ) {
inFilePosPrecent[count] = inFilePos;
2024-07-18 19:02:22 -04:00
blockIDPrecent[count] = hitID;
2022-12-01 12:50:48 -05:00
count++;
}
PrintStatus(10000);
}
2024-07-18 19:02:22 -04:00
PrintStatus(1);
2022-12-01 12:50:48 -05:00
printf("\n\n\n");
2024-07-18 19:02:22 -04:00
printf("scan complete: number of data Block : %ld\n", numHit);
2022-12-01 12:50:48 -05:00
2024-07-18 19:02:22 -04:00
rewind(inFile); ///back to the File begining
if( isOldFormat == false && isNoHeaderFormat == false ){ // only when normal format
2024-09-03 13:41:36 -04:00
FillData(&data.Header);
}
2022-12-01 12:50:48 -05:00
inFilePos = 0;
2024-07-18 19:02:22 -04:00
hitID = -1;
2022-12-01 12:50:48 -05:00
endOfFile = false;
}
void BinReader::JumptoPrecent(int precent){
if( precent < 0 || precent > 10 ) {
printf("input precent should be 0 to 10\n");
return;
}
fseek(inFile, inFilePosPrecent[precent], SEEK_SET);
2024-07-18 19:02:22 -04:00
hitID = blockIDPrecent[precent];
2022-12-01 12:50:48 -05:00
}
void BinReader::PrintStatus(int mod){
///==== event stats, print status every 10000 events
2024-07-18 19:02:22 -04:00
if ( hitID % mod == 0 ) {
2022-12-01 12:50:48 -05:00
UpdateFileSize();
gClock.Stop("timer");
double time = gClock.GetRealTime("timer");
gClock.Start("timer");
2024-07-18 19:02:22 -04:00
printf("Hit ID: \x1B[32m%llu \x1B[0m\nReading Pos: \x1B[32m %.3f/%.3f GB\x1B[0m\nTime used:%3.0f min %5.2f sec\033[A\033[A\r",
hitID, inFilePos/(1024.*1024.*1024.), inFileSize/1024./1024./1024, TMath::Floor(time/60.), time - TMath::Floor(time/60.)*60.);
2022-12-01 12:50:48 -05:00
}
}
#endif