Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

clocksource/drivers/arm_arch_timer: Work around Hisilicon erratum 161010101

Erratum Hisilicon-161010101 says that the ARM generic timer counter "has
the potential to contain an erroneous value when the timer value
changes". Accesses to TVAL (both read and write) are also affected due
to the implicit counter read. Accesses to CVAL are not affected.

The workaround is to reread the system count registers until the value
of the second read is larger than the first one by less than 32, the
system counter can be guaranteed not to return wrong value twice by
back-to-back read and the error value is always larger than the correct
one by 32. Writes to TVAL are replaced with an equivalent write to CVAL.

Signed-off-by: Ding Tianhong <dingtianhong@huawei.com>
[Mark: split patch, fix Kconfig, reword commit message]
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Acked-by: Daniel Lezcano <daniel.lezcano@linaro.org>
Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org>

authored by

Ding Tianhong and committed by
Daniel Lezcano
bb42ca47 16d10ef2

+59
+10
drivers/clocksource/Kconfig
··· 356 356 value"). The workaround will only be active if the 357 357 fsl,erratum-a008585 property is found in the timer node. 358 358 359 + config HISILICON_ERRATUM_161010101 360 + bool "Workaround for Hisilicon Erratum 161010101" 361 + default y 362 + select ARM_ARCH_TIMER_OOL_WORKAROUND 363 + depends on ARM_ARCH_TIMER && ARM64 364 + help 365 + This option enables a workaround for Hisilicon Erratum 366 + 161010101. The workaround will be active if the hisilicon,erratum-161010101 367 + property is found in the timer node. 368 + 359 369 config ARM_GLOBAL_TIMER 360 370 bool "Support for the ARM global timer" if COMPILE_TEST 361 371 select CLKSRC_OF if OF
+49
drivers/clocksource/arm_arch_timer.c
··· 130 130 } 131 131 #endif 132 132 133 + #ifdef CONFIG_HISILICON_ERRATUM_161010101 134 + /* 135 + * Verify whether the value of the second read is larger than the first by 136 + * less than 32 is the only way to confirm the value is correct, so clear the 137 + * lower 5 bits to check whether the difference is greater than 32 or not. 138 + * Theoretically the erratum should not occur more than twice in succession 139 + * when reading the system counter, but it is possible that some interrupts 140 + * may lead to more than twice read errors, triggering the warning, so setting 141 + * the number of retries far beyond the number of iterations the loop has been 142 + * observed to take. 143 + */ 144 + #define __hisi_161010101_read_reg(reg) ({ \ 145 + u64 _old, _new; \ 146 + int _retries = 50; \ 147 + \ 148 + do { \ 149 + _old = read_sysreg(reg); \ 150 + _new = read_sysreg(reg); \ 151 + _retries--; \ 152 + } while (unlikely((_new - _old) >> 5) && _retries); \ 153 + \ 154 + WARN_ON_ONCE(!_retries); \ 155 + _new; \ 156 + }) 157 + 158 + static u32 notrace hisi_161010101_read_cntp_tval_el0(void) 159 + { 160 + return __hisi_161010101_read_reg(cntp_tval_el0); 161 + } 162 + 163 + static u32 notrace hisi_161010101_read_cntv_tval_el0(void) 164 + { 165 + return __hisi_161010101_read_reg(cntv_tval_el0); 166 + } 167 + 168 + static u64 notrace hisi_161010101_read_cntvct_el0(void) 169 + { 170 + return __hisi_161010101_read_reg(cntvct_el0); 171 + } 172 + #endif 173 + 133 174 #ifdef CONFIG_ARM_ARCH_TIMER_OOL_WORKAROUND 134 175 const struct arch_timer_erratum_workaround *timer_unstable_counter_workaround = NULL; 135 176 EXPORT_SYMBOL_GPL(timer_unstable_counter_workaround); ··· 185 144 .read_cntp_tval_el0 = fsl_a008585_read_cntp_tval_el0, 186 145 .read_cntv_tval_el0 = fsl_a008585_read_cntv_tval_el0, 187 146 .read_cntvct_el0 = fsl_a008585_read_cntvct_el0, 147 + }, 148 + #endif 149 + #ifdef CONFIG_HISILICON_ERRATUM_161010101 150 + { 151 + .id = "hisilicon,erratum-161010101", 152 + .read_cntp_tval_el0 = hisi_161010101_read_cntp_tval_el0, 153 + .read_cntv_tval_el0 = hisi_161010101_read_cntv_tval_el0, 154 + .read_cntvct_el0 = hisi_161010101_read_cntvct_el0, 188 155 }, 189 156 #endif 190 157 };