regmap.c 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. /*
  2. * Copyright (c) 2015 Google, Inc
  3. * Written by Simon Glass <sjg@chromium.org>
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <dm.h>
  9. #include <errno.h>
  10. #include <libfdt.h>
  11. #include <malloc.h>
  12. #include <mapmem.h>
  13. #include <regmap.h>
  14. DECLARE_GLOBAL_DATA_PTR;
  15. static struct regmap *regmap_alloc_count(int count)
  16. {
  17. struct regmap *map;
  18. map = malloc(sizeof(struct regmap));
  19. if (!map)
  20. return NULL;
  21. if (count <= 1) {
  22. map->range = &map->base_range;
  23. } else {
  24. map->range = malloc(count * sizeof(struct regmap_range));
  25. if (!map->range) {
  26. free(map);
  27. return NULL;
  28. }
  29. }
  30. map->range_count = count;
  31. return map;
  32. }
  33. #if CONFIG_IS_ENABLED(OF_PLATDATA)
  34. int regmap_init_mem_platdata(struct udevice *dev, u32 *reg, int count,
  35. struct regmap **mapp)
  36. {
  37. struct regmap_range *range;
  38. struct regmap *map;
  39. map = regmap_alloc_count(count);
  40. if (!map)
  41. return -ENOMEM;
  42. map->base = *reg;
  43. for (range = map->range; count > 0; reg += 2, range++, count--) {
  44. range->start = *reg;
  45. range->size = reg[1];
  46. }
  47. *mapp = map;
  48. return 0;
  49. }
  50. #else
  51. int regmap_init_mem(struct udevice *dev, struct regmap **mapp)
  52. {
  53. const void *blob = gd->fdt_blob;
  54. struct regmap_range *range;
  55. const fdt32_t *cell;
  56. struct regmap *map;
  57. int count;
  58. int addr_len, size_len, both_len;
  59. int parent;
  60. int len;
  61. parent = dev->parent->of_offset;
  62. addr_len = fdt_address_cells(blob, parent);
  63. size_len = fdt_size_cells(blob, parent);
  64. both_len = addr_len + size_len;
  65. cell = fdt_getprop(blob, dev->of_offset, "reg", &len);
  66. len /= sizeof(*cell);
  67. count = len / both_len;
  68. if (!cell || !count)
  69. return -EINVAL;
  70. map = regmap_alloc_count(count);
  71. if (!map)
  72. return -ENOMEM;
  73. map->base = fdtdec_get_number(cell, addr_len);
  74. for (range = map->range; count > 0;
  75. count--, cell += both_len, range++) {
  76. range->start = fdtdec_get_number(cell, addr_len);
  77. range->size = fdtdec_get_number(cell + addr_len, size_len);
  78. }
  79. *mapp = map;
  80. return 0;
  81. }
  82. #endif
  83. void *regmap_get_range(struct regmap *map, unsigned int range_num)
  84. {
  85. struct regmap_range *range;
  86. if (range_num >= map->range_count)
  87. return NULL;
  88. range = &map->range[range_num];
  89. return map_sysmem(range->start, range->size);
  90. }
  91. int regmap_uninit(struct regmap *map)
  92. {
  93. if (map->range_count > 1)
  94. free(map->range);
  95. free(map);
  96. return 0;
  97. }