2015-09-29 23:19:01 +00:00
|
|
|
/* See LICENSE file for copyright and license details. */
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "../util.h"
|
|
|
|
|
|
|
|
off_t
|
|
|
|
parseoffset(const char *str)
|
|
|
|
{
|
|
|
|
off_t res;
|
2015-09-29 23:26:27 +00:00
|
|
|
size_t scale = 1;
|
2015-09-29 23:19:01 +00:00
|
|
|
int base = 10;
|
|
|
|
char *end;
|
|
|
|
|
2015-09-30 17:14:14 +00:00
|
|
|
if (!str || !*str) {
|
|
|
|
weprintf("parseoffset: empty string\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-09-29 23:19:01 +00:00
|
|
|
/* bases */
|
|
|
|
if (!strncasecmp(str, "0x", strlen("0x"))) {
|
|
|
|
base = 16;
|
|
|
|
} else if (*str == '0') {
|
|
|
|
str++;
|
|
|
|
base = 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = strtol(str, &end, base);
|
|
|
|
if (res < 0) {
|
2015-09-30 17:14:14 +00:00
|
|
|
weprintf("parseoffset %s: negative value\n", str);
|
2015-09-29 23:19:01 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* suffix */
|
|
|
|
if (*end) {
|
|
|
|
switch (toupper((int)*end)) {
|
|
|
|
case 'B':
|
|
|
|
scale = 512L;
|
|
|
|
break;
|
|
|
|
case 'K':
|
|
|
|
scale = 1024L;
|
|
|
|
break;
|
|
|
|
case 'M':
|
|
|
|
scale = 1024L * 1024L;
|
|
|
|
break;
|
|
|
|
case 'G':
|
|
|
|
scale = 1024L * 1024L * 1024L;
|
|
|
|
break;
|
|
|
|
default:
|
2015-09-30 17:14:14 +00:00
|
|
|
weprintf("parseoffset %s: invalid suffix\n", str);
|
2015-09-29 23:19:01 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* prevent overflow */
|
|
|
|
if (res > (SIZE_MAX / scale)) {
|
2015-09-30 17:14:14 +00:00
|
|
|
weprintf("parseoffset %s: out of range\n", str);
|
2015-09-29 23:19:01 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-09-29 23:26:27 +00:00
|
|
|
return res * scale;
|
2015-09-29 23:19:01 +00:00
|
|
|
}
|