144 : fNumPlanes(numlayers),fNumWires(numlayers)
196 if (treenode::GetObjectsAlive() > 0 || nodenode::GetObjectsAlive() > 0) {
197 QwVerbose <<
"Memory occupied by tree objects (should be close to zero when all trees cleared):" <<
QwLog::endl;
225 int *bitpattern = testnode->
fBit;
240 int tlaym1 = templayers - 1;
241 double z[templayers];
242 double binwidth, y0, dy;
250 y0 = binwidth = dy = off = 0.0;
257 for (std::vector<QwDetectorInfo*>::iterator iter =
fGeometry.begin();
258 iter !=
fGeometry.end() && layer < templayers;
267 z[layer] = zv - z[0];
268 if (z[layer] < z[0]) {
273 if(layer == templayers-1)
292 xf = bitpattern[tlaym1];
308 x0 = bitpattern[0] * binwidth;
309 xf = bitpattern[tlaym1] * binwidth + off;
310 mL = mR = (xf - x0) / z[tlaym1];
312 xmin = mL * z[layer] + x0;
313 xmax = mR * z[layer] + x0 + binwidth;
314 xiL = bitpattern[layer] * binwidth + off;
315 xiR = (bitpattern[layer] + 1) * binwidth + off;
316 if (xiL > xmax || xiR < xmin) {
319 if (xiR > xmax) mL = (xiL - x0) / z[layer];
320 if (xiL < xmin) mR = (xiR - x0) / z[layer];
322 xmin = mL * z[layer] + x0;
323 xmax = mR * z[layer] + x0 + binwidth;
324 xiL = bitpattern[layer] * binwidth;
325 xiR = (bitpattern[layer] + 1) * binwidth;
326 if (xiL > xmax || xiR < xmin) {
340 double xf = 0, zf = 0.0;
341 double z[templayers];
344 double cellwidth = 1;
345 int firstnonzero = 0;
347 for (
int layer = 0; layer < templayers; layer++) {
348 if (bitpattern[layer]) firstnonzero++;
349 if (firstnonzero && ! bitpattern[layer]) templayers = layer + 1;
356 for (
int layer = 1; layer < templayers; layer++) {
357 z[layer] = z[layer-1] + cellwidth;
361 for (
int layer = 0; layer < templayers; layer++) {
362 if (bitpattern[layer] >= xf) {
364 xf = bitpattern[layer];
378 double m_min = -((double)
fNumLayers - 1) / ((double) (1 << level) - 1);
379 double m_max = -(4.0 - 1.0) / ((double) (1 << level) - 1);
380 double m = -((double) zf) / ((double) (xf - x0));
383 if (m < m_min || m > m_max)
return 0;
391 for (
int layer = (
int) zf - 1; layer > 0; layer--) {
393 adda = (x0 - bitpattern[layer]) * dza + (x3a - x0) * z[layer];
394 if (adda <= -dza || dza <= adda)
397 adde = (x0 - bitpattern[layer]) * dze + (x3e - x0 - 1) * z[layer];
398 if (adde <= -dze || dze <= adde)
403 x3e = bitpattern[layer] + 1;
407 x3a = bitpattern[layer];
577 while (plane_combination--) {
582 for (
unsigned int plane = 0; plane <
fNumPlanes; plane++) {
584 if (plane_combination & (1 << plane)) {
585 son.
fBit[plane] = (father->
fBit[plane] << 1) + 1;
587 son.
fBit[plane] = (father->
fBit[plane] << 1);
591 max_bin = (int) std::max (max_bin, son.
fBit[plane]);
607 if (max_bin - min_bin > abs(son.
fWidth))
615 for (
unsigned int plane = 0; plane <
fNumPlanes; plane++)
625 for (
unsigned int plane = 0; plane <
fNumPlanes; plane++)
643 int insert_hitpattern = 1;
654 if (! sonptr && 0 == (sonptr =
existent (&son, hash))) {
671 sonptr->
fSon[2] = sonptr->
fSon[3] = 0;
691 marklin (sonptr, level+1, detector);
697 insert_hitpattern = 0;
728 marklin (sonptr, level+1, detector);
737 if (insert_hitpattern &&
746 father->
fSon[offs+flip] = nodptr;
766 while (wire_combination--) {
767 for (
unsigned int wire = 0; wire <
fNumWires; wire++) {
769 if (wire_combination & (1 << wire)) {
770 son.
fBit[wire] = (father->
fBit[wire] << 1) + 1;
772 son.
fBit[wire] = (father->
fBit[wire] << 1);
775 maxs = (int) std::max (maxs,son.
fBit[wire]);
789 for (
unsigned int wire = 1; wire <
fNumWires; wire++) {
790 if (son.
fBit[wire] < son.
fBit[wire-1]) {
793 if (! son.
fBit[wire]) {
798 if (son.
fBit[wire] && cutback)
815 for (
unsigned int wire = 0; wire <
fNumWires; wire++)
826 for (
unsigned int wire = 0; wire <
fNumWires; wire++)
841 int insert_hitpattern = 1;
842 if (! sonptr&& 0 == (sonptr =
existent(&son, hash))) {
848 sonptr->
fSon[2] = sonptr->
fSon[3] = 0;
857 marklin (sonptr, level+1, detector);
870 insert_hitpattern = 0;
879 marklin (sonptr, level+1, detector);
882 if (insert_hitpattern
888 father->
fSon[offs+flip] = nodptr;
902 while (layer_combination--) {
907 for (
unsigned int layer = 0; layer <
fNumLayers; layer++) {
912 if (layer_combination & (1 << layer)) {
913 son.
fBit[layer] = (father->
fBit[layer] << 1) + 1;
915 son.
fBit[layer] = (father->
fBit[layer] << 1);
918 maxs = (int) std::max (maxs, son.
fBit[layer]);
922 if (maxs - offs > abs(son.
fWidth)) {
928 for (
unsigned int layer = 0; layer <
fNumLayers; layer++)
934 for (
unsigned int layer = 0; layer <
fNumLayers; layer++)
943 int insert_hitpattern = 1;
945 if (! sonptr && 0 == (sonptr =
existent(&son, hash))) {
954 sonptr->
fSon[2] = sonptr->
fSon[3] = 0;
961 marklin (sonptr, level+1, detector);
964 insert_hitpattern = 0;
970 marklin (sonptr, level+1, detector);
973 if (insert_hitpattern &&
980 father->
fSon[offs+flip] = nodptr;
1001 const string& filename,
1011 Double_t width = 0.0;
1015 bfs::path fullfilename = bfs::path(filename);
1016 if (! bfs::exists(fullfilename)) {
1018 <<
"full file name = " << fullfilename.string() <<
QwLog::endl;
1026 if (skipreading)
return 0;
1029 file = fopen(fullfilename.string().c_str(),
"rb");
1032 <<
"full file name = " << fullfilename.string() <<
QwLog::endl;
1037 if (fread(&num,
sizeof(num), 1L, file) < 1 ||
1038 fread(&width,
sizeof(width), 1L, file) < 1 ) {
1040 <<
"full file name = " << fullfilename.string() <<
QwLog::endl;
1048 shorttree::SetDefaultSize(tlayers);
1055 if (
_readtree (file, stb, 0, tlayers) < 0) {
1057 delete [] stb; stb = 0;
1058 delete trr; trr = 0;
1060 <<
"full file name = " << fullfilename.string() <<
QwLog::endl;
1066 if (file) fclose(file);
1090 const string& filename,
1102 if (tlayer == 0)
return 0;
1116 trr =
readtree(filename, levels, tlayer, width, regenerate);
1128 if (
writetree(filename, back, levels, tlayer, width) < 0) {
1137 trr =
readtree(filename, levels, tlayer, width, 0);
1192 if (tn->
fRef == -1) {
1195 if (fputc (
REALSON, fp) == EOF ||
1196 fwrite(&tn->
fMinLevel,
sizeof(
int), 1L, fp) != 1 ||
1197 fwrite(&tn->
fWidth,
sizeof(
int), 1L, fp) != 1 ||
1198 fwrite( tn->
fBit,
sizeof(
int), (
size_t) tlayers, fp) != (
unsigned int)tlayers )
1201 for (
int i = 0; i < 4; i++) {
1208 if (fputc (
SONEND, fp) == EOF)
1212 if (fputc (
REFSON, fp) == EOF ||
1213 fwrite (&tn->
fRef,
sizeof(tn->
fRef), 1L, fp) != 1)
1228 const string& filename,
1235 bfs::path fullfilename = bfs::path(filename);
1238 FILE *file = fopen(fullfilename.string().c_str(),
"wb");
1242 <<
"full file name = " << fullfilename.string() <<
QwLog::endl;
1247 if (fwrite(&fRef,
sizeof(int32_t), 1L, file) != 1 ||
1248 fwrite(&width,
sizeof(
double), 1L, file) != 1 ||
1253 if (fputc(
SONEND, file) == EOF)
1259 if (fwrite(&fRef,
sizeof(int32_t), 1L, file) != 1) {
1294 int32_t type = fgetc(file);
1302 <<
"More nodes in file than announced in header!" <<
QwLog::endl;
1307 int32_t minlevel = 0;
1309 int32_t bit[tlayers];
1310 for (
int i = 0; i < tlayers; i++) bit[i] = 0;
1311 if (fread(&minlevel,
sizeof(int32_t), 1L, file) != 1 ||
1312 fread(&width,
sizeof(int32_t), 1L, file) != 1 ||
1313 fread(&bit,
sizeof(int32_t) * tlayers, 1L, file) != 1) {
1321 (stb+ref)->fMinLevel = minlevel;
1322 (stb+ref)->fWidth = width;
1323 for (
int i = 0; i < tlayers; i++)
1324 (stb+ref)->fBit[i] = bit[i];
1334 for (int32_t son = 0; son < 4; son++) {
1335 if (
_readtree(file, stb, stb[ref].son + son, tlayers) < 0) {
1343 }
else if (type ==
REFSON) {
1346 if (fread (&ref,
sizeof(ref), 1L, file) != 1) {
1348 <<
"Could not read reference field." <<
QwLog::endl;
1351 if (ref >=
fRef || ref < 0) {
1353 <<
"Reference field contains nonsense." <<
QwLog::endl;
1368 }
else if (type ==
SONEND) {
#define QwMessage
Predefined log drain for regular messages.
void SetNext(shortnode *next)
Set the next node.
bool IsSearchable() const
Is this region searchable?
#define QwOut
Predefined log drain for explicit output.
treenode * fFather
Father node: the main entry point to the tree.
int _readtree(FILE *file, shorttree *stb, shortnode **father, int32_t tlayers)
Recursive method to read the concise treesearch database from disk.
shortnode * GetNode()
Get the node to this tree region.
QwTrackingTreeRegion * inittree(const string &filename, int levels, int tlayer, double width, QwDetectorInfo *detector, bool regenerate)
void SetSearchable(bool searchable=true)
Set this tree region to searchable.
A container for the pattern databases for each detector region.
treenode * GetTree() const
Get the tree.
EQwRegionID GetRegion() const
#define QwVerbose
Predefined log drain for verbose messages.
int consistent(treenode *testnode, int level, QwDetectorInfo *detector)
Determines whether the pattern is geometrically possible.
void SetTree(treenode *tree)
Set the tree.
unsigned int & fNumWires
Number of wires in a region 3 VDC group.
void PrintHashTable() const
Print the hash table.
int fNumPatterns
Number of valid patterns in the tree.
treenode * _inittree(int32_t tlayer, QwDetectorInfo *detector)
Recursive method to initialize and generate the treesearch database.
#define HSHSIZ
Length of the hash table (as header define)
void SetNext(nodenode *next)
Set the next node.
double GetPlaneOffset() const
void Print(bool recursive=false, int indent=0)
Print some debugging information.
int fMinLevel
Minimum level at which this node is valid.
#define QwDebug
Predefined log drain for debugging output.
int fWidth
Width in bins of the hit pattern.
int fRef
Reference of this node when writing to file.
A logfile class, based on an identical class in the Hermes analyzer.
nodenode * fSon[4]
Each tree has four son nodes.
unsigned int fHashSize
Length of the hash table (as member field)
QwTrackingTreeRegion * readtree(const string &filename, int levels, int tlayers, double rwidth, bool skipreading)
void PrintTree() const
Print the full tree.
void SetNext(treenode *next)
Set the next node.
void SetWidth(double width)
Set the width.
EQwDetectorType GetType() const
treenode * GetNext() const
Get the next node.
treenode * nodeexists(nodenode *nd, treenode *tr)
int * fBit
Hit pattern, one bin specified per detector layer.
static std::ostream & endl(std::ostream &)
End of the line.
double fMaxSlope
Maximum allowed slope for tracks in this detector.
double GetElementSpacing() const
void SetTree(shorttree *tree, int ntrees=1)
Set the tree.
long writetree(const string &filename, treenode *tn, int levels, int tlayers, double width)
This function initializes the output file. It then calls the iterative _writetree function to fill th...
QwGeometry fGeometry
Detector (or set of detectors) represented by this search tree.
Definition of the track search tree.
nodenode * next() const
Get the next node (non-standard notation)
int fMaxLevel
Maximum level at which this node is valid.
virtual ~QwTrackingTree()
Destructor.
int _writetree(treenode *tn, FILE *fp, int32_t tlayers)
Recursive method for pulling in the concise treesearch search database.
static const std::string fgTreeDir
Name of the tree directory (in $QWSCRATCH), as static member field.
treenode * existent(treenode *node, int hash)
Search for a node in the search tree.
double GetZPosition() const
void marklin(treenode *father, int level, QwDetectorInfo *detector)
Recursively generate the treesearch pattern database.
unsigned int & fNumPlanes
Number of planes in the region 2 HDCs.
#define QwWarning
Predefined log drain for warnings.
unsigned int size() const
Get size of the bit array.
A nodenode is used as a pointer which links treenodes to their siblings.
A treenode contains the bits that make up a tree pattern.
int GetNumberOfElements() const
int fMaxRef
Maximum number of references in the cached tree file.
unsigned int fNumLayers
Number of detector planes.
nodenode * GetNext() const
Get the next node.
QwTrackingTree(unsigned int numlayers)
Default constructor.
#define QwError
Predefined log drain for errors.