/* * Copyright (C) 2013 - ARM Ltd * Author: Marc Zyngier * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * 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. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include #include #include #include #include #include #include static int fdt_psci(void *fdt) { #ifdef CONFIG_ARMV7_PSCI int nodeoff; int tmp; nodeoff = fdt_path_offset(fdt, "/cpus"); if (nodeoff < 0) { printf("couldn't find /cpus\n"); return nodeoff; } /* add 'enable-method = "psci"' to each cpu node */ for (tmp = fdt_first_subnode(fdt, nodeoff); tmp >= 0; tmp = fdt_next_subnode(fdt, tmp)) { const struct fdt_property *prop; int len; prop = fdt_get_property(fdt, tmp, "device_type", &len); if (!prop) continue; if (len < 4) continue; if (strcmp(prop->data, "cpu")) continue; fdt_setprop_string(fdt, tmp, "enable-method", "psci"); } nodeoff = fdt_path_offset(fdt, "/psci"); if (nodeoff < 0) { nodeoff = fdt_path_offset(fdt, "/"); if (nodeoff < 0) return nodeoff; nodeoff = fdt_add_subnode(fdt, nodeoff, "psci"); if (nodeoff < 0) return nodeoff; } tmp = fdt_setprop_string(fdt, nodeoff, "compatible", "arm,psci"); if (tmp) return tmp; tmp = fdt_setprop_string(fdt, nodeoff, "method", "smc"); if (tmp) return tmp; tmp = fdt_setprop_u32(fdt, nodeoff, "cpu_suspend", ARM_PSCI_FN_CPU_SUSPEND); if (tmp) return tmp; tmp = fdt_setprop_u32(fdt, nodeoff, "cpu_off", ARM_PSCI_FN_CPU_OFF); if (tmp) return tmp; tmp = fdt_setprop_u32(fdt, nodeoff, "cpu_on", ARM_PSCI_FN_CPU_ON); if (tmp) return tmp; tmp = fdt_setprop_u32(fdt, nodeoff, "migrate", ARM_PSCI_FN_MIGRATE); if (tmp) return tmp; #endif return 0; } int armv7_update_dt(void *fdt) { if (!armv7_boot_nonsec()) return 0; #ifndef CONFIG_ARMV7_SECURE_BASE /* secure code lives in RAM, keep it alive */ fdt_add_mem_rsv(fdt, (unsigned long)__secure_start, __secure_end - __secure_start); #endif return fdt_psci(fdt); }