blob: 0774e0a4c9eb8253c9b3fa3aef3a534f6152c260 [file] [log] [blame]
Tero Kristo887dde52019-10-24 15:00:46 +05301// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Texas Instruments' K3 Clas 0 Adaptive Voltage Scaling driver
4 *
Nishanth Menoneaa39c62023-11-01 15:56:03 -05005 * Copyright (C) 2019 Texas Instruments Incorporated - https://www.ti.com/
Tero Kristo887dde52019-10-24 15:00:46 +05306 * Tero Kristo <t-kristo@ti.com>
7 *
8 */
9
Tero Kristo887dde52019-10-24 15:00:46 +053010#include <dm.h>
11#include <errno.h>
12#include <asm/io.h>
13#include <i2c.h>
14#include <k3-avs.h>
Simon Glass9bc15642020-02-03 07:36:16 -070015#include <dm/device_compat.h>
Simon Glass4dcacfc2020-05-10 11:40:13 -060016#include <linux/bitops.h>
Udit Kumarbac62152023-10-19 12:57:53 +053017#include <linux/delay.h>
Tero Kristo887dde52019-10-24 15:00:46 +053018#include <power/regulator.h>
19
20#define AM6_VTM_DEVINFO(i) (priv->base + 0x100 + 0x20 * (i))
21#define AM6_VTM_OPPVID_VD(i) (priv->base + 0x104 + 0x20 * (i))
22
23#define AM6_VTM_AVS0_SUPPORTED BIT(12)
24
25#define AM6_VTM_OPP_SHIFT(opp) (8 * (opp))
26#define AM6_VTM_OPP_MASK 0xff
27
Udit Kumarbac62152023-10-19 12:57:53 +053028#define K3_VTM_DEVINFO_PWR0_OFFSET 0x4
29#define K3_VTM_DEVINFO_PWR0_TEMPSENS_CT_MASK 0xf0
30#define K3_VTM_TMPSENS0_CTRL_OFFSET 0x300
31#define K3_VTM_TMPSENS_STAT_OFFSET 0x8
32#define K3_VTM_ANYMAXT_OUTRG_ALERT_EN 0x1
33#define K3_VTM_LOW_TEMP_OFFSET 0x10
34#define K3_VTM_MISC_CTRL2_OFFSET 0x10
35#define K3_VTM_MISC_CTRL1_OFFSET 0xc
36#define K3_VTM_TMPSENS_CTRL1_SOC BIT(5)
37#define K3_VTM_TMPSENS_CTRL_CLRZ BIT(6)
38#define K3_VTM_TMPSENS_CTRL_MAXT_OUTRG_EN BIT(11)
39#define K3_VTM_ADC_COUNT_FOR_123C 0x2f8
40#define K3_VTM_ADC_COUNT_FOR_105C 0x288
41#define K3_VTM_ADC_WA_VALUE 0x2c
42#define K3_VTM_FUSE_MASK 0xc0000000
43
Tero Kristo887dde52019-10-24 15:00:46 +053044#define VD_FLAG_INIT_DONE BIT(0)
45
46struct k3_avs_privdata {
47 void *base;
48 struct vd_config *vd_config;
Udit Kumarbac62152023-10-19 12:57:53 +053049 struct udevice *dev;
Tero Kristo887dde52019-10-24 15:00:46 +053050};
51
52struct opp {
53 u32 freq;
54 u32 volt;
55};
56
57struct vd_data {
58 int id;
59 u8 opp;
60 u8 flags;
61 int dev_id;
62 int clk_id;
63 struct opp opps[NUM_OPPS];
64 struct udevice *supply;
65};
66
67struct vd_config {
68 struct vd_data *vds;
69 u32 (*efuse_xlate)(struct k3_avs_privdata *priv, int idx, int opp);
70};
71
72static struct k3_avs_privdata *k3_avs_priv;
73
74/**
75 * am6_efuse_voltage: read efuse voltage from VTM
76 * @priv: driver private data
77 * @idx: VD to read efuse for
78 * @opp: opp id to read
79 *
80 * Reads efuse value for the specified OPP, and converts the register
81 * value to a voltage. Returns the voltage in uV, or 0 if nominal voltage
82 * should be used.
83 *
84 * Efuse val to volt conversion logic:
85 *
86 * val > 171 volt increments in 20mV steps with base 171 => 1.66V
87 * val between 115 to 11 increments in 10mV steps with base 115 => 1.1V
88 * val between 15 to 115 increments in 5mV steps with base 15 => .6V
89 * val between 1 to 15 increments in 20mv steps with base 0 => .3V
90 * val 0 is invalid
91 */
92static u32 am6_efuse_xlate(struct k3_avs_privdata *priv, int idx, int opp)
93{
94 u32 val = readl(AM6_VTM_OPPVID_VD(idx));
95
96 val >>= AM6_VTM_OPP_SHIFT(opp);
97 val &= AM6_VTM_OPP_MASK;
98
99 if (!val)
100 return 0;
101
102 if (val > 171)
103 return 1660000 + 20000 * (val - 171);
104
105 if (val > 115)
106 return 1100000 + 10000 * (val - 115);
107
108 if (val > 15)
109 return 600000 + 5000 * (val - 15);
110
111 return 300000 + 20000 * val;
112}
113
114static int k3_avs_program_voltage(struct k3_avs_privdata *priv,
115 struct vd_data *vd,
116 int opp_id)
117{
118 u32 volt = vd->opps[opp_id].volt;
119 struct vd_data *vd2;
120
121 if (!vd->supply)
122 return -ENODEV;
123
Reid Tonking1a9c39f2024-11-19 06:02:57 +0530124 if (!volt) {
125 dev_err(priv->dev, "No efuse found for opp_%d\n", opp_id);
126 return -EINVAL;
127 }
128
Tero Kristo887dde52019-10-24 15:00:46 +0530129 vd->opp = opp_id;
130 vd->flags |= VD_FLAG_INIT_DONE;
131
132 /* Take care of ganged rails and pick the Max amongst them*/
133 for (vd2 = priv->vd_config->vds; vd2->id >= 0; vd2++) {
134 if (vd == vd2)
135 continue;
136
137 if (vd2->supply != vd->supply)
138 continue;
139
140 if (vd2->opps[vd2->opp].volt > volt)
141 volt = vd2->opps[vd2->opp].volt;
142
143 vd2->flags |= VD_FLAG_INIT_DONE;
144 }
145
146 return regulator_set_value(vd->supply, volt);
147}
148
149static struct vd_data *get_vd(struct k3_avs_privdata *priv, int idx)
150{
151 struct vd_data *vd;
152
153 for (vd = priv->vd_config->vds; vd->id >= 0 && vd->id != idx; vd++)
154 ;
155
156 if (vd->id < 0)
157 return NULL;
158
159 return vd;
160}
161
162/**
163 * k3_avs_set_opp: Sets the voltage for an arbitrary VD rail
164 * @dev: AVS device
165 * @vdd_id: voltage domain ID
166 * @opp_id: OPP ID
167 *
168 * Programs the desired OPP value for the defined voltage rail. This
169 * should be called from board files if reconfiguration is desired.
170 * Returns 0 on success, negative error value on failure.
171 */
172int k3_avs_set_opp(struct udevice *dev, int vdd_id, int opp_id)
173{
174 struct k3_avs_privdata *priv = dev_get_priv(dev);
175 struct vd_data *vd;
176
177 vd = get_vd(priv, vdd_id);
178 if (!vd)
179 return -EINVAL;
180
181 return k3_avs_program_voltage(priv, vd, opp_id);
182}
183
184static int match_opp(struct vd_data *vd, u32 freq)
185{
186 struct opp *opp;
187 int opp_id;
188
189 for (opp_id = 0; opp_id < NUM_OPPS; opp_id++) {
190 opp = &vd->opps[opp_id];
191 if (opp->freq == freq)
192 return opp_id;
193 }
194
195 printf("No matching OPP found for freq %d.\n", freq);
196
Reid Tonking1a9c39f2024-11-19 06:02:57 +0530197 return -EINVAL;
198}
199
200/**
201 * k3_check_opp: Check for presence of opp efuse
202 * @dev: AVS device
203 * @vdd_id: voltage domain ID
204 * @opp_id: opp id to check if voltage is present
205 *
206 * Checks to see if an opp has voltage. k3_avs probe will populate
207 * voltage data if efuse is present. Returns 0 if data is valid.
208 */
209int k3_avs_check_opp(struct udevice *dev, int vdd_id, int opp_id)
210{
211 struct k3_avs_privdata *priv = dev_get_priv(dev);
212 struct vd_data *vd;
213 int volt;
214
215 vd = get_vd(priv, vdd_id);
216 if (!vd)
217 return -EINVAL;
218
219 volt = vd->opps[opp_id].volt;
220 if (volt)
221 return 0;
222
223 printf("No efuse found for opp_%d\n", opp_id);
Tero Kristo887dde52019-10-24 15:00:46 +0530224 return -EINVAL;
225}
226
227/**
228 * k3_avs_notify_freq: Notify clock rate change towards AVS subsystem
229 * @dev_id: Device ID for the clock to be changed
230 * @clk_id: Clock ID for the clock to be changed
231 * @freq: New frequency for clock
232 *
233 * Checks if the provided clock is the MPU clock or not, if not, return
234 * immediately. If MPU clock is provided, maps the provided MPU frequency
235 * towards an MPU OPP, and programs the voltage to the regulator. Return 0
236 * on success, negative error value on failure.
237 */
238int k3_avs_notify_freq(int dev_id, int clk_id, u32 freq)
239{
240 int opp_id;
241 struct k3_avs_privdata *priv = k3_avs_priv;
242 struct vd_data *vd;
243
Vignesh Raghavendra0b6ce802020-02-14 17:52:17 +0530244 /* Driver may not be probed yet */
245 if (!priv)
246 return -EINVAL;
247
Tero Kristo887dde52019-10-24 15:00:46 +0530248 for (vd = priv->vd_config->vds; vd->id >= 0; vd++) {
249 if (vd->dev_id != dev_id || vd->clk_id != clk_id)
250 continue;
251
252 opp_id = match_opp(vd, freq);
253 if (opp_id < 0)
254 return opp_id;
255
256 vd->opp = opp_id;
257 return k3_avs_program_voltage(priv, vd, opp_id);
258 }
259
260 return -EINVAL;
261}
262
263static int k3_avs_configure(struct udevice *dev, struct k3_avs_privdata *priv)
264{
265 struct vd_config *conf;
266 int ret;
267 char pname[20];
268 struct vd_data *vd;
269
270 conf = (void *)dev_get_driver_data(dev);
271
272 priv->vd_config = conf;
273
274 for (vd = conf->vds; vd->id >= 0; vd++) {
275 sprintf(pname, "vdd-supply-%d", vd->id);
276 ret = device_get_supply_regulator(dev, pname, &vd->supply);
277 if (ret)
278 dev_warn(dev, "supply not found for VD%d.\n", vd->id);
279
280 sprintf(pname, "ti,default-opp-%d", vd->id);
281 ret = dev_read_u32_default(dev, pname, -1);
282 if (ret != -1)
283 vd->opp = ret;
284 }
285
286 return 0;
287}
288
Udit Kumarbac62152023-10-19 12:57:53 +0530289/* k3_avs_program_tshut : Program thermal shutdown value for SOC
290 * set the values corresponding to thresholds to ~123C and 105C
291 * This is optional feature, Few times OS driver takes care of
292 * tshut programing.
293 */
294
295static void k3_avs_program_tshut(struct k3_avs_privdata *priv)
296{
297 int cnt, id, val;
298 int workaround_needed = 0;
299 u32 ctrl_offset;
300 void __iomem *cfg2_base;
301 void __iomem *fuse_base;
302
303 cfg2_base = (void __iomem *)devfdt_get_addr_index(priv->dev, 1);
304 if (IS_ERR(cfg2_base)) {
305 dev_err(priv->dev, "cfg base is not defined\n");
306 return;
307 }
308
309 /*
310 * Some of TI's J721E SoCs require a software trimming procedure
311 * for the temperature monitors to function properly. To determine
312 * if this particular SoC is NOT affected, both bits in the
313 * WKUP_SPARE_FUSE0[31:30] will be set (0xC0000000) indicating
314 * when software trimming should NOT be applied.
315 *
316 * https://www.ti.com/lit/er/sprz455c/sprz455c.pdf
317 * This routine checks if workaround_needed to be applied or not
318 * based upon workaround_needed, adjust fixed value of tshut high and low
319 */
320
321 if (device_is_compatible(priv->dev, "ti,j721e-vtm")) {
322 fuse_base = (void __iomem *)devfdt_get_addr_index(priv->dev, 2);
323 if (IS_ERR(fuse_base)) {
324 dev_err(priv->dev, "fuse-base is not defined for J721E Soc\n");
325 return;
326 }
327
328 if (!((readl(fuse_base) & K3_VTM_FUSE_MASK) == K3_VTM_FUSE_MASK))
329 workaround_needed = 1;
330 }
331
332 dev_dbg(priv->dev, "Work around %sneeded\n", workaround_needed ? "" : "not ");
333
334 /* Get the sensor count in the VTM */
335 val = readl(priv->base + K3_VTM_DEVINFO_PWR0_OFFSET);
336 cnt = val & K3_VTM_DEVINFO_PWR0_TEMPSENS_CT_MASK;
337 cnt >>= __ffs(K3_VTM_DEVINFO_PWR0_TEMPSENS_CT_MASK);
338
339 /* Program the thermal sensors */
340 for (id = 0; id < cnt; id++) {
341 ctrl_offset = K3_VTM_TMPSENS0_CTRL_OFFSET + id * 0x20;
342
343 val = readl(cfg2_base + ctrl_offset);
344 val |= (K3_VTM_TMPSENS_CTRL_MAXT_OUTRG_EN |
345 K3_VTM_TMPSENS_CTRL1_SOC |
346 K3_VTM_TMPSENS_CTRL_CLRZ | BIT(4));
347 writel(val, cfg2_base + ctrl_offset);
348 }
349
350 /*
351 * Program TSHUT thresholds
352 * Step 1: set the thresholds to ~123C and 105C WKUP_VTM_MISC_CTRL2
353 * Step 2: WKUP_VTM_TMPSENS_CTRL_j set the MAXT_OUTRG_EN bit
354 * This is already taken care as per of init
355 * Step 3: WKUP_VTM_MISC_CTRL set the ANYMAXT_OUTRG_ALERT_EN bit
356 */
357
358 /* Low thresholds for tshut*/
359 val = (K3_VTM_ADC_COUNT_FOR_105C - workaround_needed * K3_VTM_ADC_WA_VALUE)
360 << K3_VTM_LOW_TEMP_OFFSET;
361 /* high thresholds */
362 val |= K3_VTM_ADC_COUNT_FOR_123C - workaround_needed * K3_VTM_ADC_WA_VALUE;
363
364 writel(val, cfg2_base + K3_VTM_MISC_CTRL2_OFFSET);
365 /* ramp-up delay from Linux code */
366 mdelay(100);
367 val = readl(cfg2_base + K3_VTM_MISC_CTRL1_OFFSET) | K3_VTM_ANYMAXT_OUTRG_ALERT_EN;
368 writel(val, cfg2_base + K3_VTM_MISC_CTRL1_OFFSET);
369}
370
Tero Kristo887dde52019-10-24 15:00:46 +0530371/**
372 * k3_avs_probe: parses VD info from VTM, and re-configures the OPP data
373 *
374 * Parses all VDs on a device calculating the AVS class-0 voltages for them,
375 * and updates the vd_data based on this. The vd_data itself shall be used
376 * to program the required OPPs later on. Returns 0 on success, negative
377 * error value on failure.
378 */
379static int k3_avs_probe(struct udevice *dev)
380{
381 int opp_id;
382 u32 volt;
383 struct opp *opp;
384 struct k3_avs_privdata *priv;
385 struct vd_data *vd;
386 int ret;
Manorit Chawdhry9234ab92024-10-15 16:22:20 +0530387 ofnode node;
388 struct ofnode_phandle_args phandle_args;
389 int i = 0;
Tero Kristo887dde52019-10-24 15:00:46 +0530390
391 priv = dev_get_priv(dev);
Udit Kumarbac62152023-10-19 12:57:53 +0530392 priv->dev = dev;
Tero Kristo887dde52019-10-24 15:00:46 +0530393
394 k3_avs_priv = priv;
395
396 ret = k3_avs_configure(dev, priv);
397 if (ret)
398 return ret;
399
400 priv->base = dev_read_addr_ptr(dev);
401 if (!priv->base)
402 return -ENODEV;
403
404 for (vd = priv->vd_config->vds; vd->id >= 0; vd++) {
Manorit Chawdhry9234ab92024-10-15 16:22:20 +0530405 /* Get the clock and dev id for Jacinto platforms */
406 if (vd->id == J721E_VDD_MPU) {
407 node = ofnode_by_compatible(ofnode_null(), "ti,am654-rproc");
408 if (!ofnode_valid(node))
409 return -ENODEV;
410
411 i = ofnode_stringlist_search(node, "clock-names", "core");
412 if (i < 0)
413 return -ENODEV;
414
415 ret = ofnode_parse_phandle_with_args(node, "clocks",
416 "#clock-cells",
417 0, i,
418 &phandle_args);
419 if (ret) {
420 printf("Couldn't get the clock node, ret = %d\n", ret);
421 return ret;
422 }
423
424 vd->dev_id = phandle_args.args[0];
425 vd->clk_id = phandle_args.args[1];
426
427 debug("%s: MPU dev_id: %d, clk_id: %d", __func__,
428 vd->dev_id, vd->clk_id);
429 }
430
Tero Kristo887dde52019-10-24 15:00:46 +0530431 if (!(readl(AM6_VTM_DEVINFO(vd->id)) &
432 AM6_VTM_AVS0_SUPPORTED)) {
433 dev_warn(dev, "AVS-class 0 not supported for VD%d\n",
434 vd->id);
435 continue;
436 }
437
438 for (opp_id = 0; opp_id < NUM_OPPS; opp_id++) {
439 opp = &vd->opps[opp_id];
440
441 if (!opp->freq)
442 continue;
443
444 volt = priv->vd_config->efuse_xlate(priv, vd->id,
445 opp_id);
446 if (volt)
447 opp->volt = volt;
448 }
449 }
450
451 for (vd = priv->vd_config->vds; vd->id >= 0; vd++) {
452 if (vd->flags & VD_FLAG_INIT_DONE)
453 continue;
454
Manorit Chawdhry0c60ab82024-10-15 16:22:18 +0530455 ret = k3_avs_program_voltage(priv, vd, vd->opp);
456 if (ret)
457 dev_warn(dev, "Could not program AVS voltage for VD%d, vd->opp=%d, ret=%d\n",
458 vd->id, vd->opp, ret);
Tero Kristo887dde52019-10-24 15:00:46 +0530459 }
460
Udit Kumarbac62152023-10-19 12:57:53 +0530461 if (!device_is_compatible(priv->dev, "ti,am654-avs"))
462 k3_avs_program_tshut(priv);
463
Tero Kristo887dde52019-10-24 15:00:46 +0530464 return 0;
465}
466
467static struct vd_data am654_vd_data[] = {
468 {
469 .id = AM6_VDD_CORE,
470 .dev_id = 82, /* AM6_DEV_CBASS0 */
471 .clk_id = 0, /* main sysclk0 */
472 .opp = AM6_OPP_NOM,
473 .opps = {
474 [AM6_OPP_NOM] = {
475 .volt = 1000000,
476 .freq = 250000000, /* CBASS0 */
477 },
478 },
479 },
480 {
481 .id = AM6_VDD_MPU0,
482 .dev_id = 202, /* AM6_DEV_COMPUTE_CLUSTER_A53_0 */
483 .clk_id = 0, /* ARM clock */
484 .opp = AM6_OPP_NOM,
485 .opps = {
486 [AM6_OPP_NOM] = {
Tero Kristo75774302020-02-14 09:05:10 +0200487 .volt = 1100000,
Tero Kristo887dde52019-10-24 15:00:46 +0530488 .freq = 800000000,
489 },
490 [AM6_OPP_OD] = {
Tero Kristo75774302020-02-14 09:05:10 +0200491 .volt = 1200000,
Tero Kristo887dde52019-10-24 15:00:46 +0530492 .freq = 1000000000,
493 },
494 [AM6_OPP_TURBO] = {
Tero Kristo75774302020-02-14 09:05:10 +0200495 .volt = 1240000,
Tero Kristo887dde52019-10-24 15:00:46 +0530496 .freq = 1100000000,
497 },
498 },
499 },
500 {
501 .id = AM6_VDD_MPU1,
502 .opp = AM6_OPP_NOM,
503 .dev_id = 204, /* AM6_DEV_COMPUTE_CLUSTER_A53_2 */
504 .clk_id = 0, /* ARM clock */
505 .opps = {
506 [AM6_OPP_NOM] = {
Tero Kristo75774302020-02-14 09:05:10 +0200507 .volt = 1100000,
Tero Kristo887dde52019-10-24 15:00:46 +0530508 .freq = 800000000,
509 },
510 [AM6_OPP_OD] = {
Tero Kristo75774302020-02-14 09:05:10 +0200511 .volt = 1200000,
Tero Kristo887dde52019-10-24 15:00:46 +0530512 .freq = 1000000000,
513 },
514 [AM6_OPP_TURBO] = {
Tero Kristo75774302020-02-14 09:05:10 +0200515 .volt = 1240000,
Tero Kristo887dde52019-10-24 15:00:46 +0530516 .freq = 1100000000,
517 },
518 },
519 },
520 { .id = -1 },
521};
522
Keerthyfda68e42019-10-24 15:00:49 +0530523static struct vd_data j721e_vd_data[] = {
524 {
525 .id = J721E_VDD_MPU,
526 .opp = AM6_OPP_NOM,
Manorit Chawdhry9234ab92024-10-15 16:22:20 +0530527 /*
528 * XXX: DEPRECATION WARNING: Around 2 u-boot versions
529 *
530 * These values will be picked up from DT, kept for backward
531 * compatibility
532 */
Keerthyfda68e42019-10-24 15:00:49 +0530533 .dev_id = 202, /* J721E_DEV_A72SS0_CORE0 */
534 .clk_id = 2, /* ARM clock */
535 .opps = {
Reid Tonkinga468cf22024-11-19 06:02:56 +0530536 [AM6_OPP_LOW] = {
537 .volt = 0, /* voltage TBD after OPP fuse reading */
538 .freq = 1000000000,
539 },
Keerthyfda68e42019-10-24 15:00:49 +0530540 [AM6_OPP_NOM] = {
541 .volt = 880000, /* TBD in DM */
542 .freq = 2000000000,
543 },
544 },
545 },
546 { .id = -1 },
547};
548
549static struct vd_config j721e_vd_config = {
550 .efuse_xlate = am6_efuse_xlate,
551 .vds = j721e_vd_data,
552};
553
Tero Kristo887dde52019-10-24 15:00:46 +0530554static struct vd_config am654_vd_config = {
555 .efuse_xlate = am6_efuse_xlate,
556 .vds = am654_vd_data,
557};
558
559static const struct udevice_id k3_avs_ids[] = {
560 { .compatible = "ti,am654-avs", .data = (ulong)&am654_vd_config },
Keerthyfda68e42019-10-24 15:00:49 +0530561 { .compatible = "ti,j721e-avs", .data = (ulong)&j721e_vd_config },
Reid Tonking8cc3ca32023-09-07 13:06:35 -0500562 { .compatible = "ti,j721e-vtm", .data = (ulong)&j721e_vd_config },
563 { .compatible = "ti,j7200-vtm", .data = (ulong)&j721e_vd_config },
Tero Kristo887dde52019-10-24 15:00:46 +0530564 {}
565};
566
567U_BOOT_DRIVER(k3_avs) = {
568 .name = "k3_avs",
569 .of_match = k3_avs_ids,
570 .id = UCLASS_MISC,
571 .probe = k3_avs_probe,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700572 .priv_auto = sizeof(struct k3_avs_privdata),
Tero Kristo887dde52019-10-24 15:00:46 +0530573};