- 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 512MB in bytes */
- if ((loffset < 0) || ((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;
- int byte, bit;
- int byteval, bitval;
- long on;
-
- if (getBitOffsetFromArgument(c,c->argv[2],&bitoffset) != REDIS_OK)
- return;
-
- if (getLongFromObjectOrReply(c,c->argv[3],&on,err) != REDIS_OK)
- return;
-
- /* Bits can only be set or cleared... */
- if (on & ~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);
- }
- }
-
- /* Grow sds value to the right length if necessary */
- byte = bitoffset >> 3;
- o->ptr = sdsgrowzero(o->ptr,byte+1);
-
- /* Get current values */
- byteval = ((char*)o->ptr)[byte];
- bit = 7 - (bitoffset & 0x7);
- bitval = byteval & (1 << bit);
-
- /* Update byte with new bit value and return original value */
- byteval &= ~(1 << bit);
- byteval |= ((on & 0x1) << bit);
- ((char*)o->ptr)[byte] = byteval;
- touchWatchedKey(c->db,c->argv[1]);