Commit 028e6e20 authored by Andy Shevchenko's avatar Andy Shevchenko Committed by Hans de Goede

platform/x86: wmi: Break possible infinite loop when parsing GUID

The while-loop may break on one of the two conditions, either ID string
is empty or GUID matches. The second one, may never be reached if the
parsed string is not correct GUID. In such a case the loop will never
advance to check the next ID.

Break possible infinite loop by factoring out guid_parse_and_compare()
helper which may be moved to the generic header for everyone later on
and preventing from similar mistake in the future.

Interestingly that firstly it appeared when WMI was turned into a bus
driver, but later when duplicated GUIDs were checked, the while-loop
has been replaced by for-loop and hence no mistake made again.

Fixes: a48e2338 ("platform/x86: wmi: add context pointer field to struct wmi_device_id")
Fixes: 844af950 ("platform/x86: wmi: Turn WMI into a bus driver")
Signed-off-by: default avatarAndy Shevchenko <andriy.shevchenko@linux.intel.com>
Link: https://lore.kernel.org/r/20230621151155.78279-1-andriy.shevchenko@linux.intel.comTested-by: default avatarArmin Wolf <W_Armin@gmx.de>
Reviewed-by: default avatarHans de Goede <hdegoede@redhat.com>
Signed-off-by: default avatarHans de Goede <hdegoede@redhat.com>
parent 06c2afb8
...@@ -136,6 +136,16 @@ static acpi_status find_guid(const char *guid_string, struct wmi_block **out) ...@@ -136,6 +136,16 @@ static acpi_status find_guid(const char *guid_string, struct wmi_block **out)
return AE_NOT_FOUND; return AE_NOT_FOUND;
} }
static bool guid_parse_and_compare(const char *string, const guid_t *guid)
{
guid_t guid_input;
if (guid_parse(string, &guid_input))
return false;
return guid_equal(&guid_input, guid);
}
static const void *find_guid_context(struct wmi_block *wblock, static const void *find_guid_context(struct wmi_block *wblock,
struct wmi_driver *wdriver) struct wmi_driver *wdriver)
{ {
...@@ -146,11 +156,7 @@ static const void *find_guid_context(struct wmi_block *wblock, ...@@ -146,11 +156,7 @@ static const void *find_guid_context(struct wmi_block *wblock,
return NULL; return NULL;
while (*id->guid_string) { while (*id->guid_string) {
guid_t guid_input; if (guid_parse_and_compare(id->guid_string, &wblock->gblock.guid))
if (guid_parse(id->guid_string, &guid_input))
continue;
if (guid_equal(&wblock->gblock.guid, &guid_input))
return id->context; return id->context;
id++; id++;
} }
...@@ -895,11 +901,7 @@ static int wmi_dev_match(struct device *dev, struct device_driver *driver) ...@@ -895,11 +901,7 @@ static int wmi_dev_match(struct device *dev, struct device_driver *driver)
return 0; return 0;
while (*id->guid_string) { while (*id->guid_string) {
guid_t driver_guid; if (guid_parse_and_compare(id->guid_string, &wblock->gblock.guid))
if (WARN_ON(guid_parse(id->guid_string, &driver_guid)))
continue;
if (guid_equal(&driver_guid, &wblock->gblock.guid))
return 1; return 1;
id++; id++;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment