Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys Refactor and simplify device_compatible_match(), and als...
details: https://anonhg.NetBSD.org/src/rev/c1134e1f288a
branches: trunk
changeset: 980136:c1134e1f288a
user: thorpej <thorpej%NetBSD.org@localhost>
date: Sun Jan 24 17:42:36 2021 +0000
description:
Refactor and simplify device_compatible_match(), and also provide
device_compatible_p{match,lookup}() which treats the strings in the
driver compatible data as pmatch(9) patterns.
Add device_compatible_{,p}{match,lookup}_strlist(), which are the same, but
take an OpenFirmware-style string list rather than an array of strings
for the device data.
diffstat:
sys/kern/subr_autoconf.c | 254 ++++++++++++++++++++++++++++++++++++++++------
sys/sys/device.h | 19 +++-
2 files changed, 239 insertions(+), 34 deletions(-)
diffs (truncated from 336 to 300 lines):
diff -r 6476fda113aa -r c1134e1f288a sys/kern/subr_autoconf.c
--- a/sys/kern/subr_autoconf.c Sun Jan 24 17:29:11 2021 +0000
+++ b/sys/kern/subr_autoconf.c Sun Jan 24 17:42:36 2021 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: subr_autoconf.c,v 1.275 2021/01/18 15:28:21 thorpej Exp $ */
+/* $NetBSD: subr_autoconf.c,v 1.276 2021/01/24 17:42:36 thorpej Exp $ */
/*
* Copyright (c) 1996, 2000 Christopher G. Demetriou
@@ -77,7 +77,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: subr_autoconf.c,v 1.275 2021/01/18 15:28:21 thorpej Exp $");
+__KERNEL_RCSID(0, "$NetBSD: subr_autoconf.c,v 1.276 2021/01/24 17:42:36 thorpej Exp $");
#ifdef _KERNEL_OPT
#include "opt_ddb.h"
@@ -2324,52 +2324,179 @@
return device_lookup(cd, unit);
}
-/*
- * device_compatible_match:
- *
- * Match a driver's "compatible" data against a device's
- * "compatible" strings. If a match is found, we return
- * a weighted match result, and optionally the matching
- * entry.
- */
+static bool
+match_strcmp(const char * const s1, const char * const s2)
+{
+ return strcmp(s1, s2) == 0;
+}
+
+static bool
+match_pmatch(const char * const s1, const char * const s2)
+{
+ return pmatch(s1, s2, NULL) == 2;
+}
+
+static bool
+strarray_match_internal(const char ** const strings,
+ unsigned int const nstrings, const char * const str,
+ unsigned int * const indexp,
+ bool (*match_fn)(const char *, const char *))
+{
+ unsigned int i;
+
+ if (strings == NULL || nstrings == 0) {
+ return 0;
+ }
+
+ for (i = 0; i < nstrings; i++) {
+ if ((*match_fn)(strings[i], str)) {
+ *indexp = i;
+ return true;
+ }
+ }
+
+ return false;
+}
+
static int
-device_compatible_match_internal(const char **device_compats,
- int ndevice_compats,
+strarray_match(const char ** const strings, unsigned int const nstrings,
+ const char * const str)
+{
+ unsigned int idx;
+
+ if (strarray_match_internal(strings, nstrings, str, &idx,
+ match_strcmp)) {
+ return (int)(nstrings - idx);
+ }
+ return 0;
+}
+
+static int
+strarray_pmatch(const char ** const strings, unsigned int const nstrings,
+ const char * const pattern)
+{
+ unsigned int idx;
+
+ if (strarray_match_internal(strings, nstrings, pattern, &idx,
+ match_pmatch)) {
+ return (int)(nstrings - idx);
+ }
+ return 0;
+}
+
+static int
+device_compatible_match_strarray_internal(
+ const char **device_compats, int ndevice_compats,
const struct device_compatible_entry *driver_compats,
- const struct device_compatible_entry **matching_entryp)
+ const struct device_compatible_entry **matching_entryp,
+ int (*match_fn)(const char **, unsigned int, const char *))
{
const struct device_compatible_entry *dce = NULL;
- int i, match_weight;
+ int rv;
if (ndevice_compats == 0 || device_compats == NULL ||
driver_compats == NULL)
return 0;
-
- /*
- * We take the first match because we start with the most-specific
- * device compatible string.
- */
- for (i = 0, match_weight = ndevice_compats - 1;
- i < ndevice_compats;
- i++, match_weight--) {
- for (dce = driver_compats; dce->compat != NULL; dce++) {
- if (strcmp(dce->compat, device_compats[i]) == 0) {
- KASSERT(match_weight >= 0);
- if (matching_entryp)
- *matching_entryp = dce;
- return 1 + match_weight;
+
+ for (dce = driver_compats; dce->compat != NULL; dce++) {
+ rv = (*match_fn)(device_compats, ndevice_compats, dce->compat);
+ if (rv != 0) {
+ if (matching_entryp != NULL) {
+ *matching_entryp = dce;
}
+ return rv;
}
}
return 0;
}
+/*
+ * device_compatible_match:
+ *
+ * Match a driver's "compatible" data against a device's
+ * "compatible" strings. Returns resulted weighted by
+ * which device "compatible" string was matched.
+ */
int
device_compatible_match(const char **device_compats, int ndevice_compats,
- const struct device_compatible_entry *driver_compats)
+ const struct device_compatible_entry *driver_compats)
+{
+ return device_compatible_match_strarray_internal(device_compats,
+ ndevice_compats, driver_compats, NULL, strarray_match);
+}
+
+/*
+ * device_compatible_pmatch:
+ *
+ * Like device_compatible_match(), but uses pmatch(9) to compare
+ * the device "compatible" strings against patterns in the
+ * driver's "compatible" data.
+ */
+int
+device_compatible_pmatch(const char **device_compats, int ndevice_compats,
+ const struct device_compatible_entry *driver_compats)
+{
+ return device_compatible_match_strarray_internal(device_compats,
+ ndevice_compats, driver_compats, NULL, strarray_pmatch);
+}
+
+static int
+device_compatible_match_strlist_internal(
+ const char * const device_compats, size_t const device_compatsize,
+ const struct device_compatible_entry *driver_compats,
+ const struct device_compatible_entry **matching_entryp,
+ int (*match_fn)(const char *, size_t, const char *))
{
- return device_compatible_match_internal(device_compats, ndevice_compats,
- driver_compats, NULL);
+ const struct device_compatible_entry *dce = NULL;
+ int rv;
+
+ if (device_compats == NULL || device_compatsize == 0 ||
+ driver_compats == NULL)
+ return 0;
+
+ for (dce = driver_compats; dce->compat != NULL; dce++) {
+ rv = (*match_fn)(device_compats, device_compatsize,
+ dce->compat);
+ if (rv != 0) {
+ if (matching_entryp != NULL) {
+ *matching_entryp = dce;
+ }
+ return rv;
+ }
+ }
+ return 0;
+}
+
+/*
+ * device_compatible_match_strlist:
+ *
+ * Like device_compatible_match(), but take the device
+ * "compatible" strings as an OpenFirmware-style string
+ * list.
+ */
+int
+device_compatible_match_strlist(
+ const char * const device_compats, size_t const device_compatsize,
+ const struct device_compatible_entry *driver_compats)
+{
+ return device_compatible_match_strlist_internal(device_compats,
+ device_compatsize, driver_compats, NULL, strlist_match);
+}
+
+/*
+ * device_compatible_pmatch_strlist:
+ *
+ * Like device_compatible_pmatch(), but take the device
+ * "compatible" strings as an OpenFirmware-style string
+ * list.
+ */
+int
+device_compatible_pmatch_strlist(
+ const char * const device_compats, size_t const device_compatsize,
+ const struct device_compatible_entry *driver_compats)
+{
+ return device_compatible_match_strlist_internal(device_compats,
+ device_compatsize, driver_compats, NULL, strlist_pmatch);
}
/*
@@ -2384,8 +2511,69 @@
{
const struct device_compatible_entry *dce;
- if (device_compatible_match_internal(device_compats, ndevice_compats,
- driver_compats, &dce)) {
+ if (device_compatible_match_strarray_internal(device_compats,
+ ndevice_compats, driver_compats, &dce, strarray_match)) {
+ return dce;
+ }
+ return NULL;
+}
+
+/*
+ * device_compatible_plookup:
+ *
+ * Look up and return the device_compatible_entry, using the
+ * same matching criteria used by device_compatible_pmatch().
+ */
+const struct device_compatible_entry *
+device_compatible_plookup(const char **device_compats, int ndevice_compats,
+ const struct device_compatible_entry *driver_compats)
+{
+ const struct device_compatible_entry *dce;
+
+ if (device_compatible_match_strarray_internal(device_compats,
+ ndevice_compats, driver_compats, &dce, strarray_pmatch)) {
+ return dce;
+ }
+ return NULL;
+}
+
+/*
+ * device_compatible_lookup_strlist:
+ *
+ * Like device_compatible_lookup(), but take the device
+ * "compatible" strings as an OpenFirmware-style string
+ * list.
+ */
+const struct device_compatible_entry *
+device_compatible_lookup_strlist(
+ const char * const device_compats, size_t const device_compatsize,
+ const struct device_compatible_entry *driver_compats)
+{
+ const struct device_compatible_entry *dce;
+
+ if (device_compatible_match_strlist_internal(device_compats,
+ device_compatsize, driver_compats, &dce, strlist_match)) {
+ return dce;
+ }
+ return NULL;
+}
+
+/*
+ * device_compatible_plookup_strlist:
+ *
+ * Like device_compatible_plookup(), but take the device
+ * "compatible" strings as an OpenFirmware-style string
+ * list.
+ */
+const struct device_compatible_entry *
+device_compatible_plookup_strlist(
+ const char * const device_compats, size_t const device_compatsize,
+ const struct device_compatible_entry *driver_compats)
+{
+ const struct device_compatible_entry *dce;
+
+ if (device_compatible_match_strlist_internal(device_compats,
+ device_compatsize, driver_compats, &dce, strlist_pmatch)) {
return dce;
}
return NULL;
diff -r 6476fda113aa -r c1134e1f288a sys/sys/device.h
Home |
Main Index |
Thread Index |
Old Index