spin_table.c 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263
  1. /*
  2. * Copyright (C) 2016 Socionext Inc.
  3. * Author: Masahiro Yamada <yamada.masahiro@socionext.com>
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <libfdt.h>
  9. #include <asm/spin_table.h>
  10. int spin_table_update_dt(void *fdt)
  11. {
  12. int cpus_offset, offset;
  13. const char *prop;
  14. int ret;
  15. unsigned long rsv_addr = (unsigned long)&spin_table_reserve_begin;
  16. unsigned long rsv_size = &spin_table_reserve_end -
  17. &spin_table_reserve_begin;
  18. cpus_offset = fdt_path_offset(fdt, "/cpus");
  19. if (cpus_offset < 0)
  20. return -ENODEV;
  21. for (offset = fdt_first_subnode(fdt, cpus_offset);
  22. offset >= 0;
  23. offset = fdt_next_subnode(fdt, offset)) {
  24. prop = fdt_getprop(fdt, offset, "device_type", NULL);
  25. if (!prop || strcmp(prop, "cpu"))
  26. continue;
  27. /*
  28. * In the first loop, we check if every CPU node specifies
  29. * spin-table. Otherwise, just return successfully to not
  30. * disturb other methods, like psci.
  31. */
  32. prop = fdt_getprop(fdt, offset, "enable-method", NULL);
  33. if (!prop || strcmp(prop, "spin-table"))
  34. return 0;
  35. }
  36. for (offset = fdt_first_subnode(fdt, cpus_offset);
  37. offset >= 0;
  38. offset = fdt_next_subnode(fdt, offset)) {
  39. prop = fdt_getprop(fdt, offset, "device_type", NULL);
  40. if (!prop || strcmp(prop, "cpu"))
  41. continue;
  42. ret = fdt_setprop_u64(fdt, offset, "cpu-release-addr",
  43. (unsigned long)&spin_table_cpu_release_addr);
  44. if (ret)
  45. return -ENOSPC;
  46. }
  47. ret = fdt_add_mem_rsv(fdt, rsv_addr, rsv_size);
  48. if (ret)
  49. return -ENOSPC;
  50. printf(" Reserved memory region for spin-table: addr=%lx size=%lx\n",
  51. rsv_addr, rsv_size);
  52. return 0;
  53. }