blob: ac46c2c24f99e214fcd18434e54abdbe90c915ec (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
|
/*
* PowerNV OPAL power control for graceful shutdown handling
*
* Copyright 2015 IBM Corp.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version
* 2 of the License, or (at your option) any later version.
*/
#include <linux/kernel.h>
#include <linux/reboot.h>
#include <linux/notifier.h>
#include <asm/opal.h>
#include <asm/machdep.h>
#define SOFT_OFF 0x00
#define SOFT_REBOOT 0x01
static int opal_power_control_event(struct notifier_block *nb,
unsigned long msg_type, void *msg)
{
struct opal_msg *power_msg = msg;
uint64_t type;
type = be64_to_cpu(power_msg->params[0]);
switch (type) {
case SOFT_REBOOT:
pr_info("OPAL: reboot requested\n");
orderly_reboot();
break;
case SOFT_OFF:
pr_info("OPAL: poweroff requested\n");
orderly_poweroff(true);
break;
default:
pr_err("OPAL: power control type unexpected %016llx\n", type);
}
return 0;
}
static struct notifier_block opal_power_control_nb = {
.notifier_call = opal_power_control_event,
.next = NULL,
.priority = 0,
};
static int __init opal_power_control_init(void)
{
int ret;
ret = opal_message_notifier_register(OPAL_MSG_SHUTDOWN,
&opal_power_control_nb);
if (ret) {
pr_err("%s: Can't register OPAL event notifier (%d)\n",
__func__, ret);
return ret;
}
return 0;
}
machine_subsys_initcall(powernv, opal_power_control_init);
|