blob: a1585b818677849665e7a768b6dd04d79817d4a0 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0
Stephen Warren73dd5c42016-08-08 09:41:34 -06002/*
3 * Copyright (c) 2016, NVIDIA CORPORATION.
Stephen Warren73dd5c42016-08-08 09:41:34 -06004 */
5
Stephen Warren73dd5c42016-08-08 09:41:34 -06006#include <dm.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -06007#include <log.h>
Simon Glass336d4612020-02-03 07:36:16 -07008#include <malloc.h>
Simon Glass10453152019-11-14 12:57:30 -07009#include <time.h>
Simon Glass401d1c42020-10-30 21:38:53 -060010#include <asm/global_data.h>
Stephen Warren73dd5c42016-08-08 09:41:34 -060011#include <dm/lists.h>
12#include <dm/root.h>
13#include <mailbox.h>
14#include <misc.h>
15#include <asm/arch-tegra/bpmp_abi.h>
16#include <asm/arch-tegra/ivc.h>
Simon Glasscd93d622020-05-10 11:40:13 -060017#include <linux/bitops.h>
Simon Glass61b29b82020-02-03 07:36:15 -070018#include <linux/err.h>
Simon Glass1e94b462023-09-14 18:21:46 -060019#include <linux/printk.h>
Stephen Warren73dd5c42016-08-08 09:41:34 -060020
21#define BPMP_IVC_FRAME_COUNT 1
22#define BPMP_IVC_FRAME_SIZE 128
23
24#define BPMP_FLAG_DO_ACK BIT(0)
25#define BPMP_FLAG_RING_DOORBELL BIT(1)
26
27DECLARE_GLOBAL_DATA_PTR;
28
29struct tegra186_bpmp {
30 struct mbox_chan mbox;
31 struct tegra_ivc ivc;
32};
33
34static int tegra186_bpmp_call(struct udevice *dev, int mrq, void *tx_msg,
35 int tx_size, void *rx_msg, int rx_size)
36{
37 struct tegra186_bpmp *priv = dev_get_priv(dev);
38 int ret, err;
39 void *ivc_frame;
40 struct mrq_request *req;
41 struct mrq_response *resp;
42 ulong start_time;
43
44 debug("%s(dev=%p, mrq=%u, tx_msg=%p, tx_size=%d, rx_msg=%p, rx_size=%d) (priv=%p)\n",
45 __func__, dev, mrq, tx_msg, tx_size, rx_msg, rx_size, priv);
46
47 if ((tx_size > BPMP_IVC_FRAME_SIZE) || (rx_size > BPMP_IVC_FRAME_SIZE))
48 return -EINVAL;
49
50 ret = tegra_ivc_write_get_next_frame(&priv->ivc, &ivc_frame);
51 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090052 pr_err("tegra_ivc_write_get_next_frame() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060053 return ret;
54 }
55
56 req = ivc_frame;
57 req->mrq = mrq;
58 req->flags = BPMP_FLAG_DO_ACK | BPMP_FLAG_RING_DOORBELL;
59 memcpy(req + 1, tx_msg, tx_size);
60
61 ret = tegra_ivc_write_advance(&priv->ivc);
62 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090063 pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060064 return ret;
65 }
66
67 start_time = timer_get_us();
68 for (;;) {
69 ret = tegra_ivc_channel_notified(&priv->ivc);
70 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090071 pr_err("tegra_ivc_channel_notified() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060072 return ret;
73 }
74
75 ret = tegra_ivc_read_get_next_frame(&priv->ivc, &ivc_frame);
76 if (!ret)
77 break;
78
79 /* Timeout 20ms; roughly 10x current max observed duration */
80 if ((timer_get_us() - start_time) > 20 * 1000) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090081 pr_err("tegra_ivc_read_get_next_frame() timed out (%d)\n",
Stephen Warren73dd5c42016-08-08 09:41:34 -060082 ret);
83 return -ETIMEDOUT;
84 }
85 }
86
87 resp = ivc_frame;
88 err = resp->err;
89 if (!err && rx_msg && rx_size)
90 memcpy(rx_msg, resp + 1, rx_size);
91
92 ret = tegra_ivc_read_advance(&priv->ivc);
93 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090094 pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060095 return ret;
96 }
97
98 if (err) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090099 pr_err("BPMP responded with error %d\n", err);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600100 /* err isn't a U-Boot error code, so don't that */
101 return -EIO;
102 }
103
104 return rx_size;
105}
106
107/**
108 * The BPMP exposes multiple different services. We create a sub-device for
109 * each separate type of service, since each device must be of the appropriate
110 * UCLASS.
111 */
112static int tegra186_bpmp_bind(struct udevice *dev)
113{
114 int ret;
115 struct udevice *child;
116
117 debug("%s(dev=%p)\n", __func__, dev);
118
119 ret = device_bind_driver_to_node(dev, "tegra186_clk", "tegra186_clk",
Simon Glass45a26862017-05-18 20:09:07 -0600120 dev_ofnode(dev), &child);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600121 if (ret)
122 return ret;
123
124 ret = device_bind_driver_to_node(dev, "tegra186_reset",
Simon Glass45a26862017-05-18 20:09:07 -0600125 "tegra186_reset", dev_ofnode(dev),
Stephen Warren73dd5c42016-08-08 09:41:34 -0600126 &child);
127 if (ret)
128 return ret;
129
130 ret = device_bind_driver_to_node(dev, "tegra186_power_domain",
131 "tegra186_power_domain",
Simon Glass45a26862017-05-18 20:09:07 -0600132 dev_ofnode(dev), &child);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600133 if (ret)
134 return ret;
135
136 ret = dm_scan_fdt_dev(dev);
137 if (ret)
138 return ret;
139
140 return 0;
141}
142
143static ulong tegra186_bpmp_get_shmem(struct udevice *dev, int index)
144{
145 int ret;
146 struct fdtdec_phandle_args args;
147 fdt_addr_t reg;
148
Simon Glasse160f7d2017-01-17 16:52:55 -0700149 ret = fdtdec_parse_phandle_with_args(gd->fdt_blob, dev_of_offset(dev),
Stephen Warren73dd5c42016-08-08 09:41:34 -0600150 "shmem", NULL, 0, index, &args);
151 if (ret < 0) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900152 pr_err("fdtdec_parse_phandle_with_args() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600153 return ret;
154 }
155
156 reg = fdtdec_get_addr_size_auto_noparent(gd->fdt_blob, args.node,
157 "reg", 0, NULL, true);
158 if (reg == FDT_ADDR_T_NONE) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900159 pr_err("fdtdec_get_addr_size_auto_noparent() failed\n");
Stephen Warren73dd5c42016-08-08 09:41:34 -0600160 return -ENODEV;
161 }
162
163 return reg;
164}
165
166static void tegra186_bpmp_ivc_notify(struct tegra_ivc *ivc)
167{
168 struct tegra186_bpmp *priv =
169 container_of(ivc, struct tegra186_bpmp, ivc);
170 int ret;
171
172 ret = mbox_send(&priv->mbox, NULL);
173 if (ret)
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900174 pr_err("mbox_send() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600175}
176
177static int tegra186_bpmp_probe(struct udevice *dev)
178{
179 struct tegra186_bpmp *priv = dev_get_priv(dev);
180 int ret;
181 ulong tx_base, rx_base, start_time;
182
183 debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
184
185 ret = mbox_get_by_index(dev, 0, &priv->mbox);
186 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900187 pr_err("mbox_get_by_index() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600188 return ret;
189 }
190
191 tx_base = tegra186_bpmp_get_shmem(dev, 0);
192 if (IS_ERR_VALUE(tx_base)) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900193 pr_err("tegra186_bpmp_get_shmem failed for tx_base\n");
Stephen Warren73dd5c42016-08-08 09:41:34 -0600194 return tx_base;
195 }
196 rx_base = tegra186_bpmp_get_shmem(dev, 1);
197 if (IS_ERR_VALUE(rx_base)) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900198 pr_err("tegra186_bpmp_get_shmem failed for rx_base\n");
Stephen Warren73dd5c42016-08-08 09:41:34 -0600199 return rx_base;
200 }
201 debug("shmem: rx=%lx, tx=%lx\n", rx_base, tx_base);
202
203 ret = tegra_ivc_init(&priv->ivc, rx_base, tx_base, BPMP_IVC_FRAME_COUNT,
204 BPMP_IVC_FRAME_SIZE, tegra186_bpmp_ivc_notify);
205 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900206 pr_err("tegra_ivc_init() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600207 return ret;
208 }
209
210 tegra_ivc_channel_reset(&priv->ivc);
211 start_time = timer_get_us();
212 for (;;) {
213 ret = tegra_ivc_channel_notified(&priv->ivc);
214 if (!ret)
215 break;
216
217 /* Timeout 100ms */
218 if ((timer_get_us() - start_time) > 100 * 1000) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900219 pr_err("Initial IVC reset timed out (%d)\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600220 ret = -ETIMEDOUT;
221 goto err_free_mbox;
222 }
223 }
224
225 return 0;
226
227err_free_mbox:
228 mbox_free(&priv->mbox);
229
230 return ret;
231}
232
233static int tegra186_bpmp_remove(struct udevice *dev)
234{
235 struct tegra186_bpmp *priv = dev_get_priv(dev);
236
237 debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
238
239 mbox_free(&priv->mbox);
240
241 return 0;
242}
243
244static struct misc_ops tegra186_bpmp_ops = {
245 .call = tegra186_bpmp_call,
246};
247
248static const struct udevice_id tegra186_bpmp_ids[] = {
249 { .compatible = "nvidia,tegra186-bpmp" },
250 { }
251};
252
253U_BOOT_DRIVER(tegra186_bpmp) = {
254 .name = "tegra186_bpmp",
255 .id = UCLASS_MISC,
256 .of_match = tegra186_bpmp_ids,
257 .bind = tegra186_bpmp_bind,
258 .probe = tegra186_bpmp_probe,
259 .remove = tegra186_bpmp_remove,
260 .ops = &tegra186_bpmp_ops,
Simon Glass41575d82020-12-03 16:55:17 -0700261 .priv_auto = sizeof(struct tegra186_bpmp),
Stephen Warren73dd5c42016-08-08 09:41:34 -0600262};