xref: /titanic_44/usr/src/uts/common/cpr/cpr_main.c (revision 4a16f9a6c1cc74aeed5ff36b4723c3e43bc67666)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 /*
27  * This module contains the guts of checkpoint-resume mechanism.
28  * All code in this module is platform independent.
29  */
30 
31 #include <sys/types.h>
32 #include <sys/errno.h>
33 #include <sys/callb.h>
34 #include <sys/processor.h>
35 #include <sys/machsystm.h>
36 #include <sys/clock.h>
37 #include <sys/vfs.h>
38 #include <sys/kmem.h>
39 #include <nfs/lm.h>
40 #include <sys/systm.h>
41 #include <sys/cpr.h>
42 #include <sys/bootconf.h>
43 #include <sys/cyclic.h>
44 #include <sys/filio.h>
45 #include <sys/fs/ufs_filio.h>
46 #include <sys/epm.h>
47 #include <sys/modctl.h>
48 #include <sys/reboot.h>
49 #include <sys/kdi.h>
50 #include <sys/promif.h>
51 #include <sys/srn.h>
52 #include <sys/cpr_impl.h>
53 
54 #define	PPM(dip) ((dev_info_t *)DEVI(dip)->devi_pm_ppm)
55 
56 extern struct cpr_terminator cpr_term;
57 
58 extern int cpr_alloc_statefile(int);
59 extern void cpr_start_kernel_threads(void);
60 extern void cpr_abbreviate_devpath(char *, char *);
61 extern void cpr_convert_promtime(cpr_time_t *);
62 extern void cpr_send_notice(void);
63 extern void cpr_set_bitmap_size(void);
64 extern void cpr_stat_init();
65 extern void cpr_statef_close(void);
66 extern void flush_windows(void);
67 extern void (*srn_signal)(int, int);
68 extern void init_cpu_syscall(struct cpu *);
69 extern void i_cpr_pre_resume_cpus();
70 extern void i_cpr_post_resume_cpus();
71 extern int cpr_is_ufs(struct vfs *);
72 
73 extern int pm_powering_down;
74 extern kmutex_t srn_clone_lock;
75 extern int srn_inuse;
76 
77 static int cpr_suspend(int);
78 static int cpr_resume(int);
79 static void cpr_suspend_init(int);
80 #if defined(__x86)
81 static int cpr_suspend_cpus(void);
82 static void cpr_resume_cpus(void);
83 #endif
84 static int cpr_all_online(void);
85 static void cpr_restore_offline(void);
86 
87 cpr_time_t wholecycle_tv;
88 int cpr_suspend_succeeded;
89 pfn_t curthreadpfn;
90 int curthreadremapped;
91 
92 extern cpuset_t cpu_ready_set;
93 extern void *(*cpu_pause_func)(void *);
94 
95 extern processorid_t i_cpr_bootcpuid(void);
96 extern cpu_t *i_cpr_bootcpu(void);
97 extern void tsc_adjust_delta(hrtime_t tdelta);
98 extern void tsc_resume(void);
99 extern int tsc_resume_in_cyclic;
100 
101 /*
102  * Set this variable to 1, to have device drivers resume in an
103  * uniprocessor environment. This is to allow drivers that assume
104  * that they resume on a UP machine to continue to work. Should be
105  * deprecated once the broken drivers are fixed
106  */
107 int cpr_resume_uniproc = 0;
108 
109 /*
110  * save or restore abort_enable;  this prevents a drop
111  * to kadb or prom during cpr_resume_devices() when
112  * there is no kbd present;  see abort_sequence_enter()
113  */
114 static void
115 cpr_sae(int stash)
116 {
117 	static int saved_ae = -1;
118 
119 	if (stash) {
120 		saved_ae = abort_enable;
121 		abort_enable = 0;
122 	} else if (saved_ae != -1) {
123 		abort_enable = saved_ae;
124 		saved_ae = -1;
125 	}
126 }
127 
128 
129 /*
130  * The main switching point for cpr, this routine starts the ckpt
131  * and state file saving routines; on resume the control is
132  * returned back to here and it then calls the resume routine.
133  */
134 int
135 cpr_main(int sleeptype)
136 {
137 	int rc, rc2;
138 	label_t saveq;
139 	klwp_t *tlwp = ttolwp(curthread);
140 
141 	if (sleeptype == CPR_TODISK) {
142 		if ((rc = cpr_default_setup(1)) != 0)
143 			return (rc);
144 		ASSERT(tlwp);
145 		saveq = tlwp->lwp_qsav;
146 	}
147 
148 	if (sleeptype == CPR_TORAM) {
149 		rc = cpr_suspend(sleeptype);
150 		PMD(PMD_SX, ("cpr_suspend rets %x\n", rc))
151 		if (rc == 0) {
152 			int i_cpr_power_down(int sleeptype);
153 
154 			/*
155 			 * From this point on, we should be at a high
156 			 * spl, interrupts disabled, and all but one
157 			 * cpu's paused (effectively UP/single threaded).
158 			 * So this is were we want to put ASSERTS()
159 			 * to let us know otherwise.
160 			 */
161 			ASSERT(cpus_paused());
162 
163 			/*
164 			 * Now do the work of actually putting this
165 			 * machine to sleep!
166 			 */
167 			rc = i_cpr_power_down(sleeptype);
168 			if (rc == 0) {
169 				PMD(PMD_SX, ("back from succssful suspend\n"))
170 			}
171 			/*
172 			 * We do care about the return value from cpr_resume
173 			 * at this point, as it will tell us if one of the
174 			 * resume functions failed (cpr_resume_devices())
175 			 * However, for this to return and _not_ panic, means
176 			 * that we must be in one of the test functions.  So
177 			 * check for that and return an appropriate message.
178 			 */
179 			rc2 = cpr_resume(sleeptype);
180 			if (rc2 != 0) {
181 				ASSERT(cpr_test_point > 0);
182 				cmn_err(CE_NOTE,
183 				    "cpr_resume returned non-zero: %d\n", rc2);
184 				PMD(PMD_SX, ("cpr_resume rets %x\n", rc2))
185 			}
186 			ASSERT(!cpus_paused());
187 		} else {
188 			PMD(PMD_SX, ("failed suspend, resuming\n"))
189 			rc = cpr_resume(sleeptype);
190 		}
191 		return (rc);
192 	}
193 	/*
194 	 * Remember where we are for resume after reboot
195 	 */
196 	if (!setjmp(&tlwp->lwp_qsav)) {
197 		/*
198 		 * try to checkpoint the system, if failed return back
199 		 * to userland, otherwise power off.
200 		 */
201 		rc = cpr_suspend(sleeptype);
202 		if (rc || cpr_reusable_mode) {
203 			/*
204 			 * We don't really want to go down, or
205 			 * something went wrong in suspend, do what we can
206 			 * to put the system back to an operable state then
207 			 * return back to userland.
208 			 */
209 			PMD(PMD_SX, ("failed suspend, resuming\n"))
210 			(void) cpr_resume(sleeptype);
211 			PMD(PMD_SX, ("back from failed suspend resume\n"))
212 		}
213 	} else {
214 		/*
215 		 * This is the resumed side of longjmp, restore the previous
216 		 * longjmp pointer if there is one so this will be transparent
217 		 * to the world.
218 		 * This path is only for CPR_TODISK, where we reboot
219 		 */
220 		ASSERT(sleeptype == CPR_TODISK);
221 		tlwp->lwp_qsav = saveq;
222 		CPR->c_flags &= ~C_SUSPENDING;
223 		CPR->c_flags |= C_RESUMING;
224 
225 		/*
226 		 * resume the system back to the original state
227 		 */
228 		rc = cpr_resume(sleeptype);
229 		PMD(PMD_SX, ("back from successful suspend; resume rets %x\n",
230 		    rc))
231 	}
232 
233 	(void) cpr_default_setup(0);
234 
235 	return (rc);
236 }
237 
238 
239 #if defined(__sparc)
240 
241 /*
242  * check/disable or re-enable UFS logging
243  */
244 static void
245 cpr_log_status(int enable, int *svstat, vnode_t *vp)
246 {
247 	int cmd, status, error;
248 	char *str, *able;
249 	fiolog_t fl;
250 	refstr_t *mntpt;
251 
252 	str = "cpr_log_status";
253 	bzero(&fl, sizeof (fl));
254 	fl.error = FIOLOG_ENONE;
255 
256 	/*
257 	 * when disabling, first get and save logging status (0 or 1)
258 	 */
259 	if (enable == 0) {
260 		if (error = VOP_IOCTL(vp, _FIOISLOG,
261 		    (uintptr_t)&status, FKIOCTL, CRED(), NULL, NULL)) {
262 			mntpt = vfs_getmntpoint(vp->v_vfsp);
263 			prom_printf("%s: \"%s\", cant get logging "
264 			    "status, error %d\n", str, refstr_value(mntpt),
265 			    error);
266 			refstr_rele(mntpt);
267 			return;
268 		}
269 		*svstat = status;
270 		if (cpr_debug & CPR_DEBUG5) {
271 			mntpt = vfs_getmntpoint(vp->v_vfsp);
272 			errp("%s: \"%s\", logging status = %d\n",
273 			    str, refstr_value(mntpt), status);
274 			refstr_rele(mntpt);
275 		};
276 
277 		able = "disable";
278 		cmd = _FIOLOGDISABLE;
279 	} else {
280 		able = "enable";
281 		cmd = _FIOLOGENABLE;
282 	}
283 
284 	/*
285 	 * disable or re-enable logging when the saved status is 1
286 	 */
287 	if (*svstat == 1) {
288 		error = VOP_IOCTL(vp, cmd, (uintptr_t)&fl,
289 		    FKIOCTL, CRED(), NULL, NULL);
290 		if (error) {
291 			mntpt = vfs_getmntpoint(vp->v_vfsp);
292 			prom_printf("%s: \"%s\", cant %s logging, error %d\n",
293 			    str, refstr_value(mntpt), able, error);
294 			refstr_rele(mntpt);
295 		} else {
296 			if (cpr_debug & CPR_DEBUG5) {
297 				mntpt = vfs_getmntpoint(vp->v_vfsp);
298 				errp("%s: \"%s\", logging is now %sd\n",
299 				    str, refstr_value(mntpt), able);
300 				refstr_rele(mntpt);
301 			};
302 		}
303 	}
304 
305 	/*
306 	 * when enabling logging, reset the saved status
307 	 * to unknown for next time
308 	 */
309 	if (enable)
310 		*svstat = -1;
311 }
312 
313 /*
314  * enable/disable UFS logging on filesystems containing cpr_default_path
315  * and cpr statefile.  since the statefile can be on any fs, that fs
316  * needs to be handled separately.  this routine and cprboot expect that
317  * CPR_CONFIG and CPR_DEFAULT both reside on the same fs, rootfs.  cprboot
318  * is loaded from the device with rootfs and uses the same device to open
319  * both CPR_CONFIG and CPR_DEFAULT (see common/support.c).  moving either
320  * file outside of rootfs would cause errors during cprboot, plus cpr and
321  * fsck problems with the new fs if logging were enabled.
322  */
323 
324 static int
325 cpr_ufs_logging(int enable)
326 {
327 	static int def_status = -1, sf_status = -1;
328 	struct vfs *vfsp;
329 	char *fname;
330 	vnode_t *vp;
331 	int error;
332 
333 	if (cpr_reusable_mode)
334 		return (0);
335 
336 	if (error = cpr_open_deffile(FREAD, &vp))
337 		return (error);
338 	vfsp = vp->v_vfsp;
339 	if (!cpr_is_ufs(vfsp)) {
340 		(void) VOP_CLOSE(vp, FREAD, 1, (offset_t)0, CRED(), NULL);
341 		VN_RELE(vp);
342 		return (0);
343 	}
344 
345 	cpr_log_status(enable, &def_status, vp);
346 	(void) VOP_CLOSE(vp, FREAD, 1, (offset_t)0, CRED(), NULL);
347 	VN_RELE(vp);
348 
349 	fname = cpr_build_statefile_path();
350 	if (fname == NULL)
351 		return (ENOENT);
352 	if (error = vn_open(fname, UIO_SYSSPACE, FCREAT|FWRITE,
353 	    0600, &vp, CRCREAT, 0)) {
354 		prom_printf("cpr_ufs_logging: cant open/create \"%s\", "
355 		    "error %d\n", fname, error);
356 		return (error);
357 	}
358 
359 	/*
360 	 * check logging status for the statefile if it resides
361 	 * on a different fs and the type is a regular file
362 	 */
363 	if (vp->v_vfsp != vfsp && vp->v_type == VREG)
364 		cpr_log_status(enable, &sf_status, vp);
365 	(void) VOP_CLOSE(vp, FWRITE, 1, (offset_t)0, CRED(), NULL);
366 	VN_RELE(vp);
367 
368 	return (0);
369 }
370 #endif
371 
372 
373 /*
374  * Check if klmmod is loaded and call a lock manager service; if klmmod
375  * is not loaded, the services aren't needed and a call would trigger a
376  * modload, which would block since another thread would never run.
377  */
378 static void
379 cpr_lock_mgr(void (*service)(void))
380 {
381 	if (mod_find_by_filename(NULL, "misc/klmmod") != NULL)
382 		(*service)();
383 }
384 
385 int
386 cpr_suspend_cpus(void)
387 {
388 	int	ret = 0;
389 	extern void *i_cpr_save_context(void *arg);
390 
391 	mutex_enter(&cpu_lock);
392 
393 	/*
394 	 * the machine could not have booted without a bootcpu
395 	 */
396 	ASSERT(i_cpr_bootcpu() != NULL);
397 
398 	/*
399 	 * bring all the offline cpus online
400 	 */
401 	if ((ret = cpr_all_online())) {
402 		mutex_exit(&cpu_lock);
403 		return (ret);
404 	}
405 
406 	/*
407 	 * Set the affinity to be the boot processor
408 	 * This is cleared in either cpr_resume_cpus() or cpr_unpause_cpus()
409 	 */
410 	affinity_set(i_cpr_bootcpuid());
411 
412 	ASSERT(CPU->cpu_id == 0);
413 
414 	PMD(PMD_SX, ("curthread running on bootcpu\n"))
415 
416 	/*
417 	 * pause all other running CPUs and save the CPU state at the sametime
418 	 */
419 	cpu_pause_func = i_cpr_save_context;
420 	pause_cpus(NULL);
421 
422 	mutex_exit(&cpu_lock);
423 
424 	return (0);
425 }
426 
427 /*
428  * Take the system down to a checkpointable state and write
429  * the state file, the following are sequentially executed:
430  *
431  *    - Request all user threads to stop themselves
432  *    - push out and invalidate user pages
433  *    - bring statefile inode incore to prevent a miss later
434  *    - request all daemons to stop
435  *    - check and make sure all threads are stopped
436  *    - sync the file system
437  *    - suspend all devices
438  *    - block intrpts
439  *    - dump system state and memory to state file
440  *    - SPARC code will not be called with CPR_TORAM, caller filters
441  */
442 static int
443 cpr_suspend(int sleeptype)
444 {
445 #if defined(__sparc)
446 	int sf_realloc, nverr;
447 #endif
448 	int	rc = 0;
449 	int	skt_rc = 0;
450 
451 	PMD(PMD_SX, ("cpr_suspend %x\n", sleeptype))
452 	cpr_set_substate(C_ST_SUSPEND_BEGIN);
453 
454 	cpr_suspend_init(sleeptype);
455 
456 	cpr_save_time();
457 
458 	cpr_tod_get(&wholecycle_tv);
459 	CPR_STAT_EVENT_START("Suspend Total");
460 
461 	i_cpr_alloc_cpus();
462 
463 #if defined(__sparc)
464 	ASSERT(sleeptype == CPR_TODISK);
465 	if (!cpr_reusable_mode) {
466 		/*
467 		 * We need to validate default file before fs
468 		 * functionality is disabled.
469 		 */
470 		if (rc = cpr_validate_definfo(0))
471 			return (rc);
472 	}
473 	i_cpr_save_machdep_info();
474 #endif
475 
476 	PMD(PMD_SX, ("cpr_suspend: stop scans\n"))
477 	/* Stop PM scans ASAP */
478 	(void) callb_execute_class(CB_CL_CPR_PM, CB_CODE_CPR_CHKPT);
479 
480 	pm_dispatch_to_dep_thread(PM_DEP_WK_CPR_SUSPEND,
481 	    NULL, NULL, PM_DEP_WAIT, NULL, 0);
482 
483 #if defined(__sparc)
484 	ASSERT(sleeptype == CPR_TODISK);
485 	cpr_set_substate(C_ST_MP_OFFLINE);
486 	if (rc = cpr_mp_offline())
487 		return (rc);
488 #endif
489 	/*
490 	 * Ask Xorg to suspend the frame buffer, and wait for it to happen
491 	 */
492 	mutex_enter(&srn_clone_lock);
493 	if (srn_signal) {
494 		PMD(PMD_SX, ("cpr_suspend: (*srn_signal)(..., "
495 		    "SRN_SUSPEND_REQ)\n"))
496 		srn_inuse = 1;	/* because *(srn_signal) cv_waits */
497 		(*srn_signal)(SRN_TYPE_APM, SRN_SUSPEND_REQ);
498 		srn_inuse = 0;
499 	} else {
500 		PMD(PMD_SX, ("cpr_suspend: srn_signal NULL\n"))
501 	}
502 	mutex_exit(&srn_clone_lock);
503 
504 	/*
505 	 * Ask the user threads to stop by themselves, but
506 	 * if they don't or can't after 3 retries, we give up on CPR.
507 	 * The 3 retry is not a random number because 2 is possible if
508 	 * a thread has been forked before the parent thread is stopped.
509 	 */
510 	CPR_DEBUG(CPR_DEBUG1, "\nstopping user threads...");
511 	CPR_STAT_EVENT_START("  stop users");
512 	cpr_set_substate(C_ST_STOP_USER_THREADS);
513 	PMD(PMD_SX, ("cpr_suspend: stop user threads\n"))
514 	if (rc = cpr_stop_user_threads())
515 		return (rc);
516 	CPR_STAT_EVENT_END("  stop users");
517 	CPR_DEBUG(CPR_DEBUG1, "done\n");
518 
519 	PMD(PMD_SX, ("cpr_suspend: save direct levels\n"))
520 	pm_save_direct_levels();
521 
522 	/*
523 	 * User threads are stopped.  We will start communicating with the
524 	 * user via prom_printf (some debug output may have already happened)
525 	 * so let anybody who cares know about this (bug 4096122)
526 	 */
527 	(void) callb_execute_class(CB_CL_CPR_PROMPRINTF, CB_CODE_CPR_CHKPT);
528 
529 	PMD(PMD_SX, ("cpr_suspend: send notice\n"))
530 #ifndef DEBUG
531 	cpr_send_notice();
532 	if (cpr_debug)
533 		prom_printf("\n");
534 #endif
535 
536 	PMD(PMD_SX, ("cpr_suspend: POST USER callback\n"))
537 	(void) callb_execute_class(CB_CL_CPR_POST_USER, CB_CODE_CPR_CHKPT);
538 
539 	/*
540 	 * Reattach any drivers which originally exported the
541 	 * no-involuntary-power-cycles property.  We need to do this before
542 	 * stopping kernel threads because modload is implemented using
543 	 * a kernel thread.
544 	 */
545 	cpr_set_substate(C_ST_PM_REATTACH_NOINVOL);
546 	PMD(PMD_SX, ("cpr_suspend: reattach noinvol\n"))
547 	if (!pm_reattach_noinvol())
548 		return (ENXIO);
549 
550 #if defined(__sparc)
551 	ASSERT(sleeptype == CPR_TODISK);
552 	/*
553 	 * if ufs logging is enabled, we need to disable before
554 	 * stopping kernel threads so that ufs delete and roll
555 	 * threads can do the work.
556 	 */
557 	cpr_set_substate(C_ST_DISABLE_UFS_LOGGING);
558 	if (rc = cpr_ufs_logging(0))
559 		return (rc);
560 
561 	/*
562 	 * Use sync_all to swap out all user pages and find out how much
563 	 * extra space needed for user pages that don't have back store
564 	 * space left.
565 	 */
566 	CPR_STAT_EVENT_START("  swapout upages");
567 	vfs_sync(SYNC_ALL);
568 	CPR_STAT_EVENT_END("  swapout upages");
569 
570 	cpr_set_bitmap_size();
571 
572 alloc_statefile:
573 	/*
574 	 * If our last state was C_ST_DUMP_NOSPC, we're trying to
575 	 * realloc the statefile, otherwise this is the first attempt.
576 	 */
577 	sf_realloc = (CPR->c_substate == C_ST_DUMP_NOSPC) ? 1 : 0;
578 
579 	CPR_STAT_EVENT_START("  alloc statefile");
580 	cpr_set_substate(C_ST_STATEF_ALLOC);
581 	if (rc = cpr_alloc_statefile(sf_realloc)) {
582 		if (sf_realloc)
583 			errp("realloc failed\n");
584 		return (rc);
585 	}
586 	CPR_STAT_EVENT_END("  alloc statefile");
587 
588 	/*
589 	 * Sync the filesystem to preserve its integrity.
590 	 *
591 	 * This sync is also used to flush out all B_DELWRI buffers
592 	 * (fs cache) which are mapped and neither dirty nor referenced
593 	 * before cpr_invalidate_pages destroys them.
594 	 * fsflush does similar thing.
595 	 */
596 	sync();
597 
598 	/*
599 	 * destroy all clean file mapped kernel pages
600 	 */
601 	CPR_STAT_EVENT_START("  clean pages");
602 	CPR_DEBUG(CPR_DEBUG1, ("cleaning up mapped pages..."));
603 	(void) callb_execute_class(CB_CL_CPR_VM, CB_CODE_CPR_CHKPT);
604 	CPR_DEBUG(CPR_DEBUG1, ("done\n"));
605 	CPR_STAT_EVENT_END("  clean pages");
606 #endif
607 
608 
609 	/*
610 	 * Hooks needed by lock manager prior to suspending.
611 	 * Refer to code for more comments.
612 	 */
613 	PMD(PMD_SX, ("cpr_suspend: lock mgr\n"))
614 	cpr_lock_mgr(lm_cprsuspend);
615 
616 	/*
617 	 * Now suspend all the devices
618 	 */
619 	CPR_STAT_EVENT_START("  stop drivers");
620 	CPR_DEBUG(CPR_DEBUG1, "suspending drivers...");
621 	cpr_set_substate(C_ST_SUSPEND_DEVICES);
622 	pm_powering_down = 1;
623 	PMD(PMD_SX, ("cpr_suspend: suspending devices\n"))
624 	rc = cpr_suspend_devices(ddi_root_node());
625 	pm_powering_down = 0;
626 	if (rc)
627 		return (rc);
628 	CPR_DEBUG(CPR_DEBUG1, "done\n");
629 	CPR_STAT_EVENT_END("  stop drivers");
630 
631 	/*
632 	 * Stop all daemon activities
633 	 */
634 	cpr_set_substate(C_ST_STOP_KERNEL_THREADS);
635 	PMD(PMD_SX, ("cpr_suspend: stopping kernel threads\n"))
636 	if (skt_rc = cpr_stop_kernel_threads())
637 		return (skt_rc);
638 
639 	PMD(PMD_SX, ("cpr_suspend: POST KERNEL callback\n"))
640 	(void) callb_execute_class(CB_CL_CPR_POST_KERNEL, CB_CODE_CPR_CHKPT);
641 
642 	PMD(PMD_SX, ("cpr_suspend: reattach noinvol fini\n"))
643 	pm_reattach_noinvol_fini();
644 
645 	cpr_sae(1);
646 
647 	PMD(PMD_SX, ("cpr_suspend: CPR CALLOUT callback\n"))
648 	(void) callb_execute_class(CB_CL_CPR_CALLOUT, CB_CODE_CPR_CHKPT);
649 
650 	if (sleeptype == CPR_TODISK) {
651 		/*
652 		 * It's safer to do tod_get before we disable all intr.
653 		 */
654 		CPR_STAT_EVENT_START("  write statefile");
655 	}
656 
657 	/*
658 	 * it's time to ignore the outside world, stop the real time
659 	 * clock and disable any further intrpt activity.
660 	 */
661 	PMD(PMD_SX, ("cpr_suspend: handle xc\n"))
662 	i_cpr_handle_xc(1);	/* turn it on to disable xc assertion */
663 
664 	mutex_enter(&cpu_lock);
665 	PMD(PMD_SX, ("cpr_suspend: cyclic suspend\n"))
666 	cyclic_suspend();
667 	mutex_exit(&cpu_lock);
668 
669 	/*
670 	 * Due to the different methods of resuming the system between
671 	 * CPR_TODISK (boot cprboot on SPARC, which reloads kernel image)
672 	 * and CPR_TORAM (restart via reset into existing kernel image)
673 	 * cpus are not suspended and restored in the SPARC case, since it
674 	 * is necessary to restart the cpus and pause them before restoring
675 	 * the OBP image
676 	 */
677 
678 #if defined(__x86)
679 
680 	/* pause aux cpus */
681 	PMD(PMD_SX, ("pause aux cpus\n"))
682 
683 	cpr_set_substate(C_ST_MP_PAUSED);
684 
685 	if ((rc = cpr_suspend_cpus()) != 0)
686 		return (rc);
687 #endif
688 
689 	PMD(PMD_SX, ("cpr_suspend: stop intr\n"))
690 	i_cpr_stop_intr();
691 	CPR_DEBUG(CPR_DEBUG1, "interrupt is stopped\n");
692 
693 	/*
694 	 * Since we will now disable the mechanism that causes prom_printfs
695 	 * to power up (if needed) the console fb/monitor, we assert that
696 	 * it must be up now.
697 	 */
698 	ASSERT(pm_cfb_is_up());
699 	PMD(PMD_SX, ("cpr_suspend: prom suspend prepost\n"))
700 	prom_suspend_prepost();
701 
702 #if defined(__sparc)
703 	/*
704 	 * getting ready to write ourself out, flush the register
705 	 * windows to make sure that our stack is good when we
706 	 * come back on the resume side.
707 	 */
708 	flush_windows();
709 #endif
710 
711 	/*
712 	 * For S3, we're done
713 	 */
714 	if (sleeptype == CPR_TORAM) {
715 		PMD(PMD_SX, ("cpr_suspend rets %x\n", rc))
716 		cpr_set_substate(C_ST_NODUMP);
717 		return (rc);
718 	}
719 #if defined(__sparc)
720 	/*
721 	 * FATAL: NO MORE MEMORY ALLOCATION ALLOWED AFTER THIS POINT!!!
722 	 *
723 	 * The system is quiesced at this point, we are ready to either dump
724 	 * to the state file for a extended sleep or a simple shutdown for
725 	 * systems with non-volatile memory.
726 	 */
727 
728 	/*
729 	 * special handling for reusable:
730 	 */
731 	if (cpr_reusable_mode) {
732 		cpr_set_substate(C_ST_SETPROPS_1);
733 		if (nverr = cpr_set_properties(1))
734 			return (nverr);
735 	}
736 
737 	cpr_set_substate(C_ST_DUMP);
738 	rc = cpr_dump(C_VP);
739 
740 	/*
741 	 * if any error occurred during dump, more
742 	 * special handling for reusable:
743 	 */
744 	if (rc && cpr_reusable_mode) {
745 		cpr_set_substate(C_ST_SETPROPS_0);
746 		if (nverr = cpr_set_properties(0))
747 			return (nverr);
748 	}
749 
750 	if (rc == ENOSPC) {
751 		cpr_set_substate(C_ST_DUMP_NOSPC);
752 		(void) cpr_resume(sleeptype);
753 		goto alloc_statefile;
754 	} else if (rc == 0) {
755 		if (cpr_reusable_mode) {
756 			cpr_set_substate(C_ST_REUSABLE);
757 			longjmp(&ttolwp(curthread)->lwp_qsav);
758 		} else
759 			rc = cpr_set_properties(1);
760 	}
761 #endif
762 	PMD(PMD_SX, ("cpr_suspend: return %d\n", rc))
763 	return (rc);
764 }
765 
766 void
767 cpr_resume_cpus(void)
768 {
769 	/*
770 	 * this is a cut down version of start_other_cpus()
771 	 * just do the initialization to wake the other cpus
772 	 */
773 
774 #if defined(__x86)
775 	/*
776 	 * Initialize our syscall handlers
777 	 */
778 	init_cpu_syscall(CPU);
779 
780 #endif
781 
782 	i_cpr_pre_resume_cpus();
783 
784 	/*
785 	 * Restart the paused cpus
786 	 */
787 	mutex_enter(&cpu_lock);
788 	start_cpus();
789 	mutex_exit(&cpu_lock);
790 
791 	i_cpr_post_resume_cpus();
792 
793 	mutex_enter(&cpu_lock);
794 	/*
795 	 * Restore this cpu to use the regular cpu_pause(), so that
796 	 * online and offline will work correctly
797 	 */
798 	cpu_pause_func = NULL;
799 
800 	/*
801 	 * clear the affinity set in cpr_suspend_cpus()
802 	 */
803 	affinity_clear();
804 
805 	/*
806 	 * offline all the cpus that were brought online during suspend
807 	 */
808 	cpr_restore_offline();
809 
810 	mutex_exit(&cpu_lock);
811 }
812 
813 void
814 cpr_unpause_cpus(void)
815 {
816 	/*
817 	 * Now restore the system back to what it was before we suspended
818 	 */
819 
820 	PMD(PMD_SX, ("cpr_unpause_cpus: restoring system\n"))
821 
822 	mutex_enter(&cpu_lock);
823 
824 	/*
825 	 * Restore this cpu to use the regular cpu_pause(), so that
826 	 * online and offline will work correctly
827 	 */
828 	cpu_pause_func = NULL;
829 
830 	/*
831 	 * Restart the paused cpus
832 	 */
833 	start_cpus();
834 
835 	/*
836 	 * clear the affinity set in cpr_suspend_cpus()
837 	 */
838 	affinity_clear();
839 
840 	/*
841 	 * offline all the cpus that were brought online during suspend
842 	 */
843 	cpr_restore_offline();
844 
845 	mutex_exit(&cpu_lock);
846 }
847 
848 /*
849  * Bring the system back up from a checkpoint, at this point
850  * the VM has been minimally restored by boot, the following
851  * are executed sequentially:
852  *
853  *    - machdep setup and enable interrupts (mp startup if it's mp)
854  *    - resume all devices
855  *    - restart daemons
856  *    - put all threads back on run queue
857  */
858 static int
859 cpr_resume(int sleeptype)
860 {
861 	cpr_time_t pwron_tv, *ctp;
862 	char *str;
863 	int rc = 0;
864 
865 	/*
866 	 * The following switch is used to resume the system
867 	 * that was suspended to a different level.
868 	 */
869 	CPR_DEBUG(CPR_DEBUG1, "\nEntering cpr_resume...\n");
870 	PMD(PMD_SX, ("cpr_resume %x\n", sleeptype))
871 
872 	/*
873 	 * Note:
874 	 *
875 	 * The rollback labels rb_xyz do not represent the cpr resume
876 	 * state when event 'xyz' has happened. Instead they represent
877 	 * the state during cpr suspend when event 'xyz' was being
878 	 * entered (and where cpr suspend failed). The actual call that
879 	 * failed may also need to be partially rolled back, since they
880 	 * aren't atomic in most cases.  In other words, rb_xyz means
881 	 * "roll back all cpr suspend events that happened before 'xyz',
882 	 * and the one that caused the failure, if necessary."
883 	 */
884 	switch (CPR->c_substate) {
885 #if defined(__sparc)
886 	case C_ST_DUMP:
887 		/*
888 		 * This is most likely a full-fledged cpr_resume after
889 		 * a complete and successful cpr suspend. Just roll back
890 		 * everything.
891 		 */
892 		ASSERT(sleeptype == CPR_TODISK);
893 		break;
894 
895 	case C_ST_REUSABLE:
896 	case C_ST_DUMP_NOSPC:
897 	case C_ST_SETPROPS_0:
898 	case C_ST_SETPROPS_1:
899 		/*
900 		 * C_ST_REUSABLE and C_ST_DUMP_NOSPC are the only two
901 		 * special switch cases here. The other two do not have
902 		 * any state change during cpr_suspend() that needs to
903 		 * be rolled back. But these are exit points from
904 		 * cpr_suspend, so theoretically (or in the future), it
905 		 * is possible that a need for roll back of a state
906 		 * change arises between these exit points.
907 		 */
908 		ASSERT(sleeptype == CPR_TODISK);
909 		goto rb_dump;
910 #endif
911 
912 	case C_ST_NODUMP:
913 		PMD(PMD_SX, ("cpr_resume: NODUMP\n"))
914 		goto rb_nodump;
915 
916 	case C_ST_STOP_KERNEL_THREADS:
917 		PMD(PMD_SX, ("cpr_resume: STOP_KERNEL_THREADS\n"))
918 		goto rb_stop_kernel_threads;
919 
920 	case C_ST_SUSPEND_DEVICES:
921 		PMD(PMD_SX, ("cpr_resume: SUSPEND_DEVICES\n"))
922 		goto rb_suspend_devices;
923 
924 #if defined(__sparc)
925 	case C_ST_STATEF_ALLOC:
926 		ASSERT(sleeptype == CPR_TODISK);
927 		goto rb_statef_alloc;
928 
929 	case C_ST_DISABLE_UFS_LOGGING:
930 		ASSERT(sleeptype == CPR_TODISK);
931 		goto rb_disable_ufs_logging;
932 #endif
933 
934 	case C_ST_PM_REATTACH_NOINVOL:
935 		PMD(PMD_SX, ("cpr_resume: REATTACH_NOINVOL\n"))
936 		goto rb_pm_reattach_noinvol;
937 
938 	case C_ST_STOP_USER_THREADS:
939 		PMD(PMD_SX, ("cpr_resume: STOP_USER_THREADS\n"))
940 		goto rb_stop_user_threads;
941 
942 #if defined(__sparc)
943 	case C_ST_MP_OFFLINE:
944 		PMD(PMD_SX, ("cpr_resume: MP_OFFLINE\n"))
945 		goto rb_mp_offline;
946 #endif
947 
948 #if defined(__x86)
949 	case C_ST_MP_PAUSED:
950 		PMD(PMD_SX, ("cpr_resume: MP_PAUSED\n"))
951 		goto rb_mp_paused;
952 #endif
953 
954 
955 	default:
956 		PMD(PMD_SX, ("cpr_resume: others\n"))
957 		goto rb_others;
958 	}
959 
960 rb_all:
961 	/*
962 	 * perform platform-dependent initialization
963 	 */
964 	if (cpr_suspend_succeeded)
965 		i_cpr_machdep_setup();
966 
967 	/*
968 	 * system did not really go down if we jump here
969 	 */
970 rb_dump:
971 	/*
972 	 * IMPORTANT:  SENSITIVE RESUME SEQUENCE
973 	 *
974 	 * DO NOT ADD ANY INITIALIZATION STEP BEFORE THIS POINT!!
975 	 */
976 rb_nodump:
977 	/*
978 	 * If we did suspend to RAM, we didn't generate a dump
979 	 */
980 	PMD(PMD_SX, ("cpr_resume: CPR DMA callback\n"))
981 	(void) callb_execute_class(CB_CL_CPR_DMA, CB_CODE_CPR_RESUME);
982 	if (cpr_suspend_succeeded) {
983 		PMD(PMD_SX, ("cpr_resume: CPR RPC callback\n"))
984 		(void) callb_execute_class(CB_CL_CPR_RPC, CB_CODE_CPR_RESUME);
985 	}
986 
987 	prom_resume_prepost();
988 #if !defined(__sparc)
989 	/*
990 	 * Need to sync the software clock with the hardware clock.
991 	 * On Sparc, this occurs in the sparc-specific cbe.  However
992 	 * on x86 this needs to be handled _before_ we bring other cpu's
993 	 * back online.  So we call a resume function in timestamp.c
994 	 */
995 	if (tsc_resume_in_cyclic == 0)
996 		tsc_resume();
997 
998 #endif
999 
1000 #if defined(__sparc)
1001 	if (cpr_suspend_succeeded && (boothowto & RB_DEBUG))
1002 		kdi_dvec_cpr_restart();
1003 #endif
1004 
1005 
1006 #if defined(__x86)
1007 rb_mp_paused:
1008 	PT(PT_RMPO);
1009 	PMD(PMD_SX, ("resume aux cpus\n"))
1010 
1011 	if (cpr_suspend_succeeded) {
1012 		cpr_resume_cpus();
1013 	} else {
1014 		cpr_unpause_cpus();
1015 	}
1016 #endif
1017 
1018 	/*
1019 	 * let the tmp callout catch up.
1020 	 */
1021 	PMD(PMD_SX, ("cpr_resume: CPR CALLOUT callback\n"))
1022 	(void) callb_execute_class(CB_CL_CPR_CALLOUT, CB_CODE_CPR_RESUME);
1023 
1024 	i_cpr_enable_intr();
1025 
1026 	mutex_enter(&cpu_lock);
1027 	PMD(PMD_SX, ("cpr_resume: cyclic resume\n"))
1028 	cyclic_resume();
1029 	mutex_exit(&cpu_lock);
1030 
1031 	PMD(PMD_SX, ("cpr_resume: handle xc\n"))
1032 	i_cpr_handle_xc(0);	/* turn it off to allow xc assertion */
1033 
1034 	PMD(PMD_SX, ("cpr_resume: CPR POST KERNEL callback\n"))
1035 	(void) callb_execute_class(CB_CL_CPR_POST_KERNEL, CB_CODE_CPR_RESUME);
1036 
1037 	/*
1038 	 * statistics gathering
1039 	 */
1040 	if (cpr_suspend_succeeded) {
1041 		/*
1042 		 * Prevent false alarm in tod_validate() due to tod
1043 		 * value change between suspend and resume
1044 		 */
1045 		cpr_tod_status_set(TOD_CPR_RESUME_DONE);
1046 
1047 		cpr_convert_promtime(&pwron_tv);
1048 
1049 		ctp = &cpr_term.tm_shutdown;
1050 		if (sleeptype == CPR_TODISK)
1051 			CPR_STAT_EVENT_END_TMZ("  write statefile", ctp);
1052 		CPR_STAT_EVENT_END_TMZ("Suspend Total", ctp);
1053 
1054 		CPR_STAT_EVENT_START_TMZ("Resume Total", &pwron_tv);
1055 
1056 		str = "  prom time";
1057 		CPR_STAT_EVENT_START_TMZ(str, &pwron_tv);
1058 		ctp = &cpr_term.tm_cprboot_start;
1059 		CPR_STAT_EVENT_END_TMZ(str, ctp);
1060 
1061 		str = "  read statefile";
1062 		CPR_STAT_EVENT_START_TMZ(str, ctp);
1063 		ctp = &cpr_term.tm_cprboot_end;
1064 		CPR_STAT_EVENT_END_TMZ(str, ctp);
1065 	}
1066 
1067 rb_stop_kernel_threads:
1068 	/*
1069 	 * Put all threads back to where they belong; get the kernel
1070 	 * daemons straightened up too. Note that the callback table
1071 	 * locked during cpr_stop_kernel_threads() is released only
1072 	 * in cpr_start_kernel_threads(). Ensure modunloading is
1073 	 * disabled before starting kernel threads, we don't want
1074 	 * modunload thread to start changing device tree underneath.
1075 	 */
1076 	PMD(PMD_SX, ("cpr_resume: modunload disable\n"))
1077 	modunload_disable();
1078 	PMD(PMD_SX, ("cpr_resume: start kernel threads\n"))
1079 	cpr_start_kernel_threads();
1080 
1081 rb_suspend_devices:
1082 	CPR_DEBUG(CPR_DEBUG1, "resuming devices...");
1083 	CPR_STAT_EVENT_START("  start drivers");
1084 
1085 	PMD(PMD_SX,
1086 	    ("cpr_resume: rb_suspend_devices: cpr_resume_uniproc = %d\n",
1087 	    cpr_resume_uniproc))
1088 
1089 #if defined(__x86)
1090 	/*
1091 	 * If cpr_resume_uniproc is set, then pause all the other cpus
1092 	 * apart from the current cpu, so that broken drivers that think
1093 	 * that they are on a uniprocessor machine will resume
1094 	 */
1095 	if (cpr_resume_uniproc) {
1096 		mutex_enter(&cpu_lock);
1097 		pause_cpus(NULL);
1098 		mutex_exit(&cpu_lock);
1099 	}
1100 #endif
1101 
1102 	/*
1103 	 * The policy here is to continue resume everything we can if we did
1104 	 * not successfully finish suspend; and panic if we are coming back
1105 	 * from a fully suspended system.
1106 	 */
1107 	PMD(PMD_SX, ("cpr_resume: resume devices\n"))
1108 	rc = cpr_resume_devices(ddi_root_node(), 0);
1109 
1110 	cpr_sae(0);
1111 
1112 	str = "Failed to resume one or more devices.";
1113 
1114 	if (rc) {
1115 		if (CPR->c_substate == C_ST_DUMP ||
1116 		    (sleeptype == CPR_TORAM &&
1117 		    CPR->c_substate == C_ST_NODUMP)) {
1118 			if (cpr_test_point == FORCE_SUSPEND_TO_RAM) {
1119 				PMD(PMD_SX, ("cpr_resume: resume device "
1120 				    "warn\n"))
1121 				cpr_err(CE_WARN, str);
1122 			} else {
1123 				PMD(PMD_SX, ("cpr_resume: resume device "
1124 				    "panic\n"))
1125 				cpr_err(CE_PANIC, str);
1126 			}
1127 		} else {
1128 			PMD(PMD_SX, ("cpr_resume: resume device warn\n"))
1129 			cpr_err(CE_WARN, str);
1130 		}
1131 	}
1132 
1133 	CPR_STAT_EVENT_END("  start drivers");
1134 	CPR_DEBUG(CPR_DEBUG1, "done\n");
1135 
1136 #if defined(__x86)
1137 	/*
1138 	 * If cpr_resume_uniproc is set, then unpause all the processors
1139 	 * that were paused before resuming the drivers
1140 	 */
1141 	if (cpr_resume_uniproc) {
1142 		mutex_enter(&cpu_lock);
1143 		start_cpus();
1144 		mutex_exit(&cpu_lock);
1145 	}
1146 #endif
1147 
1148 	/*
1149 	 * If we had disabled modunloading in this cpr resume cycle (i.e. we
1150 	 * resumed from a state earlier than C_ST_SUSPEND_DEVICES), re-enable
1151 	 * modunloading now.
1152 	 */
1153 	if (CPR->c_substate != C_ST_SUSPEND_DEVICES) {
1154 		PMD(PMD_SX, ("cpr_resume: modload enable\n"))
1155 		modunload_enable();
1156 	}
1157 
1158 	/*
1159 	 * Hooks needed by lock manager prior to resuming.
1160 	 * Refer to code for more comments.
1161 	 */
1162 	PMD(PMD_SX, ("cpr_resume: lock mgr\n"))
1163 	cpr_lock_mgr(lm_cprresume);
1164 
1165 #if defined(__sparc)
1166 	/*
1167 	 * This is a partial (half) resume during cpr suspend, we
1168 	 * haven't yet given up on the suspend. On return from here,
1169 	 * cpr_suspend() will try to reallocate and retry the suspend.
1170 	 */
1171 	if (CPR->c_substate == C_ST_DUMP_NOSPC) {
1172 		return (0);
1173 	}
1174 
1175 	if (sleeptype == CPR_TODISK) {
1176 rb_statef_alloc:
1177 		cpr_statef_close();
1178 
1179 rb_disable_ufs_logging:
1180 		/*
1181 		 * if ufs logging was disabled, re-enable
1182 		 */
1183 		(void) cpr_ufs_logging(1);
1184 	}
1185 #endif
1186 
1187 rb_pm_reattach_noinvol:
1188 	/*
1189 	 * When pm_reattach_noinvol() succeeds, modunload_thread will
1190 	 * remain disabled until after cpr suspend passes the
1191 	 * C_ST_STOP_KERNEL_THREADS state. If any failure happens before
1192 	 * cpr suspend reaches this state, we'll need to enable modunload
1193 	 * thread during rollback.
1194 	 */
1195 	if (CPR->c_substate == C_ST_DISABLE_UFS_LOGGING ||
1196 	    CPR->c_substate == C_ST_STATEF_ALLOC ||
1197 	    CPR->c_substate == C_ST_SUSPEND_DEVICES ||
1198 	    CPR->c_substate == C_ST_STOP_KERNEL_THREADS) {
1199 		PMD(PMD_SX, ("cpr_resume: reattach noinvol fini\n"))
1200 		pm_reattach_noinvol_fini();
1201 	}
1202 
1203 	PMD(PMD_SX, ("cpr_resume: CPR POST USER callback\n"))
1204 	(void) callb_execute_class(CB_CL_CPR_POST_USER, CB_CODE_CPR_RESUME);
1205 	PMD(PMD_SX, ("cpr_resume: CPR PROMPRINTF callback\n"))
1206 	(void) callb_execute_class(CB_CL_CPR_PROMPRINTF, CB_CODE_CPR_RESUME);
1207 
1208 	PMD(PMD_SX, ("cpr_resume: restore direct levels\n"))
1209 	pm_restore_direct_levels();
1210 
1211 rb_stop_user_threads:
1212 	CPR_DEBUG(CPR_DEBUG1, "starting user threads...");
1213 	PMD(PMD_SX, ("cpr_resume: starting user threads\n"))
1214 	cpr_start_user_threads();
1215 	CPR_DEBUG(CPR_DEBUG1, "done\n");
1216 	/*
1217 	 * Ask Xorg to resume the frame buffer, and wait for it to happen
1218 	 */
1219 	mutex_enter(&srn_clone_lock);
1220 	if (srn_signal) {
1221 		PMD(PMD_SX, ("cpr_suspend: (*srn_signal)(..., "
1222 		    "SRN_NORMAL_RESUME)\n"))
1223 		srn_inuse = 1;		/* because (*srn_signal) cv_waits */
1224 		(*srn_signal)(SRN_TYPE_APM, SRN_NORMAL_RESUME);
1225 		srn_inuse = 0;
1226 	} else {
1227 		PMD(PMD_SX, ("cpr_suspend: srn_signal NULL\n"))
1228 	}
1229 	mutex_exit(&srn_clone_lock);
1230 
1231 #if defined(__sparc)
1232 rb_mp_offline:
1233 	if (cpr_mp_online())
1234 		cpr_err(CE_WARN, "Failed to online all the processors.");
1235 #endif
1236 
1237 rb_others:
1238 	PMD(PMD_SX, ("cpr_resume: dep thread\n"))
1239 	pm_dispatch_to_dep_thread(PM_DEP_WK_CPR_RESUME, NULL, NULL,
1240 	    PM_DEP_WAIT, NULL, 0);
1241 
1242 	PMD(PMD_SX, ("cpr_resume: CPR PM callback\n"))
1243 	(void) callb_execute_class(CB_CL_CPR_PM, CB_CODE_CPR_RESUME);
1244 
1245 	if (cpr_suspend_succeeded) {
1246 		cpr_stat_record_events();
1247 	}
1248 
1249 #if defined(__sparc)
1250 	if (sleeptype == CPR_TODISK && !cpr_reusable_mode)
1251 		cpr_clear_definfo();
1252 #endif
1253 
1254 	i_cpr_free_cpus();
1255 	CPR_DEBUG(CPR_DEBUG1, "Sending SIGTHAW...");
1256 	PMD(PMD_SX, ("cpr_resume: SIGTHAW\n"))
1257 	cpr_signal_user(SIGTHAW);
1258 	CPR_DEBUG(CPR_DEBUG1, "done\n");
1259 
1260 	CPR_STAT_EVENT_END("Resume Total");
1261 
1262 	CPR_STAT_EVENT_START_TMZ("WHOLE CYCLE", &wholecycle_tv);
1263 	CPR_STAT_EVENT_END("WHOLE CYCLE");
1264 
1265 	if (cpr_debug & CPR_DEBUG1)
1266 		cmn_err(CE_CONT, "\nThe system is back where you left!\n");
1267 
1268 	CPR_STAT_EVENT_START("POST CPR DELAY");
1269 
1270 #ifdef CPR_STAT
1271 	ctp = &cpr_term.tm_shutdown;
1272 	CPR_STAT_EVENT_START_TMZ("PWROFF TIME", ctp);
1273 	CPR_STAT_EVENT_END_TMZ("PWROFF TIME", &pwron_tv);
1274 
1275 	CPR_STAT_EVENT_PRINT();
1276 #endif /* CPR_STAT */
1277 
1278 	PMD(PMD_SX, ("cpr_resume returns %x\n", rc))
1279 	return (rc);
1280 }
1281 
1282 static void
1283 cpr_suspend_init(int sleeptype)
1284 {
1285 	cpr_time_t *ctp;
1286 
1287 	cpr_stat_init();
1288 
1289 	/*
1290 	 * If cpr_suspend() failed before cpr_dump() gets a chance
1291 	 * to reinitialize the terminator of the statefile,
1292 	 * the values of the old terminator will still linger around.
1293 	 * Since the terminator contains information that we need to
1294 	 * decide whether suspend succeeded or not, we need to
1295 	 * reinitialize it as early as possible.
1296 	 */
1297 	cpr_term.real_statef_size = 0;
1298 	ctp = &cpr_term.tm_shutdown;
1299 	bzero(ctp, sizeof (*ctp));
1300 	ctp = &cpr_term.tm_cprboot_start;
1301 	bzero(ctp, sizeof (*ctp));
1302 	ctp = &cpr_term.tm_cprboot_end;
1303 	bzero(ctp, sizeof (*ctp));
1304 
1305 	if (sleeptype == CPR_TODISK) {
1306 		/*
1307 		 * Lookup the physical address of our thread structure.
1308 		 * This should never be invalid and the entire thread structure
1309 		 * is expected to reside within the same pfn.
1310 		 */
1311 		curthreadpfn = hat_getpfnum(kas.a_hat, (caddr_t)curthread);
1312 		ASSERT(curthreadpfn != PFN_INVALID);
1313 		ASSERT(curthreadpfn == hat_getpfnum(kas.a_hat,
1314 		    (caddr_t)curthread + sizeof (kthread_t) - 1));
1315 	}
1316 
1317 	cpr_suspend_succeeded = 0;
1318 }
1319 
1320 /*
1321  * bring all the offline cpus online
1322  */
1323 static int
1324 cpr_all_online(void)
1325 {
1326 	int	rc = 0;
1327 
1328 #ifdef	__sparc
1329 	/*
1330 	 * do nothing
1331 	 */
1332 #else
1333 
1334 	cpu_t	*cp;
1335 
1336 	ASSERT(MUTEX_HELD(&cpu_lock));
1337 
1338 	cp = cpu_list;
1339 	do {
1340 		cp->cpu_cpr_flags &= ~CPU_CPR_ONLINE;
1341 		if (!CPU_ACTIVE(cp)) {
1342 			if ((rc = cpu_online(cp)) != 0)
1343 				break;
1344 			CPU_SET_CPR_FLAGS(cp, CPU_CPR_ONLINE);
1345 		}
1346 	} while ((cp = cp->cpu_next) != cpu_list);
1347 
1348 	if (rc) {
1349 		/*
1350 		 * an online operation failed so offline the cpus
1351 		 * that were onlined above to restore the system
1352 		 * to its original state
1353 		 */
1354 		cpr_restore_offline();
1355 	}
1356 #endif
1357 	return (rc);
1358 }
1359 
1360 /*
1361  * offline all the cpus that were brought online by cpr_all_online()
1362  */
1363 static void
1364 cpr_restore_offline(void)
1365 {
1366 
1367 #ifdef	__sparc
1368 	/*
1369 	 * do nothing
1370 	 */
1371 #else
1372 
1373 	cpu_t	*cp;
1374 	int	rc = 0;
1375 
1376 	ASSERT(MUTEX_HELD(&cpu_lock));
1377 
1378 	cp = cpu_list;
1379 	do {
1380 		if (CPU_CPR_IS_ONLINE(cp)) {
1381 			rc =  cpu_offline(cp, 0);
1382 			/*
1383 			 * this offline should work, since the cpu was
1384 			 * offline originally and was successfully onlined
1385 			 * by cpr_all_online()
1386 			 */
1387 			ASSERT(rc == 0);
1388 			cp->cpu_cpr_flags &= ~CPU_CPR_ONLINE;
1389 		}
1390 	} while ((cp = cp->cpu_next) != cpu_list);
1391 
1392 #endif
1393 
1394 }
1395