mirror of
https://github.com/fluencelabs/redis
synced 2025-03-30 22:31:03 +00:00
Initial work for ziplist backed sorted sets
This commit is contained in:
parent
9e7cee0ed0
commit
21c5b508a4
314
src/t_zset.c
314
src/t_zset.c
@ -403,6 +403,157 @@ static int zslParseRange(robj *min, robj *max, zrangespec *spec) {
|
|||||||
return REDIS_OK;
|
return REDIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------------------------
|
||||||
|
* Ziplist-backed sorted set API
|
||||||
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
double zzlGetScore(unsigned char *sptr) {
|
||||||
|
unsigned char *vstr;
|
||||||
|
unsigned int vlen;
|
||||||
|
long long vlong;
|
||||||
|
char buf[128];
|
||||||
|
double score;
|
||||||
|
|
||||||
|
redisAssert(sptr != NULL);
|
||||||
|
redisAssert(ziplistGet(sptr,&vstr,&vlen,&vlong));
|
||||||
|
|
||||||
|
if (vstr) {
|
||||||
|
memcpy(buf,vstr,vlen);
|
||||||
|
buf[vlen] = '\0';
|
||||||
|
score = strtod(buf,NULL);
|
||||||
|
} else {
|
||||||
|
score = vlong;
|
||||||
|
}
|
||||||
|
|
||||||
|
return score;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Compare element in sorted set with given element. */
|
||||||
|
int zzlCompareElements(unsigned char *eptr, unsigned char *cstr, unsigned int clen) {
|
||||||
|
unsigned char *vstr;
|
||||||
|
unsigned int vlen;
|
||||||
|
long long vlong;
|
||||||
|
unsigned char vbuf[32];
|
||||||
|
int minlen, cmp;
|
||||||
|
|
||||||
|
redisAssert(ziplistGet(eptr,&vstr,&vlen,&vlong));
|
||||||
|
if (vstr == NULL) {
|
||||||
|
/* Store string representation of long long in buf. */
|
||||||
|
vlen = ll2string((char*)vbuf,sizeof(vbuf),vlong);
|
||||||
|
vstr = vbuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
minlen = (vlen < clen) ? vlen : clen;
|
||||||
|
cmp = memcmp(vstr,cstr,minlen);
|
||||||
|
if (cmp == 0) return vlen-clen;
|
||||||
|
return cmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned char *zzlFind(robj *zobj, robj *ele, double *score) {
|
||||||
|
unsigned char *zl = zobj->ptr;
|
||||||
|
unsigned char *eptr = ziplistIndex(zl,0), *sptr;
|
||||||
|
|
||||||
|
ele = getDecodedObject(ele);
|
||||||
|
while (eptr != NULL) {
|
||||||
|
sptr = ziplistNext(zl,eptr);
|
||||||
|
redisAssert(sptr != NULL);
|
||||||
|
|
||||||
|
if (ziplistCompare(eptr,ele->ptr,sdslen(ele->ptr))) {
|
||||||
|
/* Matching element, pull out score. */
|
||||||
|
*score = zzlGetScore(sptr);
|
||||||
|
decrRefCount(ele);
|
||||||
|
return eptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Move to next element. */
|
||||||
|
eptr = ziplistNext(zl,sptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
decrRefCount(ele);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Delete (element,score) pair from ziplist. Use local copy of eptr because we
|
||||||
|
* don't want to modify the one given as argument. */
|
||||||
|
int zzlDelete(robj *zobj, unsigned char *eptr) {
|
||||||
|
unsigned char *zl = zobj->ptr;
|
||||||
|
unsigned char *p = eptr;
|
||||||
|
|
||||||
|
/* TODO: add function to ziplist API to delete N elements from offset. */
|
||||||
|
zl = ziplistDelete(zl,&p);
|
||||||
|
zl = ziplistDelete(zl,&p);
|
||||||
|
zobj->ptr = zl;
|
||||||
|
return REDIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
int zzlInsertAt(robj *zobj, robj *ele, double score, unsigned char *eptr) {
|
||||||
|
unsigned char *zl = zobj->ptr;
|
||||||
|
unsigned char *sptr;
|
||||||
|
char scorebuf[128];
|
||||||
|
int scorelen;
|
||||||
|
int offset;
|
||||||
|
|
||||||
|
redisAssert(ele->encoding == REDIS_ENCODING_RAW);
|
||||||
|
scorelen = d2string(scorebuf,sizeof(scorebuf),score);
|
||||||
|
if (eptr == NULL) {
|
||||||
|
zl = ziplistPush(zl,ele->ptr,sdslen(ele->ptr),ZIPLIST_TAIL);
|
||||||
|
zl = ziplistPush(zl,(unsigned char*)scorebuf,scorelen,ZIPLIST_TAIL);
|
||||||
|
} else {
|
||||||
|
/* Keep offset relative to zl, as it might be re-allocated. */
|
||||||
|
offset = eptr-zl;
|
||||||
|
zl = ziplistInsert(zl,eptr,ele->ptr,sdslen(ele->ptr));
|
||||||
|
eptr = zl+offset;
|
||||||
|
|
||||||
|
/* Insert score after the element. */
|
||||||
|
redisAssert((sptr = ziplistNext(zl,eptr)) != NULL);
|
||||||
|
zl = ziplistInsert(zl,sptr,(unsigned char*)scorebuf,scorelen);
|
||||||
|
}
|
||||||
|
|
||||||
|
zobj->ptr = zl;
|
||||||
|
return REDIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Insert (element,score) pair in ziplist. This function assumes the element is
|
||||||
|
* not yet present in the list. */
|
||||||
|
int zzlInsert(robj *zobj, robj *ele, double score) {
|
||||||
|
unsigned char *zl = zobj->ptr;
|
||||||
|
unsigned char *eptr = ziplistIndex(zl,0), *sptr;
|
||||||
|
double s;
|
||||||
|
int insert = 0;
|
||||||
|
|
||||||
|
ele = getDecodedObject(ele);
|
||||||
|
while (eptr != NULL) {
|
||||||
|
sptr = ziplistNext(zl,eptr);
|
||||||
|
redisAssert(sptr != NULL);
|
||||||
|
s = zzlGetScore(sptr);
|
||||||
|
|
||||||
|
if (s > score) {
|
||||||
|
/* First element with score larger than score for element to be
|
||||||
|
* inserted. This means we should take its spot in the list to
|
||||||
|
* maintain ordering. */
|
||||||
|
insert = 1;
|
||||||
|
} else if (s == score) {
|
||||||
|
/* Ensure lexicographical ordering for elements. */
|
||||||
|
if (zzlCompareElements(eptr,ele->ptr,sdslen(ele->ptr)) < 0)
|
||||||
|
insert = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (insert) {
|
||||||
|
zzlInsertAt(zobj,ele,score,eptr);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Move to next element. */
|
||||||
|
eptr = ziplistNext(zl,sptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Push on tail of list when it was not yet inserted. */
|
||||||
|
if (!insert)
|
||||||
|
zzlInsertAt(zobj,ele,score,eptr);
|
||||||
|
|
||||||
|
decrRefCount(ele);
|
||||||
|
return REDIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
/*-----------------------------------------------------------------------------
|
/*-----------------------------------------------------------------------------
|
||||||
* Sorted set commands
|
* Sorted set commands
|
||||||
@ -410,90 +561,113 @@ static int zslParseRange(robj *min, robj *max, zrangespec *spec) {
|
|||||||
|
|
||||||
/* This generic command implements both ZADD and ZINCRBY. */
|
/* This generic command implements both ZADD and ZINCRBY. */
|
||||||
void zaddGenericCommand(redisClient *c, robj *key, robj *ele, double score, int incr) {
|
void zaddGenericCommand(redisClient *c, robj *key, robj *ele, double score, int incr) {
|
||||||
robj *zsetobj;
|
static char *nanerr = "resulting score is not a number (NaN)";
|
||||||
zset *zs;
|
robj *zobj;
|
||||||
zskiplistNode *znode;
|
robj *curobj;
|
||||||
|
double curscore = 0.0;
|
||||||
|
|
||||||
zsetobj = lookupKeyWrite(c->db,key);
|
zobj = lookupKeyWrite(c->db,key);
|
||||||
if (zsetobj == NULL) {
|
if (zobj == NULL) {
|
||||||
zsetobj = createZsetObject();
|
zobj = createZsetZiplistObject();
|
||||||
dbAdd(c->db,key,zsetobj);
|
dbAdd(c->db,key,zobj);
|
||||||
} else {
|
} else {
|
||||||
if (zsetobj->type != REDIS_ZSET) {
|
if (zobj->type != REDIS_ZSET) {
|
||||||
addReply(c,shared.wrongtypeerr);
|
addReply(c,shared.wrongtypeerr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
zs = zsetobj->ptr;
|
|
||||||
|
|
||||||
/* Since both ZADD and ZINCRBY are implemented here, we need to increment
|
if (zobj->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
* the score first by the current score if ZINCRBY is called. */
|
unsigned char *eptr;
|
||||||
if (incr) {
|
|
||||||
/* Read the old score. If the element was not present starts from 0 */
|
|
||||||
dictEntry *de = dictFind(zs->dict,ele);
|
|
||||||
if (de != NULL)
|
|
||||||
score += *(double*)dictGetEntryVal(de);
|
|
||||||
|
|
||||||
if (isnan(score)) {
|
if ((eptr = zzlFind(zobj,ele,&curscore)) != NULL) {
|
||||||
addReplyError(c,"resulting score is not a number (NaN)");
|
if (incr) {
|
||||||
/* Note that we don't need to check if the zset may be empty and
|
score += curscore;
|
||||||
* should be removed here, as we can only obtain Nan as score if
|
if (isnan(score)) {
|
||||||
* there was already an element in the sorted set. */
|
addReplyError(c,nanerr);
|
||||||
return;
|
/* Don't need to check if the sorted set is empty, because
|
||||||
}
|
* we know it has at least one element. */
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* We need to remove and re-insert the element when it was already present
|
/* Remove and re-insert when score changed. */
|
||||||
* in the dictionary, to update the skiplist. Note that we delay adding a
|
if (score != curscore) {
|
||||||
* pointer to the score because we want to reference the score in the
|
redisAssert(zzlDelete(zobj,eptr) == REDIS_OK);
|
||||||
* skiplist node. */
|
redisAssert(zzlInsert(zobj,ele,score) == REDIS_OK);
|
||||||
if (dictAdd(zs->dict,ele,NULL) == DICT_OK) {
|
|
||||||
dictEntry *de;
|
|
||||||
|
|
||||||
/* New element */
|
signalModifiedKey(c->db,key);
|
||||||
incrRefCount(ele); /* added to hash */
|
server.dirty++;
|
||||||
znode = zslInsert(zs->zsl,score,ele);
|
}
|
||||||
incrRefCount(ele); /* added to skiplist */
|
|
||||||
|
|
||||||
/* Update the score in the dict entry */
|
if (incr) /* ZINCRBY */
|
||||||
de = dictFind(zs->dict,ele);
|
addReplyDouble(c,score);
|
||||||
redisAssert(de != NULL);
|
else /* ZADD */
|
||||||
dictGetEntryVal(de) = &znode->score;
|
addReply(c,shared.czero);
|
||||||
signalModifiedKey(c->db,c->argv[1]);
|
} else {
|
||||||
server.dirty++;
|
redisAssert(zzlInsert(zobj,ele,score) == REDIS_OK);
|
||||||
if (incr)
|
|
||||||
addReplyDouble(c,score);
|
|
||||||
else
|
|
||||||
addReply(c,shared.cone);
|
|
||||||
} else {
|
|
||||||
dictEntry *de;
|
|
||||||
robj *curobj;
|
|
||||||
double *curscore;
|
|
||||||
int deleted;
|
|
||||||
|
|
||||||
/* Update score */
|
signalModifiedKey(c->db,key);
|
||||||
de = dictFind(zs->dict,ele);
|
|
||||||
redisAssert(de != NULL);
|
|
||||||
curobj = dictGetEntryKey(de);
|
|
||||||
curscore = dictGetEntryVal(de);
|
|
||||||
|
|
||||||
/* When the score is updated, reuse the existing string object to
|
|
||||||
* prevent extra alloc/dealloc of strings on ZINCRBY. */
|
|
||||||
if (score != *curscore) {
|
|
||||||
deleted = zslDelete(zs->zsl,*curscore,curobj);
|
|
||||||
redisAssert(deleted != 0);
|
|
||||||
znode = zslInsert(zs->zsl,score,curobj);
|
|
||||||
incrRefCount(curobj);
|
|
||||||
|
|
||||||
/* Update the score in the current dict entry */
|
|
||||||
dictGetEntryVal(de) = &znode->score;
|
|
||||||
signalModifiedKey(c->db,c->argv[1]);
|
|
||||||
server.dirty++;
|
server.dirty++;
|
||||||
|
|
||||||
|
if (incr) /* ZINCRBY */
|
||||||
|
addReplyDouble(c,score);
|
||||||
|
else /* ZADD */
|
||||||
|
addReply(c,shared.cone);
|
||||||
}
|
}
|
||||||
if (incr)
|
} else if (zobj->encoding == REDIS_ENCODING_RAW) {
|
||||||
addReplyDouble(c,score);
|
zset *zs = zobj->ptr;
|
||||||
else
|
zskiplistNode *znode;
|
||||||
addReply(c,shared.czero);
|
dictEntry *de;
|
||||||
|
|
||||||
|
de = dictFind(zs->dict,ele);
|
||||||
|
if (de != NULL) {
|
||||||
|
curobj = dictGetEntryKey(de);
|
||||||
|
curscore = *(double*)dictGetEntryVal(de);
|
||||||
|
|
||||||
|
if (incr) {
|
||||||
|
score += curscore;
|
||||||
|
if (isnan(score)) {
|
||||||
|
addReplyError(c,nanerr);
|
||||||
|
/* Don't need to check if the sorted set is empty, because
|
||||||
|
* we know it has at least one element. */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove and re-insert when score changed. We can safely delete
|
||||||
|
* the key object from the skiplist, since the dictionary still has
|
||||||
|
* a reference to it. */
|
||||||
|
if (score != curscore) {
|
||||||
|
redisAssert(zslDelete(zs->zsl,curscore,curobj));
|
||||||
|
znode = zslInsert(zs->zsl,score,curobj);
|
||||||
|
incrRefCount(curobj); /* Re-inserted in skiplist. */
|
||||||
|
dictGetEntryVal(de) = &znode->score; /* Update score ptr. */
|
||||||
|
|
||||||
|
signalModifiedKey(c->db,key);
|
||||||
|
server.dirty++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (incr) /* ZINCRBY */
|
||||||
|
addReplyDouble(c,score);
|
||||||
|
else /* ZADD */
|
||||||
|
addReply(c,shared.czero);
|
||||||
|
} else {
|
||||||
|
znode = zslInsert(zs->zsl,score,ele);
|
||||||
|
incrRefCount(ele); /* Inserted in skiplist. */
|
||||||
|
redisAssert(dictAdd(zs->dict,ele,&znode->score) == DICT_OK);
|
||||||
|
incrRefCount(ele); /* Added to dictionary. */
|
||||||
|
|
||||||
|
signalModifiedKey(c->db,key);
|
||||||
|
server.dirty++;
|
||||||
|
|
||||||
|
if (incr) /* ZINCRBY */
|
||||||
|
addReplyDouble(c,score);
|
||||||
|
else /* ZADD */
|
||||||
|
addReply(c,shared.cone);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown sorted set encoding");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user