blob: 20532c98e1d6687bbcdd9332fef756cb3802a48f [file] [log] [blame]
Brad Bishop87c6b732016-08-30 19:17:57 -04001#!/usr/bin/env python
Norman Jamese2765102015-08-19 22:00:55 -05002
Norman James6f8d0422015-09-14 18:48:00 -05003import gobject
Norman Jamese2765102015-08-19 22:00:55 -05004import dbus
5import dbus.service
6import dbus.mainloop.glib
Brad Bishop84e73b52016-05-12 15:57:52 -04007from obmc.dbuslib.bindings import get_dbus, DbusProperties, DbusObjectManager
Norman Jamese2765102015-08-19 22:00:55 -05008
Norman James3f97c5d2015-08-26 17:44:14 -05009DBUS_NAME = 'org.openbmc.control.Chassis'
Norman James8fee6f22015-10-28 12:48:43 -050010OBJ_NAME = '/org/openbmc/control/chassis0'
Norman Jamesa3e47c42015-10-18 14:43:10 -050011CONTROL_INTF = 'org.openbmc.Control'
Norman James3f97c5d2015-08-26 17:44:14 -050012
Norman Jamesad8c3d32016-02-11 15:19:01 -060013MACHINE_ID = '/etc/machine-id'
14
Norman James2a3d20b2015-08-20 07:09:33 -050015POWER_OFF = 0
16POWER_ON = 1
17
Norman James9e6acf92015-09-08 07:00:04 -050018BOOTED = 100
19
Norman James90baede2015-09-02 20:32:49 -050020
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050021class ChassisControlObject(DbusProperties, DbusObjectManager):
22 def getUuid(self):
Brad Bishop7a4d77a2016-09-08 23:26:17 -040023 uuid = ""
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050024 try:
25 with open(MACHINE_ID) as f:
26 data = f.readline().rstrip('\n')
27 if (len(data) == 32):
28 uuid = data
29 else:
30 print "ERROR: UUID is not formatted correctly: " + data
31 except:
32 print "ERROR: Unable to open uuid file: " + MACHINE_ID
Adriana Kobylak08d3bdb2015-10-20 16:59:14 -050033
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050034 return uuid
Norman Jamescfc2b442015-10-31 17:31:46 -050035
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050036 def __init__(self, bus, name):
37 self.dbus_objects = {}
38 DbusProperties.__init__(self)
39 DbusObjectManager.__init__(self)
40 dbus.service.Object.__init__(self, bus, name)
41 ## load utilized objects
42 self.dbus_objects = {
43 'power_control': {
44 'bus_name': 'org.openbmc.control.Power',
45 'object_name': '/org/openbmc/control/power0',
46 'interface_name': 'org.openbmc.control.Power'
47 },
48 'identify_led': {
49 'bus_name': 'org.openbmc.control.led',
50 'object_name': '/org/openbmc/control/led/identify',
51 'interface_name': 'org.openbmc.Led'
52 },
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050053 'host_services': {
54 'bus_name': 'org.openbmc.HostServices',
55 'object_name': '/org/openbmc/HostServices',
56 'interface_name': 'org.openbmc.HostServices'
57 },
58 'settings': {
59 'bus_name': 'org.openbmc.settings.Host',
60 'object_name': '/org/openbmc/settings/host0',
61 'interface_name': 'org.freedesktop.DBus.Properties'
62 },
Brad Bishopabe0f682016-08-22 19:27:18 -040063 'systemd': {
64 'bus_name': 'org.freedesktop.systemd1',
65 'object_name': '/org/freedesktop/systemd1',
66 'interface_name': 'org.freedesktop.systemd1.Manager'
Andrew Geissler6b63e9a2016-06-30 10:45:01 -050067 },
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050068 }
Norman James72567ba2016-01-13 16:57:48 -060069
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050070 # uuid
71 self.Set(DBUS_NAME, "uuid", self.getUuid())
72 self.Set(DBUS_NAME, "reboot", 0)
Norman James72567ba2016-01-13 16:57:48 -060073
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050074 bus.add_signal_receiver(self.power_button_signal_handler,
75 dbus_interface="org.openbmc.Button",
76 signal_name="Released",
77 path="/org/openbmc/buttons/power0")
Adriana Kobylakeeb9b442016-08-15 11:12:53 -050078 bus.add_signal_receiver(self.long_power_button_signal_handler,
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050079 dbus_interface="org.openbmc.Button",
80 signal_name="PressedLong",
81 path="/org/openbmc/buttons/power0")
82 bus.add_signal_receiver(self.softreset_button_signal_handler,
83 dbus_interface="org.openbmc.Button",
84 signal_name="Released",
85 path="/org/openbmc/buttons/reset0")
86
87 bus.add_signal_receiver(self.host_watchdog_signal_handler,
88 dbus_interface="org.openbmc.Watchdog",
89 signal_name="WatchdogError")
90
91 bus.add_signal_receiver(self.emergency_shutdown_signal_handler,
92 dbus_interface="org.openbmc.SensorThresholds",
93 signal_name="Emergency")
94
95 bus.add_signal_receiver(self.SystemStateHandler,
96 signal_name="GotoSystemState")
Norman James471ab592015-08-30 22:29:40 -050097
Andrew Geisslerb7f40b52016-06-29 15:49:26 -050098 def getInterface(self, name):
99 o = self.dbus_objects[name]
100 obj = bus.get_object(o['bus_name'], o['object_name'], introspect=False)
101 return dbus.Interface(obj, o['interface_name'])
Norman Jamese2765102015-08-19 22:00:55 -0500102
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500103 @dbus.service.method(DBUS_NAME,
104 in_signature='', out_signature='')
105 def setIdentify(self):
106 print "Turn on identify"
107 intf = self.getInterface('identify_led')
108 intf.setOn()
109 return None
Norman Jamese2765102015-08-19 22:00:55 -0500110
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500111 @dbus.service.method(DBUS_NAME,
112 in_signature='', out_signature='')
113 def clearIdentify(self):
114 print "Turn on identify"
115 intf = self.getInterface('identify_led')
116 intf.setOff()
117 return None
Norman Jamese2765102015-08-19 22:00:55 -0500118
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500119 @dbus.service.method(DBUS_NAME,
120 in_signature='', out_signature='')
121 def powerOn(self):
122 print "Turn on power and boot"
123 self.Set(DBUS_NAME, "reboot", 0)
Brad Bishopabe0f682016-08-22 19:27:18 -0400124 intf = self.getInterface('systemd')
125 f = getattr(intf, 'StartUnit')
126 f.call_async('obmc-chassis-start@0.target', 'replace')
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500127 return None
Norman Jamese2765102015-08-19 22:00:55 -0500128
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500129 @dbus.service.method(DBUS_NAME,
130 in_signature='', out_signature='')
131 def powerOff(self):
132 print "Turn off power"
Brad Bishopabe0f682016-08-22 19:27:18 -0400133
134 intf = self.getInterface('systemd')
135 f = getattr(intf, 'StartUnit')
136 f.call_async('obmc-chassis-stop@0.target', 'replace')
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500137 return None
Norman James362a80f2015-09-14 14:04:39 -0500138
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500139 @dbus.service.method(DBUS_NAME,
140 in_signature='', out_signature='')
141 def softPowerOff(self):
142 print "Soft off power"
143 intf = self.getInterface('host_services')
144 ## host services will call power off when ready
145 intf.SoftPowerOff()
146 return None
Norman James362a80f2015-09-14 14:04:39 -0500147
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500148 @dbus.service.method(DBUS_NAME,
149 in_signature='', out_signature='')
150 def reboot(self):
151 print "Rebooting"
152 if self.getPowerState() == POWER_OFF:
Brad Bishop7a4d77a2016-09-08 23:26:17 -0400153 self.powerOn()
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500154 else:
155 self.Set(DBUS_NAME, "reboot", 1)
156 self.powerOff()
157 return None
Norman Jamesb4ef3182015-12-03 17:54:35 -0600158
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500159 @dbus.service.method(DBUS_NAME,
160 in_signature='', out_signature='')
161 def softReboot(self):
162 print "Soft Rebooting"
163 if self.getPowerState() == POWER_OFF:
Brad Bishop7a4d77a2016-09-08 23:26:17 -0400164 self.powerOn()
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500165 else:
166 self.Set(DBUS_NAME, "reboot", 1)
167 self.softPowerOff()
168 return None
Norman Jamese2765102015-08-19 22:00:55 -0500169
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500170 @dbus.service.method(DBUS_NAME,
171 in_signature='', out_signature='i')
172 def getPowerState(self):
173 intf = self.getInterface('power_control')
174 return intf.getPowerState()
Norman Jamese2765102015-08-19 22:00:55 -0500175
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500176 ## Signal handler
Norman James362a80f2015-09-14 14:04:39 -0500177
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500178 def SystemStateHandler(self, state_name):
Brad Bishop7a4d77a2016-09-08 23:26:17 -0400179 if state_name in ["HOST_POWERED_OFF", "HOST_POWERED_ON"]:
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500180 intf = self.getInterface('settings')
181 intf.Set("org.openbmc.settings.Host", "system_state", state_name)
Norman Jamesf5edb9e2016-01-31 13:32:24 -0600182
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500183 if (state_name == "HOST_POWERED_OFF" and self.Get(DBUS_NAME,
184 "reboot") == 1):
185 self.powerOn()
Norman James362a80f2015-09-14 14:04:39 -0500186
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500187 def power_button_signal_handler(self):
Adriana Kobylakeeb9b442016-08-15 11:12:53 -0500188 # toggle power, power-on / soft-power-off
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500189 state = self.getPowerState()
190 if state == POWER_OFF:
191 self.powerOn()
192 elif state == POWER_ON:
Brad Bishop7a4d77a2016-09-08 23:26:17 -0400193 self.softPowerOff()
Norman Jamese2765102015-08-19 22:00:55 -0500194
Adriana Kobylakeeb9b442016-08-15 11:12:53 -0500195 def long_power_button_signal_handler(self):
196 print "Long-press button, hard power off"
Brad Bishop7a4d77a2016-09-08 23:26:17 -0400197 self.powerOff()
Kenc95eccd2015-12-19 07:02:34 +0800198
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500199 def softreset_button_signal_handler(self):
Brad Bishop7a4d77a2016-09-08 23:26:17 -0400200 self.softReboot()
Norman James72567ba2016-01-13 16:57:48 -0600201
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500202 def host_watchdog_signal_handler(self):
203 print "Watchdog Error, Hard Rebooting"
204 self.Set(DBUS_NAME, "reboot", 1)
205 self.powerOff()
206
Yi Li4eab1dc2016-07-02 22:29:45 +0800207 def emergency_shutdown_signal_handler(self, message):
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500208 print "Emergency Shutdown!"
Yi Li4eab1dc2016-07-02 22:29:45 +0800209 # Log an event.
210 try:
211 # Exception happens or not, we need to power off.
212 obj = bus.get_object("org.openbmc.records.events",
213 "/org/openbmc/records/events",
214 introspect=False)
215 intf = dbus.Interface(obj, "org.openbmc.recordlog")
216 desc = message
217 sev = "critical error"
218 details = "Get emergency shutdown signal. Shutdown the host."
219 debug = dbus.ByteArray("")
220 intf.acceptBMCMessage(desc, sev, details, debug)
221 except Exception as e:
222 print "Emergency shutdown signal handler: log event error."
223 print e
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500224 self.powerOff()
225
Norman Jamese2765102015-08-19 22:00:55 -0500226
227if __name__ == '__main__':
228 dbus.mainloop.glib.DBusGMainLoop(set_as_default=True)
229
Brad Bishop84e73b52016-05-12 15:57:52 -0400230 bus = get_dbus()
Norman James3f97c5d2015-08-26 17:44:14 -0500231 obj = ChassisControlObject(bus, OBJ_NAME)
Norman James6f8d0422015-09-14 18:48:00 -0500232 mainloop = gobject.MainLoop()
Brad Bishopf0f3efe2016-06-29 23:20:24 -0400233
234 obj.unmask_signals()
Brad Bishop70852a32016-06-29 22:58:51 -0400235 name = dbus.service.BusName(DBUS_NAME, bus)
Andrew Geisslerb7f40b52016-06-29 15:49:26 -0500236
Norman Jamese2765102015-08-19 22:00:55 -0500237 print "Running ChassisControlService"
238 mainloop.run()