1 /* SPDX-License-Identifier: GPL-2.0-only */
3 /* EXYNOS - Thermal Management Unit */
5 #include <device/mmio.h>
6 #include <console/console.h>
10 #define TRIMINFO_RELOAD 1
12 #define THERM_TRIP_EN (1 << 12)
15 #define INTEN_RISE1 (1 << 4)
16 #define INTEN_RISE2 (1 << 8)
17 #define INTEN_FALL0 (1 << 16)
18 #define INTEN_FALL1 (1 << 20)
19 #define INTEN_FALL2 (1 << 24)
21 #define TRIM_INFO_MASK 0xff
23 #define INTCLEAR_RISE0 1
24 #define INTCLEAR_RISE1 (1 << 4)
25 #define INTCLEAR_RISE2 (1 << 8)
26 #define INTCLEAR_FALL0 (1 << 16)
27 #define INTCLEAR_FALL1 (1 << 20)
28 #define INTCLEAR_FALL2 (1 << 24)
29 #define INTCLEARALL (INTCLEAR_RISE0 | INTCLEAR_RISE1 | \
30 INTCLEAR_RISE2 | INTCLEAR_FALL0 | \
31 INTCLEAR_FALL1 | INTCLEAR_FALL2)
33 struct tmu_info exynos5250_tmu_info
= {
34 .tmu_base
= 0x10060000,
41 .start_tripping
= 105,
42 .hardware_tripping
= 110,
44 .efuse_min_value
= 40,
46 .efuse_max_value
= 100,
53 * After reading temperature code from register, compensating
54 * its value and calculating celsius temperature,
55 * get current temperature.
57 * @return current temperature of the chip as sensed by TMU
59 static int get_cur_temp(struct tmu_info
*info
)
62 struct tmu_reg
*reg
= (struct tmu_reg
*)info
->tmu_base
;
64 /* Temperature code range between min 25 and max 125 */
65 cur_temp
= read32(®
->current_temp
) & 0xff;
67 /* Calibrate current temperature */
69 cur_temp
= cur_temp
- info
->te1
+ info
->dc_value
;
75 * Monitors status of the TMU device and exynos temperature
78 * @temp pointer to the current temperature value
79 * @return enum tmu_status_t value, code indicating event to execute
81 enum tmu_status_t
tmu_monitor(struct tmu_info
*info
, int *temp
)
83 if (info
->tmu_state
== TMU_STATUS_INIT
)
87 struct tmu_data
*data
= &info
->data
;
89 /* Read current temperature of the SOC */
90 cur_temp
= get_cur_temp(info
);
93 /* Temperature code lies between min 25 and max 125 */
94 if (cur_temp
>= data
->ts
.start_tripping
&&
95 cur_temp
<= data
->ts
.max_val
)
96 return TMU_STATUS_TRIPPED
;
97 else if (cur_temp
>= data
->ts
.start_warning
)
98 return TMU_STATUS_WARNING
;
99 else if (cur_temp
< data
->ts
.start_warning
&&
100 cur_temp
>= data
->ts
.min_val
)
101 return TMU_STATUS_NORMAL
;
102 /* Temperature code does not lie between min 25 and max 125 */
104 info
->tmu_state
= TMU_STATUS_INIT
;
105 printk(BIOS_DEBUG
, "EXYNOS_TMU: Thermal reading failed\n");
112 * Calibrate and calculate threshold values and
113 * enable interrupt levels
115 * @param info pointer to the tmu_info struct
117 static void tmu_setup_parameters(struct tmu_info
*info
)
119 unsigned int te_temp
, con
;
120 unsigned int warning_code
, trip_code
, hwtrip_code
;
121 unsigned int cooling_temp
;
122 unsigned int rising_value
;
123 struct tmu_data
*data
= &info
->data
;
124 struct tmu_reg
*reg
= (struct tmu_reg
*)info
->tmu_base
;
126 /* Must reload for using efuse value at EXYNOS */
127 write32(®
->triminfo_control
, TRIMINFO_RELOAD
);
129 /* Get the compensation parameter */
130 te_temp
= read32(®
->triminfo
);
131 info
->te1
= te_temp
& TRIM_INFO_MASK
;
132 info
->te2
= ((te_temp
>> 8) & TRIM_INFO_MASK
);
134 if ((data
->efuse_min_value
> info
->te1
) ||
135 (info
->te1
> data
->efuse_max_value
)
137 info
->te1
= data
->efuse_value
;
139 /* Get RISING & FALLING Threshold value */
140 warning_code
= data
->ts
.start_warning
141 + info
->te1
- info
->dc_value
;
142 trip_code
= data
->ts
.start_tripping
143 + info
->te1
- info
->dc_value
;
144 hwtrip_code
= data
->ts
.hardware_tripping
145 + info
->te1
- info
->dc_value
;
149 rising_value
= ((warning_code
<< 8) |
151 (hwtrip_code
<< 24));
153 /* Set interrupt level */
154 write32(®
->threshold_temp_rise
, rising_value
);
155 write32(®
->threshold_temp_fall
, cooling_temp
);
158 * Need to init all register settings after getting parameter info
159 * [28:23] vref [11:8] slope - Tuning parameter
161 * WARNING: this slope value writes into many bits in the tmu_control
162 * register, with the default FDT value of 268470274 (0x10008802)
163 * we are using this essentially sets the default register setting
164 * from the TRM for tmu_control.
165 * TODO(bhthompson): rewrite this code such that we are not performing
166 * a hard wipe of tmu_control and re verify functionality.
168 write32(®
->tmu_control
, data
->slope
);
170 write32(®
->intclear
, INTCLEARALL
);
171 /* TMU core enable */
172 con
= read32(®
->tmu_control
);
173 con
|= (info
->tmu_mux
<< 20) | THERM_TRIP_EN
| CORE_EN
;
175 write32(®
->tmu_control
, con
);
177 /* Enable HW thermal trip */
178 power_enable_hw_thermal_trip();
180 /* LEV1 LEV2 interrupt enable */
181 write32(®
->inten
, INTEN_RISE1
| INTEN_RISE2
);
185 * Initialize TMU device
187 * @return int value, 0 for success
189 int tmu_init(struct tmu_info
*info
)
191 info
->tmu_state
= TMU_STATUS_INIT
;
193 tmu_setup_parameters(info
);
194 info
->tmu_state
= TMU_STATUS_NORMAL
;