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 2008 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 * Copyright 2011 Joyent, Inc. All rights reserved. 25 * Copyright 2011 Nexenta Systems, Inc. All rights reserved. 26 */ 27 28 /* 29 * Power Button Driver 30 * 31 * This driver handles interrupt generated by the power button on 32 * platforms with "power" device node which has "button" property. 33 * Currently, these platforms are: 34 * 35 * ACPI-enabled x86/x64 platforms 36 * Ultra-5_10, Ultra-80, Sun-Blade-100, Sun-Blade-150, 37 * Sun-Blade-1500, Sun-Blade-2500, 38 * Sun-Fire-V210, Sun-Fire-V240, Netra-240 39 * 40 * Only one instance is allowed to attach. In order to know when 41 * an application that has opened the device is going away, a new 42 * minor clone is created for each open(9E) request. There are 43 * allocations for creating minor clones between 1 and 255. The ioctl 44 * interface is defined by pbio(4I) and approved as part of 45 * PSARC/1999/393 case. 46 */ 47 48 #include <sys/types.h> 49 #include <sys/conf.h> 50 #include <sys/ddi.h> 51 #include <sys/sunddi.h> 52 #include <sys/ddi_impldefs.h> 53 #include <sys/cmn_err.h> 54 #include <sys/errno.h> 55 #include <sys/modctl.h> 56 #include <sys/open.h> 57 #include <sys/stat.h> 58 #include <sys/poll.h> 59 #include <sys/pbio.h> 60 #include <sys/sysevent/eventdefs.h> 61 #include <sys/sysevent/pwrctl.h> 62 63 #if defined(__sparc) 64 #include <sys/machsystm.h> 65 #endif 66 67 #ifdef ACPI_POWER_BUTTON 68 69 #include <sys/acpi/acpi.h> 70 #include <sys/acpica.h> 71 72 #else 73 74 #include <sys/epic.h> 75 /* 76 * Some #defs that must be here as they differ for power.c 77 * and epic.c 78 */ 79 #define EPIC_REGS_OFFSET 0x00 80 #define EPIC_REGS_LEN 0x82 81 82 83 /* 84 * This flag, which is set for platforms, that have EPIC processor 85 * to process power button interrupt, helps in executing platform 86 * specific code. 87 */ 88 static char hasEPIC = B_FALSE; 89 #endif /* ACPI_POWER_BUTTON */ 90 91 /* 92 * Maximum number of clone minors that is allowed. This value 93 * is defined relatively low to save memory. 94 */ 95 #define POWER_MAX_CLONE 256 96 97 /* 98 * Minor number is instance << 8 + clone minor from range 1-255; clone 0 99 * is reserved for "original" minor. 100 */ 101 #define POWER_MINOR_TO_CLONE(minor) ((minor) & (POWER_MAX_CLONE - 1)) 102 103 /* 104 * Power Button Abort Delay 105 */ 106 #define ABORT_INCREMENT_DELAY 10 107 108 /* 109 * FWARC 2005/687: power device compatible property 110 */ 111 #define POWER_DEVICE_TYPE "power-device-type" 112 113 /* 114 * Driver global variables 115 */ 116 static void *power_state; 117 static int power_inst = -1; 118 119 static hrtime_t power_button_debounce = MSEC2NSEC(10); 120 static hrtime_t power_button_abort_interval = 1.5 * NANOSEC; 121 static int power_button_abort_presses = 3; 122 static int power_button_abort_enable = 1; 123 static int power_button_enable = 1; 124 125 static int power_button_pressed = 0; 126 static int power_button_cancel = 0; 127 static int power_button_timeouts = 0; 128 static int timeout_cancel = 0; 129 static int additional_presses = 0; 130 131 /* 132 * Function prototypes 133 */ 134 static int power_attach(dev_info_t *, ddi_attach_cmd_t); 135 static int power_detach(dev_info_t *, ddi_detach_cmd_t); 136 static int power_getinfo(dev_info_t *, ddi_info_cmd_t, void *, void **); 137 static int power_open(dev_t *, int, int, cred_t *); 138 static int power_close(dev_t, int, int, cred_t *); 139 static int power_ioctl(dev_t, int, intptr_t, int, cred_t *, int *); 140 static int power_chpoll(dev_t, short, int, short *, struct pollhead **); 141 #ifndef ACPI_POWER_BUTTON 142 static uint_t power_high_intr(caddr_t); 143 #endif 144 static uint_t power_soft_intr(caddr_t); 145 static uint_t power_issue_shutdown(caddr_t); 146 static void power_timeout(caddr_t); 147 static void power_log_message(void); 148 149 /* 150 * Structure used in the driver 151 */ 152 struct power_soft_state { 153 dev_info_t *dip; /* device info pointer */ 154 kmutex_t power_mutex; /* mutex lock */ 155 kmutex_t power_intr_mutex; /* interrupt mutex lock */ 156 ddi_iblock_cookie_t soft_iblock_cookie; /* holds interrupt cookie */ 157 ddi_iblock_cookie_t high_iblock_cookie; /* holds interrupt cookie */ 158 ddi_softintr_t softintr_id; /* soft interrupt id */ 159 uchar_t clones[POWER_MAX_CLONE]; /* array of minor clones */ 160 int monitor_on; /* clone monitoring the button event */ 161 /* clone 0 indicates no one is */ 162 /* monitoring the button event */ 163 pollhead_t pollhd; /* poll head struct */ 164 int events; /* bit map of occured events */ 165 int shutdown_pending; /* system shutdown in progress */ 166 #ifdef ACPI_POWER_BUTTON 167 boolean_t fixed_attached; /* true means fixed is attached */ 168 boolean_t gpe_attached; /* true means GPE is attached */ 169 ACPI_HANDLE button_obj; /* handle to device power button */ 170 #else 171 ddi_acc_handle_t power_rhandle; /* power button register handle */ 172 uint8_t *power_btn_reg; /* power button register address */ 173 uint8_t power_btn_bit; /* power button register bit */ 174 boolean_t power_regs_mapped; /* flag to tell if regs mapped */ 175 boolean_t power_btn_ioctl; /* flag to specify ioctl request */ 176 #endif 177 }; 178 179 static void power_gen_sysevent(struct power_soft_state *); 180 181 #ifdef ACPI_POWER_BUTTON 182 static int power_attach_acpi(struct power_soft_state *softsp); 183 static void power_detach_acpi(struct power_soft_state *softsp); 184 static UINT32 power_acpi_fixed_event(void *ctx); 185 #else 186 static int power_setup_regs(struct power_soft_state *softsp); 187 static void power_free_regs(struct power_soft_state *softsp); 188 #endif /* ACPI_POWER_BUTTON */ 189 190 /* 191 * Configuration data structures 192 */ 193 static struct cb_ops power_cb_ops = { 194 power_open, /* open */ 195 power_close, /* close */ 196 nodev, /* strategy */ 197 nodev, /* print */ 198 nodev, /* dump */ 199 nodev, /* read */ 200 nodev, /* write */ 201 power_ioctl, /* ioctl */ 202 nodev, /* devmap */ 203 nodev, /* mmap */ 204 nodev, /* segmap */ 205 power_chpoll, /* poll */ 206 ddi_prop_op, /* cb_prop_op */ 207 NULL, /* streamtab */ 208 D_MP | D_NEW, /* Driver compatibility flag */ 209 CB_REV, /* rev */ 210 nodev, /* cb_aread */ 211 nodev /* cb_awrite */ 212 }; 213 214 static struct dev_ops power_ops = { 215 DEVO_REV, /* devo_rev, */ 216 0, /* refcnt */ 217 power_getinfo, /* getinfo */ 218 nulldev, /* identify */ 219 nulldev, /* probe */ 220 power_attach, /* attach */ 221 power_detach, /* detach */ 222 nodev, /* reset */ 223 &power_cb_ops, /* cb_ops */ 224 (struct bus_ops *)NULL, /* bus_ops */ 225 NULL, /* power */ 226 ddi_quiesce_not_supported, /* devo_quiesce */ 227 }; 228 229 static struct modldrv modldrv = { 230 &mod_driverops, /* Type of module. This one is a driver */ 231 "power button driver", /* name of module */ 232 &power_ops, /* driver ops */ 233 }; 234 235 static struct modlinkage modlinkage = { 236 MODREV_1, 237 (void *)&modldrv, 238 NULL 239 }; 240 241 /* 242 * These are the module initialization routines. 243 */ 244 245 int 246 _init(void) 247 { 248 int error; 249 250 if ((error = ddi_soft_state_init(&power_state, 251 sizeof (struct power_soft_state), 0)) != 0) 252 return (error); 253 254 if ((error = mod_install(&modlinkage)) != 0) 255 ddi_soft_state_fini(&power_state); 256 257 return (error); 258 } 259 260 int 261 _fini(void) 262 { 263 int error; 264 265 if ((error = mod_remove(&modlinkage)) == 0) 266 ddi_soft_state_fini(&power_state); 267 268 return (error); 269 } 270 271 int 272 _info(struct modinfo *modinfop) 273 { 274 return (mod_info(&modlinkage, modinfop)); 275 } 276 277 /*ARGSUSED*/ 278 static int 279 power_getinfo(dev_info_t *dip, ddi_info_cmd_t infocmd, void *arg, 280 void **result) 281 { 282 struct power_soft_state *softsp; 283 284 if (power_inst == -1) 285 return (DDI_FAILURE); 286 287 switch (infocmd) { 288 case DDI_INFO_DEVT2DEVINFO: 289 if ((softsp = ddi_get_soft_state(power_state, power_inst)) 290 == NULL) 291 return (DDI_FAILURE); 292 *result = (void *)softsp->dip; 293 return (DDI_SUCCESS); 294 295 case DDI_INFO_DEVT2INSTANCE: 296 *result = (void *)(uintptr_t)power_inst; 297 return (DDI_SUCCESS); 298 299 default: 300 return (DDI_FAILURE); 301 } 302 } 303 304 static int 305 power_attach(dev_info_t *dip, ddi_attach_cmd_t cmd) 306 { 307 struct power_soft_state *softsp; 308 309 switch (cmd) { 310 case DDI_ATTACH: 311 break; 312 case DDI_RESUME: 313 return (DDI_SUCCESS); 314 default: 315 return (DDI_FAILURE); 316 } 317 318 /* 319 * If the power node doesn't have "button" property, quietly 320 * fail to attach. 321 */ 322 if (ddi_prop_exists(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS, 323 "button") == 0) 324 return (DDI_FAILURE); 325 326 if (power_inst != -1) 327 return (DDI_FAILURE); 328 329 power_inst = ddi_get_instance(dip); 330 331 if (ddi_soft_state_zalloc(power_state, power_inst) != DDI_SUCCESS) 332 return (DDI_FAILURE); 333 334 if (ddi_create_minor_node(dip, "power_button", S_IFCHR, 335 (power_inst << 8) + 0, "ddi_power_button", 0) != DDI_SUCCESS) 336 return (DDI_FAILURE); 337 338 softsp = ddi_get_soft_state(power_state, power_inst); 339 softsp->dip = dip; 340 341 #ifdef ACPI_POWER_BUTTON 342 (void) power_attach_acpi(softsp); 343 #else 344 if (power_setup_regs(softsp) != DDI_SUCCESS) { 345 cmn_err(CE_WARN, "power_attach: failed to setup registers"); 346 goto error; 347 } 348 349 if (ddi_get_iblock_cookie(dip, 0, 350 &softsp->high_iblock_cookie) != DDI_SUCCESS) { 351 cmn_err(CE_WARN, "power_attach: ddi_get_soft_iblock_cookie " 352 "failed."); 353 goto error; 354 } 355 mutex_init(&softsp->power_intr_mutex, NULL, MUTEX_DRIVER, 356 softsp->high_iblock_cookie); 357 358 if (ddi_add_intr(dip, 0, &softsp->high_iblock_cookie, NULL, 359 power_high_intr, (caddr_t)softsp) != DDI_SUCCESS) { 360 cmn_err(CE_WARN, "power_attach: failed to add high-level " 361 " interrupt handler."); 362 mutex_destroy(&softsp->power_intr_mutex); 363 goto error; 364 } 365 #endif /* ACPI_POWER_BUTTON */ 366 367 if (ddi_get_soft_iblock_cookie(dip, DDI_SOFTINT_LOW, 368 &softsp->soft_iblock_cookie) != DDI_SUCCESS) { 369 cmn_err(CE_WARN, "power_attach: ddi_get_soft_iblock_cookie " 370 "failed."); 371 mutex_destroy(&softsp->power_intr_mutex); 372 ddi_remove_intr(dip, 0, NULL); 373 goto error; 374 } 375 376 mutex_init(&softsp->power_mutex, NULL, MUTEX_DRIVER, 377 (void *)softsp->soft_iblock_cookie); 378 379 if (ddi_add_softintr(dip, DDI_SOFTINT_LOW, &softsp->softintr_id, 380 NULL, NULL, power_soft_intr, (caddr_t)softsp) != DDI_SUCCESS) { 381 cmn_err(CE_WARN, "power_attach: failed to add soft " 382 "interrupt handler."); 383 mutex_destroy(&softsp->power_mutex); 384 mutex_destroy(&softsp->power_intr_mutex); 385 ddi_remove_intr(dip, 0, NULL); 386 goto error; 387 } 388 389 ddi_report_dev(dip); 390 391 return (DDI_SUCCESS); 392 393 error: 394 #ifdef ACPI_POWER_BUTTON 395 /* 396 * detach ACPI power button 397 */ 398 power_detach_acpi(softsp); 399 #else 400 power_free_regs(softsp); 401 #endif /* ACPI_POWER_BUTTON */ 402 ddi_remove_minor_node(dip, "power_button"); 403 ddi_soft_state_free(power_state, power_inst); 404 return (DDI_FAILURE); 405 } 406 407 /*ARGSUSED*/ 408 /* 409 * This driver doesn't detach. 410 */ 411 static int 412 power_detach(dev_info_t *dip, ddi_detach_cmd_t cmd) 413 { 414 /* 415 * Since the "power" node has "reg" property, as part of 416 * the suspend operation, detach(9E) entry point is called. 417 * There is no state to save, since this register is used 418 * by OBP to power off the system and the state of the 419 * power off is preserved by hardware. 420 */ 421 return ((cmd == DDI_SUSPEND) ? DDI_SUCCESS : 422 DDI_FAILURE); 423 } 424 425 426 #ifndef ACPI_POWER_BUTTON 427 /* 428 * Handler for the high-level interrupt. 429 */ 430 static uint_t 431 power_high_intr(caddr_t arg) 432 { 433 struct power_soft_state *softsp = (struct power_soft_state *)arg; 434 ddi_acc_handle_t hdl = softsp->power_rhandle; 435 uint8_t reg; 436 437 hrtime_t tstamp; 438 static hrtime_t o_tstamp = 0; 439 static hrtime_t power_button_tstamp = 0; 440 static int power_button_cnt; 441 442 if (softsp->power_regs_mapped) { 443 mutex_enter(&softsp->power_intr_mutex); 444 445 /* Check if power button interrupt is delivered by EPIC HW */ 446 if (hasEPIC) { 447 /* read isr - first issue command */ 448 EPIC_WR(hdl, softsp->power_btn_reg, 449 EPIC_ATOM_INTR_READ); 450 /* next, read the reg */ 451 EPIC_RD(hdl, softsp->power_btn_reg, reg); 452 453 if (reg & EPIC_FIRE_INTERRUPT) { /* PB pressed */ 454 /* clear the interrupt */ 455 EPIC_WR(hdl, softsp->power_btn_reg, 456 EPIC_ATOM_INTR_CLEAR); 457 } else { 458 if (!softsp->power_btn_ioctl) { 459 mutex_exit(&softsp->power_intr_mutex); 460 return (DDI_INTR_CLAIMED); 461 } 462 softsp->power_btn_ioctl = B_FALSE; 463 } 464 } else { 465 reg = ddi_get8(hdl, softsp->power_btn_reg); 466 if (reg & softsp->power_btn_bit) { 467 reg &= softsp->power_btn_bit; 468 ddi_put8(hdl, softsp->power_btn_reg, reg); 469 (void) ddi_get8(hdl, softsp->power_btn_reg); 470 } else { 471 if (!softsp->power_btn_ioctl) { 472 mutex_exit(&softsp->power_intr_mutex); 473 return (DDI_INTR_CLAIMED); 474 } 475 softsp->power_btn_ioctl = B_FALSE; 476 } 477 } 478 mutex_exit(&softsp->power_intr_mutex); 479 } 480 481 tstamp = gethrtime(); 482 483 /* need to deal with power button debounce */ 484 if (o_tstamp && (tstamp - o_tstamp) < power_button_debounce) { 485 o_tstamp = tstamp; 486 return (DDI_INTR_CLAIMED); 487 } 488 o_tstamp = tstamp; 489 490 power_button_cnt++; 491 492 mutex_enter(&softsp->power_intr_mutex); 493 power_button_pressed++; 494 mutex_exit(&softsp->power_intr_mutex); 495 496 /* 497 * If power button abort is enabled and power button was pressed 498 * power_button_abort_presses times within power_button_abort_interval 499 * then call abort_sequence_enter(); 500 */ 501 if (power_button_abort_enable) { 502 if (power_button_abort_presses == 1 || 503 tstamp < (power_button_tstamp + 504 power_button_abort_interval)) { 505 if (power_button_cnt == power_button_abort_presses) { 506 mutex_enter(&softsp->power_intr_mutex); 507 power_button_cancel += power_button_timeouts; 508 power_button_pressed = 0; 509 mutex_exit(&softsp->power_intr_mutex); 510 power_button_cnt = 0; 511 abort_sequence_enter("Power Button Abort"); 512 return (DDI_INTR_CLAIMED); 513 } 514 } else { 515 power_button_cnt = 1; 516 power_button_tstamp = tstamp; 517 } 518 } 519 520 if (!power_button_enable) 521 return (DDI_INTR_CLAIMED); 522 523 /* post softint to issue timeout for power button action */ 524 if (softsp->softintr_id != NULL) 525 ddi_trigger_softintr(softsp->softintr_id); 526 527 return (DDI_INTR_CLAIMED); 528 } 529 #endif /* ifndef ACPI_POWER_BUTTON */ 530 531 /* 532 * Handle the softints.... 533 * 534 * If only one softint is posted for several button presses, record 535 * the number of additional presses just incase this was actually not quite 536 * an Abort sequence so that we can log this event later. 537 * 538 * Issue a timeout with a duration being a fraction larger than 539 * the specified Abort interval inorder to perform a power down if required. 540 */ 541 static uint_t 542 power_soft_intr(caddr_t arg) 543 { 544 struct power_soft_state *softsp = (struct power_soft_state *)arg; 545 546 if (!power_button_abort_enable) 547 return (power_issue_shutdown(arg)); 548 549 mutex_enter(&softsp->power_intr_mutex); 550 if (!power_button_pressed) { 551 mutex_exit(&softsp->power_intr_mutex); 552 return (DDI_INTR_CLAIMED); 553 } 554 555 /* 556 * Schedule a timeout to do the necessary 557 * work for shutdown, only one timeout for 558 * n presses if power button was pressed 559 * more than once before softint fired 560 */ 561 if (power_button_pressed > 1) 562 additional_presses += power_button_pressed - 1; 563 564 timeout_cancel = 0; 565 power_button_pressed = 0; 566 power_button_timeouts++; 567 mutex_exit(&softsp->power_intr_mutex); 568 (void) timeout((void(*)(void *))power_timeout, 569 softsp, NSEC_TO_TICK(power_button_abort_interval) + 570 ABORT_INCREMENT_DELAY); 571 572 return (DDI_INTR_CLAIMED); 573 } 574 575 /* 576 * Upon receiving a timeout the following is determined: 577 * 578 * If an Abort sequence was issued, then we cancel all outstanding timeouts 579 * and additional presses prior to the Abort sequence. 580 * 581 * If we had multiple timeouts issued and the abort sequence was not met, 582 * then we had more than one button press to power down the machine. We 583 * were probably trying to issue an abort. So log a message indicating this 584 * and cancel all outstanding timeouts. 585 * 586 * If we had just one timeout and the abort sequence was not met then 587 * we really did want to power down the machine, so call power_issue_shutdown() 588 * to do the work and schedule a power down 589 */ 590 static void 591 power_timeout(caddr_t arg) 592 { 593 struct power_soft_state *softsp = (struct power_soft_state *)arg; 594 static int first = 0; 595 596 /* 597 * Abort was generated cancel all outstanding power 598 * button timeouts 599 */ 600 mutex_enter(&softsp->power_intr_mutex); 601 if (power_button_cancel) { 602 power_button_cancel--; 603 power_button_timeouts--; 604 if (!first) { 605 first++; 606 additional_presses = 0; 607 } 608 mutex_exit(&softsp->power_intr_mutex); 609 return; 610 } 611 first = 0; 612 613 /* 614 * We get here if the timeout(s) have fired and they were 615 * not issued prior to an abort. 616 * 617 * If we had more than one press in the interval we were 618 * probably trying to issue an abort, but didnt press the 619 * required number within the interval. Hence cancel all 620 * timeouts and do not continue towards shutdown. 621 */ 622 if (!timeout_cancel) { 623 timeout_cancel = power_button_timeouts + 624 additional_presses; 625 626 power_button_timeouts--; 627 if (!power_button_timeouts) 628 additional_presses = 0; 629 630 if (timeout_cancel > 1) { 631 mutex_exit(&softsp->power_intr_mutex); 632 cmn_err(CE_NOTE, "Power Button pressed " 633 "%d times, cancelling all requests", 634 timeout_cancel); 635 return; 636 } 637 mutex_exit(&softsp->power_intr_mutex); 638 639 /* Go and do the work to request shutdown */ 640 (void) power_issue_shutdown((caddr_t)softsp); 641 return; 642 } 643 644 power_button_timeouts--; 645 if (!power_button_timeouts) 646 additional_presses = 0; 647 mutex_exit(&softsp->power_intr_mutex); 648 } 649 650 #ifdef ACPI_POWER_BUTTON 651 static void 652 do_shutdown(void) 653 { 654 proc_t *initpp; 655 656 /* 657 * If we're still booting and init(1) isn't set up yet, simply halt. 658 */ 659 mutex_enter(&pidlock); 660 initpp = prfind(P_INITPID); 661 mutex_exit(&pidlock); 662 if (initpp == NULL) { 663 extern void halt(char *); 664 halt("Power off the System"); /* just in case */ 665 } 666 667 /* 668 * else, graceful shutdown with inittab and all getting involved 669 */ 670 psignal(initpp, SIGPWR); 671 } 672 #endif 673 674 static uint_t 675 power_issue_shutdown(caddr_t arg) 676 { 677 struct power_soft_state *softsp = (struct power_soft_state *)arg; 678 679 mutex_enter(&softsp->power_mutex); 680 softsp->events |= PB_BUTTON_PRESS; 681 if (softsp->monitor_on != 0) { 682 mutex_exit(&softsp->power_mutex); 683 pollwakeup(&softsp->pollhd, POLLRDNORM); 684 pollwakeup(&softsp->pollhd, POLLIN); 685 power_gen_sysevent(softsp); 686 return (DDI_INTR_CLAIMED); 687 } 688 689 if (!softsp->shutdown_pending) { 690 cmn_err(CE_WARN, "Power off requested from power button or " 691 "SC, powering down the system!"); 692 softsp->shutdown_pending = 1; 693 do_shutdown(); 694 695 /* 696 * Wait a while for "do_shutdown()" to shut down the system 697 * before logging an error message. 698 */ 699 (void) timeout((void(*)(void *))power_log_message, NULL, 700 100 * hz); 701 } 702 mutex_exit(&softsp->power_mutex); 703 704 return (DDI_INTR_CLAIMED); 705 } 706 707 static void 708 power_gen_sysevent(struct power_soft_state *softsp) 709 { 710 nvlist_t *attr_list = NULL; 711 int err; 712 char pathname[MAXPATHLEN]; 713 char hid[9] = "\0"; 714 715 /* Allocate and build sysevent attribute list */ 716 err = nvlist_alloc(&attr_list, NV_UNIQUE_NAME_TYPE, DDI_NOSLEEP); 717 if (err != 0) { 718 cmn_err(CE_WARN, 719 "cannot allocate memory for sysevent attributes\n"); 720 return; 721 } 722 723 #ifdef ACPI_POWER_BUTTON 724 /* Only control method power button has HID */ 725 if (softsp->gpe_attached) { 726 (void) strlcpy(hid, "PNP0C0C", sizeof (hid)); 727 } 728 #endif 729 730 err = nvlist_add_string(attr_list, PWRCTL_DEV_HID, hid); 731 if (err != 0) { 732 cmn_err(CE_WARN, 733 "Failed to add attr [%s] for %s/%s event", 734 PWRCTL_DEV_HID, EC_PWRCTL, ESC_PWRCTL_POWER_BUTTON); 735 nvlist_free(attr_list); 736 return; 737 } 738 739 (void) ddi_pathname(softsp->dip, pathname); 740 err = nvlist_add_string(attr_list, PWRCTL_DEV_PHYS_PATH, pathname); 741 if (err != 0) { 742 cmn_err(CE_WARN, 743 "Failed to add attr [%s] for %s/%s event", 744 PWRCTL_DEV_PHYS_PATH, EC_PWRCTL, ESC_PWRCTL_POWER_BUTTON); 745 nvlist_free(attr_list); 746 return; 747 } 748 749 /* Generate/log sysevent */ 750 err = ddi_log_sysevent(softsp->dip, DDI_VENDOR_SUNW, EC_PWRCTL, 751 ESC_PWRCTL_POWER_BUTTON, attr_list, NULL, DDI_NOSLEEP); 752 if (err != DDI_SUCCESS) { 753 cmn_err(CE_WARN, 754 "cannot log sysevent, err code %x\n", err); 755 } 756 757 nvlist_free(attr_list); 758 } 759 760 /* 761 * Open the device. 762 */ 763 /*ARGSUSED*/ 764 static int 765 power_open(dev_t *devp, int openflags, int otyp, cred_t *credp) 766 { 767 struct power_soft_state *softsp; 768 int clone; 769 770 if (otyp != OTYP_CHR) 771 return (EINVAL); 772 773 if ((softsp = ddi_get_soft_state(power_state, power_inst)) == 774 NULL) 775 return (ENXIO); 776 777 mutex_enter(&softsp->power_mutex); 778 for (clone = 1; clone < POWER_MAX_CLONE; clone++) 779 if (!softsp->clones[clone]) 780 break; 781 782 if (clone == POWER_MAX_CLONE) { 783 cmn_err(CE_WARN, "power_open: No more allocation left " 784 "to create a clone minor."); 785 mutex_exit(&softsp->power_mutex); 786 return (ENXIO); 787 } 788 789 *devp = makedevice(getmajor(*devp), (power_inst << 8) + clone); 790 softsp->clones[clone] = 1; 791 mutex_exit(&softsp->power_mutex); 792 793 return (0); 794 } 795 796 /* 797 * Close the device. 798 */ 799 /*ARGSUSED*/ 800 static int 801 power_close(dev_t dev, int openflags, int otyp, cred_t *credp) 802 { 803 struct power_soft_state *softsp; 804 int clone; 805 806 if (otyp != OTYP_CHR) 807 return (EINVAL); 808 809 if ((softsp = ddi_get_soft_state(power_state, power_inst)) == 810 NULL) 811 return (ENXIO); 812 813 clone = POWER_MINOR_TO_CLONE(getminor(dev)); 814 mutex_enter(&softsp->power_mutex); 815 if (softsp->monitor_on == clone) 816 softsp->monitor_on = 0; 817 softsp->clones[clone] = 0; 818 mutex_exit(&softsp->power_mutex); 819 820 return (0); 821 } 822 823 /*ARGSUSED*/ 824 static int 825 power_ioctl(dev_t dev, int cmd, intptr_t arg, int mode, cred_t *cred_p, 826 int *rval_p) 827 { 828 struct power_soft_state *softsp; 829 int clone; 830 831 if ((softsp = ddi_get_soft_state(power_state, power_inst)) == 832 NULL) 833 return (ENXIO); 834 835 clone = POWER_MINOR_TO_CLONE(getminor(dev)); 836 switch (cmd) { 837 case PB_BEGIN_MONITOR: 838 mutex_enter(&softsp->power_mutex); 839 if (softsp->monitor_on) { 840 mutex_exit(&softsp->power_mutex); 841 return (EBUSY); 842 } 843 softsp->monitor_on = clone; 844 mutex_exit(&softsp->power_mutex); 845 return (0); 846 847 case PB_END_MONITOR: 848 mutex_enter(&softsp->power_mutex); 849 850 /* 851 * If PB_END_MONITOR is called without first 852 * calling PB_BEGIN_MONITOR, an error will be 853 * returned. 854 */ 855 if (!softsp->monitor_on) { 856 mutex_exit(&softsp->power_mutex); 857 return (ENXIO); 858 } 859 860 /* 861 * This clone is not monitoring the button. 862 */ 863 if (softsp->monitor_on != clone) { 864 mutex_exit(&softsp->power_mutex); 865 return (EINVAL); 866 } 867 softsp->monitor_on = 0; 868 mutex_exit(&softsp->power_mutex); 869 return (0); 870 871 case PB_GET_EVENTS: 872 mutex_enter(&softsp->power_mutex); 873 if (ddi_copyout((void *)&softsp->events, (void *)arg, 874 sizeof (int), mode) != 0) { 875 mutex_exit(&softsp->power_mutex); 876 return (EFAULT); 877 } 878 879 /* 880 * This ioctl returned the events detected since last 881 * call. Note that any application can get the events 882 * and clear the event register. 883 */ 884 softsp->events = 0; 885 mutex_exit(&softsp->power_mutex); 886 return (0); 887 888 /* 889 * This ioctl is used by the test suite. 890 */ 891 case PB_CREATE_BUTTON_EVENT: 892 #ifdef ACPI_POWER_BUTTON 893 (UINT32)power_acpi_fixed_event((void *)softsp); 894 #else 895 if (softsp->power_regs_mapped) { 896 mutex_enter(&softsp->power_intr_mutex); 897 softsp->power_btn_ioctl = B_TRUE; 898 mutex_exit(&softsp->power_intr_mutex); 899 } 900 (void) power_high_intr((caddr_t)softsp); 901 #endif /* ACPI_POWER_BUTTON */ 902 return (0); 903 904 default: 905 return (ENOTTY); 906 } 907 } 908 909 /*ARGSUSED*/ 910 static int 911 power_chpoll(dev_t dev, short events, int anyyet, 912 short *reventsp, struct pollhead **phpp) 913 { 914 struct power_soft_state *softsp; 915 916 if ((softsp = ddi_get_soft_state(power_state, power_inst)) == NULL) 917 return (ENXIO); 918 919 mutex_enter(&softsp->power_mutex); 920 *reventsp = 0; 921 if (softsp->events) 922 *reventsp = POLLRDNORM|POLLIN; 923 else { 924 if (!anyyet) 925 *phpp = &softsp->pollhd; 926 } 927 mutex_exit(&softsp->power_mutex); 928 929 return (0); 930 } 931 932 static void 933 power_log_message(void) 934 { 935 struct power_soft_state *softsp; 936 937 if ((softsp = ddi_get_soft_state(power_state, power_inst)) == NULL) { 938 cmn_err(CE_WARN, "Failed to get internal state!"); 939 return; 940 } 941 942 mutex_enter(&softsp->power_mutex); 943 softsp->shutdown_pending = 0; 944 cmn_err(CE_WARN, "Failed to shut down the system!"); 945 mutex_exit(&softsp->power_mutex); 946 } 947 948 #ifdef ACPI_POWER_BUTTON 949 /* 950 * 951 */ 952 /*ARGSUSED*/ 953 static ACPI_STATUS 954 acpi_device(ACPI_HANDLE obj, UINT32 nesting, void *context, void **rv) 955 { 956 957 *((ACPI_HANDLE *)context) = obj; 958 return (AE_OK); 959 } 960 961 /* 962 * 963 */ 964 static ACPI_HANDLE 965 probe_acpi_pwrbutton() 966 { 967 ACPI_HANDLE obj = NULL; 968 969 (void) AcpiGetDevices("PNP0C0C", acpi_device, (void *)&obj, NULL); 970 return (obj); 971 } 972 973 static UINT32 974 power_acpi_fixed_event(void *ctx) 975 { 976 977 mutex_enter(&((struct power_soft_state *)ctx)->power_intr_mutex); 978 power_button_pressed++; 979 mutex_exit(&((struct power_soft_state *)ctx)->power_intr_mutex); 980 981 /* post softint to issue timeout for power button action */ 982 if (((struct power_soft_state *)ctx)->softintr_id != NULL) 983 ddi_trigger_softintr( 984 ((struct power_soft_state *)ctx)->softintr_id); 985 986 return (AE_OK); 987 } 988 989 /*ARGSUSED*/ 990 static void 991 power_acpi_notify_event(ACPI_HANDLE obj, UINT32 val, void *ctx) 992 { 993 if (val == 0x80) 994 (void) power_acpi_fixed_event(ctx); 995 } 996 997 /* 998 * 999 */ 1000 static int 1001 power_probe_method_button(struct power_soft_state *softsp) 1002 { 1003 ACPI_HANDLE button_obj; 1004 1005 button_obj = probe_acpi_pwrbutton(); 1006 softsp->button_obj = button_obj; /* remember obj */ 1007 if ((button_obj != NULL) && 1008 (AcpiInstallNotifyHandler(button_obj, ACPI_DEVICE_NOTIFY, 1009 power_acpi_notify_event, (void*)softsp) == AE_OK)) 1010 return (1); 1011 return (0); 1012 } 1013 1014 /* 1015 * 1016 */ 1017 static int 1018 power_probe_fixed_button(struct power_soft_state *softsp) 1019 { 1020 ACPI_TABLE_FADT *fadt; 1021 1022 if (AcpiGetTable(ACPI_SIG_FADT, 1, (ACPI_TABLE_HEADER **) &fadt) != 1023 AE_OK) 1024 return (0); 1025 1026 if ((fadt->Flags & ACPI_FADT_POWER_BUTTON) == 0) { 1027 if (AcpiInstallFixedEventHandler(ACPI_EVENT_POWER_BUTTON, 1028 power_acpi_fixed_event, (void *)softsp) == AE_OK) 1029 return (1); 1030 } 1031 return (0); 1032 } 1033 1034 1035 /* 1036 * 1037 */ 1038 static int 1039 power_attach_acpi(struct power_soft_state *softsp) 1040 { 1041 1042 /* 1043 * If we've attached anything already, return an error 1044 */ 1045 if ((softsp->gpe_attached) || (softsp->fixed_attached)) 1046 return (DDI_FAILURE); 1047 1048 /* 1049 * attempt to attach both a fixed-event handler and a GPE 1050 * handler; remember what we got 1051 */ 1052 softsp->fixed_attached = (power_probe_fixed_button(softsp) != 0); 1053 softsp->gpe_attached = (power_probe_method_button(softsp) != 0); 1054 1055 /* 1056 * If we've attached anything now, return success 1057 */ 1058 if ((softsp->gpe_attached) || (softsp->fixed_attached)) 1059 return (DDI_SUCCESS); 1060 1061 return (DDI_FAILURE); 1062 } 1063 1064 /* 1065 * 1066 */ 1067 static void 1068 power_detach_acpi(struct power_soft_state *softsp) 1069 { 1070 if (softsp->gpe_attached) { 1071 if (AcpiRemoveNotifyHandler(softsp->button_obj, 1072 ACPI_DEVICE_NOTIFY, power_acpi_notify_event) != AE_OK) 1073 cmn_err(CE_WARN, "!power: failed to remove Notify" 1074 " handler"); 1075 } 1076 1077 if (softsp->fixed_attached) { 1078 if (AcpiRemoveFixedEventHandler(ACPI_EVENT_POWER_BUTTON, 1079 power_acpi_fixed_event) != AE_OK) 1080 cmn_err(CE_WARN, "!power: failed to remove Power" 1081 " Button handler"); 1082 } 1083 } 1084 1085 #else 1086 /* 1087 * Code for platforms that have EPIC processor for processing power 1088 * button interrupts. 1089 */ 1090 static int 1091 power_setup_epic_regs(dev_info_t *dip, struct power_soft_state *softsp) 1092 { 1093 ddi_device_acc_attr_t attr; 1094 uint8_t *reg_base; 1095 1096 attr.devacc_attr_version = DDI_DEVICE_ATTR_V0; 1097 attr.devacc_attr_endian_flags = DDI_STRUCTURE_LE_ACC; 1098 attr.devacc_attr_dataorder = DDI_STRICTORDER_ACC; 1099 if (ddi_regs_map_setup(dip, 0, (caddr_t *)®_base, 1100 EPIC_REGS_OFFSET, EPIC_REGS_LEN, &attr, 1101 &softsp->power_rhandle) != DDI_SUCCESS) { 1102 return (DDI_FAILURE); 1103 } 1104 1105 softsp->power_btn_reg = reg_base; 1106 softsp->power_regs_mapped = B_TRUE; 1107 1108 /* Clear power button interrupt first */ 1109 EPIC_WR(softsp->power_rhandle, softsp->power_btn_reg, 1110 EPIC_ATOM_INTR_CLEAR); 1111 1112 /* Enable EPIC interrupt for power button single press event */ 1113 EPIC_WR(softsp->power_rhandle, softsp->power_btn_reg, 1114 EPIC_ATOM_INTR_ENABLE); 1115 1116 /* 1117 * At this point, EPIC interrupt processing is fully initialised. 1118 */ 1119 hasEPIC = B_TRUE; 1120 return (DDI_SUCCESS); 1121 } 1122 1123 /* 1124 * 1125 * power button register definitions for acpi register on m1535d 1126 */ 1127 #define M1535D_PWR_BTN_REG_01 0x1 1128 #define M1535D_PWR_BTN_EVENT_FLAG 0x1 1129 1130 static int 1131 power_setup_m1535_regs(dev_info_t *dip, struct power_soft_state *softsp) 1132 { 1133 ddi_device_acc_attr_t attr; 1134 uint8_t *reg_base; 1135 1136 attr.devacc_attr_version = DDI_DEVICE_ATTR_V0; 1137 attr.devacc_attr_endian_flags = DDI_STRUCTURE_LE_ACC; 1138 attr.devacc_attr_dataorder = DDI_STRICTORDER_ACC; 1139 if (ddi_regs_map_setup(dip, 0, (caddr_t *)®_base, 0, 0, &attr, 1140 &softsp->power_rhandle) != DDI_SUCCESS) { 1141 return (DDI_FAILURE); 1142 } 1143 softsp->power_btn_reg = ®_base[M1535D_PWR_BTN_REG_01]; 1144 softsp->power_btn_bit = M1535D_PWR_BTN_EVENT_FLAG; 1145 softsp->power_regs_mapped = B_TRUE; 1146 return (DDI_SUCCESS); 1147 } 1148 1149 /* 1150 * MBC Fire/SSI Interrupt Status Register definitions 1151 */ 1152 #define FIRE_SSI_ISR 0x0 1153 #define FIRE_SSI_INTR_ENA 0x8 1154 #define FIRE_SSI_SHUTDOWN_REQ 0x4 1155 1156 static int 1157 power_setup_mbc_regs(dev_info_t *dip, struct power_soft_state *softsp) 1158 { 1159 ddi_device_acc_attr_t attr; 1160 uint8_t *reg_base; 1161 ddi_acc_handle_t hdl; 1162 uint8_t reg; 1163 1164 attr.devacc_attr_version = DDI_DEVICE_ATTR_V0; 1165 attr.devacc_attr_endian_flags = DDI_STRUCTURE_LE_ACC; 1166 attr.devacc_attr_dataorder = DDI_STRICTORDER_ACC; 1167 if (ddi_regs_map_setup(dip, 0, (caddr_t *)®_base, 0, 0, &attr, 1168 &softsp->power_rhandle) != DDI_SUCCESS) { 1169 return (DDI_FAILURE); 1170 } 1171 softsp->power_btn_reg = ®_base[FIRE_SSI_ISR]; 1172 softsp->power_btn_bit = FIRE_SSI_SHUTDOWN_REQ; 1173 hdl = softsp->power_rhandle; 1174 /* 1175 * Clear MBC Fire Power Button interrupt, if set. 1176 */ 1177 reg = ddi_get8(hdl, softsp->power_btn_reg); 1178 if (reg & softsp->power_btn_bit) { 1179 reg &= softsp->power_btn_bit; 1180 ddi_put8(hdl, softsp->power_btn_reg, reg); 1181 (void) ddi_get8(hdl, softsp->power_btn_reg); 1182 } 1183 /* 1184 * Enable MBC Fire Power Button interrupt. 1185 */ 1186 reg = ddi_get8(hdl, ®_base[FIRE_SSI_INTR_ENA]); 1187 reg |= FIRE_SSI_SHUTDOWN_REQ; 1188 ddi_put8(hdl, ®_base[FIRE_SSI_INTR_ENA], reg); 1189 1190 softsp->power_regs_mapped = B_TRUE; 1191 1192 return (DDI_SUCCESS); 1193 } 1194 1195 /* 1196 * Setup register map for the power button 1197 * NOTE:- we only map registers for platforms if 1198 * the OBP power device has any of the following 1199 * properties: 1200 * 1201 * a) Boston: power-device-type set to "SUNW,mbc" 1202 * b) Seattle: power-device-type set to "SUNW,pic18lf65j10" 1203 * c) Chalupa: compatible set to "ali1535d+-power" 1204 * 1205 * Cases (a) and (b) are defined in FWARC 2005/687. 1206 * If none of the above conditions are true, then we 1207 * do not need to map in any registers, and this 1208 * function can simply return DDI_SUCCESS. 1209 */ 1210 static int 1211 power_setup_regs(struct power_soft_state *softsp) 1212 { 1213 char *binding_name; 1214 char *power_type = NULL; 1215 int retval = DDI_SUCCESS; 1216 1217 softsp->power_regs_mapped = B_FALSE; 1218 softsp->power_btn_ioctl = B_FALSE; 1219 binding_name = ddi_binding_name(softsp->dip); 1220 if (ddi_prop_lookup_string(DDI_DEV_T_ANY, softsp->dip, 1221 DDI_PROP_DONTPASS, POWER_DEVICE_TYPE, 1222 &power_type) == DDI_PROP_SUCCESS) { 1223 if (strcmp(power_type, "SUNW,mbc") == 0) { 1224 retval = power_setup_mbc_regs(softsp->dip, softsp); 1225 } else if (strcmp(power_type, "SUNW,pic18lf65j10") == 0) { 1226 retval = power_setup_epic_regs(softsp->dip, softsp); 1227 } else { 1228 cmn_err(CE_WARN, "unexpected power-device-type: %s\n", 1229 power_type); 1230 retval = DDI_FAILURE; 1231 } 1232 ddi_prop_free(power_type); 1233 } else if (strcmp(binding_name, "ali1535d+-power") == 0) { 1234 retval = power_setup_m1535_regs(softsp->dip, softsp); 1235 } 1236 1237 /* 1238 * If power-device-type does not exist AND the binding name is not 1239 * "ali1535d+-power", that means there is no additional HW and hence 1240 * no extra processing is necessary. In that case, retval should still 1241 * be set to its initial value of DDI_SUCCESS. 1242 */ 1243 return (retval); 1244 } 1245 1246 static void 1247 power_free_regs(struct power_soft_state *softsp) 1248 { 1249 if (softsp->power_regs_mapped) 1250 ddi_regs_map_free(&softsp->power_rhandle); 1251 } 1252 #endif /* ACPI_POWER_BUTTON */ 1253