Blame view

kernel/freezer.c 4.46 KB
8174f1503   Matt Helsley   container freezer...
1
2
3
4
5
6
7
8
  /*
   * kernel/freezer.c - Function to freeze a process
   *
   * Originally from kernel/power/process.c
   */
  
  #include <linux/interrupt.h>
  #include <linux/suspend.h>
9984de1a5   Paul Gortmaker   kernel: Map most ...
9
  #include <linux/export.h>
8174f1503   Matt Helsley   container freezer...
10
11
  #include <linux/syscalls.h>
  #include <linux/freezer.h>
8a32c441c   Tejun Heo   freezer: implemen...
12
  #include <linux/kthread.h>
8174f1503   Matt Helsley   container freezer...
13

a3201227f   Tejun Heo   freezer: make fre...
14
15
16
  /* total number of freezing conditions in effect */
  atomic_t system_freezing_cnt = ATOMIC_INIT(0);
  EXPORT_SYMBOL(system_freezing_cnt);
55f2503c3   Pingfan Liu   PM / reboot: Elim...
17
18
19
  /* indicate whether PM freezing is in effect, protected by
   * system_transition_mutex
   */
a3201227f   Tejun Heo   freezer: make fre...
20
21
  bool pm_freezing;
  bool pm_nosig_freezing;
85fbd722a   Tejun Heo   libata, freezer: ...
22
23
24
25
26
  /*
   * Temporary export for the deadlock workaround in ata_scsi_hotplug().
   * Remove once the hack becomes unnecessary.
   */
  EXPORT_SYMBOL_GPL(pm_freezing);
0c9af0926   Tejun Heo   freezer: use dedi...
27
28
  /* protects freezing and frozen transitions */
  static DEFINE_SPINLOCK(freezer_lock);
8174f1503   Matt Helsley   container freezer...
29

a3201227f   Tejun Heo   freezer: make fre...
30
31
32
33
34
35
36
37
38
39
40
  /**
   * freezing_slow_path - slow path for testing whether a task needs to be frozen
   * @p: task to be tested
   *
   * This function is called by freezing() if system_freezing_cnt isn't zero
   * and tests whether @p needs to enter and stay in frozen state.  Can be
   * called under any context.  The freezers are responsible for ensuring the
   * target tasks see the updated state.
   */
  bool freezing_slow_path(struct task_struct *p)
  {
2b44c4db2   Colin Cross   freezer: set PF_S...
41
  	if (p->flags & (PF_NOFREEZE | PF_SUSPEND_TASK))
a3201227f   Tejun Heo   freezer: make fre...
42
  		return false;
a34c80a72   Michal Hocko   freezer, oom: che...
43
  	if (test_tsk_thread_flag(p, TIF_MEMDIE))
51fae6da6   Cong Wang   freezer: Do not f...
44
  		return false;
a3201227f   Tejun Heo   freezer: make fre...
45
46
  	if (pm_nosig_freezing || cgroup_freezing(p))
  		return true;
34b087e48   Tejun Heo   freezer: kill unu...
47
  	if (pm_freezing && !(p->flags & PF_KTHREAD))
a3201227f   Tejun Heo   freezer: make fre...
48
49
50
51
52
  		return true;
  
  	return false;
  }
  EXPORT_SYMBOL(freezing_slow_path);
8174f1503   Matt Helsley   container freezer...
53
  /* Refrigerator is place where frozen processes are stored :-). */
8a32c441c   Tejun Heo   freezer: implemen...
54
  bool __refrigerator(bool check_kthr_stop)
8174f1503   Matt Helsley   container freezer...
55
56
57
  {
  	/* Hmm, should we be allowed to suspend when there are realtime
  	   processes around? */
a0acae0e8   Tejun Heo   freezer: unexport...
58
  	bool was_frozen = false;
5ece3eae4   Tejun Heo   freezer: restruct...
59
  	long save = current->state;
8174f1503   Matt Helsley   container freezer...
60

8174f1503   Matt Helsley   container freezer...
61
62
  	pr_debug("%s entered refrigerator
  ", current->comm);
8174f1503   Matt Helsley   container freezer...
63
64
  	for (;;) {
  		set_current_state(TASK_UNINTERRUPTIBLE);
5ece3eae4   Tejun Heo   freezer: restruct...
65
66
67
  
  		spin_lock_irq(&freezer_lock);
  		current->flags |= PF_FROZEN;
6907483b4   Tejun Heo   freezer: make fre...
68
  		if (!freezing(current) ||
8a32c441c   Tejun Heo   freezer: implemen...
69
  		    (check_kthr_stop && kthread_should_stop()))
5ece3eae4   Tejun Heo   freezer: restruct...
70
71
72
73
  			current->flags &= ~PF_FROZEN;
  		spin_unlock_irq(&freezer_lock);
  
  		if (!(current->flags & PF_FROZEN))
8174f1503   Matt Helsley   container freezer...
74
  			break;
a0acae0e8   Tejun Heo   freezer: unexport...
75
  		was_frozen = true;
8174f1503   Matt Helsley   container freezer...
76
77
  		schedule();
  	}
6301cb95c   Thomas Gleixner   sched: fix nr_uni...
78

8174f1503   Matt Helsley   container freezer...
79
80
  	pr_debug("%s left refrigerator
  ", current->comm);
50fb4f7fc   Tejun Heo   freezer: fix curr...
81
82
83
84
85
86
87
  
  	/*
  	 * Restore saved task state before returning.  The mb'd version
  	 * needs to be used; otherwise, it might silently break
  	 * synchronization which depends on ordered task state change.
  	 */
  	set_current_state(save);
a0acae0e8   Tejun Heo   freezer: unexport...
88
89
  
  	return was_frozen;
8174f1503   Matt Helsley   container freezer...
90
  }
a0acae0e8   Tejun Heo   freezer: unexport...
91
  EXPORT_SYMBOL(__refrigerator);
8174f1503   Matt Helsley   container freezer...
92
93
94
95
  
  static void fake_signal_wake_up(struct task_struct *p)
  {
  	unsigned long flags;
37ad8aca9   Tejun Heo   freezer: use lock...
96
97
98
99
  	if (lock_task_sighand(p, &flags)) {
  		signal_wake_up(p, 0);
  		unlock_task_sighand(p, &flags);
  	}
8174f1503   Matt Helsley   container freezer...
100
101
102
  }
  
  /**
839e3407d   Tejun Heo   freezer: remove u...
103
104
   * freeze_task - send a freeze request to given task
   * @p: task to send the request to
8174f1503   Matt Helsley   container freezer...
105
   *
37f08be11   Marcos Paulo de Souza   PM / Freezer: Rem...
106
107
108
   * If @p is freezing, the freeze request is sent either by sending a fake
   * signal (if it's not a kernel thread) or waking it up (if it's a kernel
   * thread).
839e3407d   Tejun Heo   freezer: remove u...
109
110
111
   *
   * RETURNS:
   * %false, if @p is not freezing or already frozen; %true, otherwise
8174f1503   Matt Helsley   container freezer...
112
   */
839e3407d   Tejun Heo   freezer: remove u...
113
  bool freeze_task(struct task_struct *p)
8174f1503   Matt Helsley   container freezer...
114
  {
0c9af0926   Tejun Heo   freezer: use dedi...
115
  	unsigned long flags;
0c9af0926   Tejun Heo   freezer: use dedi...
116

613f5d13b   Colin Cross   freezer: skip wak...
117
118
119
120
121
122
123
124
125
126
127
  	/*
  	 * This check can race with freezer_do_not_count, but worst case that
  	 * will result in an extra wakeup being sent to the task.  It does not
  	 * race with freezer_count(), the barriers in freezer_count() and
  	 * freezer_should_skip() ensure that either freezer_count() sees
  	 * freezing == true in try_to_freeze() and freezes, or
  	 * freezer_should_skip() sees !PF_FREEZE_SKIP and freezes the task
  	 * normally.
  	 */
  	if (freezer_should_skip(p))
  		return false;
0c9af0926   Tejun Heo   freezer: use dedi...
128
  	spin_lock_irqsave(&freezer_lock, flags);
a3201227f   Tejun Heo   freezer: make fre...
129
130
131
132
  	if (!freezing(p) || frozen(p)) {
  		spin_unlock_irqrestore(&freezer_lock, flags);
  		return false;
  	}
8174f1503   Matt Helsley   container freezer...
133

5d8f72b55   Oleg Nesterov   freezer: change p...
134
  	if (!(p->flags & PF_KTHREAD))
8cfe400ca   Tejun Heo   Freezer: Fix a ra...
135
  		fake_signal_wake_up(p);
5d8f72b55   Oleg Nesterov   freezer: change p...
136
  	else
8174f1503   Matt Helsley   container freezer...
137
  		wake_up_state(p, TASK_INTERRUPTIBLE);
a3201227f   Tejun Heo   freezer: make fre...
138

0c9af0926   Tejun Heo   freezer: use dedi...
139
  	spin_unlock_irqrestore(&freezer_lock, flags);
a3201227f   Tejun Heo   freezer: make fre...
140
  	return true;
8174f1503   Matt Helsley   container freezer...
141
  }
a5be2d0d1   Tejun Heo   freezer: rename t...
142
  void __thaw_task(struct task_struct *p)
dc52ddc0e   Matt Helsley   container freezer...
143
  {
0c9af0926   Tejun Heo   freezer: use dedi...
144
  	unsigned long flags;
a5be2d0d1   Tejun Heo   freezer: rename t...
145

0c9af0926   Tejun Heo   freezer: use dedi...
146
  	spin_lock_irqsave(&freezer_lock, flags);
34b087e48   Tejun Heo   freezer: kill unu...
147
  	if (frozen(p))
a5be2d0d1   Tejun Heo   freezer: rename t...
148
  		wake_up_process(p);
0c9af0926   Tejun Heo   freezer: use dedi...
149
  	spin_unlock_irqrestore(&freezer_lock, flags);
dc52ddc0e   Matt Helsley   container freezer...
150
  }
96ee6d853   Tejun Heo   freezer: fix set_...
151
152
  
  /**
34b087e48   Tejun Heo   freezer: kill unu...
153
   * set_freezable - make %current freezable
96ee6d853   Tejun Heo   freezer: fix set_...
154
155
156
   *
   * Mark %current freezable and enter refrigerator if necessary.
   */
34b087e48   Tejun Heo   freezer: kill unu...
157
  bool set_freezable(void)
96ee6d853   Tejun Heo   freezer: fix set_...
158
159
160
161
162
163
164
165
166
167
  {
  	might_sleep();
  
  	/*
  	 * Modify flags while holding freezer_lock.  This ensures the
  	 * freezer notices that we aren't frozen yet or the freezing
  	 * condition is visible to try_to_freeze() below.
  	 */
  	spin_lock_irq(&freezer_lock);
  	current->flags &= ~PF_NOFREEZE;
96ee6d853   Tejun Heo   freezer: fix set_...
168
169
170
171
  	spin_unlock_irq(&freezer_lock);
  
  	return try_to_freeze();
  }
34b087e48   Tejun Heo   freezer: kill unu...
172
  EXPORT_SYMBOL(set_freezable);