Blame view

kernel/freezer.c 4.22 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
17
18
19
20
  /* total number of freezing conditions in effect */
  atomic_t system_freezing_cnt = ATOMIC_INIT(0);
  EXPORT_SYMBOL(system_freezing_cnt);
  
  /* indicate whether PM freezing is in effect, protected by pm_mutex */
  bool pm_freezing;
  bool pm_nosig_freezing;
0c9af0926   Tejun Heo   freezer: use dedi...
21
22
  /* protects freezing and frozen transitions */
  static DEFINE_SPINLOCK(freezer_lock);
8174f1503   Matt Helsley   container freezer...
23

a3201227f   Tejun Heo   freezer: make fre...
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
  /**
   * 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)
  {
  	if (p->flags & PF_NOFREEZE)
  		return false;
  
  	if (pm_nosig_freezing || cgroup_freezing(p))
  		return true;
34b087e48   Tejun Heo   freezer: kill unu...
40
  	if (pm_freezing && !(p->flags & PF_KTHREAD))
a3201227f   Tejun Heo   freezer: make fre...
41
42
43
44
45
  		return true;
  
  	return false;
  }
  EXPORT_SYMBOL(freezing_slow_path);
8174f1503   Matt Helsley   container freezer...
46
  /* Refrigerator is place where frozen processes are stored :-). */
8a32c441c   Tejun Heo   freezer: implemen...
47
  bool __refrigerator(bool check_kthr_stop)
8174f1503   Matt Helsley   container freezer...
48
49
50
  {
  	/* Hmm, should we be allowed to suspend when there are realtime
  	   processes around? */
a0acae0e8   Tejun Heo   freezer: unexport...
51
  	bool was_frozen = false;
5ece3eae4   Tejun Heo   freezer: restruct...
52
  	long save = current->state;
8174f1503   Matt Helsley   container freezer...
53

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

8174f1503   Matt Helsley   container freezer...
72
73
  	pr_debug("%s left refrigerator
  ", current->comm);
50fb4f7fc   Tejun Heo   freezer: fix curr...
74
75
76
77
78
79
80
  
  	/*
  	 * 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...
81
82
  
  	return was_frozen;
8174f1503   Matt Helsley   container freezer...
83
  }
a0acae0e8   Tejun Heo   freezer: unexport...
84
  EXPORT_SYMBOL(__refrigerator);
8174f1503   Matt Helsley   container freezer...
85
86
87
88
  
  static void fake_signal_wake_up(struct task_struct *p)
  {
  	unsigned long flags;
37ad8aca9   Tejun Heo   freezer: use lock...
89
90
91
92
  	if (lock_task_sighand(p, &flags)) {
  		signal_wake_up(p, 0);
  		unlock_task_sighand(p, &flags);
  	}
8174f1503   Matt Helsley   container freezer...
93
94
95
  }
  
  /**
839e3407d   Tejun Heo   freezer: remove u...
96
97
   * freeze_task - send a freeze request to given task
   * @p: task to send the request to
8174f1503   Matt Helsley   container freezer...
98
   *
839e3407d   Tejun Heo   freezer: remove u...
99
100
101
102
103
104
   * If @p is freezing, the freeze request is sent by setting %TIF_FREEZE
   * flag and either sending a fake signal to it or waking it up, depending
   * on whether it has %PF_FREEZER_NOSIG set.
   *
   * RETURNS:
   * %false, if @p is not freezing or already frozen; %true, otherwise
8174f1503   Matt Helsley   container freezer...
105
   */
839e3407d   Tejun Heo   freezer: remove u...
106
  bool freeze_task(struct task_struct *p)
8174f1503   Matt Helsley   container freezer...
107
  {
0c9af0926   Tejun Heo   freezer: use dedi...
108
  	unsigned long flags;
0c9af0926   Tejun Heo   freezer: use dedi...
109
110
  
  	spin_lock_irqsave(&freezer_lock, flags);
a3201227f   Tejun Heo   freezer: make fre...
111
112
113
114
  	if (!freezing(p) || frozen(p)) {
  		spin_unlock_irqrestore(&freezer_lock, flags);
  		return false;
  	}
8174f1503   Matt Helsley   container freezer...
115

34b087e48   Tejun Heo   freezer: kill unu...
116
  	if (!(p->flags & PF_KTHREAD)) {
8cfe400ca   Tejun Heo   Freezer: Fix a ra...
117
118
119
120
121
122
123
  		fake_signal_wake_up(p);
  		/*
  		 * fake_signal_wake_up() goes through p's scheduler
  		 * lock and guarantees that TASK_STOPPED/TRACED ->
  		 * TASK_RUNNING transition can't race with task state
  		 * testing in try_to_freeze_tasks().
  		 */
8174f1503   Matt Helsley   container freezer...
124
125
126
  	} else {
  		wake_up_state(p, TASK_INTERRUPTIBLE);
  	}
a3201227f   Tejun Heo   freezer: make fre...
127

0c9af0926   Tejun Heo   freezer: use dedi...
128
  	spin_unlock_irqrestore(&freezer_lock, flags);
a3201227f   Tejun Heo   freezer: make fre...
129
  	return true;
8174f1503   Matt Helsley   container freezer...
130
  }
a5be2d0d1   Tejun Heo   freezer: rename t...
131
  void __thaw_task(struct task_struct *p)
dc52ddc0e   Matt Helsley   container freezer...
132
  {
0c9af0926   Tejun Heo   freezer: use dedi...
133
  	unsigned long flags;
a5be2d0d1   Tejun Heo   freezer: rename t...
134

6907483b4   Tejun Heo   freezer: make fre...
135
136
137
138
139
140
  	/*
  	 * Clear freezing and kick @p if FROZEN.  Clearing is guaranteed to
  	 * be visible to @p as waking up implies wmb.  Waking up inside
  	 * freezer_lock also prevents wakeups from leaking outside
  	 * refrigerator.
  	 */
0c9af0926   Tejun Heo   freezer: use dedi...
141
  	spin_lock_irqsave(&freezer_lock, flags);
34b087e48   Tejun Heo   freezer: kill unu...
142
  	if (frozen(p))
a5be2d0d1   Tejun Heo   freezer: rename t...
143
  		wake_up_process(p);
0c9af0926   Tejun Heo   freezer: use dedi...
144
  	spin_unlock_irqrestore(&freezer_lock, flags);
dc52ddc0e   Matt Helsley   container freezer...
145
  }
96ee6d853   Tejun Heo   freezer: fix set_...
146
147
  
  /**
34b087e48   Tejun Heo   freezer: kill unu...
148
   * set_freezable - make %current freezable
96ee6d853   Tejun Heo   freezer: fix set_...
149
150
151
   *
   * Mark %current freezable and enter refrigerator if necessary.
   */
34b087e48   Tejun Heo   freezer: kill unu...
152
  bool set_freezable(void)
96ee6d853   Tejun Heo   freezer: fix set_...
153
154
155
156
157
158
159
160
161
162
  {
  	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_...
163
164
165
166
  	spin_unlock_irq(&freezer_lock);
  
  	return try_to_freeze();
  }
34b087e48   Tejun Heo   freezer: kill unu...
167
  EXPORT_SYMBOL(set_freezable);