You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

40 lines
1.1 KiB
C

#include "scan.h"
size_t scan_varint(const char* in,size_t len, unsigned long long* n) {
size_t i;
unsigned long long l;
if (len==0) return 0;
for (l=0, i=0; i<len; ++i) {
if (i*7+7>sizeof(l)*8) /* value too large to fit in destination integer! */
return 0;
l+=(unsigned long long)(in[i]&0x7f) << (i*7);
if (!(in[i]&0x80)) {
if (in[i]==0) return 0; /* libowfat extension: reject non-minimal encoding */
*n=l;
return i+1;
}
}
return 0;
}
#ifdef __ELF__
size_t scan_pb_type0_int(const char* dest,size_t len,unsigned long long* l) __attribute__((alias("scan_varint")));
#else
size_t scan_pb_type0_int(const char* dest,size_t len,unsigned long long* l) {
return scan_varint(dest,len,l);
}
#endif
#ifdef UNITTEST
#include <assert.h>
int main() {
unsigned long long n;
assert(scan_varint("\x05",1,&n)==1 && n==5);
assert(scan_varint("\x80\x00",2,&n)==0); // non-minimal encoding
assert(scan_varint("\x80\x80",2,&n)==0); // truncated (last byte in sequence has high bit 0)
assert(scan_varint("\x80\x80\x80\x80\x80\x80\x80\x80\x80\x01",10,&n)==0); // value too large
return 0;
}
#endif