442 lines
11 KiB
C++
442 lines
11 KiB
C++
//This is an object interface for reading HIT data files
|
||
//See HIT documentation for details and examples.
|
||
/*
|
||
.L hitreader.c
|
||
|
||
Hitdata data;
|
||
data.read(“my_file.da2”); //to load whole file at once – forget it! See below.
|
||
data.read(“my_file.da2”,1000,100,10) //to read 100 frames starting from frame 1000 and incrementing by 10 (i.e. frame 1000, 1010, 1020, ... 1990 will be read)
|
||
//Reading 10 000 frames is reasonable. Reading 100 000 frames made my VM beg for memory.
|
||
|
||
data.nrFrames //to see how many frames you have
|
||
data.frames[0].nrBoards //to see how many boards you had in the system
|
||
data.frames[0].boards[0].nrChannels //to see how many channels you have in board 0
|
||
data.frames[10].boards[0].data[100] //get signal value for frame 10, board 0, channel 100
|
||
data.frames[10].boards[0].syncframe.local_ctr //get the local synchro counter for frame 10, board 0
|
||
//same for .global_ctr, .sma_state, .dummy, .device_nr, .data_ok
|
||
|
||
*/
|
||
|
||
//*********************** Helper *************************
|
||
#include <fstream>
|
||
#include <iostream>
|
||
|
||
using namespace std;
|
||
|
||
//#define debug(str) std::cout << "HIT DEBUG: " << str << endl;
|
||
#define debug(str)
|
||
|
||
//*********************** Syncframe *************************
|
||
class Syncframe
|
||
{
|
||
public:
|
||
Syncframe()
|
||
{
|
||
debug("Syncframe()");
|
||
|
||
local_ctr = global_ctr = 0;
|
||
sma_state = dummy = 0;
|
||
device_nr = -1;
|
||
data_ok = 0;
|
||
};
|
||
|
||
~Syncframe()
|
||
{
|
||
debug("~Syncframe()");
|
||
};
|
||
|
||
int sizeInFile()
|
||
{
|
||
return 16;
|
||
};
|
||
|
||
int read(std::ifstream* file)
|
||
{
|
||
char buffer[16];
|
||
file->read(buffer,16);
|
||
if (file->fail())
|
||
return 0;
|
||
local_ctr = *(unsigned short*)(buffer+0);
|
||
global_ctr = *(unsigned short*)(buffer+2);
|
||
sma_state = *(unsigned short*)(buffer+4);
|
||
dummy = *(unsigned short*)(buffer+6);
|
||
device_nr = *(int*)(buffer+8);
|
||
data_ok = *(int*)(buffer+12);
|
||
std::cout << "Syncframe:" << local_ctr << " " << global_ctr << " " << sma_state << " " << dummy << " " << device_nr << " " << data_ok << std::endl;
|
||
|
||
return 1;
|
||
};
|
||
|
||
|
||
unsigned short local_ctr;
|
||
unsigned short global_ctr;
|
||
unsigned short sma_state;
|
||
unsigned short dummy;
|
||
int device_nr;
|
||
unsigned int data_ok;
|
||
};
|
||
|
||
//*********************** Sensorframe *************************
|
||
class Boardframe
|
||
{
|
||
public:
|
||
Boardframe(int nr_channels = 0)
|
||
{
|
||
debug("Boardframe()");
|
||
|
||
data = NULL;
|
||
resize (nr_channels);
|
||
};
|
||
|
||
Boardframe(const Boardframe& in)
|
||
{
|
||
debug("Boardframe(Boardframe&)");
|
||
|
||
data = NULL;
|
||
resize(in.nrChannels);
|
||
for (int i = 0; i < nrChannels; i++)
|
||
data[i] = in.data[i];
|
||
syncframe = in.syncframe;
|
||
};
|
||
|
||
Boardframe& operator=(const Boardframe& in)
|
||
{
|
||
debug("Boardframe::operator==");
|
||
|
||
resize(in.nrChannels);//creates an array called data of length nrChannels
|
||
for (int i = 0; i < nrChannels; i++)
|
||
data[i] = in.data[i];
|
||
syncframe = in.syncframe;
|
||
return *this;
|
||
};
|
||
|
||
~Boardframe()
|
||
{
|
||
debug("~Boardframe()");
|
||
|
||
if (data)
|
||
delete[] data;
|
||
};
|
||
|
||
void resize(int nr_channels)
|
||
{
|
||
if (data)
|
||
delete[] data;
|
||
nrChannels = nr_channels;
|
||
if (nrChannels)
|
||
data = new unsigned short[nrChannels];
|
||
else
|
||
data = NULL;
|
||
};
|
||
|
||
int sizeInFile()
|
||
{
|
||
std::cout << "boardframe.sizeInFile() = " << syncframe.sizeInFile() + nrChannels*2 << std::endl;
|
||
return syncframe.sizeInFile() + nrChannels*2;
|
||
|
||
};
|
||
|
||
int read(std::ifstream* file)
|
||
{
|
||
if (syncframe.read(file) == 0)//get the syncframe before the board data
|
||
return 0;
|
||
//I must be already resized at this point!
|
||
file->read((char*)data,2*nrChannels);
|
||
if (file->fail())
|
||
return 0;
|
||
std::cout<< "data[" << nrChannels << "]: ";
|
||
for (int i = 0;i<nrChannels;i++) std::cout << data[i] << " ";
|
||
std::cout << std::endl;
|
||
|
||
return 1;
|
||
};
|
||
|
||
unsigned short& operator[] (int index)
|
||
{
|
||
return data[index];
|
||
};
|
||
|
||
Syncframe syncframe;
|
||
int nrChannels;
|
||
unsigned short* data;
|
||
};
|
||
|
||
//*********************** Fullframe *************************
|
||
class Fullframe
|
||
{
|
||
public:
|
||
Fullframe(int nr_boards = 0)
|
||
{
|
||
debug("Fullframe()");
|
||
boards = NULL;
|
||
resize(nr_boards);
|
||
};
|
||
|
||
Fullframe(const Fullframe& in)
|
||
{
|
||
debug("Fullframe(Fullframe&)");
|
||
boards = NULL;
|
||
resize(in.nrBoards);
|
||
for (int i = 0; i < nrBoards; i++)
|
||
boards[i] = in.boards[i];
|
||
};
|
||
|
||
Fullframe& operator=(const Fullframe& in)
|
||
{
|
||
debug("Fullframe::operator==");
|
||
resize(in.nrBoards);
|
||
for (int i = 0; i < nrBoards; i++)
|
||
boards[i] = in.boards[i];
|
||
|
||
return *this;
|
||
};
|
||
|
||
~Fullframe()
|
||
{
|
||
debug("~Fullframe()");
|
||
if (boards)
|
||
delete[] boards;
|
||
};
|
||
|
||
void resize (int nr_boards)
|
||
{
|
||
if (boards)
|
||
delete[] boards;
|
||
nrBoards = nr_boards;
|
||
if (nrBoards)
|
||
boards = new Boardframe[nrBoards];
|
||
else
|
||
boards = NULL;
|
||
}
|
||
|
||
int sizeInFile()
|
||
{
|
||
if (boards){
|
||
std::cout << "Fullframe.sizeInFile() = " << 2 + nrBoards*2 + nrBoards * boards[0].sizeInFile() << std::endl;
|
||
// return 2 + nrBoards*2 + nrBoards * boards[0].sizeInFile();
|
||
//// boards[0].sizeInFile() returns 656 for every board...
|
||
return 2 + 4*2 + (16 + 320 * 2) + (16 + 128*2)*3; //1482
|
||
//board[3].sizeInFile should be zero. 1482 - 128*2;
|
||
}
|
||
else
|
||
return 0; //no boards, makes no sense...
|
||
};
|
||
|
||
int read(std::ifstream* file)
|
||
{
|
||
//Read number of boards
|
||
unsigned short nr_boards;
|
||
file->read((char*)&nr_boards,2);
|
||
if(file->fail()){
|
||
std::cerr << "File read failed." << std::endl;
|
||
return 0;
|
||
}
|
||
if (nr_boards!=4){
|
||
std::cerr << "Unrealistic number(!=) of boards to be read:"<< nr_boards << std::endl;
|
||
std::cerr << "Will try to resync frame." << std::endl;
|
||
for (int j = 0;j<741;j++){
|
||
file->read((char*)&nr_boards,2);
|
||
if (nr_boards==4) break;
|
||
}
|
||
if ( nr_boards!=4){
|
||
std::cerr << "Resync failed." << std::endl;
|
||
return 0;
|
||
}
|
||
|
||
}
|
||
//std::cout << " nr_boards: " << nr_boards << std::endl;
|
||
//Read channel counts
|
||
unsigned short* channel_counts = new unsigned short[nr_boards];
|
||
file->read((char*)channel_counts,nr_boards*2);
|
||
if (file->fail())
|
||
{
|
||
delete[] channel_counts;
|
||
return 0;
|
||
}
|
||
|
||
|
||
|
||
//Read board frames
|
||
resize(nr_boards);
|
||
for (int board_nr = 0; board_nr < nr_boards; board_nr++)
|
||
{
|
||
// std::cout << " channel_counts[" << board_nr << "]: "<< channel_counts[board_nr] << std::endl;
|
||
|
||
boards[board_nr].resize(channel_counts[board_nr]);
|
||
if (boards[board_nr].read(file) == 0)//read the board
|
||
{
|
||
delete[] channel_counts;
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
delete[] channel_counts;
|
||
return 1;
|
||
};
|
||
|
||
int nrChannels()
|
||
{
|
||
int result = 0;
|
||
for (int board_nr = 0; board_nr < nrBoards; board_nr++)
|
||
result += boards[board_nr].nrChannels;
|
||
return result;
|
||
};
|
||
|
||
unsigned short& operator[] (int index)
|
||
{
|
||
for (int board_nr = 0; board_nr < nrBoards; board_nr++)
|
||
{
|
||
if (index >= boards[board_nr].nrChannels)
|
||
index -= boards[board_nr].nrChannels;
|
||
else
|
||
return boards[board_nr][index];
|
||
}
|
||
|
||
std::cerr << " ### Fullframe::operator[]: index out of range!" << std::endl;
|
||
// return (*NULL); //this will cause crash (intended).
|
||
return boards[nrBoards][index];
|
||
};
|
||
|
||
int nrBoards;
|
||
Boardframe* boards;
|
||
};
|
||
|
||
//*********************** Hitdata *************************
|
||
|
||
class Hitdata
|
||
{
|
||
public:
|
||
Hitdata(int nr_frames = 0)
|
||
{
|
||
frames = NULL;
|
||
resize(nr_frames);
|
||
};
|
||
|
||
Hitdata(const Hitdata& in)
|
||
{
|
||
frames = NULL;
|
||
resize(in.nrFrames);
|
||
for (int i = 0; i < nrFrames; i++)
|
||
frames[i] = in.frames[i];
|
||
};
|
||
|
||
Hitdata& operator=(const Hitdata& in)
|
||
{
|
||
resize(in.nrFrames);
|
||
for (int i = 0; i < nrFrames; i++)
|
||
frames[i] = in.frames[i];
|
||
|
||
return *this;
|
||
};
|
||
|
||
~Hitdata()
|
||
{
|
||
if (nrFrames)
|
||
delete[] frames;
|
||
};
|
||
|
||
void resize (int nr_frames)
|
||
{
|
||
if (nrFrames)
|
||
delete[] frames;
|
||
nrFrames = nr_frames;
|
||
if (nrFrames)
|
||
frames = new Fullframe[nrFrames];
|
||
else
|
||
frames = NULL;
|
||
};
|
||
|
||
//Read data from a given file.
|
||
//first_frame is the number of first frame to be read
|
||
//nr_frames is the maximum number of frames to be read
|
||
//-1 to read all of them
|
||
//increment allows you reading once every nth sample
|
||
//Return number of frames read or 0 in case of failure
|
||
int readFile(char* filename, int first_frame = 1, int nr_frames = -1, int increment = 1)
|
||
{
|
||
std::ifstream file;
|
||
//Open the file
|
||
file.open(filename, ios_base::in | ios_base::binary);
|
||
if (!file.is_open())
|
||
{
|
||
std::cerr << " ### Hitdata: File could not be open!" << std::endl;
|
||
return 0; //file could not be opened
|
||
}
|
||
|
||
//Read first record to find board configuration
|
||
Fullframe sampleframe;
|
||
if (sampleframe.read(&file) == 0)
|
||
{
|
||
std::cerr << " ### Hitdata: First frame could not be read!" << std::endl;
|
||
file.close();
|
||
return 0;
|
||
}
|
||
else {
|
||
std::cout << "Sample frame size (bytes): " << sampleframe.sizeInFile() << std::endl;
|
||
}
|
||
|
||
//Check file size
|
||
file.seekg(0, std::ios::beg);
|
||
std::streamsize fsize = file.tellg();
|
||
file.seekg(0, std::ios::end);
|
||
fsize = file.tellg() - fsize;
|
||
|
||
//Determine real frames to read
|
||
unsigned int max_frames = fsize / sampleframe.sizeInFile();
|
||
if ((max_frames == -1) || (max_frames < nr_frames))
|
||
nr_frames = max_frames;
|
||
|
||
std::cout << " Hitdata: Nr frames to be read: " << nr_frames << std::endl;
|
||
|
||
//Read!
|
||
// resize(nr_frames); //make an array of Fullframes called frames of size nr_frames
|
||
file.seekg(first_frame * sampleframe.sizeInFile(), std::ios::beg);
|
||
for (int frame_nr = first_frame; frame_nr < nr_frames; frame_nr++)
|
||
{
|
||
if ((frame_nr%10000) == 0)
|
||
std::cout << " Frame " << frame_nr << std::endl;
|
||
|
||
file.seekg((frame_nr*increment) * sampleframe.sizeInFile() , std::ios::beg);
|
||
if (file.eof()) {
|
||
std::cerr<< "end of file reached." << std::endl;
|
||
return frame_nr;
|
||
}
|
||
if ( sampleframe.read(&file) == 0) //read the next frame
|
||
{
|
||
std::cerr << " ### Hitdata: Frame " << frame_nr << " could not be read!" << std::endl;
|
||
file.close(); //read error, finish!
|
||
// frames = frame_nr; //Kinky! We decrease nr_frames, but the actual array size remains unchanged!
|
||
///???? I don't know what the above line does.
|
||
return frame_nr;
|
||
}
|
||
// std::cout << frames[frame_nr].nrBoards << std::endl;
|
||
}
|
||
|
||
//Finished
|
||
file.close();
|
||
return nr_frames;
|
||
};
|
||
|
||
Fullframe& operator[] (int index)
|
||
{
|
||
if (index < nrFrames)
|
||
return frames[index];
|
||
else
|
||
{
|
||
std::cerr << " ### Hitdata::operator[]: index out of range!" << std::endl;
|
||
// return (*NULL); //this will cause crash (intended).
|
||
return frames[index];
|
||
|
||
}
|
||
};
|
||
|
||
int nrFrames;
|
||
Fullframe* frames;
|
||
};
|
||
|
||
|
||
|
||
|
||
|
||
|