"\\var  = " #n "\n\t"                   \
        ".endif\n\t"
 
-__asm__(".macro        parse_r var r\n\t"
-       "\\var  = -1\n\t"
-       _IFC_REG(0)  _IFC_REG(1)  _IFC_REG(2)  _IFC_REG(3)
-       _IFC_REG(4)  _IFC_REG(5)  _IFC_REG(6)  _IFC_REG(7)
-       _IFC_REG(8)  _IFC_REG(9)  _IFC_REG(10) _IFC_REG(11)
-       _IFC_REG(12) _IFC_REG(13) _IFC_REG(14) _IFC_REG(15)
-       _IFC_REG(16) _IFC_REG(17) _IFC_REG(18) _IFC_REG(19)
-       _IFC_REG(20) _IFC_REG(21) _IFC_REG(22) _IFC_REG(23)
-       _IFC_REG(24) _IFC_REG(25) _IFC_REG(26) _IFC_REG(27)
-       _IFC_REG(28) _IFC_REG(29) _IFC_REG(30) _IFC_REG(31)
-       ".iflt  \\var\n\t"
-       ".error \"Unable to parse register name \\r\"\n\t"
-       ".endif\n\t"
-       ".endm");
-
-#undef _IFC_REG
+#define _ASM_SET_PARSE_R                                               \
+       ".macro parse_r var r\n\t"                                      \
+       "\\var  = -1\n\t"                                               \
+       _IFC_REG(0)  _IFC_REG(1)  _IFC_REG(2)  _IFC_REG(3)              \
+       _IFC_REG(4)  _IFC_REG(5)  _IFC_REG(6)  _IFC_REG(7)              \
+       _IFC_REG(8)  _IFC_REG(9)  _IFC_REG(10) _IFC_REG(11)             \
+       _IFC_REG(12) _IFC_REG(13) _IFC_REG(14) _IFC_REG(15)             \
+       _IFC_REG(16) _IFC_REG(17) _IFC_REG(18) _IFC_REG(19)             \
+       _IFC_REG(20) _IFC_REG(21) _IFC_REG(22) _IFC_REG(23)             \
+       _IFC_REG(24) _IFC_REG(25) _IFC_REG(26) _IFC_REG(27)             \
+       _IFC_REG(28) _IFC_REG(29) _IFC_REG(30) _IFC_REG(31)             \
+       ".iflt  \\var\n\t"                                              \
+       ".error \"Unable to parse register name \\r\"\n\t"              \
+       ".endif\n\t"                                                    \
+       ".endm\n\t"
+#define _ASM_UNSET_PARSE_R ".purgem parse_r\n\t"
 
 /*
  * C macros for generating assembler macros for common instruction formats.
  * the ENC encodings.
  */
 
-/* Instructions with no operands */
-#define _ASM_MACRO_0(OP, ENC)                                          \
-       __asm__(".macro " #OP "\n\t"                                    \
-               ENC                                                     \
-               ".endm")
-
 /* Instructions with 1 register operand & 1 immediate operand */
 #define _ASM_MACRO_1R1I(OP, R1, I2, ENC)                               \
-       __asm__(".macro " #OP " " #R1 ", " #I2 "\n\t"                   \
+               ".macro " #OP " " #R1 ", " #I2 "\n\t"                   \
+               _ASM_SET_PARSE_R                                        \
                "parse_r __" #R1 ", \\" #R1 "\n\t"                      \
                ENC                                                     \
-               ".endm")
+               _ASM_UNSET_PARSE_R                                      \
+               ".endm\n\t"
 
 /* Instructions with 2 register operands */
 #define _ASM_MACRO_2R(OP, R1, R2, ENC)                                 \
-       __asm__(".macro " #OP " " #R1 ", " #R2 "\n\t"                   \
+               ".macro " #OP " " #R1 ", " #R2 "\n\t"                   \
+               _ASM_SET_PARSE_R                                        \
                "parse_r __" #R1 ", \\" #R1 "\n\t"                      \
                "parse_r __" #R2 ", \\" #R2 "\n\t"                      \
                ENC                                                     \
-               ".endm")
+               _ASM_UNSET_PARSE_R                                      \
+               ".endm\n\t"
 
 /* Instructions with 3 register operands */
 #define _ASM_MACRO_3R(OP, R1, R2, R3, ENC)                             \
-       __asm__(".macro " #OP " " #R1 ", " #R2 ", " #R3 "\n\t"          \
+               ".macro " #OP " " #R1 ", " #R2 ", " #R3 "\n\t"          \
+               _ASM_SET_PARSE_R                                        \
                "parse_r __" #R1 ", \\" #R1 "\n\t"                      \
                "parse_r __" #R2 ", \\" #R2 "\n\t"                      \
                "parse_r __" #R3 ", \\" #R3 "\n\t"                      \
                ENC                                                     \
-               ".endm")
+               _ASM_UNSET_PARSE_R                                      \
+               ".endm\n\t"
 
 /* Instructions with 2 register operands and 1 optional select operand */
 #define _ASM_MACRO_2R_1S(OP, R1, R2, SEL3, ENC)                                \
-       __asm__(".macro " #OP " " #R1 ", " #R2 ", " #SEL3 " = 0\n\t"    \
+               ".macro " #OP " " #R1 ", " #R2 ", " #SEL3 " = 0\n\t"    \
+               _ASM_SET_PARSE_R                                        \
                "parse_r __" #R1 ", \\" #R1 "\n\t"                      \
                "parse_r __" #R2 ", \\" #R2 "\n\t"                      \
                ENC                                                     \
-               ".endm")
+               _ASM_UNSET_PARSE_R                                      \
+               ".endm\n\t"
 
 /*
  * TLB Invalidate Flush
 } while (0)
 
 #ifndef TOOLCHAIN_SUPPORTS_XPA
-_ASM_MACRO_2R_1S(mfhc0, rt, rs, sel,
-       _ASM_INSN_IF_MIPS(0x40400000 | __rt << 16 | __rs << 11 | \\sel)
-       _ASM_INSN32_IF_MM(0x000000f4 | __rt << 21 | __rs << 16 | \\sel << 11));
-_ASM_MACRO_2R_1S(mthc0, rt, rd, sel,
-       _ASM_INSN_IF_MIPS(0x40c00000 | __rt << 16 | __rd << 11 | \\sel)
-       _ASM_INSN32_IF_MM(0x000002f4 | __rt << 21 | __rd << 16 | \\sel << 11));
-#define _ASM_SET_XPA ""
+#define _ASM_SET_MFHC0                                                 \
+       _ASM_MACRO_2R_1S(mfhc0, rt, rs, sel,                            \
+                        _ASM_INSN_IF_MIPS(0x40400000 | __rt << 16 | __rs << 11 | \\sel)        \
+                        _ASM_INSN32_IF_MM(0x000000f4 | __rt << 21 | __rs << 16 | \\sel << 11))
+#define _ASM_UNSET_MFHC0 ".purgem mfhc0\n\t"
+#define _ASM_SET_MTHC0                                                 \
+       _ASM_MACRO_2R_1S(mthc0, rt, rd, sel,                            \
+                        _ASM_INSN_IF_MIPS(0x40c00000 | __rt << 16 | __rd << 11 | \\sel)        \
+                        _ASM_INSN32_IF_MM(0x000002f4 | __rt << 21 | __rd << 16 | \\sel << 11))
+#define _ASM_UNSET_MTHC0 ".purgem mthc0\n\t"
 #else  /* !TOOLCHAIN_SUPPORTS_XPA */
-#define _ASM_SET_XPA ".set\txpa\n\t"
+#define _ASM_SET_MFHC0 ".set\txpa\n\t"
+#define _ASM_SET_MTHC0 ".set\txpa\n\t"
+#define _ASM_UNSET_MFHC0
+#define _ASM_UNSET_MTHC0
 #endif
 
 #define __readx_32bit_c0_register(source, sel)                         \
        __asm__ __volatile__(                                           \
        "       .set    push                                    \n"     \
        "       .set    mips32r2                                \n"     \
-       _ASM_SET_XPA                                                    \
+       _ASM_SET_MFHC0                                                  \
        "       mfhc0   %0, " #source ", %1                     \n"     \
+       _ASM_UNSET_MFHC0                                                \
        "       .set    pop                                     \n"     \
        : "=r" (__res)                                                  \
        : "i" (sel));                                                   \
        __asm__ __volatile__(                                           \
        "       .set    push                                    \n"     \
        "       .set    mips32r2                                \n"     \
-       _ASM_SET_XPA                                                    \
+       _ASM_SET_MTHC0                                                  \
        "       mthc0   %z0, " #register ", %1                  \n"     \
+       _ASM_UNSET_MTHC0                                                \
        "       .set    pop                                     \n"     \
        :                                                               \
        : "Jr" (value), "i" (sel));                                     \
  */
 
 #ifndef TOOLCHAIN_SUPPORTS_VIRT
-_ASM_MACRO_2R_1S(mfgc0, rt, rs, sel,
-       _ASM_INSN_IF_MIPS(0x40600000 | __rt << 16 | __rs << 11 | \\sel)
-       _ASM_INSN32_IF_MM(0x000004fc | __rt << 21 | __rs << 16 | \\sel << 11));
-_ASM_MACRO_2R_1S(dmfgc0, rt, rs, sel,
-       _ASM_INSN_IF_MIPS(0x40600100 | __rt << 16 | __rs << 11 | \\sel)
-       _ASM_INSN32_IF_MM(0x580004fc | __rt << 21 | __rs << 16 | \\sel << 11));
-_ASM_MACRO_2R_1S(mtgc0, rt, rd, sel,
-       _ASM_INSN_IF_MIPS(0x40600200 | __rt << 16 | __rd << 11 | \\sel)
-       _ASM_INSN32_IF_MM(0x000006fc | __rt << 21 | __rd << 16 | \\sel << 11));
-_ASM_MACRO_2R_1S(dmtgc0, rt, rd, sel,
-       _ASM_INSN_IF_MIPS(0x40600300 | __rt << 16 | __rd << 11 | \\sel)
-       _ASM_INSN32_IF_MM(0x580006fc | __rt << 21 | __rd << 16 | \\sel << 11));
-_ASM_MACRO_0(tlbgp,    _ASM_INSN_IF_MIPS(0x42000010)
-                      _ASM_INSN32_IF_MM(0x0000017c));
-_ASM_MACRO_0(tlbgr,    _ASM_INSN_IF_MIPS(0x42000009)
-                      _ASM_INSN32_IF_MM(0x0000117c));
-_ASM_MACRO_0(tlbgwi,   _ASM_INSN_IF_MIPS(0x4200000a)
-                      _ASM_INSN32_IF_MM(0x0000217c));
-_ASM_MACRO_0(tlbgwr,   _ASM_INSN_IF_MIPS(0x4200000e)
-                      _ASM_INSN32_IF_MM(0x0000317c));
-_ASM_MACRO_0(tlbginvf, _ASM_INSN_IF_MIPS(0x4200000c)
-                      _ASM_INSN32_IF_MM(0x0000517c));
-#define _ASM_SET_VIRT ""
+#define _ASM_SET_MFGC0                                                 \
+       _ASM_MACRO_2R_1S(mfgc0, rt, rs, sel,                            \
+                        _ASM_INSN_IF_MIPS(0x40600000 | __rt << 16 | __rs << 11 | \\sel)        \
+                        _ASM_INSN32_IF_MM(0x000004fc | __rt << 21 | __rs << 16 | \\sel << 11))
+#define _ASM_UNSET_MFGC0 ".purgem mfgc0\n\t"
+#define _ASM_SET_DMFGC0                                                        \
+       _ASM_MACRO_2R_1S(dmfgc0, rt, rs, sel,                           \
+                        _ASM_INSN_IF_MIPS(0x40600100 | __rt << 16 | __rs << 11 | \\sel)        \
+                        _ASM_INSN32_IF_MM(0x580004fc | __rt << 21 | __rs << 16 | \\sel << 11))
+#define _ASM_UNSET_DMFGC0 ".purgem dmfgc0\n\t"
+#define _ASM_SET_MTGC0                                                 \
+       _ASM_MACRO_2R_1S(mtgc0, rt, rd, sel,                            \
+                        _ASM_INSN_IF_MIPS(0x40600200 | __rt << 16 | __rd << 11 | \\sel)        \
+                        _ASM_INSN32_IF_MM(0x000006fc | __rt << 21 | __rd << 16 | \\sel << 11))
+#define _ASM_UNSET_MTGC0 ".purgem mtgc0\n\t"
+#define _ASM_SET_DMTGC0                                                        \
+       _ASM_MACRO_2R_1S(dmtgc0, rt, rd, sel,                           \
+                        _ASM_INSN_IF_MIPS(0x40600300 | __rt << 16 | __rd << 11 | \\sel)        \
+                        _ASM_INSN32_IF_MM(0x580006fc | __rt << 21 | __rd << 16 | \\sel << 11))
+#define _ASM_UNSET_DMTGC0 ".purgem dmtgc0\n\t"
+
+#define __tlbgp()                                                      \
+               _ASM_INSN_IF_MIPS(0x42000010)                           \
+               _ASM_INSN32_IF_MM(0x0000017c)
+#define __tlbgr()                                                      \
+               _ASM_INSN_IF_MIPS(0x42000009)                           \
+               _ASM_INSN32_IF_MM(0x0000117c)
+#define __tlbgwi()                                                     \
+               _ASM_INSN_IF_MIPS(0x4200000a)                           \
+               _ASM_INSN32_IF_MM(0x0000217c)
+#define __tlbgwr()                                                     \
+               _ASM_INSN_IF_MIPS(0x4200000e)                           \
+               _ASM_INSN32_IF_MM(0x0000317c)
+#define __tlbginvf()                                                   \
+               _ASM_INSN_IF_MIPS(0x4200000c)                           \
+               _ASM_INSN32_IF_MM(0x0000517c)
 #else  /* !TOOLCHAIN_SUPPORTS_VIRT */
 #define _ASM_SET_VIRT ".set\tvirt\n\t"
+#define _ASM_SET_MFGC0 _ASM_SET_VIRT
+#define _ASM_SET_DMFGC0        _ASM_SET_VIRT
+#define _ASM_SET_MTGC0 _ASM_SET_VIRT
+#define _ASM_SET_DMTGC0        _ASM_SET_VIRT
+#define _ASM_UNSET_MFGC0
+#define _ASM_UNSET_DMFGC0
+#define _ASM_UNSET_MTGC0
+#define _ASM_UNSET_DMTGC0
+
+#define __tlbgp()      _ASM_SET_VIRT "tlbgp\n\t"
+#define __tlbgr()      _ASM_SET_VIRT "tlbgr\n\t"
+#define __tlbgwi()     _ASM_SET_VIRT "tlbgwi\n\t"
+#define __tlbgwr()     _ASM_SET_VIRT "tlbgwr\n\t"
+#define __tlbginvf()   _ASM_SET_VIRT "tlbginvf\n\t"
 #endif
 
 #define __read_32bit_gc0_register(source, sel)                         \
        __asm__ __volatile__(                                           \
                ".set\tpush\n\t"                                        \
                ".set\tmips32r5\n\t"                                    \
-               _ASM_SET_VIRT                                           \
+               _ASM_SET_MFGC0                                          \
                "mfgc0\t%0, " #source ", %1\n\t"                        \
+               _ASM_UNSET_MFGC0                                        \
                ".set\tpop"                                             \
                : "=r" (__res)                                          \
                : "i" (sel));                                           \
        __asm__ __volatile__(                                           \
                ".set\tpush\n\t"                                        \
                ".set\tmips64r5\n\t"                                    \
-               _ASM_SET_VIRT                                           \
+               _ASM_SET_DMFGC0                                         \
                "dmfgc0\t%0, " #source ", %1\n\t"                       \
+               _ASM_UNSET_DMFGC0                                       \
                ".set\tpop"                                             \
                : "=r" (__res)                                          \
                : "i" (sel));                                           \
        __asm__ __volatile__(                                           \
                ".set\tpush\n\t"                                        \
                ".set\tmips32r5\n\t"                                    \
-               _ASM_SET_VIRT                                           \
+               _ASM_SET_MTGC0                                          \
                "mtgc0\t%z0, " #register ", %1\n\t"                     \
+               _ASM_UNSET_MTGC0                                        \
                ".set\tpop"                                             \
                : : "Jr" ((unsigned int)(value)),                       \
                    "i" (sel));                                         \
        __asm__ __volatile__(                                           \
                ".set\tpush\n\t"                                        \
                ".set\tmips64r5\n\t"                                    \
-               _ASM_SET_VIRT                                           \
+               _ASM_SET_DMTGC0                                         \
                "dmtgc0\t%z0, " #register ", %1\n\t"                    \
+               _ASM_UNSET_DMTGC0                                       \
                ".set\tpop"                                             \
                : : "Jr" (value),                                       \
                    "i" (sel));                                         \
        __asm__ __volatile__(
                ".set push\n\t"
                ".set noreorder\n\t"
-               _ASM_SET_VIRT
-               "tlbgp\n\t"
+               __tlbgp()
                ".set pop");
 }
 
        __asm__ __volatile__(
                ".set push\n\t"
                ".set noreorder\n\t"
-               _ASM_SET_VIRT
-               "tlbgr\n\t"
+               __tlbgr()
                ".set pop");
 }
 
        __asm__ __volatile__(
                ".set push\n\t"
                ".set noreorder\n\t"
-               _ASM_SET_VIRT
-               "tlbgwi\n\t"
+               __tlbgwi()
                ".set pop");
 }
 
        __asm__ __volatile__(
                ".set push\n\t"
                ".set noreorder\n\t"
-               _ASM_SET_VIRT
-               "tlbgwr\n\t"
+               __tlbgwr()
                ".set pop");
 }
 
        __asm__ __volatile__(
                ".set push\n\t"
                ".set noreorder\n\t"
-               _ASM_SET_VIRT
-               "tlbginvf\n\t"
+               __tlbginvf()
                ".set pop");
 }