libowfat/scan/scan_varint.c

40 lines
1.1 KiB
C
Raw Normal View History

#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;
2016-04-25 19:23:50 +00:00
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;
}
2016-05-23 09:44:14 +00:00
#ifdef __ELF__
size_t scan_pb_type0_int(const char* dest,size_t len,unsigned long long* l) __attribute__((alias("scan_varint")));
2016-05-23 09:44:14 +00:00
#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