xref: /openbmc/linux/kernel/freezer.c (revision a0acae0e886d44bd5ce6d2f173c1ace0fcf0d9f6)
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 :-). */
26*a0acae0eSTejun Heo bool __refrigerator(void)
278174f150SMatt Helsley {
288174f150SMatt Helsley 	/* Hmm, should we be allowed to suspend when there are realtime
298174f150SMatt Helsley 	   processes around? */
30*a0acae0eSTejun Heo 	bool was_frozen = false;
318174f150SMatt Helsley 	long save;
328174f150SMatt Helsley 
338174f150SMatt Helsley 	task_lock(current);
348174f150SMatt Helsley 	if (freezing(current)) {
358174f150SMatt Helsley 		frozen_process();
368174f150SMatt Helsley 		task_unlock(current);
378174f150SMatt Helsley 	} else {
388174f150SMatt Helsley 		task_unlock(current);
39*a0acae0eSTejun Heo 		return was_frozen;
408174f150SMatt Helsley 	}
418174f150SMatt Helsley 	save = current->state;
428174f150SMatt Helsley 	pr_debug("%s entered refrigerator\n", current->comm);
438174f150SMatt Helsley 
448174f150SMatt Helsley 	spin_lock_irq(&current->sighand->siglock);
458174f150SMatt Helsley 	recalc_sigpending(); /* We sent fake signal, clean it up */
468174f150SMatt Helsley 	spin_unlock_irq(&current->sighand->siglock);
478174f150SMatt Helsley 
486301cb95SThomas Gleixner 	/* prevent accounting of that task to load */
496301cb95SThomas Gleixner 	current->flags |= PF_FREEZING;
506301cb95SThomas Gleixner 
518174f150SMatt Helsley 	for (;;) {
528174f150SMatt Helsley 		set_current_state(TASK_UNINTERRUPTIBLE);
538174f150SMatt Helsley 		if (!frozen(current))
548174f150SMatt Helsley 			break;
55*a0acae0eSTejun Heo 		was_frozen = true;
568174f150SMatt Helsley 		schedule();
578174f150SMatt Helsley 	}
586301cb95SThomas Gleixner 
596301cb95SThomas Gleixner 	/* Remove the accounting blocker */
606301cb95SThomas Gleixner 	current->flags &= ~PF_FREEZING;
616301cb95SThomas Gleixner 
628174f150SMatt Helsley 	pr_debug("%s left refrigerator\n", current->comm);
6350fb4f7fSTejun Heo 
6450fb4f7fSTejun Heo 	/*
6550fb4f7fSTejun Heo 	 * Restore saved task state before returning.  The mb'd version
6650fb4f7fSTejun Heo 	 * needs to be used; otherwise, it might silently break
6750fb4f7fSTejun Heo 	 * synchronization which depends on ordered task state change.
6850fb4f7fSTejun Heo 	 */
6950fb4f7fSTejun Heo 	set_current_state(save);
70*a0acae0eSTejun Heo 
71*a0acae0eSTejun Heo 	return was_frozen;
728174f150SMatt Helsley }
73*a0acae0eSTejun Heo EXPORT_SYMBOL(__refrigerator);
748174f150SMatt Helsley 
758174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p)
768174f150SMatt Helsley {
778174f150SMatt Helsley 	unsigned long flags;
788174f150SMatt Helsley 
798174f150SMatt Helsley 	spin_lock_irqsave(&p->sighand->siglock, flags);
80d6cc7685STejun Heo 	signal_wake_up(p, 0);
818174f150SMatt Helsley 	spin_unlock_irqrestore(&p->sighand->siglock, flags);
828174f150SMatt Helsley }
838174f150SMatt Helsley 
848174f150SMatt Helsley /**
858174f150SMatt Helsley  *	freeze_task - send a freeze request to given task
868174f150SMatt Helsley  *	@p: task to send the request to
878174f150SMatt Helsley  *	@sig_only: if set, the request will only be sent if the task has the
888174f150SMatt Helsley  *		PF_FREEZER_NOSIG flag unset
898174f150SMatt Helsley  *	Return value: 'false', if @sig_only is set and the task has
908174f150SMatt Helsley  *		PF_FREEZER_NOSIG set or the task is frozen, 'true', otherwise
918174f150SMatt Helsley  *
928174f150SMatt Helsley  *	The freeze request is sent by setting the tasks's TIF_FREEZE flag and
938174f150SMatt Helsley  *	either sending a fake signal to it or waking it up, depending on whether
948174f150SMatt Helsley  *	or not it has PF_FREEZER_NOSIG set.  If @sig_only is set and the task
958174f150SMatt Helsley  *	has PF_FREEZER_NOSIG set (ie. it is a typical kernel thread), its
968174f150SMatt Helsley  *	TIF_FREEZE flag will not be set.
978174f150SMatt Helsley  */
988174f150SMatt Helsley bool freeze_task(struct task_struct *p, bool sig_only)
998174f150SMatt Helsley {
1008174f150SMatt Helsley 	/*
1018174f150SMatt Helsley 	 * We first check if the task is freezing and next if it has already
1028174f150SMatt Helsley 	 * been frozen to avoid the race with frozen_process() which first marks
1038174f150SMatt Helsley 	 * the task as frozen and next clears its TIF_FREEZE.
1048174f150SMatt Helsley 	 */
1058174f150SMatt Helsley 	if (!freezing(p)) {
106ee940d8dSMike Frysinger 		smp_rmb();
1078174f150SMatt Helsley 		if (frozen(p))
1088174f150SMatt Helsley 			return false;
1098174f150SMatt Helsley 
1108174f150SMatt Helsley 		if (!sig_only || should_send_signal(p))
1118174f150SMatt Helsley 			set_freeze_flag(p);
1128174f150SMatt Helsley 		else
1138174f150SMatt Helsley 			return false;
1148174f150SMatt Helsley 	}
1158174f150SMatt Helsley 
1168174f150SMatt Helsley 	if (should_send_signal(p)) {
1178174f150SMatt Helsley 		fake_signal_wake_up(p);
1188cfe400cSTejun Heo 		/*
1198cfe400cSTejun Heo 		 * fake_signal_wake_up() goes through p's scheduler
1208cfe400cSTejun Heo 		 * lock and guarantees that TASK_STOPPED/TRACED ->
1218cfe400cSTejun Heo 		 * TASK_RUNNING transition can't race with task state
1228cfe400cSTejun Heo 		 * testing in try_to_freeze_tasks().
1238cfe400cSTejun Heo 		 */
1248174f150SMatt Helsley 	} else if (sig_only) {
1258174f150SMatt Helsley 		return false;
1268174f150SMatt Helsley 	} else {
1278174f150SMatt Helsley 		wake_up_state(p, TASK_INTERRUPTIBLE);
1288174f150SMatt Helsley 	}
1298174f150SMatt Helsley 
1308174f150SMatt Helsley 	return true;
1318174f150SMatt Helsley }
1328174f150SMatt Helsley 
1338174f150SMatt Helsley void cancel_freezing(struct task_struct *p)
1348174f150SMatt Helsley {
1358174f150SMatt Helsley 	unsigned long flags;
1368174f150SMatt Helsley 
1378174f150SMatt Helsley 	if (freezing(p)) {
1388174f150SMatt Helsley 		pr_debug("  clean up: %s\n", p->comm);
1398174f150SMatt Helsley 		clear_freeze_flag(p);
1408174f150SMatt Helsley 		spin_lock_irqsave(&p->sighand->siglock, flags);
1418174f150SMatt Helsley 		recalc_sigpending_and_wake(p);
1428174f150SMatt Helsley 		spin_unlock_irqrestore(&p->sighand->siglock, flags);
1438174f150SMatt Helsley 	}
1448174f150SMatt Helsley }
145dc52ddc0SMatt Helsley 
14600c2e63cSLi Zefan static int __thaw_process(struct task_struct *p)
147dc52ddc0SMatt Helsley {
148dc52ddc0SMatt Helsley 	if (frozen(p)) {
149dc52ddc0SMatt Helsley 		p->flags &= ~PF_FROZEN;
150dc52ddc0SMatt Helsley 		return 1;
151dc52ddc0SMatt Helsley 	}
152dc52ddc0SMatt Helsley 	clear_freeze_flag(p);
153dc52ddc0SMatt Helsley 	return 0;
154dc52ddc0SMatt Helsley }
155dc52ddc0SMatt Helsley 
15600c2e63cSLi Zefan /*
15700c2e63cSLi Zefan  * Wake up a frozen process
15800c2e63cSLi Zefan  *
15900c2e63cSLi Zefan  * task_lock() is needed to prevent the race with refrigerator() which may
16000c2e63cSLi Zefan  * occur if the freezing of tasks fails.  Namely, without the lock, if the
16100c2e63cSLi Zefan  * freezing of tasks failed, thaw_tasks() might have run before a task in
16200c2e63cSLi Zefan  * refrigerator() could call frozen_process(), in which case the task would be
16300c2e63cSLi Zefan  * frozen and no one would thaw it.
16400c2e63cSLi Zefan  */
165dc52ddc0SMatt Helsley int thaw_process(struct task_struct *p)
166dc52ddc0SMatt Helsley {
167dc52ddc0SMatt Helsley 	task_lock(p);
168dc52ddc0SMatt Helsley 	if (__thaw_process(p) == 1) {
169dc52ddc0SMatt Helsley 		task_unlock(p);
170dc52ddc0SMatt Helsley 		wake_up_process(p);
171dc52ddc0SMatt Helsley 		return 1;
172dc52ddc0SMatt Helsley 	}
173dc52ddc0SMatt Helsley 	task_unlock(p);
174dc52ddc0SMatt Helsley 	return 0;
175dc52ddc0SMatt Helsley }
176dc52ddc0SMatt Helsley EXPORT_SYMBOL(thaw_process);
177