tegra210-emc-table.c 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2020, NVIDIA CORPORATION. All rights reserved.
  4. */
  5. #include <linux/of_reserved_mem.h>
  6. #include "tegra210-emc.h"
  7. #define TEGRA_EMC_MAX_FREQS 16
  8. static int tegra210_emc_table_device_init(struct reserved_mem *rmem,
  9. struct device *dev)
  10. {
  11. struct tegra210_emc *emc = dev_get_drvdata(dev);
  12. struct tegra210_emc_timing *timings;
  13. unsigned int i, count = 0;
  14. timings = memremap(rmem->base, rmem->size, MEMREMAP_WB);
  15. if (!timings) {
  16. dev_err(dev, "failed to map EMC table\n");
  17. return -ENOMEM;
  18. }
  19. count = 0;
  20. for (i = 0; i < TEGRA_EMC_MAX_FREQS; i++) {
  21. if (timings[i].revision == 0)
  22. break;
  23. count++;
  24. }
  25. /* only the nominal and derated tables are expected */
  26. if (emc->derated) {
  27. dev_warn(dev, "excess EMC table '%s'\n", rmem->name);
  28. goto out;
  29. }
  30. if (emc->nominal) {
  31. if (count != emc->num_timings) {
  32. dev_warn(dev, "%u derated vs. %u nominal entries\n",
  33. count, emc->num_timings);
  34. memunmap(timings);
  35. return -EINVAL;
  36. }
  37. emc->derated = timings;
  38. } else {
  39. emc->num_timings = count;
  40. emc->nominal = timings;
  41. }
  42. out:
  43. /* keep track of which table this is */
  44. rmem->priv = timings;
  45. return 0;
  46. }
  47. static void tegra210_emc_table_device_release(struct reserved_mem *rmem,
  48. struct device *dev)
  49. {
  50. struct tegra210_emc_timing *timings = rmem->priv;
  51. struct tegra210_emc *emc = dev_get_drvdata(dev);
  52. if ((emc->nominal && timings != emc->nominal) &&
  53. (emc->derated && timings != emc->derated))
  54. dev_warn(dev, "trying to release unassigned EMC table '%s'\n",
  55. rmem->name);
  56. memunmap(timings);
  57. }
  58. static const struct reserved_mem_ops tegra210_emc_table_ops = {
  59. .device_init = tegra210_emc_table_device_init,
  60. .device_release = tegra210_emc_table_device_release,
  61. };
  62. static int tegra210_emc_table_init(struct reserved_mem *rmem)
  63. {
  64. pr_debug("Tegra210 EMC table at %pa, size %lu bytes\n", &rmem->base,
  65. (unsigned long)rmem->size);
  66. rmem->ops = &tegra210_emc_table_ops;
  67. return 0;
  68. }
  69. RESERVEDMEM_OF_DECLARE(tegra210_emc_table, "nvidia,tegra210-emc-table",
  70. tegra210_emc_table_init);