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 }