spin_table.c 1.6 KB

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