00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00026
00027 #include "vicinity.h"
00028 #include <math.h>
00029 #include <algorithm>
00030 #include <iostream>
00031
00032 namespace siscone{
00033
00034 using namespace std;
00035
00036
00037
00038
00039
00040
00041
00042
00043
00044
00045 bool ve_less(Cvicinity_elm *ve1, Cvicinity_elm *ve2){
00046 return ve1->angle < ve2->angle;
00047 }
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060 Cvicinity::Cvicinity(){
00061 n_part = 0;
00062
00063 ve_list = NULL;
00064 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00065 quadtree = NULL;
00066 #endif
00067
00068 parent = NULL;
00069 VR2 = VR = 0.0;
00070
00071 }
00072
00073
00074
00075 Cvicinity::Cvicinity(vector<Cmomentum> &_particle_list){
00076 parent = NULL;
00077 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00078 quadtree = NULL;
00079 #endif
00080 VR2 = VR = 0.0;
00081
00082 set_particle_list(_particle_list);
00083 }
00084
00085
00086
00087 Cvicinity::~Cvicinity(){
00088 if (ve_list!=NULL)
00089 delete[] ve_list;
00090
00091 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00092 if (quadtree!=NULL)
00093 delete quadtree;
00094 #endif
00095 }
00096
00097
00098
00099
00100
00101
00102 void Cvicinity::set_particle_list(vector<Cmomentum> &_particle_list){
00103 int i,j;
00104 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00105 double eta_max=0.0;
00106 #endif
00107
00108
00109 if (ve_list!=NULL){
00110 delete[] ve_list;
00111 }
00112 vicinity.clear();
00113 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00114 if (quadtree!=NULL)
00115 delete quadtree;
00116 #endif
00117
00118
00119
00120
00121 n_part = 0;
00122 plist.clear();
00123 pincluded.clear();
00124 for (i=0;i<(int) _particle_list.size();i++){
00125
00126
00127 if (fabs(_particle_list[i].pz)!=_particle_list[i].E){
00128 plist.push_back(_particle_list[i]);
00129 pincluded.push_back(Cvicinity_inclusion());
00130
00131
00132
00133
00134
00135 plist[n_part].index = n_part;
00136
00137
00138 plist[n_part].ref.randomize();
00139
00140 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00141 if (fabs(plist[n_part].eta)>eta_max) eta_max=fabs(plist[n_part].eta);
00142 #endif
00143
00144 n_part++;
00145 }
00146 }
00147
00148
00149
00150 ve_list = new Cvicinity_elm[2*n_part];
00151 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00152 eta_max+=0.1;
00153 quadtree = new Cquadtree(0.0, 0.0, eta_max, M_PI);
00154 #endif
00155
00156
00157 j = 0;
00158 for (i=0;i<n_part;i++){
00159 #ifdef USE_QUADTREE_FOR_STABILITY_TEST
00160 quadtree->add(&plist[i]);
00161 #endif
00162 ve_list[j].v = ve_list[j+1].v = &plist[i];
00163 ve_list[j].is_inside = ve_list[j+1].is_inside = &(pincluded[i]);
00164 j+=2;
00165 }
00166
00167 }
00168
00169
00170
00171
00172
00173
00174
00175 void Cvicinity::build(Cmomentum *_parent, double _VR){
00176 int i;
00177
00178
00179 parent = _parent;
00180 VR = _VR;
00181 VR2 = VR*VR;
00182 R2 = 0.25*VR2;
00183 R = 0.5*VR;
00184 inv_R_EPS_COCIRC = 1.0 / R / EPSILON_COCIRCULAR;
00185 inv_R_2EPS_COCIRC = 0.5 / R / EPSILON_COCIRCULAR;
00186
00187
00188 vicinity.clear();
00189
00190
00191 pcx = parent->eta;
00192 pcy = parent->phi;
00193
00194
00195 for (i=0;i<n_part;i++){
00196 append_to_vicinity(&plist[i]);
00197 }
00198
00199
00200 sort(vicinity.begin(), vicinity.end(), ve_less);
00201
00202 vicinity_size = vicinity.size();
00203 }
00204
00205
00207 inline double sort_angle(double s, double c){
00208 if (s==0) return (c>0) ? 0.0 : 2.0;
00209 double t=c/s;
00210 return (s>0) ? 1-t/(1+fabs(t)) : 3-t/(1+fabs(t));
00211 }
00212
00213
00214
00215
00216
00217
00218
00219 void Cvicinity::append_to_vicinity(Cmomentum *v){
00220 double dx, dy, d2;
00221
00222
00223 if (v==parent)
00224 return;
00225
00226 int i=2*(v->index);
00227
00228
00229 dx = v->eta - pcx;
00230 dy = v->phi - pcy;
00231
00232
00233 if (dy>M_PI)
00234 dy -= twopi;
00235 else if (dy<-M_PI)
00236 dy += twopi;
00237
00238 d2 = dx*dx+dy*dy;
00239
00240
00241 if (d2<VR2){
00242 double s,c,tmp;
00243
00244
00245
00246
00247 tmp = sqrt(VR2/d2-1);
00248
00249
00250 c = 0.5*(dx-dy*tmp);
00251 s = 0.5*(dy+dx*tmp);
00252 ve_list[i].angle = sort_angle(s,c);
00253 ve_list[i].eta = pcx+c;
00254 ve_list[i].phi = phi_in_range(pcy+s);
00255 ve_list[i].side = true;
00256 ve_list[i].cocircular.clear();
00257 vicinity.push_back(&(ve_list[i]));
00258
00259
00260 c = 0.5*(dx+dy*tmp);
00261 s = 0.5*(dy-dx*tmp);
00262 ve_list[i+1].angle = sort_angle(s,c);
00263 ve_list[i+1].eta = pcx+c;
00264 ve_list[i+1].phi = phi_in_range(pcy+s);
00265 ve_list[i+1].side = false;
00266 ve_list[i+1].cocircular.clear();
00267 vicinity.push_back(&(ve_list[i+1]));
00268
00269
00270
00271
00272
00273 Ctwovect OP(pcx - ve_list[i+1].eta, phi_in_range(pcy-ve_list[i+1].phi));
00274 Ctwovect OC(v->eta - ve_list[i+1].eta,
00275 phi_in_range(v->phi-ve_list[i+1].phi));
00276
00277
00278
00279
00280
00281
00282
00283
00284 c = dot_product(OP,OC);
00285 s = fabs(cross_product(OP,OC));
00286 double inv_err1 = s * inv_R_EPS_COCIRC;
00287 double inv_err2_sq = (R2-c) * inv_R_2EPS_COCIRC;
00288 ve_list[i].cocircular_range = pow2(inv_err1) > inv_err2_sq ?
00289 1.0/inv_err1 :
00290 sqrt(1.0/inv_err2_sq);
00291 ve_list[i+1].cocircular_range = ve_list[i].cocircular_range;
00292 }
00293 }
00294
00295 }