blob: b3ed03e9dab38ccdfae69cb3c02c1a6c8e7e4037 [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
6#include <common.h>
7#include <dm.h>
8#include <dm/lists.h>
9#include <dm/root.h>
10#include <mailbox.h>
11#include <misc.h>
12#include <asm/arch-tegra/bpmp_abi.h>
13#include <asm/arch-tegra/ivc.h>
14
15#define BPMP_IVC_FRAME_COUNT 1
16#define BPMP_IVC_FRAME_SIZE 128
17
18#define BPMP_FLAG_DO_ACK BIT(0)
19#define BPMP_FLAG_RING_DOORBELL BIT(1)
20
21DECLARE_GLOBAL_DATA_PTR;
22
23struct tegra186_bpmp {
24 struct mbox_chan mbox;
25 struct tegra_ivc ivc;
26};
27
28static int tegra186_bpmp_call(struct udevice *dev, int mrq, void *tx_msg,
29 int tx_size, void *rx_msg, int rx_size)
30{
31 struct tegra186_bpmp *priv = dev_get_priv(dev);
32 int ret, err;
33 void *ivc_frame;
34 struct mrq_request *req;
35 struct mrq_response *resp;
36 ulong start_time;
37
38 debug("%s(dev=%p, mrq=%u, tx_msg=%p, tx_size=%d, rx_msg=%p, rx_size=%d) (priv=%p)\n",
39 __func__, dev, mrq, tx_msg, tx_size, rx_msg, rx_size, priv);
40
41 if ((tx_size > BPMP_IVC_FRAME_SIZE) || (rx_size > BPMP_IVC_FRAME_SIZE))
42 return -EINVAL;
43
44 ret = tegra_ivc_write_get_next_frame(&priv->ivc, &ivc_frame);
45 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090046 pr_err("tegra_ivc_write_get_next_frame() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060047 return ret;
48 }
49
50 req = ivc_frame;
51 req->mrq = mrq;
52 req->flags = BPMP_FLAG_DO_ACK | BPMP_FLAG_RING_DOORBELL;
53 memcpy(req + 1, tx_msg, tx_size);
54
55 ret = tegra_ivc_write_advance(&priv->ivc);
56 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090057 pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060058 return ret;
59 }
60
61 start_time = timer_get_us();
62 for (;;) {
63 ret = tegra_ivc_channel_notified(&priv->ivc);
64 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090065 pr_err("tegra_ivc_channel_notified() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060066 return ret;
67 }
68
69 ret = tegra_ivc_read_get_next_frame(&priv->ivc, &ivc_frame);
70 if (!ret)
71 break;
72
73 /* Timeout 20ms; roughly 10x current max observed duration */
74 if ((timer_get_us() - start_time) > 20 * 1000) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090075 pr_err("tegra_ivc_read_get_next_frame() timed out (%d)\n",
Stephen Warren73dd5c42016-08-08 09:41:34 -060076 ret);
77 return -ETIMEDOUT;
78 }
79 }
80
81 resp = ivc_frame;
82 err = resp->err;
83 if (!err && rx_msg && rx_size)
84 memcpy(rx_msg, resp + 1, rx_size);
85
86 ret = tegra_ivc_read_advance(&priv->ivc);
87 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090088 pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -060089 return ret;
90 }
91
92 if (err) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090093 pr_err("BPMP responded with error %d\n", err);
Stephen Warren73dd5c42016-08-08 09:41:34 -060094 /* err isn't a U-Boot error code, so don't that */
95 return -EIO;
96 }
97
98 return rx_size;
99}
100
101/**
102 * The BPMP exposes multiple different services. We create a sub-device for
103 * each separate type of service, since each device must be of the appropriate
104 * UCLASS.
105 */
106static int tegra186_bpmp_bind(struct udevice *dev)
107{
108 int ret;
109 struct udevice *child;
110
111 debug("%s(dev=%p)\n", __func__, dev);
112
113 ret = device_bind_driver_to_node(dev, "tegra186_clk", "tegra186_clk",
Simon Glass45a26862017-05-18 20:09:07 -0600114 dev_ofnode(dev), &child);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600115 if (ret)
116 return ret;
117
118 ret = device_bind_driver_to_node(dev, "tegra186_reset",
Simon Glass45a26862017-05-18 20:09:07 -0600119 "tegra186_reset", dev_ofnode(dev),
Stephen Warren73dd5c42016-08-08 09:41:34 -0600120 &child);
121 if (ret)
122 return ret;
123
124 ret = device_bind_driver_to_node(dev, "tegra186_power_domain",
125 "tegra186_power_domain",
Simon Glass45a26862017-05-18 20:09:07 -0600126 dev_ofnode(dev), &child);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600127 if (ret)
128 return ret;
129
130 ret = dm_scan_fdt_dev(dev);
131 if (ret)
132 return ret;
133
134 return 0;
135}
136
137static ulong tegra186_bpmp_get_shmem(struct udevice *dev, int index)
138{
139 int ret;
140 struct fdtdec_phandle_args args;
141 fdt_addr_t reg;
142
Simon Glasse160f7d2017-01-17 16:52:55 -0700143 ret = fdtdec_parse_phandle_with_args(gd->fdt_blob, dev_of_offset(dev),
Stephen Warren73dd5c42016-08-08 09:41:34 -0600144 "shmem", NULL, 0, index, &args);
145 if (ret < 0) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900146 pr_err("fdtdec_parse_phandle_with_args() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600147 return ret;
148 }
149
150 reg = fdtdec_get_addr_size_auto_noparent(gd->fdt_blob, args.node,
151 "reg", 0, NULL, true);
152 if (reg == FDT_ADDR_T_NONE) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900153 pr_err("fdtdec_get_addr_size_auto_noparent() failed\n");
Stephen Warren73dd5c42016-08-08 09:41:34 -0600154 return -ENODEV;
155 }
156
157 return reg;
158}
159
160static void tegra186_bpmp_ivc_notify(struct tegra_ivc *ivc)
161{
162 struct tegra186_bpmp *priv =
163 container_of(ivc, struct tegra186_bpmp, ivc);
164 int ret;
165
166 ret = mbox_send(&priv->mbox, NULL);
167 if (ret)
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900168 pr_err("mbox_send() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600169}
170
171static int tegra186_bpmp_probe(struct udevice *dev)
172{
173 struct tegra186_bpmp *priv = dev_get_priv(dev);
174 int ret;
175 ulong tx_base, rx_base, start_time;
176
177 debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
178
179 ret = mbox_get_by_index(dev, 0, &priv->mbox);
180 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900181 pr_err("mbox_get_by_index() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600182 return ret;
183 }
184
185 tx_base = tegra186_bpmp_get_shmem(dev, 0);
186 if (IS_ERR_VALUE(tx_base)) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900187 pr_err("tegra186_bpmp_get_shmem failed for tx_base\n");
Stephen Warren73dd5c42016-08-08 09:41:34 -0600188 return tx_base;
189 }
190 rx_base = tegra186_bpmp_get_shmem(dev, 1);
191 if (IS_ERR_VALUE(rx_base)) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900192 pr_err("tegra186_bpmp_get_shmem failed for rx_base\n");
Stephen Warren73dd5c42016-08-08 09:41:34 -0600193 return rx_base;
194 }
195 debug("shmem: rx=%lx, tx=%lx\n", rx_base, tx_base);
196
197 ret = tegra_ivc_init(&priv->ivc, rx_base, tx_base, BPMP_IVC_FRAME_COUNT,
198 BPMP_IVC_FRAME_SIZE, tegra186_bpmp_ivc_notify);
199 if (ret) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900200 pr_err("tegra_ivc_init() failed: %d\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600201 return ret;
202 }
203
204 tegra_ivc_channel_reset(&priv->ivc);
205 start_time = timer_get_us();
206 for (;;) {
207 ret = tegra_ivc_channel_notified(&priv->ivc);
208 if (!ret)
209 break;
210
211 /* Timeout 100ms */
212 if ((timer_get_us() - start_time) > 100 * 1000) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +0900213 pr_err("Initial IVC reset timed out (%d)\n", ret);
Stephen Warren73dd5c42016-08-08 09:41:34 -0600214 ret = -ETIMEDOUT;
215 goto err_free_mbox;
216 }
217 }
218
219 return 0;
220
221err_free_mbox:
222 mbox_free(&priv->mbox);
223
224 return ret;
225}
226
227static int tegra186_bpmp_remove(struct udevice *dev)
228{
229 struct tegra186_bpmp *priv = dev_get_priv(dev);
230
231 debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
232
233 mbox_free(&priv->mbox);
234
235 return 0;
236}
237
238static struct misc_ops tegra186_bpmp_ops = {
239 .call = tegra186_bpmp_call,
240};
241
242static const struct udevice_id tegra186_bpmp_ids[] = {
243 { .compatible = "nvidia,tegra186-bpmp" },
244 { }
245};
246
247U_BOOT_DRIVER(tegra186_bpmp) = {
248 .name = "tegra186_bpmp",
249 .id = UCLASS_MISC,
250 .of_match = tegra186_bpmp_ids,
251 .bind = tegra186_bpmp_bind,
252 .probe = tegra186_bpmp_probe,
253 .remove = tegra186_bpmp_remove,
254 .ops = &tegra186_bpmp_ops,
255 .priv_auto_alloc_size = sizeof(struct tegra186_bpmp),
256};