ae0989784d
Why I did it Nvidia platform API does not support set LED to orange How I did it Allow user to set LED to orange How to verify it Added unit test Manual test
315 lines
12 KiB
Python
315 lines
12 KiB
Python
#
|
|
# Copyright (c) 2020-2021 NVIDIA CORPORATION & AFFILIATES.
|
|
# Apache-2.0
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
#
|
|
import os
|
|
import time
|
|
|
|
from sonic_py_common.logger import Logger
|
|
from . import utils
|
|
|
|
logger = Logger()
|
|
|
|
class Led(object):
|
|
STATUS_LED_COLOR_GREEN = 'green'
|
|
STATUS_LED_COLOR_GREEN_BLINK = 'green_blink'
|
|
STATUS_LED_COLOR_RED = 'red'
|
|
STATUS_LED_COLOR_RED_BLINK = 'red_blink'
|
|
STATUS_LED_COLOR_ORANGE = 'orange'
|
|
STATUS_LED_COLOR_ORANGE_BLINK = 'orange_blink'
|
|
STATUS_LED_COLOR_OFF = 'off'
|
|
|
|
LED_ON = '255'
|
|
LED_OFF = '0'
|
|
LED_BLINK = '50'
|
|
|
|
LED_PATH = "/var/run/hw-management/led/"
|
|
|
|
def set_status(self, color):
|
|
led_cap_list = self.get_capability()
|
|
if led_cap_list is None:
|
|
return False
|
|
|
|
status = False
|
|
try:
|
|
self._stop_blink(led_cap_list)
|
|
blink_pos = color.find('blink')
|
|
if blink_pos != -1:
|
|
return self._set_status_blink(color, led_cap_list)
|
|
|
|
if color == Led.STATUS_LED_COLOR_GREEN:
|
|
utils.write_file(self.get_green_led_path(), Led.LED_ON)
|
|
status = True
|
|
elif color == Led.STATUS_LED_COLOR_RED:
|
|
# Some led don't support red led but support orange led, in this case we set led to orange
|
|
if Led.STATUS_LED_COLOR_RED in led_cap_list:
|
|
led_path = self.get_red_led_path()
|
|
elif Led.STATUS_LED_COLOR_ORANGE in led_cap_list:
|
|
led_path = self.get_orange_led_path()
|
|
else:
|
|
return False
|
|
|
|
utils.write_file(led_path, Led.LED_ON)
|
|
status = True
|
|
elif color == Led.STATUS_LED_COLOR_ORANGE:
|
|
if Led.STATUS_LED_COLOR_ORANGE in led_cap_list:
|
|
led_path = self.get_orange_led_path()
|
|
elif Led.STATUS_LED_COLOR_RED in led_cap_list:
|
|
led_path = self.get_red_led_path()
|
|
else:
|
|
return False
|
|
|
|
utils.write_file(led_path, Led.LED_ON)
|
|
status = True
|
|
elif color == Led.STATUS_LED_COLOR_OFF:
|
|
if Led.STATUS_LED_COLOR_GREEN in led_cap_list:
|
|
utils.write_file(self.get_green_led_path(), Led.LED_OFF)
|
|
if Led.STATUS_LED_COLOR_RED in led_cap_list:
|
|
utils.write_file(self.get_red_led_path(), Led.LED_OFF)
|
|
if Led.STATUS_LED_COLOR_ORANGE in led_cap_list:
|
|
utils.write_file(self.get_orange_led_path(), Led.LED_OFF)
|
|
|
|
status = True
|
|
else:
|
|
status = False
|
|
except (ValueError, IOError):
|
|
status = False
|
|
|
|
return status
|
|
|
|
def _set_status_blink(self, color, led_cap_list):
|
|
if color not in led_cap_list:
|
|
if color == Led.STATUS_LED_COLOR_RED_BLINK and Led.STATUS_LED_COLOR_ORANGE_BLINK in led_cap_list:
|
|
color = Led.STATUS_LED_COLOR_ORANGE_BLINK
|
|
elif color == Led.STATUS_LED_COLOR_ORANGE_BLINK and Led.STATUS_LED_COLOR_RED_BLINK in led_cap_list:
|
|
color = Led.STATUS_LED_COLOR_RED_BLINK
|
|
else:
|
|
return False
|
|
|
|
if Led.STATUS_LED_COLOR_GREEN_BLINK == color:
|
|
self._trigger_blink(self.get_green_led_trigger())
|
|
return self._set_led_blink_status(self.get_green_led_delay_on_path(), self.get_green_led_delay_off_path(), Led.LED_BLINK)
|
|
elif Led.STATUS_LED_COLOR_RED_BLINK == color:
|
|
self._trigger_blink(self.get_red_led_trigger())
|
|
return self._set_led_blink_status(self.get_red_led_delay_on_path(), self.get_red_led_delay_off_path(), Led.LED_BLINK)
|
|
elif Led.STATUS_LED_COLOR_ORANGE_BLINK == color:
|
|
self._trigger_blink(self.get_orange_led_trigger())
|
|
return self._set_led_blink_status(self.get_orange_led_delay_on_path(), self.get_orange_led_delay_off_path(), Led.LED_BLINK)
|
|
else:
|
|
return False
|
|
|
|
def _stop_blink(self, led_cap_list):
|
|
try:
|
|
if Led.STATUS_LED_COLOR_GREEN_BLINK in led_cap_list:
|
|
self._untrigger_blink(self.get_green_led_trigger())
|
|
if Led.STATUS_LED_COLOR_RED_BLINK in led_cap_list:
|
|
self._untrigger_blink(self.get_red_led_trigger())
|
|
if Led.STATUS_LED_COLOR_ORANGE_BLINK in led_cap_list:
|
|
self._untrigger_blink(self.get_orange_led_trigger())
|
|
except Exception as e:
|
|
return
|
|
|
|
def _trigger_blink(self, blink_trigger_file):
|
|
utils.write_file(blink_trigger_file, 'timer')
|
|
|
|
def _untrigger_blink(self, blink_trigger_file):
|
|
utils.write_file(blink_trigger_file, 'none')
|
|
|
|
def _set_led_blink_status(self, delay_on_file, delay_off_file, value):
|
|
if not self._wait_files_ready((delay_on_file, delay_off_file)):
|
|
return False
|
|
|
|
utils.write_file(delay_on_file, value)
|
|
utils.write_file(delay_off_file, value)
|
|
return True
|
|
|
|
def _wait_files_ready(self, file_list):
|
|
"""delay_off and delay_on sysfs will be available only if _trigger_blink is called. And once
|
|
_trigger_blink is called, driver might need time to prepare delay_off and delay_on. So,
|
|
need wait a few seconds here to make sure the sysfs is ready
|
|
|
|
Args:
|
|
file_list (list of str): files to be checked
|
|
"""
|
|
wait_time = 5.0
|
|
initial_sleep = 0.01
|
|
while wait_time > 0:
|
|
if all([os.path.exists(x) for x in file_list]):
|
|
return True
|
|
time.sleep(initial_sleep)
|
|
wait_time -= initial_sleep
|
|
initial_sleep = initial_sleep * 2
|
|
|
|
return False
|
|
|
|
def get_status(self):
|
|
led_cap_list = self.get_capability()
|
|
if led_cap_list is None:
|
|
return Led.STATUS_LED_COLOR_OFF
|
|
|
|
try:
|
|
blink_status = self._get_blink_status(led_cap_list)
|
|
if blink_status is not None:
|
|
return blink_status
|
|
|
|
if utils.read_str_from_file(self.get_green_led_path()) != Led.LED_OFF:
|
|
return Led.STATUS_LED_COLOR_GREEN
|
|
|
|
if Led.STATUS_LED_COLOR_RED in led_cap_list:
|
|
if utils.read_str_from_file(self.get_red_led_path()) != Led.LED_OFF:
|
|
return Led.STATUS_LED_COLOR_RED
|
|
if Led.STATUS_LED_COLOR_ORANGE in led_cap_list:
|
|
if utils.read_str_from_file(self.get_orange_led_path()) != Led.LED_OFF:
|
|
return Led.STATUS_LED_COLOR_RED
|
|
except (ValueError, IOError) as e:
|
|
raise RuntimeError("Failed to read led status due to {}".format(repr(e)))
|
|
|
|
return Led.STATUS_LED_COLOR_OFF
|
|
|
|
def _get_blink_status(self, led_cap_list):
|
|
try:
|
|
if Led.STATUS_LED_COLOR_GREEN_BLINK in led_cap_list:
|
|
if self._is_led_blinking(self.get_green_led_delay_on_path(), self.get_green_led_delay_off_path()):
|
|
return Led.STATUS_LED_COLOR_GREEN_BLINK
|
|
|
|
if Led.STATUS_LED_COLOR_RED_BLINK in led_cap_list:
|
|
if self._is_led_blinking(self.get_red_led_delay_on_path(), self.get_red_led_delay_off_path()):
|
|
return Led.STATUS_LED_COLOR_RED_BLINK
|
|
if Led.STATUS_LED_COLOR_ORANGE_BLINK in led_cap_list:
|
|
if self._is_led_blinking(self.get_orange_led_delay_on_path(), self.get_orange_led_delay_off_path()):
|
|
return Led.STATUS_LED_COLOR_ORANGE_BLINK
|
|
except Exception as e:
|
|
return None
|
|
|
|
return None
|
|
|
|
def _is_led_blinking(self, delay_on_file, delay_off_file):
|
|
delay_on = utils.read_str_from_file(delay_on_file, default=Led.LED_OFF, log_func=None)
|
|
delay_off = utils.read_str_from_file(delay_off_file, default=Led.LED_OFF, log_func=None)
|
|
return delay_on != Led.LED_OFF and delay_off != Led.LED_OFF
|
|
|
|
def get_capability(self):
|
|
caps = utils.read_str_from_file(self.get_led_cap_path())
|
|
return set(caps.split())
|
|
|
|
def get_green_led_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_green'.format(self._led_id))
|
|
|
|
def get_green_led_delay_off_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_green_delay_off'.format(self._led_id))
|
|
|
|
def get_green_led_delay_on_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_green_delay_on'.format(self._led_id))
|
|
|
|
def get_green_led_trigger(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_green_trigger'.format(self._led_id))
|
|
|
|
def get_red_led_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_red'.format(self._led_id))
|
|
|
|
def get_red_led_delay_off_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_red_delay_off'.format(self._led_id))
|
|
|
|
def get_red_led_delay_on_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_red_delay_on'.format(self._led_id))
|
|
|
|
def get_red_led_trigger(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_red_trigger'.format(self._led_id))
|
|
|
|
def get_orange_led_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_orange'.format(self._led_id))
|
|
|
|
def get_orange_led_delay_off_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_orange_delay_off'.format(self._led_id))
|
|
|
|
def get_orange_led_delay_on_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_orange_delay_on'.format(self._led_id))
|
|
|
|
def get_orange_led_trigger(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_orange_trigger'.format(self._led_id))
|
|
|
|
def get_led_cap_path(self):
|
|
return os.path.join(Led.LED_PATH, 'led_{}_capability'.format(self._led_id))
|
|
|
|
|
|
class FanLed(Led):
|
|
def __init__(self, index):
|
|
if index is not None:
|
|
self._led_id = 'fan{}'.format(index)
|
|
else:
|
|
self._led_id = 'fan'
|
|
|
|
|
|
class PsuLed(Led):
|
|
def __init__(self, index):
|
|
if index is not None:
|
|
self._led_id = 'psu{}'.format(index)
|
|
else:
|
|
self._led_id = 'psu'
|
|
|
|
|
|
class SystemLed(Led):
|
|
def __init__(self):
|
|
self._led_id = 'status'
|
|
|
|
|
|
class SharedLed(object):
|
|
LED_PRIORITY = {
|
|
Led.STATUS_LED_COLOR_RED: 0,
|
|
Led.STATUS_LED_COLOR_GREEN: 1
|
|
}
|
|
|
|
def __init__(self, led):
|
|
self._led = led
|
|
self._virtual_leds = []
|
|
|
|
def add_virtual_leds(self, led):
|
|
self._virtual_leds.append(led)
|
|
|
|
def update_status_led(self):
|
|
target_color = Led.STATUS_LED_COLOR_GREEN
|
|
for virtual_led in self._virtual_leds:
|
|
try:
|
|
if SharedLed.LED_PRIORITY[virtual_led.get_led_color()] < SharedLed.LED_PRIORITY[target_color]:
|
|
target_color = virtual_led.get_led_color()
|
|
except KeyError:
|
|
return False
|
|
return self._led.set_status(target_color)
|
|
|
|
def get_status(self):
|
|
return self._led.get_status()
|
|
|
|
|
|
class ComponentFaultyIndicator(object):
|
|
def __init__(self, shared_led):
|
|
self._color = Led.STATUS_LED_COLOR_GREEN
|
|
self._shared_led = shared_led
|
|
self._shared_led.add_virtual_leds(self)
|
|
|
|
def set_status(self, color):
|
|
current_color = self._color
|
|
self._color = color
|
|
if self._shared_led.update_status_led():
|
|
return True
|
|
else:
|
|
self._color = current_color
|
|
return False
|
|
|
|
def get_led_color(self):
|
|
return self._color
|
|
|
|
def get_status(self):
|
|
return self._shared_led.get_status()
|