blob: c29ed571ae49e92c6424da7f4d89f1ab10254f94 [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001/* SPDX-License-Identifier: GPL-2.0 */
Sam Ravnborga439fe52008-07-27 23:00:59 +02002/*
3 * include/asm/irqflags.h
4 *
5 * IRQ flags handling
6 *
7 * This file gets included from lowlevel asm headers too, to provide
8 * wrapped versions of the local_irq_*() APIs, based on the
David Howellsdf9ee292010-10-07 14:08:55 +01009 * arch_local_irq_*() functions from the lowlevel headers.
Sam Ravnborga439fe52008-07-27 23:00:59 +020010 */
11#ifndef _ASM_IRQFLAGS_H
12#define _ASM_IRQFLAGS_H
13
David S. Millerb4f43722008-11-23 21:55:29 -080014#include <asm/pil.h>
15
Sam Ravnborga439fe52008-07-27 23:00:59 +020016#ifndef __ASSEMBLY__
17
Steven Rostedt18d85bc2011-07-06 08:00:29 -070018static inline notrace unsigned long arch_local_save_flags(void)
Sam Ravnborga439fe52008-07-27 23:00:59 +020019{
20 unsigned long flags;
21
22 __asm__ __volatile__(
23 "rdpr %%pil, %0"
24 : "=r" (flags)
25 );
26
27 return flags;
28}
29
Steven Rostedt18d85bc2011-07-06 08:00:29 -070030static inline notrace void arch_local_irq_restore(unsigned long flags)
Sam Ravnborga439fe52008-07-27 23:00:59 +020031{
32 __asm__ __volatile__(
33 "wrpr %0, %%pil"
34 : /* no output */
35 : "r" (flags)
36 : "memory"
37 );
38}
39
Steven Rostedt18d85bc2011-07-06 08:00:29 -070040static inline notrace void arch_local_irq_disable(void)
Sam Ravnborga439fe52008-07-27 23:00:59 +020041{
42 __asm__ __volatile__(
David S. Millerb4f43722008-11-23 21:55:29 -080043 "wrpr %0, %%pil"
Sam Ravnborga439fe52008-07-27 23:00:59 +020044 : /* no outputs */
David S. Millerb4f43722008-11-23 21:55:29 -080045 : "i" (PIL_NORMAL_MAX)
Sam Ravnborga439fe52008-07-27 23:00:59 +020046 : "memory"
47 );
48}
49
Steven Rostedt18d85bc2011-07-06 08:00:29 -070050static inline notrace void arch_local_irq_enable(void)
Sam Ravnborga439fe52008-07-27 23:00:59 +020051{
52 __asm__ __volatile__(
53 "wrpr 0, %%pil"
54 : /* no outputs */
55 : /* no inputs */
56 : "memory"
57 );
58}
59
Steven Rostedt18d85bc2011-07-06 08:00:29 -070060static inline notrace int arch_irqs_disabled_flags(unsigned long flags)
Sam Ravnborga439fe52008-07-27 23:00:59 +020061{
62 return (flags > 0);
63}
64
Steven Rostedt18d85bc2011-07-06 08:00:29 -070065static inline notrace int arch_irqs_disabled(void)
Sam Ravnborga439fe52008-07-27 23:00:59 +020066{
David Howellsdf9ee292010-10-07 14:08:55 +010067 return arch_irqs_disabled_flags(arch_local_save_flags());
Sam Ravnborga439fe52008-07-27 23:00:59 +020068}
69
Steven Rostedt18d85bc2011-07-06 08:00:29 -070070static inline notrace unsigned long arch_local_irq_save(void)
Sam Ravnborga439fe52008-07-27 23:00:59 +020071{
David S. Miller0c25e9e2010-04-12 22:21:52 -070072 unsigned long flags, tmp;
Sam Ravnborga439fe52008-07-27 23:00:59 +020073
David S. Miller0c25e9e2010-04-12 22:21:52 -070074 /* Disable interrupts to PIL_NORMAL_MAX unless we already
75 * are using PIL_NMI, in which case PIL_NMI is retained.
David S. Millerc011f802010-04-13 01:50:43 -070076 *
77 * The only values we ever program into the %pil are 0,
78 * PIL_NORMAL_MAX and PIL_NMI.
79 *
80 * Since PIL_NMI is the largest %pil value and all bits are
81 * set in it (0xf), it doesn't matter what PIL_NORMAL_MAX
82 * actually is.
David S. Miller0c25e9e2010-04-12 22:21:52 -070083 */
84 __asm__ __volatile__(
85 "rdpr %%pil, %0\n\t"
86 "or %0, %2, %1\n\t"
87 "wrpr %1, 0x0, %%pil"
88 : "=r" (flags), "=r" (tmp)
89 : "i" (PIL_NORMAL_MAX)
90 : "memory"
91 );
Sam Ravnborga439fe52008-07-27 23:00:59 +020092
93 return flags;
94}
95
Sam Ravnborga439fe52008-07-27 23:00:59 +020096#endif /* (__ASSEMBLY__) */
97
98#endif /* !(_ASM_IRQFLAGS_H) */