blob: 9bee886561c5296cdd8a226ef7d6e1478db0c9bd [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0
Stephen Warrene0e2b262016-06-17 09:43:57 -06002/*
3 * Copyright (c) 2016, NVIDIA CORPORATION.
Stephen Warrene0e2b262016-06-17 09:43:57 -06004 */
5
6#include <common.h>
7#include <asm/io.h>
8#include <dm.h>
9#include <mailbox-uclass.h>
Stephen Warrenc5317312016-07-27 15:24:49 -060010#include <dt-bindings/mailbox/tegra186-hsp.h>
11
12#define TEGRA_HSP_INT_DIMENSIONING 0x380
13#define TEGRA_HSP_INT_DIMENSIONING_NSI_SHIFT 16
14#define TEGRA_HSP_INT_DIMENSIONING_NSI_MASK 0xf
15#define TEGRA_HSP_INT_DIMENSIONING_NDB_SHIFT 12
16#define TEGRA_HSP_INT_DIMENSIONING_NDB_MASK 0xf
17#define TEGRA_HSP_INT_DIMENSIONING_NAS_SHIFT 8
18#define TEGRA_HSP_INT_DIMENSIONING_NAS_MASK 0xf
19#define TEGRA_HSP_INT_DIMENSIONING_NSS_SHIFT 4
20#define TEGRA_HSP_INT_DIMENSIONING_NSS_MASK 0xf
21#define TEGRA_HSP_INT_DIMENSIONING_NSM_SHIFT 0
22#define TEGRA_HSP_INT_DIMENSIONING_NSM_MASK 0xf
Stephen Warrene0e2b262016-06-17 09:43:57 -060023
24#define TEGRA_HSP_DB_REG_TRIGGER 0x0
25#define TEGRA_HSP_DB_REG_ENABLE 0x4
26#define TEGRA_HSP_DB_REG_RAW 0x8
27#define TEGRA_HSP_DB_REG_PENDING 0xc
28
29#define TEGRA_HSP_DB_ID_CCPLEX 1
30#define TEGRA_HSP_DB_ID_BPMP 3
31#define TEGRA_HSP_DB_ID_NUM 7
32
33struct tegra_hsp {
34 fdt_addr_t regs;
35 uint32_t db_base;
36};
37
Stephen Warrene0e2b262016-06-17 09:43:57 -060038static uint32_t *tegra_hsp_reg(struct tegra_hsp *thsp, uint32_t db_id,
39 uint32_t reg)
40{
41 return (uint32_t *)(thsp->regs + thsp->db_base + (db_id * 0x100) + reg);
42}
43
44static uint32_t tegra_hsp_readl(struct tegra_hsp *thsp, uint32_t db_id,
45 uint32_t reg)
46{
47 uint32_t *r = tegra_hsp_reg(thsp, db_id, reg);
48 return readl(r);
49}
50
51static void tegra_hsp_writel(struct tegra_hsp *thsp, uint32_t val,
52 uint32_t db_id, uint32_t reg)
53{
54 uint32_t *r = tegra_hsp_reg(thsp, db_id, reg);
55
56 writel(val, r);
57 readl(r);
58}
59
60static int tegra_hsp_db_id(ulong chan_id)
61{
62 switch (chan_id) {
Stephen Warrenc5317312016-07-27 15:24:49 -060063 case (HSP_MBOX_TYPE_DB << 16) | HSP_DB_MASTER_BPMP:
Stephen Warrene0e2b262016-06-17 09:43:57 -060064 return TEGRA_HSP_DB_ID_BPMP;
65 default:
66 debug("Invalid channel ID\n");
67 return -EINVAL;
68 }
Stephen Warrenc5317312016-07-27 15:24:49 -060069}
70
71static int tegra_hsp_of_xlate(struct mbox_chan *chan,
Simon Glass91dbd532017-05-18 20:09:46 -060072 struct ofnode_phandle_args *args)
Stephen Warrenc5317312016-07-27 15:24:49 -060073{
74 debug("%s(chan=%p)\n", __func__, chan);
75
76 if (args->args_count != 2) {
77 debug("Invaild args_count: %d\n", args->args_count);
78 return -EINVAL;
79 }
80
81 chan->id = (args->args[0] << 16) | args->args[1];
82
83 return 0;
Stephen Warrene0e2b262016-06-17 09:43:57 -060084}
85
86static int tegra_hsp_request(struct mbox_chan *chan)
87{
88 int db_id;
89
90 debug("%s(chan=%p)\n", __func__, chan);
91
92 db_id = tegra_hsp_db_id(chan->id);
93 if (db_id < 0) {
94 debug("tegra_hsp_db_id() failed: %d\n", db_id);
95 return -EINVAL;
96 }
97
98 return 0;
99}
100
101static int tegra_hsp_free(struct mbox_chan *chan)
102{
103 debug("%s(chan=%p)\n", __func__, chan);
104
105 return 0;
106}
107
108static int tegra_hsp_send(struct mbox_chan *chan, const void *data)
109{
110 struct tegra_hsp *thsp = dev_get_priv(chan->dev);
111 int db_id;
112
113 debug("%s(chan=%p, data=%p)\n", __func__, chan, data);
114
115 db_id = tegra_hsp_db_id(chan->id);
116 tegra_hsp_writel(thsp, 1, db_id, TEGRA_HSP_DB_REG_TRIGGER);
117
118 return 0;
119}
120
121static int tegra_hsp_recv(struct mbox_chan *chan, void *data)
122{
123 struct tegra_hsp *thsp = dev_get_priv(chan->dev);
124 uint32_t db_id = TEGRA_HSP_DB_ID_CCPLEX;
125 uint32_t val;
126
127 debug("%s(chan=%p, data=%p)\n", __func__, chan, data);
128
129 val = tegra_hsp_readl(thsp, db_id, TEGRA_HSP_DB_REG_RAW);
130 if (!(val & BIT(chan->id)))
131 return -ENODATA;
132
133 tegra_hsp_writel(thsp, BIT(chan->id), db_id, TEGRA_HSP_DB_REG_RAW);
134
135 return 0;
136}
137
138static int tegra_hsp_bind(struct udevice *dev)
139{
140 debug("%s(dev=%p)\n", __func__, dev);
141
142 return 0;
143}
144
145static int tegra_hsp_probe(struct udevice *dev)
146{
147 struct tegra_hsp *thsp = dev_get_priv(dev);
Stephen Warrenc5317312016-07-27 15:24:49 -0600148 u32 val;
Stephen Warrene0e2b262016-06-17 09:43:57 -0600149 int nr_sm, nr_ss, nr_as;
150
151 debug("%s(dev=%p)\n", __func__, dev);
152
Simon Glassba1dea42017-05-17 17:18:05 -0600153 thsp->regs = devfdt_get_addr(dev);
Stephen Warrene0e2b262016-06-17 09:43:57 -0600154 if (thsp->regs == FDT_ADDR_T_NONE)
155 return -ENODEV;
156
Stephen Warrenc5317312016-07-27 15:24:49 -0600157 val = readl(thsp->regs + TEGRA_HSP_INT_DIMENSIONING);
158 nr_sm = (val >> TEGRA_HSP_INT_DIMENSIONING_NSM_SHIFT) &
159 TEGRA_HSP_INT_DIMENSIONING_NSM_MASK;
160 nr_ss = (val >> TEGRA_HSP_INT_DIMENSIONING_NSS_SHIFT) &
161 TEGRA_HSP_INT_DIMENSIONING_NSS_MASK;
162 nr_as = (val >> TEGRA_HSP_INT_DIMENSIONING_NAS_SHIFT) &
163 TEGRA_HSP_INT_DIMENSIONING_NAS_MASK;
164
Stephen Warrene0e2b262016-06-17 09:43:57 -0600165 thsp->db_base = (1 + (nr_sm >> 1) + nr_ss + nr_as) << 16;
166
167 return 0;
168}
169
170static const struct udevice_id tegra_hsp_ids[] = {
171 { .compatible = "nvidia,tegra186-hsp" },
172 { }
173};
174
175struct mbox_ops tegra_hsp_mbox_ops = {
Stephen Warrenc5317312016-07-27 15:24:49 -0600176 .of_xlate = tegra_hsp_of_xlate,
Stephen Warrene0e2b262016-06-17 09:43:57 -0600177 .request = tegra_hsp_request,
178 .free = tegra_hsp_free,
179 .send = tegra_hsp_send,
180 .recv = tegra_hsp_recv,
181};
182
183U_BOOT_DRIVER(tegra_hsp) = {
184 .name = "tegra-hsp",
185 .id = UCLASS_MAILBOX,
186 .of_match = tegra_hsp_ids,
187 .bind = tegra_hsp_bind,
188 .probe = tegra_hsp_probe,
189 .priv_auto_alloc_size = sizeof(struct tegra_hsp),
190 .ops = &tegra_hsp_mbox_ops,
191};