| /* SPDX-License-Identifier: GPL-2.0 */ |
| #ifndef _ASM_ALTERNATIVE_ASM_H |
| #define _ASM_ALTERNATIVE_ASM_H |
| |
| #ifdef __ASSEMBLY__ |
| |
| #include <asm/asm.h> |
| |
| /* |
| * Issue one struct alt_instr descriptor entry (need to put it into |
| * the section .altinstructions, see below). This entry contains |
| * enough information for the alternatives patching code to patch an |
| * instruction. See apply_alternatives(). |
| */ |
| .macro altinstruction_entry orig alt feature orig_len alt_len |
| .long \orig - . |
| .long \alt - . |
| .short \feature |
| .byte \orig_len |
| .byte \alt_len |
| .endm |
| |
| /* |
| * Define an alternative between two instructions. If @feature is |
| * present, early code in apply_alternatives() replaces @oldinstr with |
| * @newinstr. ".fill" directive takes care of proper instruction padding |
| * in case @newinstr is longer than @oldinstr. |
| */ |
| .macro ALTERNATIVE oldinstr, newinstr, feature |
| 140 : |
| \oldinstr |
| 141 : |
| .fill - (((144f-143f)-(141b-140b)) > 0) * ((144f-143f)-(141b-140b)) / 4, 4, 0x03400000 |
| 142 : |
| |
| .pushsection .altinstructions, "a" |
| altinstruction_entry 140b, 143f, \feature, 142b-140b, 144f-143f |
| .popsection |
| |
| .subsection 1 |
| 143 : |
| \newinstr |
| 144 : |
| .previous |
| .endm |
| |
| #define old_len (141b-140b) |
| #define new_len1 (144f-143f) |
| #define new_len2 (145f-144f) |
| |
| #define alt_max_short(a, b) ((a) ^ (((a) ^ (b)) & -(-((a) < (b))))) |
| |
| /* |
| * Same as ALTERNATIVE macro above but for two alternatives. If CPU |
| * has @feature1, it replaces @oldinstr with @newinstr1. If CPU has |
| * @feature2, it replaces @oldinstr with @feature2. |
| */ |
| .macro ALTERNATIVE_2 oldinstr, newinstr1, feature1, newinstr2, feature2 |
| 140 : |
| \oldinstr |
| 141 : |
| .fill - ((alt_max_short(new_len1, new_len2) - (old_len)) > 0) * \ |
| (alt_max_short(new_len1, new_len2) - (old_len)) / 4, 4, 0x03400000 |
| 142 : |
| |
| .pushsection .altinstructions, "a" |
| altinstruction_entry 140b, 143f, \feature1, 142b-140b, 144f-143f, 142b-141b |
| altinstruction_entry 140b, 144f, \feature2, 142b-140b, 145f-144f, 142b-141b |
| .popsection |
| |
| .subsection 1 |
| 143 : |
| \newinstr1 |
| 144 : |
| \newinstr2 |
| 145 : |
| .previous |
| .endm |
| |
| #endif /* __ASSEMBLY__ */ |
| |
| #endif /* _ASM_ALTERNATIVE_ASM_H */ |