of_extra.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (c) 2017 Google, Inc
  4. * Written by Simon Glass <sjg@chromium.org>
  5. */
  6. #include <common.h>
  7. #include <linux/libfdt.h>
  8. #include <dm/of_access.h>
  9. #include <dm/of_extra.h>
  10. #include <dm/ofnode.h>
  11. int ofnode_read_fmap_entry(ofnode node, struct fmap_entry *entry)
  12. {
  13. const char *prop;
  14. u32 reg[2];
  15. if (ofnode_read_u32_array(node, "reg", reg, 2)) {
  16. debug("Node '%s' has bad/missing 'reg' property\n",
  17. ofnode_get_name(node));
  18. return -log_ret(ENOENT);
  19. }
  20. entry->offset = reg[0];
  21. entry->length = reg[1];
  22. entry->used = ofnode_read_s32_default(node, "used", entry->length);
  23. prop = ofnode_read_string(node, "compress");
  24. entry->compress_algo = prop && !strcmp(prop, "lzo") ?
  25. FMAP_COMPRESS_LZO : FMAP_COMPRESS_NONE;
  26. prop = ofnode_read_string(node, "hash");
  27. if (prop)
  28. entry->hash_size = strlen(prop);
  29. entry->hash_algo = prop ? FMAP_HASH_SHA256 : FMAP_HASH_NONE;
  30. entry->hash = (uint8_t *)prop;
  31. return 0;
  32. }
  33. int ofnode_decode_region(ofnode node, const char *prop_name, fdt_addr_t *basep,
  34. fdt_size_t *sizep)
  35. {
  36. const fdt_addr_t *cell;
  37. int len;
  38. debug("%s: %s: %s\n", __func__, ofnode_get_name(node), prop_name);
  39. cell = ofnode_get_property(node, prop_name, &len);
  40. if (!cell || (len < sizeof(fdt_addr_t) * 2)) {
  41. debug("cell=%p, len=%d\n", cell, len);
  42. return -1;
  43. }
  44. *basep = fdt_addr_to_cpu(*cell);
  45. *sizep = fdt_size_to_cpu(cell[1]);
  46. debug("%s: base=%08lx, size=%lx\n", __func__, (ulong)*basep,
  47. (ulong)*sizep);
  48. return 0;
  49. }
  50. int ofnode_decode_memory_region(ofnode config_node, const char *mem_type,
  51. const char *suffix, fdt_addr_t *basep,
  52. fdt_size_t *sizep)
  53. {
  54. char prop_name[50];
  55. const char *mem;
  56. fdt_size_t size, offset_size;
  57. fdt_addr_t base, offset;
  58. ofnode node;
  59. if (!ofnode_valid(config_node)) {
  60. config_node = ofnode_path("/config");
  61. if (!ofnode_valid(config_node)) {
  62. debug("%s: Cannot find /config node\n", __func__);
  63. return -ENOENT;
  64. }
  65. }
  66. if (!suffix)
  67. suffix = "";
  68. snprintf(prop_name, sizeof(prop_name), "%s-memory%s", mem_type,
  69. suffix);
  70. mem = ofnode_read_string(config_node, prop_name);
  71. if (!mem) {
  72. debug("%s: No memory type for '%s', using /memory\n", __func__,
  73. prop_name);
  74. mem = "/memory";
  75. }
  76. node = ofnode_path(mem);
  77. if (!ofnode_valid(node)) {
  78. debug("%s: Failed to find node '%s'\n", __func__, mem);
  79. return -ENOENT;
  80. }
  81. /*
  82. * Not strictly correct - the memory may have multiple banks. We just
  83. * use the first
  84. */
  85. if (ofnode_decode_region(node, "reg", &base, &size)) {
  86. debug("%s: Failed to decode memory region %s\n", __func__,
  87. mem);
  88. return -EINVAL;
  89. }
  90. snprintf(prop_name, sizeof(prop_name), "%s-offset%s", mem_type,
  91. suffix);
  92. if (ofnode_decode_region(config_node, prop_name, &offset,
  93. &offset_size)) {
  94. debug("%s: Failed to decode memory region '%s'\n", __func__,
  95. prop_name);
  96. return -EINVAL;
  97. }
  98. *basep = base + offset;
  99. *sizep = offset_size;
  100. return 0;
  101. }