xref: /openbmc/linux/kernel/freezer.c (revision 50fb4f7fc907efff65eadb0b74387a9ffed6e849)
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>
99984de1aSPaul Gortmaker #include <linux/export.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;
20ee940d8dSMike Frysinger 		smp_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 
476301cb95SThomas Gleixner 	/* prevent accounting of that task to load */
486301cb95SThomas Gleixner 	current->flags |= PF_FREEZING;
496301cb95SThomas Gleixner 
508174f150SMatt Helsley 	for (;;) {
518174f150SMatt Helsley 		set_current_state(TASK_UNINTERRUPTIBLE);
528174f150SMatt Helsley 		if (!frozen(current))
538174f150SMatt Helsley 			break;
548174f150SMatt Helsley 		schedule();
558174f150SMatt Helsley 	}
566301cb95SThomas Gleixner 
576301cb95SThomas Gleixner 	/* Remove the accounting blocker */
586301cb95SThomas Gleixner 	current->flags &= ~PF_FREEZING;
596301cb95SThomas Gleixner 
608174f150SMatt Helsley 	pr_debug("%s left refrigerator\n", current->comm);
61*50fb4f7fSTejun Heo 
62*50fb4f7fSTejun Heo 	/*
63*50fb4f7fSTejun Heo 	 * Restore saved task state before returning.  The mb'd version
64*50fb4f7fSTejun Heo 	 * needs to be used; otherwise, it might silently break
65*50fb4f7fSTejun Heo 	 * synchronization which depends on ordered task state change.
66*50fb4f7fSTejun Heo 	 */
67*50fb4f7fSTejun Heo 	set_current_state(save);
688174f150SMatt Helsley }
698174f150SMatt Helsley EXPORT_SYMBOL(refrigerator);
708174f150SMatt Helsley 
718174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p)
728174f150SMatt Helsley {
738174f150SMatt Helsley 	unsigned long flags;
748174f150SMatt Helsley 
758174f150SMatt Helsley 	spin_lock_irqsave(&p->sighand->siglock, flags);
76d6cc7685STejun Heo 	signal_wake_up(p, 0);
778174f150SMatt Helsley 	spin_unlock_irqrestore(&p->sighand->siglock, flags);
788174f150SMatt Helsley }
798174f150SMatt Helsley 
808174f150SMatt Helsley /**
818174f150SMatt Helsley  *	freeze_task - send a freeze request to given task
828174f150SMatt Helsley  *	@p: task to send the request to
838174f150SMatt Helsley  *	@sig_only: if set, the request will only be sent if the task has the
848174f150SMatt Helsley  *		PF_FREEZER_NOSIG flag unset
858174f150SMatt Helsley  *	Return value: 'false', if @sig_only is set and the task has
868174f150SMatt Helsley  *		PF_FREEZER_NOSIG set or the task is frozen, 'true', otherwise
878174f150SMatt Helsley  *
888174f150SMatt Helsley  *	The freeze request is sent by setting the tasks's TIF_FREEZE flag and
898174f150SMatt Helsley  *	either sending a fake signal to it or waking it up, depending on whether
908174f150SMatt Helsley  *	or not it has PF_FREEZER_NOSIG set.  If @sig_only is set and the task
918174f150SMatt Helsley  *	has PF_FREEZER_NOSIG set (ie. it is a typical kernel thread), its
928174f150SMatt Helsley  *	TIF_FREEZE flag will not be set.
938174f150SMatt Helsley  */
948174f150SMatt Helsley bool freeze_task(struct task_struct *p, bool sig_only)
958174f150SMatt Helsley {
968174f150SMatt Helsley 	/*
978174f150SMatt Helsley 	 * We first check if the task is freezing and next if it has already
988174f150SMatt Helsley 	 * been frozen to avoid the race with frozen_process() which first marks
998174f150SMatt Helsley 	 * the task as frozen and next clears its TIF_FREEZE.
1008174f150SMatt Helsley 	 */
1018174f150SMatt Helsley 	if (!freezing(p)) {
102ee940d8dSMike Frysinger 		smp_rmb();
1038174f150SMatt Helsley 		if (frozen(p))
1048174f150SMatt Helsley 			return false;
1058174f150SMatt Helsley 
1068174f150SMatt Helsley 		if (!sig_only || should_send_signal(p))
1078174f150SMatt Helsley 			set_freeze_flag(p);
1088174f150SMatt Helsley 		else
1098174f150SMatt Helsley 			return false;
1108174f150SMatt Helsley 	}
1118174f150SMatt Helsley 
1128174f150SMatt Helsley 	if (should_send_signal(p)) {
1138174f150SMatt Helsley 		fake_signal_wake_up(p);
1148cfe400cSTejun Heo 		/*
1158cfe400cSTejun Heo 		 * fake_signal_wake_up() goes through p's scheduler
1168cfe400cSTejun Heo 		 * lock and guarantees that TASK_STOPPED/TRACED ->
1178cfe400cSTejun Heo 		 * TASK_RUNNING transition can't race with task state
1188cfe400cSTejun Heo 		 * testing in try_to_freeze_tasks().
1198cfe400cSTejun Heo 		 */
1208174f150SMatt Helsley 	} else if (sig_only) {
1218174f150SMatt Helsley 		return false;
1228174f150SMatt Helsley 	} else {
1238174f150SMatt Helsley 		wake_up_state(p, TASK_INTERRUPTIBLE);
1248174f150SMatt Helsley 	}
1258174f150SMatt Helsley 
1268174f150SMatt Helsley 	return true;
1278174f150SMatt Helsley }
1288174f150SMatt Helsley 
1298174f150SMatt Helsley void cancel_freezing(struct task_struct *p)
1308174f150SMatt Helsley {
1318174f150SMatt Helsley 	unsigned long flags;
1328174f150SMatt Helsley 
1338174f150SMatt Helsley 	if (freezing(p)) {
1348174f150SMatt Helsley 		pr_debug("  clean up: %s\n", p->comm);
1358174f150SMatt Helsley 		clear_freeze_flag(p);
1368174f150SMatt Helsley 		spin_lock_irqsave(&p->sighand->siglock, flags);
1378174f150SMatt Helsley 		recalc_sigpending_and_wake(p);
1388174f150SMatt Helsley 		spin_unlock_irqrestore(&p->sighand->siglock, flags);
1398174f150SMatt Helsley 	}
1408174f150SMatt Helsley }
141dc52ddc0SMatt Helsley 
14200c2e63cSLi Zefan static int __thaw_process(struct task_struct *p)
143dc52ddc0SMatt Helsley {
144dc52ddc0SMatt Helsley 	if (frozen(p)) {
145dc52ddc0SMatt Helsley 		p->flags &= ~PF_FROZEN;
146dc52ddc0SMatt Helsley 		return 1;
147dc52ddc0SMatt Helsley 	}
148dc52ddc0SMatt Helsley 	clear_freeze_flag(p);
149dc52ddc0SMatt Helsley 	return 0;
150dc52ddc0SMatt Helsley }
151dc52ddc0SMatt Helsley 
15200c2e63cSLi Zefan /*
15300c2e63cSLi Zefan  * Wake up a frozen process
15400c2e63cSLi Zefan  *
15500c2e63cSLi Zefan  * task_lock() is needed to prevent the race with refrigerator() which may
15600c2e63cSLi Zefan  * occur if the freezing of tasks fails.  Namely, without the lock, if the
15700c2e63cSLi Zefan  * freezing of tasks failed, thaw_tasks() might have run before a task in
15800c2e63cSLi Zefan  * refrigerator() could call frozen_process(), in which case the task would be
15900c2e63cSLi Zefan  * frozen and no one would thaw it.
16000c2e63cSLi Zefan  */
161dc52ddc0SMatt Helsley int thaw_process(struct task_struct *p)
162dc52ddc0SMatt Helsley {
163dc52ddc0SMatt Helsley 	task_lock(p);
164dc52ddc0SMatt Helsley 	if (__thaw_process(p) == 1) {
165dc52ddc0SMatt Helsley 		task_unlock(p);
166dc52ddc0SMatt Helsley 		wake_up_process(p);
167dc52ddc0SMatt Helsley 		return 1;
168dc52ddc0SMatt Helsley 	}
169dc52ddc0SMatt Helsley 	task_unlock(p);
170dc52ddc0SMatt Helsley 	return 0;
171dc52ddc0SMatt Helsley }
172dc52ddc0SMatt Helsley EXPORT_SYMBOL(thaw_process);
173