modem-helpers: new uint list parser, including interval support

This commit is contained in:
Aleksander Morgado
2016-08-08 09:40:17 +02:00
parent 37bd614212
commit 3a886d848d
3 changed files with 147 additions and 0 deletions

View File

@@ -131,6 +131,105 @@ mm_split_string_groups (const gchar *str)
/*****************************************************************************/
static int uint_compare_func (gconstpointer a, gconstpointer b)
{
return (*(guint *)a - *(guint *)b);
}
GArray *
mm_parse_uint_list (const gchar *str,
GError **error)
{
GArray *array;
gchar *dupstr;
gchar *aux;
GError *inner_error = NULL;
if (!str || !str[0])
return NULL;
/* Parses into a GArray of guints, the list of numbers given in the string,
* also supporting number intervals.
* E.g.:
* 1-6 --> 1,2,3,4,5,6
* 1,2,4-6 --> 1,2,4,5,6
*/
array = g_array_new (FALSE, FALSE, sizeof (guint));
aux = dupstr = g_strdup (str);
while (aux) {
gchar *next;
gchar *interval;
next = strchr (aux, ',');
if (next) {
*next = '\0';
next++;
}
interval = strchr (aux, '-');
if (interval) {
guint start = 0;
guint stop = 0;
*interval = '\0';
interval++;
if (!mm_get_uint_from_str (aux, &start)) {
inner_error = g_error_new (MM_CORE_ERROR, MM_CORE_ERROR_FAILED,
"couldn't parse interval start integer: '%s'", aux);
goto out;
}
if (!mm_get_uint_from_str (interval, &stop)) {
inner_error = g_error_new (MM_CORE_ERROR, MM_CORE_ERROR_FAILED,
"couldn't parse interval stop integer: '%s'", interval);
goto out;
}
if (start > stop) {
inner_error = g_error_new (MM_CORE_ERROR, MM_CORE_ERROR_FAILED,
"interval start (%u) cannot be bigger than interval stop (%u)", start, stop);
goto out;
}
for (; start <= stop; start++)
g_array_append_val (array, start);
} else {
guint num;
if (!mm_get_uint_from_str (aux, &num)) {
inner_error = g_error_new (MM_CORE_ERROR, MM_CORE_ERROR_FAILED,
"couldn't parse integer: '%s'", aux);
goto out;
}
g_array_append_val (array, num);
}
aux = next;
}
if (!array->len)
inner_error = g_error_new (MM_CORE_ERROR,
MM_CORE_ERROR_FAILED,
"couldn't parse list of integers: '%s'", str);
else
g_array_sort (array, uint_compare_func);
out:
g_free (dupstr);
if (inner_error) {
g_propagate_error (error, inner_error);
g_array_unref (array);
return NULL;
}
return array;
}
/*****************************************************************************/
guint
mm_count_bits_set (gulong number)
{

View File

@@ -53,6 +53,9 @@ const gchar *mm_strip_tag (const gchar *str,
gchar **mm_split_string_groups (const gchar *str);
GArray *mm_parse_uint_list (const gchar *str,
GError **error);
guint mm_count_bits_set (gulong number);
gchar *mm_create_device_identifier (guint vid,

View File

@@ -3060,6 +3060,49 @@ test_cgcontrdp_response (void)
/*****************************************************************************/
typedef struct {
gchar *str;
gint expected_number_list[9];
} TestParseNumberList;
static const TestParseNumberList test_parse_number_list_item [] = {
{ "1-6", { 1, 2, 3, 4, 5, 6, -1 } },
{ "0,1,2,4,6", { 0, 1, 2, 4, 6, -1 } },
{ "1,3-5,7,9-11", { 1, 3, 4, 5, 7, 9, 10, 11, -1 } },
{ "9-11,7,3-5", { 3, 4, 5, 7, 9, 10, 11, -1 } },
{ "", { -1 } },
{ NULL, { -1 } },
};
static void
test_parse_uint_list (void)
{
guint i;
for (i = 0; i < G_N_ELEMENTS (test_parse_number_list_item); i++) {
GArray *array;
GError *error = NULL;
guint j;
array = mm_parse_uint_list (test_parse_number_list_item[i].str, &error);
g_assert_no_error (error);
if (test_parse_number_list_item[i].expected_number_list[0] == -1) {
g_assert (!array);
continue;
}
g_assert (array);
for (j = 0; j < array->len; j++) {
g_assert_cmpint (test_parse_number_list_item[i].expected_number_list[j], !=, -1);
g_assert_cmpuint (test_parse_number_list_item[i].expected_number_list[j], ==, g_array_index (array, guint, j));
}
g_assert_cmpint (test_parse_number_list_item[i].expected_number_list[array->len], ==, -1);
g_array_unref (array);
}
}
/*****************************************************************************/
void
_mm_log (const char *loc,
const char *func,
@@ -3245,6 +3288,8 @@ int main (int argc, char **argv)
g_test_suite_add (suite, TESTCASE (test_cgcontrdp_response, NULL));
g_test_suite_add (suite, TESTCASE (test_parse_uint_list, NULL));
result = g_test_run ();
reg_test_data_free (reg_data);