interrupt.h 5.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * This file is subject to the terms and conditions of the GNU General Public
 * License.  See the file "COPYING" in the main directory of this archive
 * for more details.
 *
 * Copyright (C) 1994, 95, 96, 97, 98, 99, 2003 by Ralf Baechle
 * Copyright (C) 1996 by Paul M. Antoine
 * Copyright (C) 1999 Silicon Graphics
 * Copyright (C) 2000 MIPS Technologies, Inc.
 */
#ifndef _ASM_INTERRUPT_H
#define _ASM_INTERRUPT_H

14
#include <linux/config.h>
L
Linus Torvalds 已提交
15 16 17
#include <asm/hazards.h>

__asm__ (
18 19 20 21
	"	.macro	local_irq_enable				\n"
	"	.set	push						\n"
	"	.set	reorder						\n"
	"	.set	noat						\n"
22 23 24 25 26 27
#ifdef CONFIG_MIPS_MT_SMTC
	"	mfc0	$1, $2, 1	# SMTC - clear TCStatus.IXMT	\n"
	"	ori	$1, 0x400					\n"
	"	xori	$1, 0x400					\n"
	"	mtc0	$1, $2, 1					\n"
#elif defined(CONFIG_CPU_MIPSR2)
28 29 30 31 32 33 34 35 36 37
	"	ei							\n"
#else
	"	mfc0	$1,$12						\n"
	"	ori	$1,0x1f						\n"
	"	xori	$1,0x1e						\n"
	"	mtc0	$1,$12						\n"
#endif
	"	irq_enable_hazard					\n"
	"	.set	pop						\n"
	"	.endm");
L
Linus Torvalds 已提交
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54

static inline void local_irq_enable(void)
{
	__asm__ __volatile__(
		"local_irq_enable"
		: /* no outputs */
		: /* no inputs */
		: "memory");
}

/*
 * For cli() we have to insert nops to make sure that the new value
 * has actually arrived in the status register before the end of this
 * macro.
 * R4000/R4400 need three nops, the R4600 two nops and the R10000 needs
 * no nops at all.
 */
A
Atsushi Nemoto 已提交
55 56 57 58 59 60 61 62 63 64 65
/*
 * For TX49, operating only IE bit is not enough.
 *
 * If mfc0 $12 follows store and the mfc0 is last instruction of a
 * page and fetching the next instruction causes TLB miss, the result
 * of the mfc0 might wrongly contain EXL bit.
 *
 * ERT-TX49H2-027, ERT-TX49H3-012, ERT-TX49HL3-006, ERT-TX49H4-008
 *
 * Workaround: mask EXL bit of the result or place a nop before mfc0.
 */
L
Linus Torvalds 已提交
66
__asm__ (
67 68 69
	"	.macro	local_irq_disable\n"
	"	.set	push						\n"
	"	.set	noat						\n"
70 71 72 73 74 75
#ifdef CONFIG_MIPS_MT_SMTC
	"	mfc0	$1, $2, 1					\n"
	"	ori	$1, 0x400					\n"
	"	.set	noreorder					\n"
	"	mtc0	$1, $2, 1					\n"
#elif defined(CONFIG_CPU_MIPSR2)
76 77 78
	"	di							\n"
#else
	"	mfc0	$1,$12						\n"
A
Atsushi Nemoto 已提交
79 80
	"	ori	$1,0x1f						\n"
	"	xori	$1,0x1f						\n"
81 82 83 84 85 86
	"	.set	noreorder					\n"
	"	mtc0	$1,$12						\n"
#endif
	"	irq_disable_hazard					\n"
	"	.set	pop						\n"
	"	.endm							\n");
L
Linus Torvalds 已提交
87 88 89 90 91 92 93 94 95 96 97

static inline void local_irq_disable(void)
{
	__asm__ __volatile__(
		"local_irq_disable"
		: /* no outputs */
		: /* no inputs */
		: "memory");
}

__asm__ (
98 99 100
	"	.macro	local_save_flags flags				\n"
	"	.set	push						\n"
	"	.set	reorder						\n"
101 102 103
#ifdef CONFIG_MIPS_MT_SMTC
	"	mfc0	\\flags, $2, 1					\n"
#else
104
	"	mfc0	\\flags, $12					\n"
105
#endif
106 107
	"	.set	pop						\n"
	"	.endm							\n");
L
Linus Torvalds 已提交
108 109 110 111 112 113 114

#define local_save_flags(x)						\
__asm__ __volatile__(							\
	"local_save_flags %0"						\
	: "=r" (x))

__asm__ (
115 116 117 118
	"	.macro	local_irq_save result				\n"
	"	.set	push						\n"
	"	.set	reorder						\n"
	"	.set	noat						\n"
119 120 121 122 123 124 125
#ifdef CONFIG_MIPS_MT_SMTC
	"	mfc0	\\result, $2, 1					\n"
	"	ori	$1, \\result, 0x400				\n"
	"	.set	noreorder					\n"
	"	mtc0	$1, $2, 1					\n"
	"	andi	\\result, \\result, 0x400			\n"
#elif defined(CONFIG_CPU_MIPSR2)
126
	"	di	\\result					\n"
M
Maxime Bizon 已提交
127
	"	andi	\\result, 1					\n"
128 129
#else
	"	mfc0	\\result, $12					\n"
A
Atsushi Nemoto 已提交
130 131
	"	ori	$1, \\result, 0x1f				\n"
	"	xori	$1, 0x1f					\n"
132 133 134 135 136 137
	"	.set	noreorder					\n"
	"	mtc0	$1, $12						\n"
#endif
	"	irq_disable_hazard					\n"
	"	.set	pop						\n"
	"	.endm							\n");
L
Linus Torvalds 已提交
138 139 140 141 142 143 144 145 146

#define local_irq_save(x)						\
__asm__ __volatile__(							\
	"local_irq_save\t%0"						\
	: "=r" (x)							\
	: /* no inputs */						\
	: "memory")

__asm__ (
147
	"	.macro	local_irq_restore flags				\n"
148
	"	.set	push						\n"
149 150
	"	.set	noreorder					\n"
	"	.set	noat						\n"
151 152 153 154 155 156 157 158
#ifdef CONFIG_MIPS_MT_SMTC
	"mfc0	$1, $2, 1						\n"
	"andi	\\flags, 0x400						\n"
	"ori	$1, 0x400						\n"
	"xori	$1, 0x400						\n"
	"or	\\flags, $1						\n"
	"mtc0	\\flags, $2, 1						\n"
#elif defined(CONFIG_CPU_MIPSR2) && defined(CONFIG_IRQ_CPU)
159 160 161 162 163 164 165 166
	/*
	 * Slow, but doesn't suffer from a relativly unlikely race
	 * condition we're having since days 1.
	 */
	"	beqz	\\flags, 1f					\n"
	"	 di							\n"
	"	ei							\n"
	"1:								\n"
167
#elif defined(CONFIG_CPU_MIPSR2)
168 169 170 171 172 173 174 175 176
	/*
	 * Fast, dangerous.  Life is fun, life is good.
	 */
	"	mfc0	$1, $12						\n"
	"	ins	$1, \\flags, 0, 1				\n"
	"	mtc0	$1, $12						\n"
#else
	"	mfc0	$1, $12						\n"
	"	andi	\\flags, 1					\n"
A
Atsushi Nemoto 已提交
177 178
	"	ori	$1, 0x1f					\n"
	"	xori	$1, 0x1f					\n"
179 180 181 182
	"	or	\\flags, $1					\n"
	"	mtc0	\\flags, $12					\n"
#endif
	"	irq_disable_hazard					\n"
183
	"	.set	pop						\n"
184
	"	.endm							\n");
L
Linus Torvalds 已提交
185 186 187 188 189 190 191 192 193 194 195 196

#define local_irq_restore(flags)					\
do {									\
	unsigned long __tmp1;						\
									\
	__asm__ __volatile__(						\
		"local_irq_restore\t%0"					\
		: "=r" (__tmp1)						\
		: "0" (flags)						\
		: "memory");						\
} while(0)

197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220
static inline int irqs_disabled(void)
{
#ifdef CONFIG_MIPS_MT_SMTC
	/*
	 * SMTC model uses TCStatus.IXMT to disable interrupts for a thread/CPU
	 */
	unsigned long __result;

	__asm__ __volatile__(
	"	.set	noreorder					\n"
	"	mfc0	%0, $2, 1					\n"
	"	andi	%0, 0x400					\n"
	"	slt	%0, $0, %0					\n"
	"	.set	reorder						\n"
	: "=r" (__result));

	return __result;
#else
	unsigned long flags;
	local_save_flags(flags);

	return !(flags & 1);
#endif
}
L
Linus Torvalds 已提交
221 222

#endif /* _ASM_INTERRUPT_H */