summaryrefslogtreecommitdiff
path: root/src/geo.c
diff options
context:
space:
mode:
authorfilipe oliveira <filipecosta.90@gmail.com>2022-11-28 08:37:41 +0000
committerGitHub <noreply@github.com>2022-11-28 10:37:41 +0200
commit376b689b03e22889b62e60013d7622cbf98d3dc7 (patch)
treeee712427f49f90a948b8685d521e8b8497e999bf /src/geo.c
parentcb7447b3875efd4b4b00fa48f1da444dbd4980df (diff)
downloadredis-376b689b03e22889b62e60013d7622cbf98d3dc7.tar.gz
Simplified geoAppendIfWithinShape() and removed spurious calls do sdsdup and sdsfree (#11522)
In scenarios in which we have large datasets and the elements are not contained within the range we do spurious calls do sdsdup and sdsfree. I.e. instead of pre-creating an sds before we know if we're gonna use it or not, change the role of geoAppendIfWithinShape to just do geoWithinShape, and let the caller create the string only when needed. Co-authored-by: Oran Agra <oran@redislabs.com>
Diffstat (limited to 'src/geo.c')
-rw-r--r--src/geo.c68
1 files changed, 38 insertions, 30 deletions
diff --git a/src/geo.c b/src/geo.c
index acfca2d08..f4cd44b91 100644
--- a/src/geo.c
+++ b/src/geo.c
@@ -60,14 +60,20 @@ geoArray *geoArrayCreate(void) {
return ga;
}
-/* Add a new entry and return its pointer so that the caller can populate
- * it with data. */
-geoPoint *geoArrayAppend(geoArray *ga) {
+/* Add and populate with data a new entry to the geoArray. */
+geoPoint *geoArrayAppend(geoArray *ga, double *xy, double dist,
+ double score, char *member)
+{
if (ga->used == ga->buckets) {
ga->buckets = (ga->buckets == 0) ? 8 : ga->buckets*2;
ga->array = zrealloc(ga->array,sizeof(geoPoint)*ga->buckets);
}
geoPoint *gp = ga->array+ga->used;
+ gp->longitude = xy[0];
+ gp->latitude = xy[1];
+ gp->dist = dist;
+ gp->member = member;
+ gp->score = score;
ga->used++;
return gp;
}
@@ -210,33 +216,33 @@ void addReplyDoubleDistance(client *c, double d) {
}
/* Helper function for geoGetPointsInRange(): given a sorted set score
- * representing a point, and a GeoShape, appends this entry as a geoPoint
- * into the specified geoArray only if the point is within the search area.
+ * representing a point, and a GeoShape, checks if the point is within the search area.
*
- * returns C_OK if the point is included, or C_ERR if it is outside. */
-int geoAppendIfWithinShape(geoArray *ga, GeoShape *shape, double score, sds member) {
- double distance = 0, xy[2];
-
+ * shape: the rectangle
+ * score: the encoded version of lat,long
+ * xy: output variable, the decoded lat,long
+ * distance: output variable, the distance between the center of the shape and the point
+ *
+ * Return values:
+ *
+ * The return value is C_OK if the point is within search area, or C_ERR if it is outside.
+ * "*xy" is populated with the decoded lat,long.
+ * "*distance" is populated with the distance between the center of the shape and the point.
+ */
+int geoWithinShape(GeoShape *shape, double score, double *xy, double *distance) {
if (!decodeGeohash(score,xy)) return C_ERR; /* Can't decode. */
/* Note that geohashGetDistanceIfInRadiusWGS84() takes arguments in
* reverse order: longitude first, latitude later. */
if (shape->type == CIRCULAR_TYPE) {
if (!geohashGetDistanceIfInRadiusWGS84(shape->xy[0], shape->xy[1], xy[0], xy[1],
- shape->t.radius*shape->conversion, &distance)) return C_ERR;
+ shape->t.radius*shape->conversion, distance))
+ return C_ERR;
} else if (shape->type == RECTANGLE_TYPE) {
if (!geohashGetDistanceIfInRectangle(shape->t.r.width * shape->conversion,
shape->t.r.height * shape->conversion,
- shape->xy[0], shape->xy[1], xy[0], xy[1], &distance))
+ shape->xy[0], shape->xy[1], xy[0], xy[1], distance))
return C_ERR;
}
-
- /* Append the new element. */
- geoPoint *gp = geoArrayAppend(ga);
- gp->longitude = xy[0];
- gp->latitude = xy[1];
- gp->dist = distance;
- gp->member = member;
- gp->score = score;
return C_OK;
}
@@ -257,8 +263,6 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
/* That's: min <= val < max */
zrangespec range = { .min = min, .max = max, .minex = 0, .maxex = 1 };
size_t origincount = ga->used;
- sds member;
-
if (zobj->encoding == OBJ_ENCODING_LISTPACK) {
unsigned char *zl = zobj->ptr;
unsigned char *eptr, *sptr;
@@ -274,6 +278,8 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
sptr = lpNext(zl, eptr);
while (eptr) {
+ double xy[2];
+ double distance = 0;
score = zzlGetScore(sptr);
/* If we fell out of range, break. */
@@ -281,10 +287,11 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
break;
vstr = lpGetValue(eptr, &vlen, &vlong);
- member = (vstr == NULL) ? sdsfromlonglong(vlong) :
- sdsnewlen(vstr,vlen);
- if (geoAppendIfWithinShape(ga,shape,score,member)
- == C_ERR) sdsfree(member);
+ if (geoWithinShape(shape, score, xy, &distance) == C_OK) {
+ /* Append the new element. */
+ char *member = (vstr == NULL) ? sdsfromlonglong(vlong) : sdsnewlen(vstr, vlen);
+ geoArrayAppend(ga, xy, distance, score, member);
+ }
if (ga->used && limit && ga->used >= limit) break;
zzlNext(zl, &eptr, &sptr);
}
@@ -299,14 +306,15 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
}
while (ln) {
- sds ele = ln->ele;
+ double xy[2];
+ double distance = 0;
/* Abort when the node is no longer in range. */
if (!zslValueLteMax(ln->score, &range))
break;
-
- ele = sdsdup(ele);
- if (geoAppendIfWithinShape(ga,shape,ln->score,ele)
- == C_ERR) sdsfree(ele);
+ if (geoWithinShape(shape, ln->score, xy, &distance) == C_OK) {
+ /* Append the new element. */
+ geoArrayAppend(ga, xy, distance, ln->score, sdsdup(ln->ele));
+ }
if (ga->used && limit && ga->used >= limit) break;
ln = ln->level[0].forward;
}