]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
memory: tegra20-emc: fix an OF node reference bug in tegra_emc_find_node_by_ram_code()
authorJoe Hattori <joe@pf.is.s.u-tokyo.ac.jp>
Tue, 17 Dec 2024 09:14:34 +0000 (18:14 +0900)
committerKrzysztof Kozlowski <krzysztof.kozlowski@linaro.org>
Sun, 22 Dec 2024 10:02:39 +0000 (11:02 +0100)
As of_find_node_by_name() release the reference of the argument device
node, tegra_emc_find_node_by_ram_code() releases some device nodes while
still in use, resulting in possible UAFs. According to the bindings and
the in-tree DTS files, the "emc-tables" node is always device's child
node with the property "nvidia,use-ram-code", and the "lpddr2" node is a
child of the "emc-tables" node. Thus utilize the
for_each_child_of_node() macro and of_get_child_by_name() instead of
of_find_node_by_name() to simplify the code.

This bug was found by an experimental verification tool that I am
developing.

Fixes: 96e5da7c8424 ("memory: tegra: Introduce Tegra20 EMC driver")
Signed-off-by: Joe Hattori <joe@pf.is.s.u-tokyo.ac.jp>
Link: https://lore.kernel.org/r/20241217091434.1993597-1-joe@pf.is.s.u-tokyo.ac.jp
Link: https://lore.kernel.org/r/20241218024415.2494267-3-joe@pf.is.s.u-tokyo.ac.jp
[krzysztof: applied v1, adjust the commit msg to incorporate v2 parts]
Signed-off-by: Krzysztof Kozlowski <krzysztof.kozlowski@linaro.org>
drivers/memory/tegra/tegra20-emc.c

index 7193f848d17e66b9227d0f4e40bb1cf3e035a37e..9b7d30a21a5bd016cfa9539100d334227b2ad982 100644 (file)
@@ -474,14 +474,15 @@ tegra_emc_find_node_by_ram_code(struct tegra_emc *emc)
 
        ram_code = tegra_read_ram_code();
 
-       for (np = of_find_node_by_name(dev->of_node, "emc-tables"); np;
-            np = of_find_node_by_name(np, "emc-tables")) {
+       for_each_child_of_node(dev->of_node, np) {
+               if (!of_node_name_eq(np, "emc-tables"))
+                       continue;
                err = of_property_read_u32(np, "nvidia,ram-code", &value);
                if (err || value != ram_code) {
                        struct device_node *lpddr2_np;
                        bool cfg_mismatches = false;
 
-                       lpddr2_np = of_find_node_by_name(np, "lpddr2");
+                       lpddr2_np = of_get_child_by_name(np, "lpddr2");
                        if (lpddr2_np) {
                                const struct lpddr2_info *info;
 
@@ -518,7 +519,6 @@ tegra_emc_find_node_by_ram_code(struct tegra_emc *emc)
                        }
 
                        if (cfg_mismatches) {
-                               of_node_put(np);
                                continue;
                        }
                }