diff --git a/Makefile.dep b/Makefile.dep index e0f0641169df1fb615f71c7685d888472b58d72e..4011310d126be4d53da0c9e0a42ed585baf3d527 100644 --- a/Makefile.dep +++ b/Makefile.dep @@ -789,6 +789,7 @@ endif ifneq (,$(filter uuid,$(USEMODULE))) USEMODULE += hashes USEMODULE += random + USEMODULE += fmt endif # Enable periph_gpio when periph_gpio_irq is enabled diff --git a/sys/fmt/fmt.c b/sys/fmt/fmt.c index 58aa01483b45a4cc13ef744371e622d09a4af3af..df79a40f4e50a5c377c9e6d86cdc8832e0e23b1f 100644 --- a/sys/fmt/fmt.c +++ b/sys/fmt/fmt.c @@ -53,6 +53,16 @@ static inline int _is_digit(char c) return (c >= '0' && c <= '9'); } +static inline int _is_upper(char c) +{ + return (c >= 'A' && c <= 'Z'); +} + +static inline char _to_lower(char c) +{ + return 'a' + (c - 'A'); +} + size_t fmt_byte_hex(char *out, uint8_t byte) { if (out) { @@ -153,6 +163,11 @@ size_t fmt_hex_bytes(uint8_t *out, const char *hex) return final_len; } +size_t fmt_u16_hex(char *out, uint16_t val) +{ + return fmt_bytes_hex_reverse(out, (uint8_t*) &val, 2); +} + size_t fmt_u32_hex(char *out, uint32_t val) { return fmt_bytes_hex_reverse(out, (uint8_t*) &val, 4); @@ -400,6 +415,35 @@ size_t fmt_lpad(char *out, size_t in_len, size_t pad_len, char pad_char) return pad_len; } +size_t fmt_char(char *out, char c) +{ + if (out) { + *out = c; + } + + return 1; +} + +size_t fmt_to_lower(char *out, const char *str) +{ + size_t len = 0; + + while (str && *str) { + if (_is_upper(*str)) { + if (out) { + *out++ = _to_lower(*str); + } + } + else if (out) { + *out++ = *str; + } + str++; + len++; + } + + return len; +} + uint32_t scn_u32_dec(const char *str, size_t n) { uint32_t res = 0; @@ -416,6 +460,30 @@ uint32_t scn_u32_dec(const char *str, size_t n) return res; } +uint32_t scn_u32_hex(const char *str, size_t n) +{ + uint32_t res = 0; + + while (n--) { + char c = *str++; + if (!_is_digit(c)) { + if (_is_upper(c)) { + c = _to_lower(c); + } + if (c == '\0' || c > 'f') { + break; + } + res <<= 4; + res |= c - 'a' + 0xa; + } + else { + res <<= 4; + res |= c - '0'; + } + } + return res; +} + void print(const char *s, size_t n) { #ifdef __WITH_AVRLIBC__ diff --git a/sys/include/fmt.h b/sys/include/fmt.h index 7176494d3098f6071807d8d75e69f8071f5a718f..b1e0a286bb2f8beffb8fa14bf19cbb6bf70d91ca 100644 --- a/sys/include/fmt.h +++ b/sys/include/fmt.h @@ -119,6 +119,20 @@ uint8_t fmt_hex_byte(const char *hex); */ size_t fmt_hex_bytes(uint8_t *out, const char *hex); +/** + * @brief Convert a uint16 value to hex string. + * + * Will write 4 bytes to @p out. + * If @p out is NULL, will only return the number of bytes that would have + * been written. + * + * @param[out] out Pointer to output buffer, or NULL + * @param[in] val Value to convert + * + * @return 4 + */ +size_t fmt_u16_hex(char *out, uint16_t val); + /** * @brief Convert a uint32 value to hex string. * @@ -296,6 +310,19 @@ size_t fmt_s32_dfp(char *out, int32_t val, int fp_digits); */ size_t fmt_float(char *out, float f, unsigned precision); +/** + * @brief Copy @p in char to string (without terminating '\0') + * + * If @p out is NULL, will only return the number of bytes that would have + * been written. + * + * @param[out] out string to write to (or NULL) + * @param[in] c char value to append + * + * @return nr of bytes the function did or would write to out + */ +size_t fmt_char(char *out, char c); + /** * @brief Count characters until '\0' (exclusive) in @p str * @@ -329,6 +356,14 @@ size_t fmt_strnlen(const char *str, size_t maxlen); */ size_t fmt_str(char *out, const char *str); +/** + * @brief Copy null-terminated string to a lowercase string (excluding terminating \0) + * + * @param[out] out Pointer to output buffer, or NULL + * @param[in] str Pointer to null-terminated source string + */ +size_t fmt_to_lower(char *out, const char *str); + /** * @brief Convert digits to uint32 * @@ -341,6 +376,18 @@ size_t fmt_str(char *out, const char *str); */ uint32_t scn_u32_dec(const char *str, size_t n); +/** + * @brief Convert hexadecimal characters to uin32_t + * + * Will convert up to @p n char. Stop at any non-hexadecimal or '\0' character + * + * @param[in] str Pointer to tring to read from + * @param[in] n Maximum number of characters to consider + * + * @return converted uint32_t value + */ +uint32_t scn_u32_hex(const char *str, size_t n); + /** * @brief Print string to stdout * diff --git a/sys/include/uuid.h b/sys/include/uuid.h index 2616ebd3b4beab5f77afa2c10b61dd442a64dfd5..ac050278aa5c0983dbb55620ce2ceefc6db60c89 100644 --- a/sys/include/uuid.h +++ b/sys/include/uuid.h @@ -38,6 +38,8 @@ extern "C" { #define UUID_NODE_LEN (6U) /**< Size of the node identifier in bytes */ +#define UUID_STR_LEN (36U) /**< Size of a string UUID without null character */ + /** * @name UUID version identifiers * @{ @@ -120,7 +122,7 @@ void uuid_v5(uuid_t *uuid, const uuid_t *ns, const uint8_t *name, size_t len); * * @return Version number */ -static inline unsigned uuid_version(uuid_t *uuid) +static inline unsigned uuid_version(const uuid_t *uuid) { uint16_t time_hi_vers = byteorder_ntohs(uuid->time_hi); @@ -135,11 +137,29 @@ static inline unsigned uuid_version(uuid_t *uuid) * * @return True when equal */ -static inline bool uuid_equal(uuid_t *uuid1, uuid_t *uuid2) +static inline bool uuid_equal(const uuid_t *uuid1, const uuid_t *uuid2) { return (memcmp(uuid1, uuid2, sizeof(uuid_t)) == 0); } +/** + * @brief Generate an UUID string from an UUID structure + * + * @param[in] uuid UUID + * @param[out] str null-terminated UUID string, must be at least UUID_STR_LEN + 1 bytes + */ +void uuid_to_string(const uuid_t *uuid, char *str); + +/** + * @brief Populate an UUID structure from an UUID string + * + * @param[out] uuid out UUID + * @param[in] str null-terminated input UUID string, must be UUID_STR_LEN bytes + * + * @return 0 on succes, < 0 if @p str is not valid + */ +int uuid_from_string(uuid_t *uuid, const char *str); + #ifdef __cplusplus } #endif diff --git a/sys/uuid/uuid.c b/sys/uuid/uuid.c index a7c9c95e4073c5413c414c627d25328bc881dba3..464586dcbefcd84b1bd35f48229476485bd1f5e4 100644 --- a/sys/uuid/uuid.c +++ b/sys/uuid/uuid.c @@ -23,6 +23,7 @@ #include "hashes/sha1.h" #include "random.h" #include "uuid.h" +#include "fmt.h" const uuid_t uuid_namespace_dns = { /* 6ba7b810-9dad-11d1-80b4-00c04fd430c8 */ .time_low.u8 = { 0x6b, 0xa7, 0xb8, 0x10 }, @@ -110,3 +111,59 @@ void uuid_v5(uuid_t *uuid, const uuid_t *ns, const uint8_t *name, size_t len) _set_version(uuid, UUID_V5); _set_reserved(uuid); } + +void uuid_to_string(const uuid_t *uuid, char *str) +{ + char *p = str; + p += fmt_u32_hex(p, byteorder_ntohl(uuid->time_low)); + p += fmt_char(p, '-'); + p += fmt_u16_hex(p, byteorder_ntohs(uuid->time_mid)); + p += fmt_char(p, '-'); + p += fmt_u16_hex(p, byteorder_ntohs(uuid->time_hi)); + p += fmt_char(p, '-'); + p += fmt_byte_hex(p, uuid->clk_seq_hi_res); + p += fmt_byte_hex(p, uuid->clk_seq_low); + p += fmt_char(p, '-'); + p += fmt_bytes_hex(p, uuid->node, UUID_NODE_LEN); + *p = '\0'; + fmt_to_lower(str, str); +} + +int uuid_from_string(uuid_t *uuid, const char *str) +{ + uint32_t tmp; + if (fmt_strlen(str) < UUID_STR_LEN) { + return -1; + } + tmp = scn_u32_hex(str, 8); + uuid->time_low = byteorder_htonl(tmp); + str += 8; + if (*str++ != '-') { + return -2; + } + tmp = scn_u32_hex(str, 4); + uuid->time_mid = byteorder_htons(tmp); + str += 4; + if (*str++ != '-') { + return -2; + } + tmp = scn_u32_hex(str, 4); + uuid->time_hi = byteorder_htons(tmp); + str += 4; + if (*str++ != '-') { + return -2; + } + uuid->clk_seq_hi_res = scn_u32_hex(str, 2); + str += 2; + uuid->clk_seq_low = scn_u32_hex(str, 2); + str += 2; + if (*str++ != '-') { + return -2; + } + for (unsigned i = 0; i < UUID_NODE_LEN; i++) { + uuid->node[i] = scn_u32_hex(str, 2); + str += 2; + } + + return 0; +} diff --git a/tests/unittests/tests-fmt/tests-fmt.c b/tests/unittests/tests-fmt/tests-fmt.c index 0a7d537828c19432d58f74bee427413fbd907cd6..8126beafccd3ced62edb818e8ce4325cc071f846 100644 --- a/tests/unittests/tests-fmt/tests-fmt.c +++ b/tests/unittests/tests-fmt/tests-fmt.c @@ -749,6 +749,26 @@ static void test_fmt_str(void) TEST_ASSERT_EQUAL_STRING(string1, &string2[0]); } +static void test_fmt_char(void) +{ + char string[] = "zzzzzzzzz"; + + TEST_ASSERT_EQUAL_INT(1, fmt_char(NULL, 'c')); + TEST_ASSERT_EQUAL_INT(1, fmt_char(string, 'c')); + string[1] = '\0'; + TEST_ASSERT_EQUAL_STRING("c", &string[0]); +} + +static void test_fmt_to_lower(void) +{ + const char string_up[] = "AbCdeFGHijkLM"; + char string[] = "zzzzzzzzzzzzzzz"; + + TEST_ASSERT_EQUAL_INT(fmt_strlen(string_up), fmt_to_lower(string, string_up)); + string[fmt_strlen(string_up)] = '\0'; + TEST_ASSERT_EQUAL_STRING("abcdefghijklm", &string[0]); +} + static void test_scn_u32_dec(void) { const char *string1 = "123456789"; @@ -759,6 +779,17 @@ static void test_scn_u32_dec(void) TEST_ASSERT_EQUAL_INT(val2, scn_u32_dec(string1, 5)); } +static void test_scn_u32_hex(void) +{ + const char *string1 = "aB12cE4F"; + uint32_t val1 = 0xab12ce4f; + uint32_t val2 = 0xab1; + + TEST_ASSERT_EQUAL_INT(val1, scn_u32_hex(string1, 8)); + TEST_ASSERT_EQUAL_INT(val2, scn_u32_hex(string1, 3)); + TEST_ASSERT_EQUAL_INT(val1, scn_u32_hex(string1, 9)); +} + static void test_fmt_lpad(void) { const char base[] = "abcd"; @@ -817,7 +848,10 @@ Test *tests_fmt_tests(void) new_TestFixture(test_fmt_strlen), new_TestFixture(test_fmt_strnlen), new_TestFixture(test_fmt_str), + new_TestFixture(test_fmt_char), + new_TestFixture(test_fmt_to_lower), new_TestFixture(test_scn_u32_dec), + new_TestFixture(test_scn_u32_hex), new_TestFixture(test_fmt_lpad), }; diff --git a/tests/unittests/tests-uuid/tests-uuid.c b/tests/unittests/tests-uuid/tests-uuid.c index 0552d24ae877ed175d31f26f21bcfedde12539da..f0b98bd8c9cc74740e7a350db167e9cb9826808b 100644 --- a/tests/unittests/tests-uuid/tests-uuid.c +++ b/tests/unittests/tests-uuid/tests-uuid.c @@ -95,12 +95,46 @@ void test_uuid_v5(void) TEST_ASSERT_EQUAL_INT(uuid_version(&uuid_next), UUID_V5); } +void test_uuid_str(void) +{ + char str[40]; + const char dns[] = "6ba7b810-9dad-11d1-80b4-00c04fd430c8"; + uuid_to_string(&uuid_namespace_dns, str); + TEST_ASSERT_EQUAL_INT(0, memcmp(dns, str, sizeof(dns))); + + const char url[] = "6ba7b811-9dad-11d1-80b4-00c04fd430c8"; + uuid_to_string(&uuid_namespace_url, str); + TEST_ASSERT_EQUAL_INT(0, memcmp(url, str, sizeof(dns))); + + const char iso[] = "6ba7b812-9dad-11d1-80b4-00c04fd430c8"; + uuid_to_string(&uuid_namespace_iso, str); + TEST_ASSERT_EQUAL_INT(0, memcmp(iso, str, sizeof(dns))); + + const char x500[] = "6ba7b814-9dad-11d1-80b4-00c04fd430c8"; + uuid_to_string(&uuid_namespace_x500, str); + TEST_ASSERT_EQUAL_INT(0, memcmp(x500, str, sizeof(dns))); + + uuid_t uuid; + TEST_ASSERT_EQUAL_INT(0, uuid_from_string(&uuid, dns)); + TEST_ASSERT_EQUAL_INT(true, uuid_equal(&uuid, &uuid_namespace_dns)); + + TEST_ASSERT_EQUAL_INT(0, uuid_from_string(&uuid, url)); + TEST_ASSERT_EQUAL_INT(true, uuid_equal(&uuid, &uuid_namespace_url)); + + TEST_ASSERT_EQUAL_INT(0, uuid_from_string(&uuid, iso)); + TEST_ASSERT_EQUAL_INT(true, uuid_equal(&uuid, &uuid_namespace_iso)); + + TEST_ASSERT_EQUAL_INT(0, uuid_from_string(&uuid, x500)); + TEST_ASSERT_EQUAL_INT(true, uuid_equal(&uuid, &uuid_namespace_x500)); +} + Test *tests_uuid_all(void) { EMB_UNIT_TESTFIXTURES(fixtures) { new_TestFixture(test_uuid_v3), new_TestFixture(test_uuid_v4), new_TestFixture(test_uuid_v5), + new_TestFixture(test_uuid_str), }; EMB_UNIT_TESTCALLER(uuid_tests, NULL, NULL, fixtures);