123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132 |
- /* SPDX-License-Identifier: GPL-2.0-only */
- /*
- * Copyright 2015, Michael Ellerman, IBM Corp.
- */
- #ifndef _SELFTESTS_POWERPC_TM_TM_H
- #define _SELFTESTS_POWERPC_TM_TM_H
- #include <stdbool.h>
- #include <asm/tm.h>
- #include "utils.h"
- #include "reg.h"
- #define TM_RETRIES 100
- static inline bool have_htm(void)
- {
- #ifdef PPC_FEATURE2_HTM
- return have_hwcap2(PPC_FEATURE2_HTM);
- #else
- printf("PPC_FEATURE2_HTM not defined, can't check AT_HWCAP2\n");
- return false;
- #endif
- }
- static inline bool have_htm_nosc(void)
- {
- #ifdef PPC_FEATURE2_HTM_NOSC
- return have_hwcap2(PPC_FEATURE2_HTM_NOSC);
- #else
- printf("PPC_FEATURE2_HTM_NOSC not defined, can't check AT_HWCAP2\n");
- return false;
- #endif
- }
- /*
- * Transactional Memory was removed in ISA 3.1. A synthetic TM implementation
- * is provided on P10 for threads running in P8/P9 compatibility mode. The
- * synthetic implementation immediately fails after tbegin. This failure sets
- * Bit 7 (Failure Persistent) and Bit 15 (Implementation-specific).
- */
- static inline bool htm_is_synthetic(void)
- {
- int i;
- /*
- * Per the ISA, the Failure Persistent bit may be incorrect. Try a few
- * times in case we got an Implementation-specific failure on a non ISA
- * v3.1 system. On these systems the Implementation-specific failure
- * should not be persistent.
- */
- for (i = 0; i < TM_RETRIES; i++) {
- asm volatile(
- "tbegin.;"
- "beq 1f;"
- "tend.;"
- "1:"
- :
- :
- : "memory");
- if ((__builtin_get_texasr() & (TEXASR_FP | TEXASR_IC)) !=
- (TEXASR_FP | TEXASR_IC))
- break;
- }
- return i == TM_RETRIES;
- }
- static inline long failure_code(void)
- {
- return __builtin_get_texasru() >> 24;
- }
- static inline bool failure_is_persistent(void)
- {
- return (failure_code() & TM_CAUSE_PERSISTENT) == TM_CAUSE_PERSISTENT;
- }
- static inline bool failure_is_syscall(void)
- {
- return (failure_code() & TM_CAUSE_SYSCALL) == TM_CAUSE_SYSCALL;
- }
- static inline bool failure_is_unavailable(void)
- {
- return (failure_code() & TM_CAUSE_FAC_UNAV) == TM_CAUSE_FAC_UNAV;
- }
- static inline bool failure_is_reschedule(void)
- {
- if ((failure_code() & TM_CAUSE_RESCHED) == TM_CAUSE_RESCHED ||
- (failure_code() & TM_CAUSE_KVM_RESCHED) == TM_CAUSE_KVM_RESCHED ||
- (failure_code() & TM_CAUSE_KVM_FAC_UNAV) == TM_CAUSE_KVM_FAC_UNAV)
- return true;
- return false;
- }
- static inline bool failure_is_nesting(void)
- {
- return (__builtin_get_texasru() & 0x400000);
- }
- static inline int tcheck(void)
- {
- long cr;
- asm volatile ("tcheck 0" : "=r"(cr) : : "cr0");
- return (cr >> 28) & 4;
- }
- static inline bool tcheck_doomed(void)
- {
- return tcheck() & 8;
- }
- static inline bool tcheck_active(void)
- {
- return tcheck() & 4;
- }
- static inline bool tcheck_suspended(void)
- {
- return tcheck() & 2;
- }
- static inline bool tcheck_transactional(void)
- {
- return tcheck() & 6;
- }
- #endif /* _SELFTESTS_POWERPC_TM_TM_H */
|