/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details: * * Copyright (C) 2021 Aleksander Morgado */ #include #include #include #include #include "mm-log-object.h" #include "mm-kernel-device-helpers.h" /******************************************************************************/ gchar * mm_kernel_device_get_lower_device_name (const gchar *sysfs_path) { g_autoptr(GFile) dirfile = NULL; g_autoptr(GFileEnumerator) direnum = NULL; dirfile = g_file_new_for_path (sysfs_path); direnum = g_file_enumerate_children (dirfile, G_FILE_ATTRIBUTE_STANDARD_NAME, G_FILE_QUERY_INFO_NONE, NULL, NULL); if (!direnum) return NULL; while (TRUE) { GFileInfo *info; g_autofree gchar *filename = NULL; g_autofree gchar *link_path = NULL; g_autofree gchar *real_path = NULL; if (!g_file_enumerator_iterate (direnum, &info, NULL, NULL, NULL) || !info) break; filename = g_file_info_get_attribute_as_string (info, G_FILE_ATTRIBUTE_STANDARD_NAME); if (!filename || !g_str_has_prefix (filename, "lower_")) continue; link_path = g_strdup_printf ("%s/%s", sysfs_path, filename); real_path = realpath (link_path, NULL); if (!real_path) continue; return g_path_get_basename (real_path); } return NULL; } /******************************************************************************/ static gchar * build_string_match_pattern (const gchar *str) { GString *regex_pattern; const gchar *str_start; gsize len; g_autofree gchar *aux = NULL; gboolean prefix_match = FALSE; gboolean suffix_match = FALSE; /* We allow prefix and suffix matches given as input, by means of the * single '*' character given either at the beginning or the end of the * string. If given in another place, it will assumed to be explicitly * the '*' character, not a catch-all indication. */ regex_pattern = g_string_new (NULL); /* suffix match? */ if (str[0] == '*') { str_start = &str[1]; suffix_match = TRUE; } else str_start = str; /* prefix match? */ len = strlen (str_start); if (len > 0 && str_start[len - 1] == '*') { len--; prefix_match = TRUE; } /* match start of string */ g_string_append (regex_pattern, "^"); if (suffix_match) g_string_append (regex_pattern, ".*"); aux = g_regex_escape_string (str_start, len); g_string_append (regex_pattern, aux); if (prefix_match) g_string_append (regex_pattern, ".*"); /* match end of string */ g_string_append (regex_pattern, "$"); return g_string_free (regex_pattern, FALSE); } gboolean mm_kernel_device_generic_string_match (const gchar *str, const gchar *pattern, gpointer log_object) { g_autoptr(GError) inner_error = NULL; g_autoptr(GRegex) regex = NULL; g_autoptr(GMatchInfo) match_info = NULL; g_autofree gchar *regex_pattern = NULL; regex_pattern = build_string_match_pattern (pattern); regex = g_regex_new (regex_pattern, G_REGEX_UNGREEDY, 0, &inner_error); if (!regex) { mm_obj_warn (log_object, "invalid pattern in rule '%s': %s", regex_pattern, inner_error->message); return FALSE; } g_regex_match_full (regex, str, -1, 0, 0, &match_info, &inner_error); if (inner_error) { mm_obj_warn (log_object, "couldn't apply pattern match in rule '%s': %s", regex_pattern, inner_error->message); return FALSE; } if (!g_match_info_matches (match_info)) return FALSE; mm_obj_dbg (log_object, "pattern '%s' matched: '%s'", regex_pattern, str); return TRUE; }