alternative-asm.h 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. #ifndef _ASM_X86_ALTERNATIVE_ASM_H
  2. #define _ASM_X86_ALTERNATIVE_ASM_H
  3. #ifdef __ASSEMBLY__
  4. #include <asm/asm.h>
  5. #ifdef CONFIG_SMP
  6. .macro LOCK_PREFIX
  7. 672: lock
  8. .pushsection .smp_locks,"a"
  9. .balign 4
  10. .long 672b - .
  11. .popsection
  12. .endm
  13. #else
  14. .macro LOCK_PREFIX
  15. .endm
  16. #endif
  17. /*
  18. * Issue one struct alt_instr descriptor entry (need to put it into
  19. * the section .altinstructions, see below). This entry contains
  20. * enough information for the alternatives patching code to patch an
  21. * instruction. See apply_alternatives().
  22. */
  23. .macro altinstruction_entry orig alt feature orig_len alt_len pad_len
  24. .long \orig - .
  25. .long \alt - .
  26. .word \feature
  27. .byte \orig_len
  28. .byte \alt_len
  29. .byte \pad_len
  30. .endm
  31. /*
  32. * Define an alternative between two instructions. If @feature is
  33. * present, early code in apply_alternatives() replaces @oldinstr with
  34. * @newinstr. ".skip" directive takes care of proper instruction padding
  35. * in case @newinstr is longer than @oldinstr.
  36. */
  37. .macro ALTERNATIVE oldinstr, newinstr, feature
  38. 140:
  39. \oldinstr
  40. 141:
  41. .skip -(((144f-143f)-(141b-140b)) > 0) * ((144f-143f)-(141b-140b)),0x90
  42. 142:
  43. .pushsection .altinstructions,"a"
  44. altinstruction_entry 140b,143f,\feature,142b-140b,144f-143f,142b-141b
  45. .popsection
  46. .pushsection .altinstr_replacement,"ax"
  47. 143:
  48. \newinstr
  49. 144:
  50. .popsection
  51. .endm
  52. #define old_len 141b-140b
  53. #define new_len1 144f-143f
  54. #define new_len2 145f-144f
  55. /*
  56. * gas compatible max based on the idea from:
  57. * http://graphics.stanford.edu/~seander/bithacks.html#IntegerMinOrMax
  58. *
  59. * The additional "-" is needed because gas uses a "true" value of -1.
  60. */
  61. #define alt_max_short(a, b) ((a) ^ (((a) ^ (b)) & -(-((a) < (b)))))
  62. /*
  63. * Same as ALTERNATIVE macro above but for two alternatives. If CPU
  64. * has @feature1, it replaces @oldinstr with @newinstr1. If CPU has
  65. * @feature2, it replaces @oldinstr with @feature2.
  66. */
  67. .macro ALTERNATIVE_2 oldinstr, newinstr1, feature1, newinstr2, feature2
  68. 140:
  69. \oldinstr
  70. 141:
  71. .skip -((alt_max_short(new_len1, new_len2) - (old_len)) > 0) * \
  72. (alt_max_short(new_len1, new_len2) - (old_len)),0x90
  73. 142:
  74. .pushsection .altinstructions,"a"
  75. altinstruction_entry 140b,143f,\feature1,142b-140b,144f-143f,142b-141b
  76. altinstruction_entry 140b,144f,\feature2,142b-140b,145f-144f,142b-141b
  77. .popsection
  78. .pushsection .altinstr_replacement,"ax"
  79. 143:
  80. \newinstr1
  81. 144:
  82. \newinstr2
  83. 145:
  84. .popsection
  85. .endm
  86. #endif /* __ASSEMBLY__ */
  87. #endif /* _ASM_X86_ALTERNATIVE_ASM_H */