pm.c 5.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11
/*
 * linux/kernel/irq/pm.c
 *
 * Copyright (C) 2009 Rafael J. Wysocki <rjw@sisk.pl>, Novell Inc.
 *
 * This file contains power management functions related to interrupts.
 */

#include <linux/irq.h>
#include <linux/module.h>
#include <linux/interrupt.h>
12
#include <linux/suspend.h>
13
#include <linux/syscore_ops.h>
14 15 16

#include "internals.h"

17 18 19 20 21 22 23
bool irq_pm_check_wakeup(struct irq_desc *desc)
{
	if (irqd_is_wakeup_armed(&desc->irq_data)) {
		irqd_clear(&desc->irq_data, IRQD_WAKEUP_ARMED);
		desc->istate |= IRQS_SUSPENDED | IRQS_PENDING;
		desc->depth++;
		irq_disable(desc);
24
		pm_system_irq_wakeup(irq_desc_get_irq(desc));
25 26 27 28 29
		return true;
	}
	return false;
}

30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
/*
 * Called from __setup_irq() with desc->lock held after @action has
 * been installed in the action chain.
 */
void irq_pm_install_action(struct irq_desc *desc, struct irqaction *action)
{
	desc->nr_actions++;

	if (action->flags & IRQF_FORCE_RESUME)
		desc->force_resume_depth++;

	WARN_ON_ONCE(desc->force_resume_depth &&
		     desc->force_resume_depth != desc->nr_actions);

	if (action->flags & IRQF_NO_SUSPEND)
		desc->no_suspend_depth++;
46 47
	else if (action->flags & IRQF_COND_SUSPEND)
		desc->cond_suspend_depth++;
48 49

	WARN_ON_ONCE(desc->no_suspend_depth &&
50 51
		     (desc->no_suspend_depth +
			desc->cond_suspend_depth) != desc->nr_actions);
52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
}

/*
 * Called from __free_irq() with desc->lock held after @action has
 * been removed from the action chain.
 */
void irq_pm_remove_action(struct irq_desc *desc, struct irqaction *action)
{
	desc->nr_actions--;

	if (action->flags & IRQF_FORCE_RESUME)
		desc->force_resume_depth--;

	if (action->flags & IRQF_NO_SUSPEND)
		desc->no_suspend_depth--;
67 68
	else if (action->flags & IRQF_COND_SUSPEND)
		desc->cond_suspend_depth--;
69 70
}

71
static bool suspend_device_irq(struct irq_desc *desc)
72
{
73 74
	if (!desc->action || irq_desc_is_chained(desc) ||
	    desc->no_suspend_depth)
75
		return false;
76

77
	if (irqd_is_wakeup_set(&desc->irq_data)) {
78
		irqd_set(&desc->irq_data, IRQD_WAKEUP_ARMED);
79 80 81 82 83 84 85 86
		/*
		 * We return true here to force the caller to issue
		 * synchronize_irq(). We need to make sure that the
		 * IRQD_WAKEUP_ARMED is visible before we return from
		 * suspend_device_irqs().
		 */
		return true;
	}
87

88
	desc->istate |= IRQS_SUSPENDED;
89
	__disable_irq(desc);
90 91 92 93 94 95 96 97 98

	/*
	 * Hardware which has no wakeup source configuration facility
	 * requires that the non wakeup interrupts are masked at the
	 * chip level. The chip implementation indicates that with
	 * IRQCHIP_MASK_ON_SUSPEND.
	 */
	if (irq_desc_get_chip(desc)->flags & IRQCHIP_MASK_ON_SUSPEND)
		mask_irq(desc);
99
	return true;
100 101
}

102 103 104
/**
 * suspend_device_irqs - disable all currently enabled interrupt lines
 *
105 106 107 108 109
 * During system-wide suspend or hibernation device drivers need to be
 * prevented from receiving interrupts and this function is provided
 * for this purpose.
 *
 * So we disable all interrupts and mark them IRQS_SUSPENDED except
110
 * for those which are unused, those which are marked as not
111
 * suspendable via an interrupt request with the flag IRQF_NO_SUSPEND
112 113 114 115 116
 * set and those which are marked as active wakeup sources.
 *
 * The active wakeup sources are handled by the flow handler entry
 * code which checks for the IRQD_WAKEUP_ARMED flag, suspends the
 * interrupt and notifies the pm core about the wakeup.
117 118 119 120 121 122 123 124
 */
void suspend_device_irqs(void)
{
	struct irq_desc *desc;
	int irq;

	for_each_irq_desc(irq, desc) {
		unsigned long flags;
125
		bool sync;
126

127 128
		if (irq_settings_is_nested_thread(desc))
			continue;
129
		raw_spin_lock_irqsave(&desc->lock, flags);
130
		sync = suspend_device_irq(desc);
131
		raw_spin_unlock_irqrestore(&desc->lock, flags);
132

133
		if (sync)
134
			synchronize_irq(irq);
135
	}
136 137 138
}
EXPORT_SYMBOL_GPL(suspend_device_irqs);

139
static void resume_irq(struct irq_desc *desc)
140
{
141 142
	irqd_clear(&desc->irq_data, IRQD_WAKEUP_ARMED);

143 144 145
	if (desc->istate & IRQS_SUSPENDED)
		goto resume;

146 147
	/* Force resume the interrupt? */
	if (!desc->force_resume_depth)
148 149 150 151 152 153
		return;

	/* Pretend that it got disabled ! */
	desc->depth++;
resume:
	desc->istate &= ~IRQS_SUSPENDED;
154
	__enable_irq(desc);
155 156
}

157
static void resume_irqs(bool want_early)
158 159 160 161 162 163
{
	struct irq_desc *desc;
	int irq;

	for_each_irq_desc(irq, desc) {
		unsigned long flags;
164 165 166
		bool is_early = desc->action &&
			desc->action->flags & IRQF_EARLY_RESUME;

167
		if (!is_early && want_early)
168
			continue;
169 170
		if (irq_settings_is_nested_thread(desc))
			continue;
171

172
		raw_spin_lock_irqsave(&desc->lock, flags);
173
		resume_irq(desc);
174
		raw_spin_unlock_irqrestore(&desc->lock, flags);
175 176
	}
}
177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210

/**
 * irq_pm_syscore_ops - enable interrupt lines early
 *
 * Enable all interrupt lines with %IRQF_EARLY_RESUME set.
 */
static void irq_pm_syscore_resume(void)
{
	resume_irqs(true);
}

static struct syscore_ops irq_pm_syscore_ops = {
	.resume		= irq_pm_syscore_resume,
};

static int __init irq_pm_init_ops(void)
{
	register_syscore_ops(&irq_pm_syscore_ops);
	return 0;
}

device_initcall(irq_pm_init_ops);

/**
 * resume_device_irqs - enable interrupt lines disabled by suspend_device_irqs()
 *
 * Enable all non-%IRQF_EARLY_RESUME interrupt lines previously
 * disabled by suspend_device_irqs() that have the IRQS_SUSPENDED flag
 * set as well as those with %IRQF_FORCE_RESUME.
 */
void resume_device_irqs(void)
{
	resume_irqs(false);
}
211
EXPORT_SYMBOL_GPL(resume_device_irqs);