aboutsummaryrefslogtreecommitdiff
path: root/src/core/strs.c
diff options
context:
space:
mode:
authorGarrett D'Amore <garrett@damore.org>2018-08-30 11:08:06 -0700
committerGarrett D'Amore <garrett@damore.org>2018-08-30 15:28:38 -0700
commit629002efac8085a9c6321a15b43a47a43ca0e084 (patch)
tree8d540d491a6583405e625dd11e12f993457837a7 /src/core/strs.c
parente7e980c228c76e108bbc4812f40645d93626757f (diff)
downloadnng-629002efac8085a9c6321a15b43a47a43ca0e084.tar.gz
nng-629002efac8085a9c6321a15b43a47a43ca0e084.tar.bz2
nng-629002efac8085a9c6321a15b43a47a43ca0e084.zip
fixes #686 strtoull() not present on Windows
This both makes new functions available to the core, and addresses a bug which would have prevented building the ZeroTier transport on Windows.
Diffstat (limited to 'src/core/strs.c')
-rw-r--r--src/core/strs.c74
1 files changed, 74 insertions, 0 deletions
diff --git a/src/core/strs.c b/src/core/strs.c
index 98c1a5fa..ed403a3b 100644
--- a/src/core/strs.c
+++ b/src/core/strs.c
@@ -202,3 +202,77 @@ nni_asprintf(char **sp, const char *fmt, ...)
*sp = s;
return (0);
}
+
+int
+nni_strtou64(const char *s, uint64_t *u)
+{
+ uint64_t v = 0;
+
+ // Arguably we could use strtoull, but Windows doesn't conform
+ // to C99, and so lacks it.
+
+ if ((s == NULL) || (*s == '\0')) {
+ // Require a non-empty string.
+ return (NNG_EINVAL);
+ }
+ while (*s) {
+ uint64_t last = v;
+ if (isdigit(*s)) {
+ v *= 10;
+ v += (*s - '0');
+ } else {
+ return (NNG_EINVAL);
+ }
+ if (v < last) {
+ // Overflow!
+ return (NNG_EINVAL);
+ }
+ s++;
+ }
+ *u = v;
+ return (0);
+}
+
+int
+nni_strtox64(const char *s, uint64_t *u)
+{
+ uint64_t v = 0;
+
+ // Arguably we could use strtoull, but Windows doesn't conform
+ // to C99, and so lacks it.
+
+ if (s == NULL) {
+ return (NNG_EINVAL);
+ }
+ // Skip over 0x if present.
+ if ((s[0] == '0') && ((s[1] == 'x') || (s[1] == 'X'))) {
+ s += 2;
+ }
+ if (*s == '\0') {
+ // Require a non-empty string.
+ return (NNG_EINVAL);
+ }
+
+ while (*s) {
+ uint64_t last = v;
+ if (isdigit(*s)) {
+ v *= 16;
+ v += (*s - '0');
+ } else if ((*s >= 'a') && (*s <= 'f')) {
+ v *= 16;
+ v += (*s - 'a') + 10;
+ } else if ((*s >= 'A') && (*s <= 'F')) {
+ v *= 16;
+ v += (*s - 'A') + 10;
+ } else {
+ return (NNG_EINVAL);
+ }
+ if (v < last) {
+ // Overflow!
+ return (NNG_EINVAL);
+ }
+ s++;
+ }
+ *u = v;
+ return (0);
+}