Changeset 2980


Ignore:
Timestamp:
02/08/10 11:00:45 (15 years ago)
Author:
Mathieu Morlighem
Message:

minor comments

Location:
issm/trunk/src/c/Bamgx
Files:
2 edited

Legend:

Unmodified
Added
Removed
  • issm/trunk/src/c/Bamgx/QuadTree.h

    r2910 r2980  
    77        const int MaxDeep = 30;
    88        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);
    1011        class Triangles;
    1112        class Vertex;
    1213        class QuadTree {
    13                 public:
     14                private:
    1415                        class QuadTreeBox {
    1516                                public:
  • issm/trunk/src/c/Bamgx/objects/QuadTree.cpp

    r2910 r2980  
    2323        //NORM(i1,j1,i2,j2) returns max(|i1-j1|,|i2-j2|)
    2424#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         */
    3183
    3284        /*Constructors/Destructors*/
     
    76128                QuadTreeBox** pb;
    77129                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)
    79134                pb = &root;
     135
     136                //Find the smallest box where w is located
    80137                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
    85151                if  (b) {     
    86152                        if (b->n > 3 &&  b->v[3] == &w) return;
     
    90156                }
    91157
    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
    96165                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];
    99169                        v4[0]= b->v[0];
    100170                        v4[1]= b->v[1];
    101171                        v4[2]= b->v[2];
    102172                        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++){
    107185                                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
    111192                                bb->v[bb->n++] = v4[k];
    112193                        }
    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)
    117206                NbVertices++;   
    118207        }
     
    126215                Icoor1   ii[MaxDeep];
    127216                Icoor1   jj[MaxDeep];
    128                 register int l=0; // level
     217                register int level=0; // levelevelevel
     218                register Int4 n0;
    129219                register QuadTreeBox* b;
    130220                IntQuad  h=MaxISize,h0;
     
    138228                Vertex*  vn=NULL;
    139229
    140                 //initialization for optimization
     230                //Get initial Quadtree box (largest)
    141231                b = root;
    142                 register Int4 n0;
    143232
    144233                //if the tree is empty, return NULL pointer
    145234                if (!root->n) return vn;
    146235
    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
    154250                        NbQuadTreeBoxSearch++;
    155251                        b=b0;   
     
    159255                }
    160256
    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
    163260                        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){
    167264                                        h = h0;
    168                                         vn = b->v[k];}
    169                                         NbVerticesSearch++;
     265                                        vn = b->v[k];
     266                                }
     267                                NbVerticesSearch++;
    170268                        }
    171269                        return vn;
    172270                }
    173271
    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
    179282                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,
    185297                                if (b->n>0){ // Vertex QuadTreeBox not empty
    186298                                        NbVerticesSearch++;
    187299                                        I2 i2 =  b->v[k]->i;
    188300                                        h0 = NORM(iplus,i2.x,jplus,i2.y);
    189                                         if (h0 <h)
    190                                           {
     301                                        if (h0 <h){
    191302                                                h = h0;
    192303                                                vn = b->v[k];
    193                                           }
     304                                        }
    194305                                }
    195                                 else{ // Pointer QuadTreeBox
    196                                         register QuadTreeBox *b0=b;
     306
     307                                else{
     308                                        register QuadTreeBox* b0=b;
    197309                                        NbQuadTreeBoxSearch++;
     310
     311                                        //if the next box exists:
    198312                                        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
    203319                                                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;
    208324                                                }
     325
     326                                                //else go backwards
    209327                                                else{
    210                                                         b=b0, hb <<=1 ;
     328                                                        //shifted righ by one bit: hb=001000000 -> 01000000
     329                                                        b=b0;
     330                                                        hb<<=1;
    211331                                                }
    212332                                        }
     333                                        //Go backwards
    213334                                        else b=b0;
    214335                                }
    215336                        }
    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
    218344                return vn;
    219345        }
Note: See TracChangeset for help on using the changeset viewer.