tegra-hsp.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. /*
  2. * Copyright (c) 2016, NVIDIA CORPORATION.
  3. *
  4. * SPDX-License-Identifier: GPL-2.0
  5. */
  6. #include <common.h>
  7. #include <asm/io.h>
  8. #include <dm.h>
  9. #include <mailbox-uclass.h>
  10. #include <dt-bindings/mailbox/tegra186-hsp.h>
  11. #define TEGRA_HSP_INT_DIMENSIONING 0x380
  12. #define TEGRA_HSP_INT_DIMENSIONING_NSI_SHIFT 16
  13. #define TEGRA_HSP_INT_DIMENSIONING_NSI_MASK 0xf
  14. #define TEGRA_HSP_INT_DIMENSIONING_NDB_SHIFT 12
  15. #define TEGRA_HSP_INT_DIMENSIONING_NDB_MASK 0xf
  16. #define TEGRA_HSP_INT_DIMENSIONING_NAS_SHIFT 8
  17. #define TEGRA_HSP_INT_DIMENSIONING_NAS_MASK 0xf
  18. #define TEGRA_HSP_INT_DIMENSIONING_NSS_SHIFT 4
  19. #define TEGRA_HSP_INT_DIMENSIONING_NSS_MASK 0xf
  20. #define TEGRA_HSP_INT_DIMENSIONING_NSM_SHIFT 0
  21. #define TEGRA_HSP_INT_DIMENSIONING_NSM_MASK 0xf
  22. #define TEGRA_HSP_DB_REG_TRIGGER 0x0
  23. #define TEGRA_HSP_DB_REG_ENABLE 0x4
  24. #define TEGRA_HSP_DB_REG_RAW 0x8
  25. #define TEGRA_HSP_DB_REG_PENDING 0xc
  26. #define TEGRA_HSP_DB_ID_CCPLEX 1
  27. #define TEGRA_HSP_DB_ID_BPMP 3
  28. #define TEGRA_HSP_DB_ID_NUM 7
  29. struct tegra_hsp {
  30. fdt_addr_t regs;
  31. uint32_t db_base;
  32. };
  33. DECLARE_GLOBAL_DATA_PTR;
  34. static uint32_t *tegra_hsp_reg(struct tegra_hsp *thsp, uint32_t db_id,
  35. uint32_t reg)
  36. {
  37. return (uint32_t *)(thsp->regs + thsp->db_base + (db_id * 0x100) + reg);
  38. }
  39. static uint32_t tegra_hsp_readl(struct tegra_hsp *thsp, uint32_t db_id,
  40. uint32_t reg)
  41. {
  42. uint32_t *r = tegra_hsp_reg(thsp, db_id, reg);
  43. return readl(r);
  44. }
  45. static void tegra_hsp_writel(struct tegra_hsp *thsp, uint32_t val,
  46. uint32_t db_id, uint32_t reg)
  47. {
  48. uint32_t *r = tegra_hsp_reg(thsp, db_id, reg);
  49. writel(val, r);
  50. readl(r);
  51. }
  52. static int tegra_hsp_db_id(ulong chan_id)
  53. {
  54. switch (chan_id) {
  55. case (HSP_MBOX_TYPE_DB << 16) | HSP_DB_MASTER_BPMP:
  56. return TEGRA_HSP_DB_ID_BPMP;
  57. default:
  58. debug("Invalid channel ID\n");
  59. return -EINVAL;
  60. }
  61. }
  62. static int tegra_hsp_of_xlate(struct mbox_chan *chan,
  63. struct fdtdec_phandle_args *args)
  64. {
  65. debug("%s(chan=%p)\n", __func__, chan);
  66. if (args->args_count != 2) {
  67. debug("Invaild args_count: %d\n", args->args_count);
  68. return -EINVAL;
  69. }
  70. chan->id = (args->args[0] << 16) | args->args[1];
  71. return 0;
  72. }
  73. static int tegra_hsp_request(struct mbox_chan *chan)
  74. {
  75. int db_id;
  76. debug("%s(chan=%p)\n", __func__, chan);
  77. db_id = tegra_hsp_db_id(chan->id);
  78. if (db_id < 0) {
  79. debug("tegra_hsp_db_id() failed: %d\n", db_id);
  80. return -EINVAL;
  81. }
  82. return 0;
  83. }
  84. static int tegra_hsp_free(struct mbox_chan *chan)
  85. {
  86. debug("%s(chan=%p)\n", __func__, chan);
  87. return 0;
  88. }
  89. static int tegra_hsp_send(struct mbox_chan *chan, const void *data)
  90. {
  91. struct tegra_hsp *thsp = dev_get_priv(chan->dev);
  92. int db_id;
  93. debug("%s(chan=%p, data=%p)\n", __func__, chan, data);
  94. db_id = tegra_hsp_db_id(chan->id);
  95. tegra_hsp_writel(thsp, 1, db_id, TEGRA_HSP_DB_REG_TRIGGER);
  96. return 0;
  97. }
  98. static int tegra_hsp_recv(struct mbox_chan *chan, void *data)
  99. {
  100. struct tegra_hsp *thsp = dev_get_priv(chan->dev);
  101. uint32_t db_id = TEGRA_HSP_DB_ID_CCPLEX;
  102. uint32_t val;
  103. debug("%s(chan=%p, data=%p)\n", __func__, chan, data);
  104. val = tegra_hsp_readl(thsp, db_id, TEGRA_HSP_DB_REG_RAW);
  105. if (!(val & BIT(chan->id)))
  106. return -ENODATA;
  107. tegra_hsp_writel(thsp, BIT(chan->id), db_id, TEGRA_HSP_DB_REG_RAW);
  108. return 0;
  109. }
  110. static int tegra_hsp_bind(struct udevice *dev)
  111. {
  112. debug("%s(dev=%p)\n", __func__, dev);
  113. return 0;
  114. }
  115. static int tegra_hsp_probe(struct udevice *dev)
  116. {
  117. struct tegra_hsp *thsp = dev_get_priv(dev);
  118. u32 val;
  119. int nr_sm, nr_ss, nr_as;
  120. debug("%s(dev=%p)\n", __func__, dev);
  121. thsp->regs = dev_get_addr(dev);
  122. if (thsp->regs == FDT_ADDR_T_NONE)
  123. return -ENODEV;
  124. val = readl(thsp->regs + TEGRA_HSP_INT_DIMENSIONING);
  125. nr_sm = (val >> TEGRA_HSP_INT_DIMENSIONING_NSM_SHIFT) &
  126. TEGRA_HSP_INT_DIMENSIONING_NSM_MASK;
  127. nr_ss = (val >> TEGRA_HSP_INT_DIMENSIONING_NSS_SHIFT) &
  128. TEGRA_HSP_INT_DIMENSIONING_NSS_MASK;
  129. nr_as = (val >> TEGRA_HSP_INT_DIMENSIONING_NAS_SHIFT) &
  130. TEGRA_HSP_INT_DIMENSIONING_NAS_MASK;
  131. thsp->db_base = (1 + (nr_sm >> 1) + nr_ss + nr_as) << 16;
  132. return 0;
  133. }
  134. static const struct udevice_id tegra_hsp_ids[] = {
  135. { .compatible = "nvidia,tegra186-hsp" },
  136. { }
  137. };
  138. struct mbox_ops tegra_hsp_mbox_ops = {
  139. .of_xlate = tegra_hsp_of_xlate,
  140. .request = tegra_hsp_request,
  141. .free = tegra_hsp_free,
  142. .send = tegra_hsp_send,
  143. .recv = tegra_hsp_recv,
  144. };
  145. U_BOOT_DRIVER(tegra_hsp) = {
  146. .name = "tegra-hsp",
  147. .id = UCLASS_MAILBOX,
  148. .of_match = tegra_hsp_ids,
  149. .bind = tegra_hsp_bind,
  150. .probe = tegra_hsp_probe,
  151. .priv_auto_alloc_size = sizeof(struct tegra_hsp),
  152. .ops = &tegra_hsp_mbox_ops,
  153. };