summaryrefslogtreecommitdiff
path: root/lib/utils/fdt/fdt_driver.c
blob: f6707c0eef99a0b5423b4e871821bdd7284869e1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
/*
 * SPDX-License-Identifier: BSD-2-Clause
 *
 * Copyright (c) 2024 SiFive
 */

#include <libfdt.h>
#include <sbi/sbi_console.h>
#include <sbi/sbi_error.h>
#include <sbi_utils/fdt/fdt_driver.h>
#include <sbi_utils/fdt/fdt_helper.h>

int fdt_driver_init_by_offset(const void *fdt, int nodeoff,
			      const struct fdt_driver *const *drivers)
{
	const struct fdt_driver *driver;
	const struct fdt_match *match;
	const void *prop;
	int len, rc;

	if (!fdt_node_is_enabled(fdt, nodeoff))
		return SBI_ENODEV;

	prop = fdt_getprop(fdt, nodeoff, "compatible", &len);
	if (!prop)
		return SBI_ENODEV;

	while ((driver = *drivers++)) {
		for (match = driver->match_table; match->compatible; match++) {
			if (!fdt_stringlist_contains(prop, len, match->compatible))
				continue;

			if (driver->experimental)
				sbi_printf("WARNING: %s driver is experimental and may change\n",
					   match->compatible);

			rc = driver->init(fdt, nodeoff, match);
			if (rc < 0) {
				const char *name;

				name = fdt_get_name(fdt, nodeoff, NULL);
				sbi_printf("%s: %s (%s) init failed: %d\n",
					   __func__, name, match->compatible, rc);
			}

			return rc;
		}
	}

	return SBI_ENODEV;
}

static int fdt_driver_init_scan(const void *fdt,
				const struct fdt_driver *const *drivers,
				bool one)
{
	int nodeoff, rc;

	for (nodeoff = fdt_next_node(fdt, -1, NULL);
	     nodeoff >= 0;
	     nodeoff = fdt_next_node(fdt, nodeoff, NULL)) {
		rc = fdt_driver_init_by_offset(fdt, nodeoff, drivers);
		if (rc == SBI_ENODEV)
			continue;
		if (rc < 0)
			return rc;
		if (one)
			return 0;
	}

	return one ? SBI_ENODEV : 0;
}

int fdt_driver_init_all(const void *fdt,
			const struct fdt_driver *const *drivers)
{
	return fdt_driver_init_scan(fdt, drivers, false);
}

int fdt_driver_init_one(const void *fdt,
			const struct fdt_driver *const *drivers)
{
	return fdt_driver_init_scan(fdt, drivers, true);
}