alternative-asm.h 1.6 KB
Newer Older
1 2 3
#ifndef _ASM_X86_ALTERNATIVE_ASM_H
#define _ASM_X86_ALTERNATIVE_ASM_H

4 5
#ifdef __ASSEMBLY__

6
#include <asm/asm.h>
7 8 9

#ifdef CONFIG_SMP
	.macro LOCK_PREFIX
10
672:	lock
11
	.pushsection .smp_locks,"a"
12
	.balign 4
13
	.long 672b - .
14
	.popsection
15 16 17 18 19 20
	.endm
#else
	.macro LOCK_PREFIX
	.endm
#endif

21
.macro altinstruction_entry orig alt feature orig_len alt_len pad_len
22 23
	.long \orig - .
	.long \alt - .
24 25 26
	.word \feature
	.byte \orig_len
	.byte \alt_len
27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
	.byte \pad_len
.endm

.macro ALTERNATIVE oldinstr, newinstr, feature
140:
	\oldinstr
141:
	.skip -(((144f-143f)-(141b-140b)) > 0) * ((144f-143f)-(141b-140b)),0x90
142:

	.pushsection .altinstructions,"a"
	altinstruction_entry 140b,143f,\feature,142b-140b,144f-143f,142b-141b
	.popsection

	.pushsection .altinstr_replacement,"ax"
143:
	\newinstr
144:
	.popsection
.endm

48 49 50 51 52 53 54 55 56 57
#define old_len			141b-140b
#define new_len1		144f-143f
#define new_len2		145f-144f

/*
 * max without conditionals. Idea adapted from:
 * http://graphics.stanford.edu/~seander/bithacks.html#IntegerMinOrMax
 */
#define alt_max_short(a, b)	((a) ^ (((a) ^ (b)) & -(-((a) < (b)))))

58 59 60 61
.macro ALTERNATIVE_2 oldinstr, newinstr1, feature1, newinstr2, feature2
140:
	\oldinstr
141:
62 63
	.skip -((alt_max_short(new_len1, new_len2) - (old_len)) > 0) * \
		(alt_max_short(new_len1, new_len2) - (old_len)),0x90
64 65 66 67 68 69 70 71 72 73 74 75 76 77
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

	.pushsection .altinstr_replacement,"ax"
143:
	\newinstr1
144:
	\newinstr2
145:
	.popsection
78 79
.endm

80
#endif  /*  __ASSEMBLY__  */
81 82

#endif /* _ASM_X86_ALTERNATIVE_ASM_H */