]>
git.saurik.com Git - redis.git/blob - src/bitop.c
1eec63ab96116723a354386aadb7240bf9c7f5af
3 /* -----------------------------------------------------------------------------
4 * Bits related string commands: GETBIT, SETBIT, BITCOUNT, BITOP.
5 * -------------------------------------------------------------------------- */
12 /* This helper function used by GETBIT / SETBIT parses the bit offset arguemnt
13 * making sure an error is returned if it is negative or if it overflows
14 * Redis 512 MB limit for the string value. */
15 static int getBitOffsetFromArgument(redisClient
*c
, robj
*o
, size_t *offset
) {
17 char *err
= "bit offset is not an integer or out of range";
19 if (getLongLongFromObjectOrReply(c
,o
,&loffset
,err
) != REDIS_OK
)
22 /* Limit offset to 512MB in bytes */
23 if ((loffset
< 0) || ((unsigned long long)loffset
>> 3) >= (512*1024*1024))
29 *offset
= (size_t)loffset
;
33 /* SETBIT key offset bitvalue */
34 void setbitCommand(redisClient
*c
) {
36 char *err
= "bit is not an integer or out of range";
42 if (getBitOffsetFromArgument(c
,c
->argv
[2],&bitoffset
) != REDIS_OK
)
45 if (getLongFromObjectOrReply(c
,c
->argv
[3],&on
,err
) != REDIS_OK
)
48 /* Bits can only be set or cleared... */
54 o
= lookupKeyWrite(c
->db
,c
->argv
[1]);
56 o
= createObject(REDIS_STRING
,sdsempty());
57 dbAdd(c
->db
,c
->argv
[1],o
);
59 if (checkType(c
,o
,REDIS_STRING
)) return;
61 /* Create a copy when the object is shared or encoded. */
62 if (o
->refcount
!= 1 || o
->encoding
!= REDIS_ENCODING_RAW
) {
63 robj
*decoded
= getDecodedObject(o
);
64 o
= createStringObject(decoded
->ptr
, sdslen(decoded
->ptr
));
65 decrRefCount(decoded
);
66 dbOverwrite(c
->db
,c
->argv
[1],o
);
70 /* Grow sds value to the right length if necessary */
71 byte
= bitoffset
>> 3;
72 o
->ptr
= sdsgrowzero(o
->ptr
,byte
+1);
74 /* Get current values */
75 byteval
= ((char*)o
->ptr
)[byte
];
76 bit
= 7 - (bitoffset
& 0x7);
77 bitval
= byteval
& (1 << bit
);
79 /* Update byte with new bit value and return original value */
80 byteval
&= ~(1 << bit
);
81 byteval
|= ((on
& 0x1) << bit
);
82 ((char*)o
->ptr
)[byte
] = byteval
;
83 signalModifiedKey(c
->db
,c
->argv
[1]);
85 addReply(c
, bitval
? shared
.cone
: shared
.czero
);
88 /* GETBIT key offset */
89 void getbitCommand(redisClient
*c
) {
96 if (getBitOffsetFromArgument(c
,c
->argv
[2],&bitoffset
) != REDIS_OK
)
99 if ((o
= lookupKeyReadOrReply(c
,c
->argv
[1],shared
.czero
)) == NULL
||
100 checkType(c
,o
,REDIS_STRING
)) return;
102 byte
= bitoffset
>> 3;
103 bit
= 7 - (bitoffset
& 0x7);
104 if (o
->encoding
!= REDIS_ENCODING_RAW
) {
105 if (byte
< (size_t)ll2string(llbuf
,sizeof(llbuf
),(long)o
->ptr
))
106 bitval
= llbuf
[byte
] & (1 << bit
);
108 if (byte
< sdslen(o
->ptr
))
109 bitval
= ((char*)o
->ptr
)[byte
] & (1 << bit
);
112 addReply(c
, bitval
? shared
.cone
: shared
.czero
);
115 /* BITOP op_name target_key src_key1 src_key2 src_key3 ... src_keyN */
116 void bitopCommand(redisClient
*c
) {
117 char *opname
= c
->argv
[1]->ptr
;
118 robj
*o
, *targetkey
= c
->argv
[2];
120 unsigned char **src
; /* Array of source strings pointers. */
121 long *len
, maxlen
= 0; /* Array of length of src strings, and max len. */
122 unsigned char *res
= NULL
; /* Resulting string. */
124 /* Parse the operation name. */
125 if ((opname
[0] == 'a' || opname
[0] == 'A') && !strcasecmp(opname
,"and"))
127 else if((opname
[0] == 'o' || opname
[0] == 'O') && !strcasecmp(opname
,"or"))
129 else if((opname
[0] == 'x' || opname
[0] == 'X') && !strcasecmp(opname
,"xor"))
131 else if((opname
[0] == 'n' || opname
[0] == 'N') && !strcasecmp(opname
,"not"))
134 addReply(c
,shared
.syntaxerr
);
138 /* Sanity check: NOT accepts only a single key argument. */
139 if (op
== BITOP_NOT
&& c
->argc
!= 4) {
140 addReplyError(c
,"BITOP NOT must be called with a single source key.");
144 /* Lookup keys, and store pointers to the string objects into an array. */
145 numkeys
= c
->argc
- 3;
146 src
= zmalloc(sizeof(unsigned char*) * numkeys
);
147 len
= zmalloc(sizeof(long) * numkeys
);
148 for (j
= 0; j
< numkeys
; j
++) {
149 o
= lookupKeyRead(c
->db
,c
->argv
[j
+3]);
150 /* Handle non-existing keys as empty strings. */
156 /* Return an error if one of the keys is not a string. */
157 if (checkType(c
,o
,REDIS_STRING
)) {
163 len
[j
] = sdslen(o
->ptr
);
164 if (len
[j
] > maxlen
) maxlen
= len
[j
];
167 /* Compute the bit operation, if at least one string is not empty. */
169 res
= (unsigned char*) sdsnewlen(NULL
,maxlen
);
170 unsigned char output
, byte
;
173 for (j
= 0; j
< maxlen
; j
++) {
174 output
= (len
[0] <= j
) ? 0 : src
[0][j
];
175 if (op
== BITOP_NOT
) output
= ~output
;
176 for (i
= 1; i
< numkeys
; i
++) {
177 byte
= (len
[i
] <= j
) ? 0 : src
[i
][j
];
179 case BITOP_AND
: output
&= byte
; break;
180 case BITOP_OR
: output
|= byte
; break;
181 case BITOP_XOR
: output
^= byte
; break;
190 /* Store the computed value into the target key */
192 o
= createObject(REDIS_STRING
,res
);
193 setKey(c
->db
,targetkey
,o
);
195 } else if (dbDelete(c
->db
,targetkey
)) {
196 signalModifiedKey(c
->db
,targetkey
);
199 addReplyLongLong(c
,maxlen
); /* Return the output string length in bytes. */
202 /* BITCOUNT key [start end] */
203 void bitcountCommand(redisClient
*c
) {
204 static const unsigned char bitsinbyte
[256] = {0,1,1,2,1,2,2,3,1,2,2,3,2,3,3,4,1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,4,5,5,6,5,6,6,7,5,6,6,7,6,7,7,8};
211 /* Lookup, check for type, and return 0 for non existing keys. */
212 if ((o
= lookupKeyReadOrReply(c
,c
->argv
[1],shared
.czero
)) == NULL
||
213 checkType(c
,o
,REDIS_STRING
)) return;
215 /* Set the 'p' pointer to the string, that can be just a stack allocated
216 * array if our string was integer encoded. */
217 if (o
->encoding
== REDIS_ENCODING_INT
) {
218 p
= (unsigned char*) llbuf
;
219 strlen
= ll2string(llbuf
,sizeof(llbuf
),(long)o
->ptr
);
221 p
= (unsigned char*) o
->ptr
;
222 strlen
= sdslen(o
->ptr
);
225 /* Parse start/end range if any. */
227 if (getLongFromObjectOrReply(c
,c
->argv
[2],&start
,NULL
) != REDIS_OK
)
229 if (getLongFromObjectOrReply(c
,c
->argv
[3],&end
,NULL
) != REDIS_OK
)
231 /* Convert negative indexes */
232 if (start
< 0) start
= strlen
+start
;
233 if (end
< 0) end
= strlen
+end
;
234 if (start
< 0) start
= 0;
235 if (end
< 0) end
= 0;
236 if ((unsigned)end
>= strlen
) end
= strlen
-1;
237 } else if (c
->argc
== 2) {
238 /* The whole string. */
243 addReply(c
,shared
.syntaxerr
);
247 /* Precondition: end >= 0 && end < strlen, so the only condition where
248 * zero can be returned is: start > end. */
250 addReply(c
,shared
.czero
);
252 long bits
= 0, bytes
= end
-start
+1;
254 /* We can finally count bits. */
256 while(bytes
--) bits
+= bitsinbyte
[*p
++];
257 addReplyLongLong(c
,bits
);