Changeset 2980
- Timestamp:
- 02/08/10 11:00:45 (15 years ago)
- Location:
- issm/trunk/src/c/Bamgx
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
issm/trunk/src/c/Bamgx/QuadTree.h
r2910 r2980 7 7 const int MaxDeep = 30; 8 8 typedef long IntQuad; 9 const IntQuad MaxISize = ( 1L << MaxDeep); //long int 1, bitwise operation: 8L = 00001000 << 2L -> 00100000 shifted left by 2 9 //long int 1, bitwise operation: 8L = 00001000 << 2L -> 00100000 shifted left by 2 10 const IntQuad MaxISize = ( 1L << MaxDeep); 10 11 class Triangles; 11 12 class Vertex; 12 13 class QuadTree { 13 p ublic:14 private: 14 15 class QuadTreeBox { 15 16 public: -
issm/trunk/src/c/Bamgx/objects/QuadTree.cpp
r2910 r2980 23 23 //NORM(i1,j1,i2,j2) returns max(|i1-j1|,|i2-j2|) 24 24 #define NORM(i1,j1,i2,j2) MAX1(ABS((i1)-(j1)),ABS((i2)-(j2))) 25 //IJ(i,j,l) returns 3 if (1,1,1), 2 if (0,1,1), 1 if (1,0,1), else 0 26 #define IJ(i,j,l) ( ( j & l) ? (( i & l) ? 3 : 2 ) :( ( i & l)? 1 : 0 )) 27 //I_IJ(k,l) retruns l if first bit of k is 1, else 0 28 #define I_IJ(k,l) (( k&1) ? l : 0) 29 //J_IJ(k,l) retruns l if second bit of k is 1, else 0 30 #define J_IJ(k,l) (( k&2) ? l : 0) 25 //IJ(i,j,l) returns the box number of i and j with respect to l 26 //if !j&l and !i$l -> 0 (box zero: lower left ) 27 //if !j&l and i$l -> 1 (box one: lower right) 28 //if j&l and !i$l -> 2 (box two: upper left ) 29 //if j&l and i$l -> 3 (box three:upper right) 30 #define IJ(i,j,l) ((j&l) ? ((i&l) ? 3:2 ) :((i&l) ? 1:0 )) 31 //I_IJ(k,l) returns l if first bit of k is 1, else 0 32 #define I_IJ(k,l) ((k&1) ? l:0) 33 //J_IJ(k,l) returns l if second bit of k is 1, else 0 34 #define J_IJ(k,l) ((k&2) ? l:0) 35 36 /*What is a QuadTree? 37 * A Quadtree is a very simple way to group the vertices according 38 * to their location. A square that holds all the points of the mesh 39 * (or the geometry) is divided into 4 boxes. As soom as one box 40 * hold more than 4 vertices, it is divided into 4 new boxes, etc... 41 * There cannot be more than MAXDEEP (=30) subdivision. 42 * This process is like a Dichotomy in dimension 2 43 * 44 * + - - - - - - - - + - - + - + - + - - - - + 45 * | | | | X | | 46 * + - + - + 47 * | | | | | | 48 * + - - + - + - + + 49 * | | | | | 50 * 51 * | | | | | 52 * + - - - - - - - - + - - + - - + - - - - + 53 * | | | | 54 * 55 * | | | | 56 * 57 * | | | | 58 * | | | | 59 * + - - - - - - - - + - - - - + - - - - + 60 * | | | 61 * 62 * | | | 63 * 64 * | | | 65 * 66 * | | | 67 * | | | 68 * | | | 69 * | | | 70 * | | | 71 * + - - - - - - - - + - - - - - - - - + 72 * 73 * The coordinate system used in a quadtree are integers to avoid 74 * round-off errors. The vertex in the lower left box has the coordinates 75 * (0 0) 76 * The upper right vertex has the follwing coordinates: 77 * 2^30 -1 2^30 -1 in decimal 78 * 0 1 1 1 .... 1 0 1 1 1 .... 1 in binary 79 * \-- 29 --/ \-- 29 --/ 80 * Using the binaries is therefor very easy to locate a vertex in a box: 81 * we just need to look at the bits from the left to the right (See ::Add) 82 */ 31 83 32 84 /*Constructors/Destructors*/ … … 76 128 QuadTreeBox** pb; 77 129 QuadTreeBox* b; 78 register long i=w.i.x, j=w.i.y,l=MaxISize; 130 register long i=w.i.x, j=w.i.y; 131 register long level=MaxISize; 132 133 //Get inital box (the largest) 79 134 pb = &root; 135 136 //Find the smallest box where w is located 80 137 while((b=*pb) && (b->n<0)){ 81 b->n--; //n=n-1 in b->n 82 l >>= 1; //shifted righ by one bit: l=00000010 -> 00000001 83 pb = &b->b[IJ(i,j,l)]; //pointer toward b 84 } 138 139 //shift b->n by -1 140 b->n--; 141 142 //shifted righ by one bit: level=00000010 -> 00000001 143 level >>= 1; 144 145 //Get next subbox according to the bit value (level) 146 pb = &b->b[IJ(i,j,level)]; 147 } 148 149 //OK, we have found b, a Subbox holding vertices (might be full) 150 //check that the vertex is not already in the box 85 151 if (b) { 86 152 if (b->n > 3 && b->v[3] == &w) return; … … 90 156 } 91 157 92 //check that l is still non zero 93 if (l==0){ 94 throw ErrorException(__FUNCT__,exprintf("l==0 cannot be true as it has been initialized as MaxISize = %i",MaxISize)); 95 } 158 //check that l is not 0 (this should not happen as MaxDeep = 30) 159 if (level==0){ 160 throw ErrorException(__FUNCT__,exprintf("level==0 cannot be true as it has been initialized as MaxISize = %i",MaxISize)); 161 } 162 163 //Now, try to add the vertex, if the subbox is full (n=4), we have to divide it 164 //in 4 new subboxes 96 165 while ((b= *pb) && (b->n == 4)){ // the QuadTreeBox is full 97 Vertex *v4[4]; // copy of the QuadTreeBox vertices 98 166 167 //Copy the 4 vertices in the current QuadTreebox 168 Vertex* v4[4]; 99 169 v4[0]= b->v[0]; 100 170 v4[1]= b->v[1]; 101 171 v4[2]= b->v[2]; 102 172 v4[3]= b->v[3]; 103 b->n = -b->n; // mark is pointer QuadTreeBox 104 b->b[0]=b->b[1]=b->b[2]=b->b[3]=0; // set empty QuadTreeBox ptr 105 l >>= 1; // div the size by 2 106 for (register int k=0;k<4;k++){ // for the 4 vertices find the sub QuadTreeBox ij 173 174 //set n as negative (box full -> holds 4 pointers toward subboxes and not 4 vertices) 175 b->n = -b->n; 176 177 //Initialize the 4 pointers toward the 4 subboxes 178 b->b[0]=b->b[1]=b->b[2]=b->b[3]=NULL; 179 180 // div the size by 2 181 level >>= 1; 182 183 //Put the four vertices in the new boxes 184 for (register int k=0;k<4;k++){ 107 185 register int ij; 108 register QuadTreeBox * bb = b->b[ij=IJ(v4[k]->i.x,v4[k]->i.y,l)]; 109 if (!bb) 110 bb=b->b[ij]=NewQuadTreeBox(); // alloc the QuadTreeBox 186 register QuadTreeBox* bb = b->b[ij=IJ(v4[k]->i.x,v4[k]->i.y,level)]; 187 188 // alloc the QuadTreeBox 189 if (!bb) bb=b->b[ij]=NewQuadTreeBox(); 190 191 //Copy the 4 vertices 111 192 bb->v[bb->n++] = v4[k]; 112 193 } 113 pb = &b->b[IJ(i,j,l)]; 114 } 115 if (!(b = *pb)) b=*pb= NewQuadTreeBox(); // alloc the QuadTreeBox 116 b->v[b->n++]=&w; // we add the vertex 194 195 //Get the subbox where w (i,j) is located 196 pb = &b->b[IJ(i,j,level)]; 197 } 198 199 // alloc the QuadTreeBox 200 if (!(b = *pb)) b=*pb= NewQuadTreeBox(); 201 202 //Add w 203 b->v[b->n++]=&w; 204 205 //Increase NbVertices by one (we have one new vertex) 117 206 NbVertices++; 118 207 } … … 126 215 Icoor1 ii[MaxDeep]; 127 216 Icoor1 jj[MaxDeep]; 128 register int l=0; // level 217 register int level=0; // levelevelevel 218 register Int4 n0; 129 219 register QuadTreeBox* b; 130 220 IntQuad h=MaxISize,h0; … … 138 228 Vertex* vn=NULL; 139 229 140 // initialization for optimization230 //Get initial Quadtree box (largest) 141 231 b = root; 142 register Int4 n0;143 232 144 233 //if the tree is empty, return NULL pointer 145 234 if (!root->n) return vn; 146 235 147 //else: find the index n of the non empty 148 //QuadTreeBox containing the point (i,j) 149 while( (n0 = b->n) < 0){ 150 register Icoor1 hb2 = hb >> 1 ; 151 register int k = IJ(iplus,jplus,hb2);// QuadTreeBox number of size hb2 containing i;j (Macro) 152 register QuadTreeBox* b0= b->b[k]; 153 if (( b0 == 0) || (b0->n == 0)) break;// null box or empty => break 236 //else, find the non empty QuadTreeBox containing the point (i,j) 237 while((n0=b->n)<0){ 238 239 //shifted righ by one bit: hb2=01000000 -> 00100000 240 register Icoor1 hb2 = hb >> 1; 241 //Get QuadTreeBox number of size hb2 containing i;j (Macro) 242 register int k = IJ(iplus,jplus,hb2); 243 //Get the corresponding box b0 244 register QuadTreeBox* b0=b->b[k]; 245 246 // break if NULL box or empty 247 if (( b0 == NULL) || (b0->n == 0)) break; 248 249 //Get next Qudtree box 154 250 NbQuadTreeBoxSearch++; 155 251 b=b0; … … 159 255 } 160 256 161 // if n0>0 ??? 162 if ( n0 > 0){ 257 // if the current subbox is holding vertices, we are almost done 258 if ( n0>0 ){ 259 //loop over the vertices of the box and find the closest vertex 163 260 for(register int k=0;k<n0;k++){ 164 I2 i2 =b->v[k]->i;165 h0 =NORM(iplus,i2.x,jplus,i2.y);166 if (h0 <h){261 I2 i2=b->v[k]->i; 262 h0=NORM(iplus,i2.x,jplus,i2.y); 263 if (h0<h){ 167 264 h = h0; 168 vn = b->v[k];} 169 NbVerticesSearch++; 265 vn = b->v[k]; 266 } 267 NbVerticesSearch++; 170 268 } 171 269 return vn; 172 270 } 173 271 174 // general case 175 pb[0]= b; 176 pi[0]=b->n>0 ?(int) b->n : 4 ; 177 ii[0]=i0; 178 jj[0]=j0; 272 /* general case: the current box is empty, we have to get backwards 273 and find the closest not-empty box and find the closest vertex*/ 274 275 //initialize pb pi ii and jj 276 pb[0]=b; //pointer toward the box b 277 pi[0]=b->n>0? (int)b->n:4;//number of vertices in the box 278 ii[0]=i0; // i coordinate of the box 279 jj[0]=j0; // j coordinate of the box 280 281 //initialize h as hb 179 282 h=hb; 180 do { 181 b= pb[l]; 182 while (pi[l]--){ 183 register int k = pi[l]; 184 283 284 //loop, until level=0 285 do { 286 //get current box 287 b= pb[level]; 288 289 //Loop over the vertices in current box (if not empty!) 290 while (pi[level]--){ 291 292 //k = number of vertices in the box if there are vertices 293 //k = 4 if the current box is pointing toward 4 other boxes 294 register int k=pi[level]; 295 296 //if the current subbox is holding vertices, 185 297 if (b->n>0){ // Vertex QuadTreeBox not empty 186 298 NbVerticesSearch++; 187 299 I2 i2 = b->v[k]->i; 188 300 h0 = NORM(iplus,i2.x,jplus,i2.y); 189 if (h0 <h) 190 { 301 if (h0 <h){ 191 302 h = h0; 192 303 vn = b->v[k]; 193 304 } 194 305 } 195 else{ // Pointer QuadTreeBox 196 register QuadTreeBox *b0=b; 306 307 else{ 308 register QuadTreeBox* b0=b; 197 309 NbQuadTreeBoxSearch++; 310 311 //if the next box exists: 198 312 if ((b=b->b[k])){ 199 hb >>=1 ; // div by 2 200 register Icoor1 iii = ii[l]+I_IJ(k,hb); 201 register Icoor1 jjj = jj[l]+J_IJ(k,hb); 202 313 //shifted righ by one bit: hb2=01000000 -> 00100000 314 hb>>=1; 315 register Icoor1 iii = ii[level]+I_IJ(k,hb); 316 register Icoor1 jjj = jj[level]+J_IJ(k,hb); 317 318 //if the current point is in b,go to next box 203 319 if (INTER_SEG(iii,iii+hb,iplus-h,iplus+h) && INTER_SEG(jjj,jjj+hb,jplus-h,jplus+h)){ 204 pb[++l ]= b;205 pi[l ]= b->n>0 ?(int) b->n : 4 ;206 ii[l ]= iii;207 jj[l ]= jjj;320 pb[++level]= b; 321 pi[level]= b->n>0 ?(int) b->n : 4 ; 322 ii[level]= iii; 323 jj[level]= jjj; 208 324 } 325 326 //else go backwards 209 327 else{ 210 b=b0, hb <<=1 ; 328 //shifted righ by one bit: hb=001000000 -> 01000000 329 b=b0; 330 hb<<=1; 211 331 } 212 332 } 333 //Go backwards 213 334 else b=b0; 214 335 } 215 336 } 216 hb <<= 1; // mul by 2 217 } while (l--); 337 338 //else go backwards 339 //shifted righ by one bit: hb=001000000 -> 01000000 340 hb <<= 1; 341 } while (level--); 342 343 //return vn, nearest vertex 218 344 return vn; 219 345 }
Note:
See TracChangeset
for help on using the changeset viewer.