libnm: merge hwaddr_aton() and nm_utils_hexstr2bin()
Have nm_utils_hexstr2bin() take over the allocated buffer via g_bytes_new_take().
This commit is contained in:
@@ -3014,10 +3014,16 @@ nm_utils_hwaddr_len (int type)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static guint8 *
|
static guint8 *
|
||||||
hwaddr_aton (const char *asc, guint8 *buffer, gsize buffer_length, gsize *out_len)
|
_str2bin (const char *asc,
|
||||||
|
gboolean delimiter_required,
|
||||||
|
const char *delimiter_candidates,
|
||||||
|
guint8 *buffer,
|
||||||
|
gsize buffer_length,
|
||||||
|
gsize *out_len)
|
||||||
{
|
{
|
||||||
const char *in = asc;
|
const char *in = asc;
|
||||||
guint8 *out = buffer;
|
guint8 *out = buffer;
|
||||||
|
gboolean delimiter_has = TRUE;
|
||||||
guint8 delimiter = '\0';
|
guint8 delimiter = '\0';
|
||||||
|
|
||||||
nm_assert (asc);
|
nm_assert (asc);
|
||||||
@@ -3039,24 +3045,43 @@ hwaddr_aton (const char *asc, guint8 *buffer, gsize buffer_length, gsize *out_le
|
|||||||
if (d2 && g_ascii_isxdigit (d2)) {
|
if (d2 && g_ascii_isxdigit (d2)) {
|
||||||
*out++ = (HEXVAL (d1) << 4) + HEXVAL (d2);
|
*out++ = (HEXVAL (d1) << 4) + HEXVAL (d2);
|
||||||
d2 = in[2];
|
d2 = in[2];
|
||||||
in += 3;
|
if (!d2)
|
||||||
|
break;
|
||||||
|
in += 2;
|
||||||
} else {
|
} else {
|
||||||
/* Fake leading zero */
|
/* Fake leading zero */
|
||||||
*out++ = HEXVAL (d1);
|
*out++ = HEXVAL (d1);
|
||||||
in += 2;
|
if (!d2) {
|
||||||
|
if (!delimiter_has) {
|
||||||
|
/* when using no delimiter, there must be pairs of hex chars */
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
in += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!d2)
|
|
||||||
break;
|
|
||||||
if (--buffer_length == 0)
|
if (--buffer_length == 0)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
if (delimiter_has) {
|
||||||
if (d2 != delimiter) {
|
if (d2 != delimiter) {
|
||||||
if ( delimiter == '\0'
|
if (delimiter)
|
||||||
&& (d2 == ':' || d2 == '-'))
|
|
||||||
delimiter = d2;
|
|
||||||
else
|
|
||||||
return NULL;
|
return NULL;
|
||||||
|
if (delimiter_candidates) {
|
||||||
|
while (delimiter_candidates[0]) {
|
||||||
|
if (delimiter_candidates++[0] == d2)
|
||||||
|
delimiter = d2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!delimiter) {
|
||||||
|
if (delimiter_required)
|
||||||
|
return NULL;
|
||||||
|
delimiter_has = FALSE;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
in++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -3064,6 +3089,8 @@ hwaddr_aton (const char *asc, guint8 *buffer, gsize buffer_length, gsize *out_le
|
|||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define hwaddr_aton(asc, buffer, buffer_length, out_len) _str2bin ((asc), TRUE, ":-", (buffer), (buffer_length), (out_len))
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* nm_utils_hexstr2bin:
|
* nm_utils_hexstr2bin:
|
||||||
* @hex: a string of hexadecimal characters with optional ':' separators
|
* @hex: a string of hexadecimal characters with optional ':' separators
|
||||||
@@ -3078,46 +3105,22 @@ hwaddr_aton (const char *asc, guint8 *buffer, gsize buffer_length, gsize *out_le
|
|||||||
GBytes *
|
GBytes *
|
||||||
nm_utils_hexstr2bin (const char *hex)
|
nm_utils_hexstr2bin (const char *hex)
|
||||||
{
|
{
|
||||||
guint i = 0, x = 0;
|
guint8 *buffer;
|
||||||
gs_free guint8 *c = NULL;
|
gsize buffer_length, len;
|
||||||
int a, b;
|
|
||||||
gboolean found_colon = FALSE;
|
|
||||||
|
|
||||||
g_return_val_if_fail (hex != NULL, NULL);
|
g_return_val_if_fail (hex != NULL, NULL);
|
||||||
|
|
||||||
if (strncasecmp (hex, "0x", 2) == 0)
|
if (hex[0] == '0' && hex[1] == 'x')
|
||||||
hex += 2;
|
hex += 2;
|
||||||
found_colon = !!strchr (hex, ':');
|
|
||||||
|
|
||||||
c = g_malloc (strlen (hex) / 2 + 1);
|
buffer_length = strlen (hex) / 2 + 3;
|
||||||
for (;;) {
|
buffer = g_malloc (buffer_length);
|
||||||
a = g_ascii_xdigit_value (hex[i++]);
|
if (!_str2bin (hex, FALSE, ":", buffer, buffer_length, &len)) {
|
||||||
if (a < 0)
|
g_free (buffer);
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if (hex[i] && hex[i] != ':') {
|
|
||||||
b = g_ascii_xdigit_value (hex[i++]);
|
|
||||||
if (b < 0)
|
|
||||||
return NULL;
|
|
||||||
c[x++] = ((guint) a << 4) | ((guint) b);
|
|
||||||
} else
|
|
||||||
c[x++] = (guint) a;
|
|
||||||
|
|
||||||
if (!hex[i])
|
|
||||||
break;
|
|
||||||
if (hex[i] == ':') {
|
|
||||||
if (!hex[i + 1]) {
|
|
||||||
/* trailing ':' is invalid */
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
i++;
|
buffer = g_realloc (buffer, len);
|
||||||
} else if (found_colon) {
|
return g_bytes_new_take (buffer, len);
|
||||||
/* If colons exist, they must delimit 1 or 2 hex chars */
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return g_bytes_new (c, x);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -4294,6 +4294,10 @@ test_hexstr2bin (void)
|
|||||||
{ "0xccddeeff", { 0xcc, 0xdd, 0xee, 0xff }, 4 },
|
{ "0xccddeeff", { 0xcc, 0xdd, 0xee, 0xff }, 4 },
|
||||||
{ "1:2:66:77:80", { 0x01, 0x02, 0x66, 0x77, 0x80 }, 5 },
|
{ "1:2:66:77:80", { 0x01, 0x02, 0x66, 0x77, 0x80 }, 5 },
|
||||||
{ "e", { 0x0e }, 1 },
|
{ "e", { 0x0e }, 1 },
|
||||||
|
{ "ef", { 0xef }, 1 },
|
||||||
|
{ "efa" },
|
||||||
|
{ "efad", { 0xef, 0xad }, 2 },
|
||||||
|
{ "ef:a", { 0xef, 0x0a }, 2 },
|
||||||
{ "aabb1199:" },
|
{ "aabb1199:" },
|
||||||
{ ":aabb1199" },
|
{ ":aabb1199" },
|
||||||
{ "aabb$$dd" },
|
{ "aabb$$dd" },
|
||||||
|
Reference in New Issue
Block a user