- dbReplace(c->db,c->argv[1],c->argv[2]);
- incrRefCount(c->argv[2]);
- touchWatchedKey(c->db,c->argv[1]);
- server.dirty++;
- removeExpire(c->db,c->argv[1]);
-}
-
-static int getBitOffsetFromArgument(redisClient *c, robj *o, size_t *offset) {
- long long loffset;
- char *err = "bit offset is not an integer or out of range";
-
- if (getLongLongFromObjectOrReply(c,o,&loffset,err) != REDIS_OK)
- return REDIS_ERR;
-
- /* Limit offset to SIZE_T_MAX or 512MB in bytes */
- if ((loffset < 0) ||
- ((unsigned long long)loffset >= (unsigned)SIZE_T_MAX) ||
- ((unsigned long long)loffset >> 3) >= (512*1024*1024))
- {
- addReplyError(c,err);
- return REDIS_ERR;
- }
-
- *offset = (size_t)loffset;
- return REDIS_OK;
-}
-
-void setbitCommand(redisClient *c) {
- robj *o;
- char *err = "bit is not an integer or out of range";
- size_t bitoffset;
- long long bitvalue;
- int byte, bit, on;
-
- if (getBitOffsetFromArgument(c,c->argv[2],&bitoffset) != REDIS_OK)
- return;
-
- if (getLongLongFromObjectOrReply(c,c->argv[3],&bitvalue,err) != REDIS_OK)
- return;
-
- /* A bit can only be set to be on or off... */
- if (bitvalue & ~1) {
- addReplyError(c,err);
- return;
- }
-
- o = lookupKeyWrite(c->db,c->argv[1]);
- if (o == NULL) {
- o = createObject(REDIS_STRING,sdsempty());
- dbAdd(c->db,c->argv[1],o);
- } else {
- if (checkType(c,o,REDIS_STRING)) return;
-
- /* Create a copy when the object is shared or encoded. */
- if (o->refcount != 1 || o->encoding != REDIS_ENCODING_RAW) {
- robj *decoded = getDecodedObject(o);
- o = createStringObject(decoded->ptr, sdslen(decoded->ptr));
- decrRefCount(decoded);
- dbReplace(c->db,c->argv[1],o);
- }
- }
-
- byte = bitoffset >> 3;
- bit = 7 - (bitoffset & 0x7);
- on = bitvalue & 0x1;
- o->ptr = sdsgrowzero(o->ptr,byte+1);
- ((char*)o->ptr)[byte] |= on << bit;
- ((char*)o->ptr)[byte] &= ~((!on) << bit);
-
- touchWatchedKey(c->db,c->argv[1]);