mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-04 03:11:28 +00:00
d6d453b4a1
this API has been borrowed from the awesome Frida project
128 lines
2.4 KiB
C
128 lines
2.4 KiB
C
/* radare - LGPL - Copyright 2014-2015 - pancake */
|
|
|
|
#include <r_util.h>
|
|
|
|
/* dex/dwarf uleb128 implementation */
|
|
|
|
R_API const ut8 *r_uleb128(const ut8 *data, int datalen, ut64 *v) {
|
|
ut8 c;
|
|
ut64 s, sum = 0;
|
|
const ut8 *data_end;
|
|
if (v) *v = 0LL;
|
|
if (datalen == ST32_MAX) {
|
|
// WARNING; possible overflow
|
|
datalen = 0xffff;
|
|
}
|
|
if (datalen < 0) {
|
|
return NULL;
|
|
}
|
|
data_end = data + datalen;
|
|
if (data && datalen > 0) {
|
|
if (*data) {
|
|
for (s = 0; data < data_end; s += 7) {
|
|
c = *(data++) & 0xff;
|
|
sum |= ((ut32) (c & 0x7f) << s);
|
|
if (!(c & 0x80)) break;
|
|
}
|
|
} else {
|
|
data++;
|
|
}
|
|
}
|
|
if (v) *v = sum;
|
|
return data;
|
|
}
|
|
|
|
/* data is the char array containing the uleb number
|
|
* datalen will point (if not NULL) to the length of the uleb number
|
|
* v (if not NULL) will point to the data's value (if fitting the size of an ut64)
|
|
*/
|
|
R_API const ut8 *r_uleb128_decode(const ut8 *data, int *datalen, ut64 *v) {
|
|
ut8 c = 0xff;
|
|
ut64 s = 0, sum = 0, l = 0;
|
|
if (data && *data) {
|
|
do {
|
|
c = *(data++) & 0xff;
|
|
sum |= ((ut32) (c&0x7f) << s);
|
|
s += 7;
|
|
l++;
|
|
} while (c & 0x80);
|
|
}
|
|
if (v) *v = sum;
|
|
if (datalen) *datalen = l;
|
|
return data;
|
|
}
|
|
|
|
R_API const ut8 *r_uleb128_encode(const ut64 s, int *len) {
|
|
ut8 c = 0;
|
|
int l = 0;
|
|
ut8 *otarget = NULL, *target = NULL;
|
|
ut64 source = s;
|
|
do {
|
|
l++;
|
|
if (!(otarget = realloc (otarget, l))) {
|
|
l = 0;
|
|
break;
|
|
}
|
|
target = otarget+l-1;
|
|
c = 0; //May not be necessary
|
|
c = source & 0x7f;
|
|
source >>= 7;
|
|
if (source) c |= 0x80;
|
|
*(target) = c;
|
|
} while (source);
|
|
if (len) *len = l;
|
|
return otarget;
|
|
}
|
|
|
|
R_API const ut8 *r_leb128(const ut8 *data, st64 *v) {
|
|
ut8 c = 0;
|
|
st64 s = 0, sum = 0;
|
|
if (data) {
|
|
for (s = 0; *data;) {
|
|
c = *(data++) & 0x0ff;
|
|
sum |= ((st64) (c & 0x7f) << s);
|
|
s += 7;
|
|
if (!(c & 0x80)) break;
|
|
}
|
|
}
|
|
if ((s < (8 * sizeof (sum))) && (c & 0x40)) {
|
|
sum |= -((st64)1 << s);
|
|
}
|
|
if (v) *v = sum;
|
|
return data;
|
|
}
|
|
|
|
|
|
//API borrowed from frida
|
|
#define G_GINT64_CONSTANT(val) (val##L)
|
|
|
|
R_API st64 r_sleb128(const ut8 **data, const ut8 *end) {
|
|
const ut8 *p = *data;
|
|
st64 result = 0;
|
|
int offset = 0;
|
|
ut8 value;
|
|
do {
|
|
st64 chunk;
|
|
value = *p;
|
|
chunk = value & 0x7f;
|
|
result |= (chunk << offset);
|
|
offset += 7;
|
|
}
|
|
while (*p++ & 0x80);
|
|
|
|
if ((value & 0x40) != 0) {
|
|
result |= G_GINT64_CONSTANT (-1) << offset;
|
|
}
|
|
*data = p;
|
|
return result;
|
|
}
|
|
|
|
#if 0
|
|
main() {
|
|
ut32 n;
|
|
ut8 *buf = "\x10\x02\x90\x88";
|
|
r_uleb128 (buf, &n);
|
|
printf ("n = %d\n", n);
|
|
}
|
|
#endif
|