xref: /openbmc/linux/kernel/freezer.c (revision 6301cb95c119ebf324bb96ee226fa9ddffad80a7)
18174f150SMatt Helsley /*
28174f150SMatt Helsley  * kernel/freezer.c - Function to freeze a process
38174f150SMatt Helsley  *
48174f150SMatt Helsley  * Originally from kernel/power/process.c
58174f150SMatt Helsley  */
68174f150SMatt Helsley 
78174f150SMatt Helsley #include <linux/interrupt.h>
88174f150SMatt Helsley #include <linux/suspend.h>
98174f150SMatt Helsley #include <linux/module.h>
108174f150SMatt Helsley #include <linux/syscalls.h>
118174f150SMatt Helsley #include <linux/freezer.h>
128174f150SMatt Helsley 
138174f150SMatt Helsley /*
148174f150SMatt Helsley  * freezing is complete, mark current process as frozen
158174f150SMatt Helsley  */
168174f150SMatt Helsley static inline void frozen_process(void)
178174f150SMatt Helsley {
188174f150SMatt Helsley 	if (!unlikely(current->flags & PF_NOFREEZE)) {
198174f150SMatt Helsley 		current->flags |= PF_FROZEN;
208174f150SMatt Helsley 		wmb();
218174f150SMatt Helsley 	}
228174f150SMatt Helsley 	clear_freeze_flag(current);
238174f150SMatt Helsley }
248174f150SMatt Helsley 
258174f150SMatt Helsley /* Refrigerator is place where frozen processes are stored :-). */
268174f150SMatt Helsley void refrigerator(void)
278174f150SMatt Helsley {
288174f150SMatt Helsley 	/* Hmm, should we be allowed to suspend when there are realtime
298174f150SMatt Helsley 	   processes around? */
308174f150SMatt Helsley 	long save;
318174f150SMatt Helsley 
328174f150SMatt Helsley 	task_lock(current);
338174f150SMatt Helsley 	if (freezing(current)) {
348174f150SMatt Helsley 		frozen_process();
358174f150SMatt Helsley 		task_unlock(current);
368174f150SMatt Helsley 	} else {
378174f150SMatt Helsley 		task_unlock(current);
388174f150SMatt Helsley 		return;
398174f150SMatt Helsley 	}
408174f150SMatt Helsley 	save = current->state;
418174f150SMatt Helsley 	pr_debug("%s entered refrigerator\n", current->comm);
428174f150SMatt Helsley 
438174f150SMatt Helsley 	spin_lock_irq(&current->sighand->siglock);
448174f150SMatt Helsley 	recalc_sigpending(); /* We sent fake signal, clean it up */
458174f150SMatt Helsley 	spin_unlock_irq(&current->sighand->siglock);
468174f150SMatt Helsley 
47*6301cb95SThomas Gleixner 	/* prevent accounting of that task to load */
48*6301cb95SThomas Gleixner 	current->flags |= PF_FREEZING;
49*6301cb95SThomas Gleixner 
508174f150SMatt Helsley 	for (;;) {
518174f150SMatt Helsley 		set_current_state(TASK_UNINTERRUPTIBLE);
528174f150SMatt Helsley 		if (!frozen(current))
538174f150SMatt Helsley 			break;
548174f150SMatt Helsley 		schedule();
558174f150SMatt Helsley 	}
56*6301cb95SThomas Gleixner 
57*6301cb95SThomas Gleixner 	/* Remove the accounting blocker */
58*6301cb95SThomas Gleixner 	current->flags &= ~PF_FREEZING;
59*6301cb95SThomas Gleixner 
608174f150SMatt Helsley 	pr_debug("%s left refrigerator\n", current->comm);
618174f150SMatt Helsley 	__set_current_state(save);
628174f150SMatt Helsley }
638174f150SMatt Helsley EXPORT_SYMBOL(refrigerator);
648174f150SMatt Helsley 
658174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p)
668174f150SMatt Helsley {
678174f150SMatt Helsley 	unsigned long flags;
688174f150SMatt Helsley 
698174f150SMatt Helsley 	spin_lock_irqsave(&p->sighand->siglock, flags);
708174f150SMatt Helsley 	signal_wake_up(p, 0);
718174f150SMatt Helsley 	spin_unlock_irqrestore(&p->sighand->siglock, flags);
728174f150SMatt Helsley }
738174f150SMatt Helsley 
748174f150SMatt Helsley /**
758174f150SMatt Helsley  *	freeze_task - send a freeze request to given task
768174f150SMatt Helsley  *	@p: task to send the request to
778174f150SMatt Helsley  *	@sig_only: if set, the request will only be sent if the task has the
788174f150SMatt Helsley  *		PF_FREEZER_NOSIG flag unset
798174f150SMatt Helsley  *	Return value: 'false', if @sig_only is set and the task has
808174f150SMatt Helsley  *		PF_FREEZER_NOSIG set or the task is frozen, 'true', otherwise
818174f150SMatt Helsley  *
828174f150SMatt Helsley  *	The freeze request is sent by setting the tasks's TIF_FREEZE flag and
838174f150SMatt Helsley  *	either sending a fake signal to it or waking it up, depending on whether
848174f150SMatt Helsley  *	or not it has PF_FREEZER_NOSIG set.  If @sig_only is set and the task
858174f150SMatt Helsley  *	has PF_FREEZER_NOSIG set (ie. it is a typical kernel thread), its
868174f150SMatt Helsley  *	TIF_FREEZE flag will not be set.
878174f150SMatt Helsley  */
888174f150SMatt Helsley bool freeze_task(struct task_struct *p, bool sig_only)
898174f150SMatt Helsley {
908174f150SMatt Helsley 	/*
918174f150SMatt Helsley 	 * We first check if the task is freezing and next if it has already
928174f150SMatt Helsley 	 * been frozen to avoid the race with frozen_process() which first marks
938174f150SMatt Helsley 	 * the task as frozen and next clears its TIF_FREEZE.
948174f150SMatt Helsley 	 */
958174f150SMatt Helsley 	if (!freezing(p)) {
968174f150SMatt Helsley 		rmb();
978174f150SMatt Helsley 		if (frozen(p))
988174f150SMatt Helsley 			return false;
998174f150SMatt Helsley 
1008174f150SMatt Helsley 		if (!sig_only || should_send_signal(p))
1018174f150SMatt Helsley 			set_freeze_flag(p);
1028174f150SMatt Helsley 		else
1038174f150SMatt Helsley 			return false;
1048174f150SMatt Helsley 	}
1058174f150SMatt Helsley 
1068174f150SMatt Helsley 	if (should_send_signal(p)) {
1078174f150SMatt Helsley 		if (!signal_pending(p))
1088174f150SMatt Helsley 			fake_signal_wake_up(p);
1098174f150SMatt Helsley 	} else if (sig_only) {
1108174f150SMatt Helsley 		return false;
1118174f150SMatt Helsley 	} else {
1128174f150SMatt Helsley 		wake_up_state(p, TASK_INTERRUPTIBLE);
1138174f150SMatt Helsley 	}
1148174f150SMatt Helsley 
1158174f150SMatt Helsley 	return true;
1168174f150SMatt Helsley }
1178174f150SMatt Helsley 
1188174f150SMatt Helsley void cancel_freezing(struct task_struct *p)
1198174f150SMatt Helsley {
1208174f150SMatt Helsley 	unsigned long flags;
1218174f150SMatt Helsley 
1228174f150SMatt Helsley 	if (freezing(p)) {
1238174f150SMatt Helsley 		pr_debug("  clean up: %s\n", p->comm);
1248174f150SMatt Helsley 		clear_freeze_flag(p);
1258174f150SMatt Helsley 		spin_lock_irqsave(&p->sighand->siglock, flags);
1268174f150SMatt Helsley 		recalc_sigpending_and_wake(p);
1278174f150SMatt Helsley 		spin_unlock_irqrestore(&p->sighand->siglock, flags);
1288174f150SMatt Helsley 	}
1298174f150SMatt Helsley }
130dc52ddc0SMatt Helsley 
13100c2e63cSLi Zefan static int __thaw_process(struct task_struct *p)
132dc52ddc0SMatt Helsley {
133dc52ddc0SMatt Helsley 	if (frozen(p)) {
134dc52ddc0SMatt Helsley 		p->flags &= ~PF_FROZEN;
135dc52ddc0SMatt Helsley 		return 1;
136dc52ddc0SMatt Helsley 	}
137dc52ddc0SMatt Helsley 	clear_freeze_flag(p);
138dc52ddc0SMatt Helsley 	return 0;
139dc52ddc0SMatt Helsley }
140dc52ddc0SMatt Helsley 
14100c2e63cSLi Zefan /*
14200c2e63cSLi Zefan  * Wake up a frozen process
14300c2e63cSLi Zefan  *
14400c2e63cSLi Zefan  * task_lock() is needed to prevent the race with refrigerator() which may
14500c2e63cSLi Zefan  * occur if the freezing of tasks fails.  Namely, without the lock, if the
14600c2e63cSLi Zefan  * freezing of tasks failed, thaw_tasks() might have run before a task in
14700c2e63cSLi Zefan  * refrigerator() could call frozen_process(), in which case the task would be
14800c2e63cSLi Zefan  * frozen and no one would thaw it.
14900c2e63cSLi Zefan  */
150dc52ddc0SMatt Helsley int thaw_process(struct task_struct *p)
151dc52ddc0SMatt Helsley {
152dc52ddc0SMatt Helsley 	task_lock(p);
153dc52ddc0SMatt Helsley 	if (__thaw_process(p) == 1) {
154dc52ddc0SMatt Helsley 		task_unlock(p);
155dc52ddc0SMatt Helsley 		wake_up_process(p);
156dc52ddc0SMatt Helsley 		return 1;
157dc52ddc0SMatt Helsley 	}
158dc52ddc0SMatt Helsley 	task_unlock(p);
159dc52ddc0SMatt Helsley 	return 0;
160dc52ddc0SMatt Helsley }
161dc52ddc0SMatt Helsley EXPORT_SYMBOL(thaw_process);
162