regmap.c 1.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  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. #if CONFIG_IS_ENABLED(OF_PLATDATA)
  16. int regmap_init_mem_platdata(struct udevice *dev, fdt32_t *reg, int size,
  17. struct regmap **mapp)
  18. {
  19. /* TODO(sjg@chromium.org): Implement this when needed */
  20. return 0;
  21. }
  22. #else
  23. int regmap_init_mem(struct udevice *dev, struct regmap **mapp)
  24. {
  25. const void *blob = gd->fdt_blob;
  26. struct regmap_range *range;
  27. const fdt32_t *cell;
  28. struct regmap *map;
  29. int count;
  30. int addr_len, size_len, both_len;
  31. int parent;
  32. int len;
  33. parent = dev->parent->of_offset;
  34. addr_len = fdt_address_cells(blob, parent);
  35. size_len = fdt_size_cells(blob, parent);
  36. both_len = addr_len + size_len;
  37. cell = fdt_getprop(blob, dev->of_offset, "reg", &len);
  38. len /= sizeof(*cell);
  39. count = len / both_len;
  40. if (!cell || !count)
  41. return -EINVAL;
  42. map = malloc(sizeof(struct regmap));
  43. if (!map)
  44. return -ENOMEM;
  45. if (count <= 1) {
  46. map->range = &map->base_range;
  47. } else {
  48. map->range = malloc(count * sizeof(struct regmap_range));
  49. if (!map->range) {
  50. free(map);
  51. return -ENOMEM;
  52. }
  53. }
  54. map->base = fdtdec_get_number(cell, addr_len);
  55. map->range_count = count;
  56. for (range = map->range; count > 0;
  57. count--, cell += both_len, range++) {
  58. range->start = fdtdec_get_number(cell, addr_len);
  59. range->size = fdtdec_get_number(cell + addr_len, size_len);
  60. }
  61. *mapp = map;
  62. return 0;
  63. }
  64. #endif
  65. void *regmap_get_range(struct regmap *map, unsigned int range_num)
  66. {
  67. struct regmap_range *range;
  68. if (range_num >= map->range_count)
  69. return NULL;
  70. range = &map->range[range_num];
  71. return map_sysmem(range->start, range->size);
  72. }
  73. int regmap_uninit(struct regmap *map)
  74. {
  75. if (map->range_count > 1)
  76. free(map->range);
  77. free(map);
  78. return 0;
  79. }