| Index: third_party/sqlite/src/src/test_rtree.c | 
| diff --git a/third_party/sqlite/src/src/test_rtree.c b/third_party/sqlite/src/src/test_rtree.c | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..5fc994ddf204c9d1771fc95a5589363045aeb720 | 
| --- /dev/null | 
| +++ b/third_party/sqlite/src/src/test_rtree.c | 
| @@ -0,0 +1,294 @@ | 
| +/* | 
| +** 2010 August 28 | 
| +** | 
| +** The author disclaims copyright to this source code.  In place of | 
| +** a legal notice, here is a blessing: | 
| +** | 
| +**    May you do good and not evil. | 
| +**    May you find forgiveness for yourself and forgive others. | 
| +**    May you share freely, never taking more than you give. | 
| +** | 
| +************************************************************************* | 
| +** Code for testing all sorts of SQLite interfaces. This code | 
| +** is not included in the SQLite library. | 
| +*/ | 
| + | 
| +#include <sqlite3.h> | 
| + | 
| +/* Solely for the UNUSED_PARAMETER() macro. */ | 
| +#include "sqliteInt.h" | 
| + | 
| +/* | 
| +** Type used to cache parameter information for the "circle" r-tree geometry | 
| +** callback. | 
| +*/ | 
| +typedef struct Circle Circle; | 
| +struct Circle { | 
| +  struct Box { | 
| +    double xmin; | 
| +    double xmax; | 
| +    double ymin; | 
| +    double ymax; | 
| +  } aBox[2]; | 
| +  double centerx; | 
| +  double centery; | 
| +  double radius; | 
| +}; | 
| + | 
| +/* | 
| +** Destructor function for Circle objects allocated by circle_geom(). | 
| +*/ | 
| +static void circle_del(void *p){ | 
| +  sqlite3_free(p); | 
| +} | 
| + | 
| +/* | 
| +** Implementation of "circle" r-tree geometry callback. | 
| +*/ | 
| +static int circle_geom( | 
| +  sqlite3_rtree_geometry *p, | 
| +  int nCoord, | 
| +  double *aCoord, | 
| +  int *pRes | 
| +){ | 
| +  int i;                          /* Iterator variable */ | 
| +  Circle *pCircle;                /* Structure defining circular region */ | 
| +  double xmin, xmax;              /* X dimensions of box being tested */ | 
| +  double ymin, ymax;              /* X dimensions of box being tested */ | 
| + | 
| +  if( p->pUser==0 ){ | 
| +    /* If pUser is still 0, then the parameter values have not been tested | 
| +    ** for correctness or stored into a Circle structure yet. Do this now. */ | 
| + | 
| +    /* This geometry callback is for use with a 2-dimensional r-tree table. | 
| +    ** Return an error if the table does not have exactly 2 dimensions. */ | 
| +    if( nCoord!=4 ) return SQLITE_ERROR; | 
| + | 
| +    /* Test that the correct number of parameters (3) have been supplied, | 
| +    ** and that the parameters are in range (that the radius of the circle | 
| +    ** radius is greater than zero). */ | 
| +    if( p->nParam!=3 || p->aParam[2]<0.0 ) return SQLITE_ERROR; | 
| + | 
| +    /* Allocate a structure to cache parameter data in. Return SQLITE_NOMEM | 
| +    ** if the allocation fails. */ | 
| +    pCircle = (Circle *)(p->pUser = sqlite3_malloc(sizeof(Circle))); | 
| +    if( !pCircle ) return SQLITE_NOMEM; | 
| +    p->xDelUser = circle_del; | 
| + | 
| +    /* Record the center and radius of the circular region. One way that | 
| +    ** tested bounding boxes that intersect the circular region are detected | 
| +    ** is by testing if each corner of the bounding box lies within radius | 
| +    ** units of the center of the circle. */ | 
| +    pCircle->centerx = p->aParam[0]; | 
| +    pCircle->centery = p->aParam[1]; | 
| +    pCircle->radius = p->aParam[2]; | 
| + | 
| +    /* Define two bounding box regions. The first, aBox[0], extends to | 
| +    ** infinity in the X dimension. It covers the same range of the Y dimension | 
| +    ** as the circular region. The second, aBox[1], extends to infinity in | 
| +    ** the Y dimension and is constrained to the range of the circle in the | 
| +    ** X dimension. | 
| +    ** | 
| +    ** Then imagine each box is split in half along its short axis by a line | 
| +    ** that intersects the center of the circular region. A bounding box | 
| +    ** being tested can be said to intersect the circular region if it contains | 
| +    ** points from each half of either of the two infinite bounding boxes. | 
| +    */ | 
| +    pCircle->aBox[0].xmin = pCircle->centerx; | 
| +    pCircle->aBox[0].xmax = pCircle->centerx; | 
| +    pCircle->aBox[0].ymin = pCircle->centery + pCircle->radius; | 
| +    pCircle->aBox[0].ymax = pCircle->centery - pCircle->radius; | 
| +    pCircle->aBox[1].xmin = pCircle->centerx + pCircle->radius; | 
| +    pCircle->aBox[1].xmax = pCircle->centerx - pCircle->radius; | 
| +    pCircle->aBox[1].ymin = pCircle->centery; | 
| +    pCircle->aBox[1].ymax = pCircle->centery; | 
| +  } | 
| + | 
| +  pCircle = (Circle *)p->pUser; | 
| +  xmin = aCoord[0]; | 
| +  xmax = aCoord[1]; | 
| +  ymin = aCoord[2]; | 
| +  ymax = aCoord[3]; | 
| + | 
| +  /* Check if any of the 4 corners of the bounding-box being tested lie | 
| +  ** inside the circular region. If they do, then the bounding-box does | 
| +  ** intersect the region of interest. Set the output variable to true and | 
| +  ** return SQLITE_OK in this case. */ | 
| +  for(i=0; i<4; i++){ | 
| +    double x = (i&0x01) ? xmax : xmin; | 
| +    double y = (i&0x02) ? ymax : ymin; | 
| +    double d2; | 
| + | 
| +    d2  = (x-pCircle->centerx)*(x-pCircle->centerx); | 
| +    d2 += (y-pCircle->centery)*(y-pCircle->centery); | 
| +    if( d2<(pCircle->radius*pCircle->radius) ){ | 
| +      *pRes = 1; | 
| +      return SQLITE_OK; | 
| +    } | 
| +  } | 
| + | 
| +  /* Check if the bounding box covers any other part of the circular region. | 
| +  ** See comments above for a description of how this test works. If it does | 
| +  ** cover part of the circular region, set the output variable to true | 
| +  ** and return SQLITE_OK. */ | 
| +  for(i=0; i<2; i++){ | 
| +    if( xmin<=pCircle->aBox[i].xmin | 
| +     && xmax>=pCircle->aBox[i].xmax | 
| +     && ymin<=pCircle->aBox[i].ymin | 
| +     && ymax>=pCircle->aBox[i].ymax | 
| +    ){ | 
| +      *pRes = 1; | 
| +      return SQLITE_OK; | 
| +    } | 
| +  } | 
| + | 
| +  /* The specified bounding box does not intersect the circular region. Set | 
| +  ** the output variable to zero and return SQLITE_OK. */ | 
| +  *pRes = 0; | 
| +  return SQLITE_OK; | 
| +} | 
| + | 
| +/* END of implementation of "circle" geometry callback. | 
| +************************************************************************** | 
| +*************************************************************************/ | 
| + | 
| +#include <assert.h> | 
| +#include "tcl.h" | 
| + | 
| +typedef struct Cube Cube; | 
| +struct Cube { | 
| +  double x; | 
| +  double y; | 
| +  double z; | 
| +  double width; | 
| +  double height; | 
| +  double depth; | 
| +}; | 
| + | 
| +static void cube_context_free(void *p){ | 
| +  sqlite3_free(p); | 
| +} | 
| + | 
| +/* | 
| +** The context pointer registered along with the 'cube' callback is | 
| +** always ((void *)&gHere). This is just to facilitate testing, it is not | 
| +** actually used for anything. | 
| +*/ | 
| +static int gHere = 42; | 
| + | 
| +/* | 
| +** Implementation of a simple r-tree geom callback to test for intersection | 
| +** of r-tree rows with a "cube" shape. Cubes are defined by six scalar | 
| +** coordinates as follows: | 
| +** | 
| +**   cube(x, y, z, width, height, depth) | 
| +** | 
| +** The width, height and depth parameters must all be greater than zero. | 
| +*/ | 
| +static int cube_geom( | 
| +  sqlite3_rtree_geometry *p, | 
| +  int nCoord, | 
| +  double *aCoord, | 
| +  int *piRes | 
| +){ | 
| +  Cube *pCube = (Cube *)p->pUser; | 
| + | 
| +  assert( p->pContext==(void *)&gHere ); | 
| + | 
| +  if( pCube==0 ){ | 
| +    if( p->nParam!=6 || nCoord!=6 | 
| +     || p->aParam[3]<=0.0 || p->aParam[4]<=0.0 || p->aParam[5]<=0.0 | 
| +    ){ | 
| +      return SQLITE_ERROR; | 
| +    } | 
| +    pCube = (Cube *)sqlite3_malloc(sizeof(Cube)); | 
| +    if( !pCube ){ | 
| +      return SQLITE_NOMEM; | 
| +    } | 
| +    pCube->x = p->aParam[0]; | 
| +    pCube->y = p->aParam[1]; | 
| +    pCube->z = p->aParam[2]; | 
| +    pCube->width = p->aParam[3]; | 
| +    pCube->height = p->aParam[4]; | 
| +    pCube->depth = p->aParam[5]; | 
| + | 
| +    p->pUser = (void *)pCube; | 
| +    p->xDelUser = cube_context_free; | 
| +  } | 
| + | 
| +  assert( nCoord==6 ); | 
| +  *piRes = 0; | 
| +  if( aCoord[0]<=(pCube->x+pCube->width) | 
| +   && aCoord[1]>=pCube->x | 
| +   && aCoord[2]<=(pCube->y+pCube->height) | 
| +   && aCoord[3]>=pCube->y | 
| +   && aCoord[4]<=(pCube->z+pCube->depth) | 
| +   && aCoord[5]>=pCube->z | 
| +  ){ | 
| +    *piRes = 1; | 
| +  } | 
| + | 
| +  return SQLITE_OK; | 
| +} | 
| + | 
| +static int register_cube_geom( | 
| +  void * clientData, | 
| +  Tcl_Interp *interp, | 
| +  int objc, | 
| +  Tcl_Obj *CONST objv[] | 
| +){ | 
| +#ifndef SQLITE_ENABLE_RTREE | 
| +  UNUSED_PARAMETER(clientData); | 
| +  UNUSED_PARAMETER(interp); | 
| +  UNUSED_PARAMETER(objc); | 
| +  UNUSED_PARAMETER(objv); | 
| +#else | 
| +  extern int getDbPointer(Tcl_Interp*, const char*, sqlite3**); | 
| +  extern const char *sqlite3TestErrorName(int); | 
| +  sqlite3 *db; | 
| +  int rc; | 
| + | 
| +  if( objc!=2 ){ | 
| +    Tcl_WrongNumArgs(interp, 1, objv, "DB"); | 
| +    return TCL_ERROR; | 
| +  } | 
| +  if( getDbPointer(interp, Tcl_GetString(objv[1]), &db) ) return TCL_ERROR; | 
| +  rc = sqlite3_rtree_geometry_callback(db, "cube", cube_geom, (void *)&gHere); | 
| +  Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_STATIC); | 
| +#endif | 
| +  return TCL_OK; | 
| +} | 
| + | 
| +static int register_circle_geom( | 
| +  void * clientData, | 
| +  Tcl_Interp *interp, | 
| +  int objc, | 
| +  Tcl_Obj *CONST objv[] | 
| +){ | 
| +#ifndef SQLITE_ENABLE_RTREE | 
| +  UNUSED_PARAMETER(clientData); | 
| +  UNUSED_PARAMETER(interp); | 
| +  UNUSED_PARAMETER(objc); | 
| +  UNUSED_PARAMETER(objv); | 
| +#else | 
| +  extern int getDbPointer(Tcl_Interp*, const char*, sqlite3**); | 
| +  extern const char *sqlite3TestErrorName(int); | 
| +  sqlite3 *db; | 
| +  int rc; | 
| + | 
| +  if( objc!=2 ){ | 
| +    Tcl_WrongNumArgs(interp, 1, objv, "DB"); | 
| +    return TCL_ERROR; | 
| +  } | 
| +  if( getDbPointer(interp, Tcl_GetString(objv[1]), &db) ) return TCL_ERROR; | 
| +  rc = sqlite3_rtree_geometry_callback(db, "circle", circle_geom, 0); | 
| +  Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_STATIC); | 
| +#endif | 
| +  return TCL_OK; | 
| +} | 
| + | 
| +int Sqlitetestrtree_Init(Tcl_Interp *interp){ | 
| +  Tcl_CreateObjCommand(interp, "register_cube_geom", register_cube_geom, 0, 0); | 
| +  Tcl_CreateObjCommand(interp, "register_circle_geom",register_circle_geom,0,0); | 
| +  return TCL_OK; | 
| +} | 
|  |