[as9716-32d] Add to support PDDF (#6902)

Add PDDF support for Accton as9716-32d platform

Co-authored-by: Jostar Yang <jostar_yang@accton.com.tw>
This commit is contained in:
jostar-yang 2021-04-15 05:43:52 +08:00 committed by GitHub
parent b4b9e4234f
commit 6641a6b1a1
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
22 changed files with 3079 additions and 1 deletions

View File

@ -0,0 +1,66 @@
{
"XCVR":
{
"xcvr_present":
{
"i2c":
{
"valmap-SFP28": {"1":true, "0":false },
"valmap-QSFP28": {"1":true, "0":false}
}
}
},
"PSU":
{
"psu_present":
{
"i2c":
{
"valmap": { "1":true, "0":false }
}
},
"psu_power_good":
{
"i2c":
{
"valmap": { "1": true, "0":false }
}
},
"psu_fan_dir":
{
"i2c":
{
"valmap": { "F2B":"EXHAUST", "B2F":"INTAKE" }
}
},
"PSU_FAN_MAX_SPEED":"18000"
},
"FAN":
{
"direction":
{
"i2c":
{
"valmap": {"1":"INTAKE", "0":"EXHAUST"}
}
},
"present":
{
"i2c":
{
"valmap": {"1":true, "0":false}
}
},
"duty_cycle_to_pwm": "lambda dc: ((dc*100)/625 - 1)",
"pwm_to_duty_cycle": "lambda pwm: (((pwm+1)*625)/100)"
}
}

File diff suppressed because it is too large Load Diff

View File

@ -1,6 +1,7 @@
ifneq ($(KERNELRELEASE),)
obj-m:= accton_as9716_32d_cpld.o accton_as9716_32d_fan.o \
accton_as9716_32d_leds.o accton_as9716_32d_psu.o accton_i2c_psu.o
accton_as9716_32d_leds.o accton_as9716_32d_psu.o accton_i2c_psu.o \
pddf_custom_psu.o
else
ifeq (,$(KERNEL_SRC))

View File

@ -0,0 +1,125 @@
#include <linux/module.h>
#include <linux/jiffies.h>
#include <linux/i2c.h>
#include <linux/hwmon.h>
#include <linux/hwmon-sysfs.h>
#include <linux/err.h>
#include <linux/delay.h>
#include <linux/mutex.h>
#include <linux/sysfs.h>
#include <linux/slab.h>
#include <linux/dmi.h>
#include "../../../../pddf/i2c/modules/include/pddf_psu_defs.h"
ssize_t pddf_show_custom_psu_v_out(struct device *dev, struct device_attribute *da, char *buf);
extern PSU_SYSFS_ATTR_DATA access_psu_v_out;
static int two_complement_to_int(u16 data, u8 valid_bit, int mask)
{
u16 valid_data = data & mask;
bool is_negative = valid_data >> (valid_bit - 1);
return is_negative ? (-(((~valid_data) & mask) + 1)) : valid_data;
}
static u8 psu_get_vout_mode(struct i2c_client *client)
{
u8 status = 0, retry = 10;
uint8_t offset = 0x20; // VOUT_MODE
while (retry) {
status = i2c_smbus_read_byte_data((struct i2c_client *)client, offset);
if (unlikely(status < 0)) {
msleep(60);
retry--;
continue;
}
break;
}
if (status < 0)
{
printk(KERN_ERR "%s: Get PSU Vout mode failed\n", __func__);
return 0;
}
else
{
/*printk(KERN_ERR "%s: vout_mode reg value 0x%x\n", __func__, status);*/
return status;
}
}
static u16 psu_get_v_out(struct i2c_client *client)
{
u16 status = 0, retry = 10;
uint8_t offset = 0x8b; // READ_VOUT
while (retry) {
status = i2c_smbus_read_word_data((struct i2c_client *)client, offset);
if (unlikely(status < 0)) {
msleep(60);
retry--;
continue;
}
break;
}
if (status < 0)
{
printk(KERN_ERR "%s: Get PSU Vout failed\n", __func__);
return 0;
}
else
{
/*printk(KERN_ERR "%s: vout reg value 0x%x\n", __func__, status);*/
return status;
}
}
ssize_t pddf_show_custom_psu_v_out(struct device *dev, struct device_attribute *da, char *buf)
{
struct i2c_client *client = to_i2c_client(dev);
struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
int exponent, mantissa;
int multiplier = 1000;
u16 value = psu_get_v_out(client);
u8 vout_mode = psu_get_vout_mode(client);
if ((vout_mode >> 5) == 0)
exponent = two_complement_to_int(vout_mode & 0x1f, 5, 0x1f);
else
{
printk(KERN_ERR "%s: Only support linear mode for vout mode\n", __func__);
exponent = 0;
}
mantissa = value;
if (exponent >= 0)
return sprintf(buf, "%d\n", (mantissa << exponent) * multiplier);
else
return sprintf(buf, "%d\n", (mantissa * multiplier) / (1 << -exponent));
}
static int __init pddf_custom_psu_init(void)
{
access_psu_v_out.show = pddf_show_custom_psu_v_out;
access_psu_v_out.do_get = NULL;
printk(KERN_ERR "pddf_custom_psu_init\n");
return 0;
}
static void __exit pddf_custom_psu_exit(void)
{
printk(KERN_ERR "pddf_custom_psu_exit\n");
return;
}
MODULE_AUTHOR("Broadcom");
MODULE_DESCRIPTION("pddf custom psu api");
MODULE_LICENSE("GPL");
module_init(pddf_custom_psu_init);
module_exit(pddf_custom_psu_exit);

View File

@ -0,0 +1,16 @@
[Unit]
Description=Accton AS9716-32D Platform Monitoring service
Before=pmon.service
After=pddf-platform-init.service
DefaultDependencies=no
[Service]
ExecStart=/usr/local/bin/accton_as9716_32d_pddf_monitor.py
KillSignal=SIGKILL
SuccessExitStatus=SIGKILL
# Resource Limitations
LimitCORE=infinity
[Install]
WantedBy=multi-user.target

View File

@ -0,0 +1 @@
../../../../pddf/i2c/service/pddf-platform-init.service

View File

@ -0,0 +1,4 @@
# All the derived classes for PDDF
__all__ = ["platform", "chassis", "sfp", "psu", "thermal", "fan"]
from . import platform

View File

@ -0,0 +1,61 @@
#!/usr/bin/env python
#############################################################################
# PDDF
# Module contains an implementation of SONiC Chassis API
#
#############################################################################
try:
import time
from sonic_platform_pddf_base.pddf_chassis import PddfChassis
except ImportError as e:
raise ImportError(str(e) + "- required module not found")
class Chassis(PddfChassis):
"""
PDDF Platform-specific Chassis class
"""
def __init__(self, pddf_data=None, pddf_plugin_data=None):
PddfChassis.__init__(self, pddf_data, pddf_plugin_data)
# Provide the functions/variables below for which implementation is to be overwritten
sfp_change_event_data = {'valid': 0, 'last': 0, 'present': 0}
def get_change_event(self, timeout=2000):
now = time.time()
port_dict = {}
change_dict = {}
change_dict['sfp'] = port_dict
if timeout < 1000:
timeout = 1000
timeout = timeout / float(1000) # Convert to secs
if now < (self.sfp_change_event_data['last'] + timeout) and self.sfp_change_event_data['valid']:
return True, change_dict
bitmap = 0
for i in range(34):
modpres = self.get_sfp(i).get_presence()
if modpres:
bitmap = bitmap | (1 << i)
changed_ports = self.sfp_change_event_data['present'] ^ bitmap
if changed_ports:
for i in range(34):
if (changed_ports & (1 << i)):
if (bitmap & (1 << i)) == 0:
port_dict[i+1] = '0'
else:
port_dict[i+1] = '1'
# Update teh cache dict
self.sfp_change_event_data['present'] = bitmap
self.sfp_change_event_data['last'] = now
self.sfp_change_event_data['valid'] = 1
return True, change_dict
else:
return True, change_dict

View File

@ -0,0 +1,14 @@
#!/usr/bin/env python
try:
from sonic_platform_pddf_base.pddf_eeprom import PddfEeprom
except ImportError as e:
raise ImportError(str(e) + "- required module not found")
class Eeprom(PddfEeprom):
def __init__(self, pddf_data=None, pddf_plugin_data=None):
PddfEeprom.__init__(self, pddf_data, pddf_plugin_data)
# Provide the functions/variables below for which implementation is to be overwritten

View File

@ -0,0 +1,48 @@
#!/usr/bin/env python
try:
from sonic_platform_pddf_base.pddf_fan import PddfFan
except ImportError as e:
raise ImportError(str(e) + "- required module not found")
class Fan(PddfFan):
"""PDDF Platform-Specific Fan class"""
def __init__(self, tray_idx, fan_idx=0, pddf_data=None, pddf_plugin_data=None, is_psu_fan=False, psu_index=0):
# idx is 0-based
PddfFan.__init__(self, tray_idx, fan_idx, pddf_data, pddf_plugin_data, is_psu_fan, psu_index)
# Provide the functions/variables below for which implementation is to be overwritten
# Since psu_fan airflow direction cant be read from sysfs, it is fixed as 'F2B' or 'intake'
def get_direction(self):
"""
Retrieves the direction of fan
Returns:
A string, either FAN_DIRECTION_INTAKE or FAN_DIRECTION_EXHAUST
depending on fan direction
"""
if self.is_psu_fan:
direction = self.FAN_DIRECTION_EXHAUST
else:
idx = (self.fantray_index-1)*self.platform['num_fans_pertray'] + self.fan_index
attr = "fan" + str(idx) + "_direction"
output = self.pddf_obj.get_attr_name_output("FAN-CTRL", attr)
if not output:
return False
mode = output['mode']
val = output['status']
val = val.rstrip()
vmap = self.plugin_data['FAN']['direction'][mode]['valmap']
if val in vmap:
direction = vmap[val]
else:
direction = val
return direction

View File

@ -0,0 +1,25 @@
#!/usr/bin/env python
#############################################################################
# PDDF
# Module contains an implementation of SONiC Platform Base API and
# provides the platform information
#
#############################################################################
try:
from sonic_platform_pddf_base.pddf_platform import PddfPlatform
except ImportError as e:
raise ImportError(str(e) + "- required module not found")
class Platform(PddfPlatform):
"""
PDDF Platform-Specific Platform Class
"""
def __init__(self):
PddfPlatform.__init__(self)
# Provide the functions/variables below for which implementation is to be overwritten

View File

@ -0,0 +1,36 @@
#!/usr/bin/env python
#
try:
from sonic_platform_pddf_base.pddf_psu import PddfPsu
except ImportError as e:
raise ImportError (str(e) + "- required module not found")
class Psu(PddfPsu):
"""PDDF Platform-Specific PSU class"""
PLATFORM_PSU_CAPACITY = 1200
def __init__(self, index, pddf_data=None, pddf_plugin_data=None):
PddfPsu.__init__(self, index, pddf_data, pddf_plugin_data)
# Provide the functions/variables below for which implementation is to be overwritten
def get_maximum_supplied_power(self):
"""
Retrieves the maximum supplied power by PSU (or PSU capacity)
Returns:
A float number, the maximum power output in Watts.
e.g. 1200.1
"""
return float(self.PLATFORM_PSU_CAPACITY)
def get_type(self):
"""
Gets the type of the PSU
Returns:
A string, the type of PSU (AC/DC)
"""
return "DC"

View File

@ -0,0 +1,17 @@
#!/usr/bin/env python
try:
from sonic_platform_pddf_base.pddf_sfp import PddfSfp
except ImportError as e:
raise ImportError (str(e) + "- required module not found")
class Sfp(PddfSfp):
"""
PDDF Platform-Specific Sfp class
"""
def __init__(self, index, pddf_data=None, pddf_plugin_data=None):
PddfSfp.__init__(self, index, pddf_data, pddf_plugin_data)
# Provide the functions/variables below for which implementation is to be overwritten

View File

@ -0,0 +1,17 @@
#!/usr/bin/env python
try:
from sonic_platform_pddf_base.pddf_thermal import PddfThermal
except ImportError as e:
raise ImportError(str(e) + "- required module not found")
class Thermal(PddfThermal):
"""PDDF Platform-Specific Thermal class"""
def __init__(self, index, pddf_data=None, pddf_plugin_data=None):
PddfThermal.__init__(self, index, pddf_data, pddf_plugin_data)
# Provide the functions/variables below for which implementation is to be overwritten

View File

@ -0,0 +1,23 @@
#!/usr/bin/env python
#############################################################################
#
# Module contains an implementation of platform specific watchdog API's
#
#############################################################################
try:
from sonic_platform_pddf_base.pddf_watchdog import PddfWatchdog
except ImportError as e:
raise ImportError(str(e) + "- required module not found")
class Watchdog(PddfWatchdog):
"""
PDDF Platform-specific Chassis class
"""
def __init__(self):
PddfWatchdog.__init__(self)
self.timeout= 180
# Provide the functions/variables below for which implementation is to be overwritten

View File

@ -0,0 +1,25 @@
from setuptools import setup
setup(
name='sonic-platform',
version='1.0',
description='SONiC platform API implementation on Accton Platforms',
license='Apache 2.0',
author='SONiC Team',
author_email='linuxnetdev@microsoft.com',
url='https://github.com/Azure/sonic-buildimage',
packages=['sonic_platform'],
classifiers=[
'Development Status :: 3 - Alpha',
'Environment :: Plugins',
'Intended Audience :: Developers',
'Intended Audience :: Information Technology',
'Intended Audience :: System Administrators',
'License :: OSI Approved :: Apache Software License',
'Natural Language :: English',
'Operating System :: POSIX :: Linux',
'Programming Language :: Python :: 3.7',
'Topic :: Utilities',
],
keywords='sonic SONiC platform PLATFORM',
)

View File

@ -0,0 +1,518 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*
# Copyright (c) 2019 Edgecore Networks Corporation
#
# 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
#
# THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR
# CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT
# LIMITATION ANY IMPLIED WARRANTIES OR CONDITIONS OF TITLE, FITNESS
# FOR A PARTICULAR PURPOSE, MERCHANTABLITY OR NON-INFRINGEMENT.
#
# See the Apache Version 2.0 License for specific language governing
# permissions and limitations under the License.
#
# HISTORY:
# mm/dd/yyyy (A.D.)#
# 8/27/2019:Jostar create for as9716_32d thermal plan
# ------------------------------------------------------------------
try:
import subprocess
import sys
import getopt
import logging
import logging.config
import logging.handlers
import time
from sonic_platform import platform
except ImportError as e:
raise ImportError('%s - required module not found' % str(e))
# Deafults
VERSION = '1.0'
FUNCTION_NAME = '/usr/local/bin/accton_as9716_32d_pddf_monitor'
class switch(object):
def __init__(self, value):
self.value = value
self.fall = False
def __iter__(self):
"""Return the match method once, then stop"""
yield self.match
raise StopIteration
def match(self, *args):
"""Indicate whether or not to enter a case suite"""
if self.fall or not args:
return True
elif self.value in args: # changed for v1.5, see below
self.fall = True
return True
else:
return False
# Read fanN_direction=1: The air flow of Fan is AFI-Back to Front
# 0: The air flow of Fan is AFO-Front to Back
# Thermal policy:
# a.Defaut fan duty_cycle=100%
# b.One fan fail, set to fan duty_cycle=100%
# 1.For AFI:
# Default fan duty_cycle will be 100%(fan_policy_state=LEVEL_FAN_MAX).
# If all below case meet with, set to 75%(LEVEL_FAN_MID).
# MB board
# LM75-1(0X48)<=57
# LM75-2(0X49)<=47.3
# LM75-3(0X4A)<=45
# LM75-4(0X4C)<=45.1
# LM75-5(0X4E)<=40.75
# LM75-6(0X4F)<=42.1
# CPU board
# Core<=44
# LM75-1(0X4B)<=35
# When fan_policy_state=LEVEL_FAN_MID, meet with below case, Fan duty_cycle will be 100%(LEVEL_FAN_DAX)
# (MB board)
# LM75-1(0X48)>=61.5
# LM75-2(0X49)>=51.5
# LM75-3(0X4A)>=49.4
# LM75-4(0X4C)>=49.4
# LM75-5(0X4E)>=45.1
# LM75-6(0X4F)>=46.75
# (CPU board)
# Core>=48
# LM75-1(0X4B)>=38.5
# 2. For AFO:
# At default, FAN duty_cycle was 100%(LEVEL_FAN_MAX). If all below case meet with, set to 75%(LEVEL_FAN_MID).
# (MB board)
# LM75-1(0X48)<=59
# LM75-2(0X49)<=53.5
# LM75-3(0X4A)<=55.3
# LM75-4(0X4C)<=50.3
# LM75-5(0X4E)<=50
# LM75-6(0X4F)<=52.5
# (CPU board)
# Core<=59
# LM75-1(0X4B)<=41.1
# When FAN duty_cycle was 75%(LEVEL_FAN_MID). If all below case meet with, set to 50%(LEVEL_FAN_DEF).
# (MB board)
# LM75-1(0X48)<=55.8
# LM75-2(0X49)<=50.5
# LM75-3(0X4A)<=51.1
# LM75-4(0X4C)<=47.6
# LM75-5(0X4E)<=45.75
# LM75-6(0X4F)<=50.1
# (CPU board)
# Core<=57
# LM75-1(0X4B)<=36.6
# When fan_speed 50%(LEVEL_FAN_DEF).
# Meet with below case, Fan duty_cycle will be 75%(LEVEL_FAN_MID)
# (MB board)
# LM75-1(0X48)>=70
# LM75-2(0X49)>=66
# LM75-3(0X4A)>=68
# LM75-4(0X4C)>=62
# LM75-5(0X4E)>=62
# LM75-6(0X4F)>=67
# (CPU board)
# Core>=77
# LM75-1(0X4B)>=50
# When FAN duty_cycle was 75%(LEVEL_FAN_MID). If all below case meet with, set to 100%(LEVEL_FAN_MAX).
# (MB board)
# LM75-1(0X48)>=67
# LM75-2(0X49)>=62.5
# LM75-3(0X4A)>=65
# LM75-4(0X4C)>=59
# LM75-5(0X4E)>=58.5
# LM75-6(0X4F)>=63
# (CPU board)
# Core >=69
# LM75-1(0X4B)>=49
# Yellow Alarm
# MB board
# LM75-1(0X48)>=68
# LM75-2(0X49)>=64
# LM75-3(0X4A)>=65
# LM75-4(0X4C)>=61
# LM75-5(0X4E)>=60
# LM75-6(0X4F)>=64
# CPU Board
# Core>=70
# LM75-1(0X4B)>=68
# Red Alarm
# MB board
# LM75-1(0X48)>=72
# LM75-2(0X49)>=68
# LM75-3(0X4A)>=69
# LM75-4(0X4C)>=65
# LM75-5(0X4E)>=64
# LM75-6(0X4F)>=68
# CPU Board
# Core>=74
# LM75-1(0X4B)>=72
# Shut down
# MB board
# LM75-1(0X48)>=77
# LM75-2(0X49)>=73
# LM75-3(0X4A)>=74
# LM75-4(0X4C)>=70
# LM75-5(0X4E)>=69
# LM75-6(0X4F)>=73
# CPU Board
# Core>=79
# LM75-1(0X4B)>=77
def as9716_32d_set_fan_speed(pwm):
if pwm < 0 or pwm > 100:
print("Error: Wrong duty cycle value %d" % (pwm))
return -1
platform_chassis.get_fan(0).set_speed(pwm)
time.sleep(1)
return 0
def power_off_dut():
cmd_str = "i2cset -y -f 19 0x60 0x60 0x10"
status, output = subprocess.getstatusoutput(cmd_str)
return status
# If only one PSU insert(or one of PSU pwoer fail), and watt >800w. Must let DUT fan pwm >= 75% in AFO.
# Because the psu temp is high.
# Return 1: full load
# Return 0: Not full load
def check_psu_loading():
global platform_chassis
check_psu_watt = 0
for i in range(2):
if platform_chassis.get_psu(i).get_powergood_status() == 0:
check_psu_watt = 1 # If one psu power is not good, need to do another psu checking
if check_psu_watt:
for i in range(2):
if platform_chassis.get_psu(i).get_powergood_status() == 1:
psu_p_out = platform_chassis.get_psu(i).get_power()
if psu_p_out > 800:
return True
else:
return False
return False
fan_policy_state = 0
fan_policy_alarm = 0
send_yellow_alarm = 0
send_red_alarm = 0
fan_fail = 0
count_check = 0
test_temp = 0
test_temp_list = [0, 0, 0, 0, 0, 0, 0, 0]
temp_test_data = 0
test_temp_revert = 0
platform_chassis = None
# Make a class we can use to capture stdout and sterr in the log
class device_monitor(object):
# static temp var
temp = 0
new_duty_cycle = 0
duty_cycle = 0
ori_duty_cycle = 0
def __init__(self, log_file, log_level):
"""Needs a logger and a logger level."""
# set up logging to file
logging.basicConfig(
filename=log_file,
filemode='w',
level=log_level,
format='[%(asctime)s] {%(pathname)s:%(lineno)d} %(levelname)s - %(message)s',
datefmt='%H:%M:%S'
)
# set up logging to console
if log_level == logging.DEBUG:
console = logging.StreamHandler()
console.setLevel(log_level)
formatter = logging.Formatter('%(name)-12s: %(levelname)-8s %(message)s')
console.setFormatter(formatter)
logging.getLogger('').addHandler(console)
sys_handler = logging.handlers.SysLogHandler(address='/dev/log')
sys_handler.setLevel(logging.WARNING)
logging.getLogger('').addHandler(sys_handler)
def manage_fans(self):
global fan_policy_state
global fan_policy_alarm
global send_yellow_alarm
global send_red_alarm
global fan_fail
global count_check
global test_temp
global test_temp_list
global temp_test_data
global test_temp_revert
global platform_chassis
CHECK_TIMES = 3
LEVEL_FAN_INIT = 0
LEVEL_FAN_MIN = 1
LEVEL_FAN_MID = 2
LEVEL_FAN_MAX = 3 # LEVEL_FAN_DEF
LEVEL_FAN_YELLOW_ALARM = 4
LEVEL_FAN_RED_ALARM = 5
LEVEL_FAN_SHUTDOWN = 6
THERMAL_NUM_MAX = 7
FAN_NUM = 2
FAN_TRAY_NUM = 6
fan_policy_f2b = { # AFO
LEVEL_FAN_MIN: [50, 0x7],
LEVEL_FAN_MID: [75, 0xb],
LEVEL_FAN_MAX: [100, 0xf]
}
fan_policy_b2f = { # AFI
LEVEL_FAN_MID: [75, 0xb],
LEVEL_FAN_MAX: [100, 0xf]
}
afi_thermal_spec = {
"mid_to_max_temp": [61500, 51500, 49400, 49400, 45100, 46750, 48000, 38500],
"max_to_mid_temp": [57000, 47300, 45000, 45100, 40750, 42100, 44000, 35000]
}
afo_thermal_spec = {
"min_to_mid_temp": [70000, 66000, 68000, 62000, 62000, 67000, 77000, 50000],
"mid_to_max_temp": [67000, 62000, 65000, 59000, 58500, 63000, 69000, 49000],
"max_to_mid_temp": [59000, 53500, 55300, 50300, 50000, 52500, 59000, 41100],
"mid_to_min_temp": [55800, 50500, 51100, 47600, 45750, 50100, 57000, 36600],
"max_to_yellow_alarm": [68000, 64000, 65000, 61000, 60000, 64000, 70000, 68000],
"yellow_to_red_alarm": [72000, 68000, 69000, 65000, 64000, 68000, 74000, 72000],
"red_alarm_to_shutdown": [77000, 73000, 74000, 70000, 69000, 73000, 79000, 77000]
}
thermal_val = [0, 0, 0, 0, 0, 0, 0, 0]
max_to_mid = 0
mid_to_min = 0
if fan_policy_state == LEVEL_FAN_INIT:
fan_policy_state = LEVEL_FAN_MAX # This is default state
logging.debug("fan_policy_state=LEVEL_FAN_MAX")
return
count_check = count_check+1
if count_check < CHECK_TIMES:
return
else:
count_check = 0
fan_dir = platform_chassis.get_fan(1).get_direction()
if fan_dir == "INTAKE": # AFI
fan_thermal_spec = afi_thermal_spec
fan_policy = fan_policy_b2f
elif fan_dir == "EXHAUST": # AFO
fan_thermal_spec = afo_thermal_spec
fan_policy = fan_policy_f2b
else:
logging.debug("NULL case, not meet with fan_dir=0 or 1")
ori_duty_cycle = platform_chassis.get_fan(0).get_speed()
new_duty_cycle = 0
if test_temp_revert == 0:
temp_test_data = temp_test_data+2000
else:
temp_test_data = temp_test_data-2000
if test_temp == 0:
for i in range(THERMAL_NUM_MAX):
thermal_val[i] = platform_chassis.get_thermal(i).get_temperature()*1000
else:
for i in range(THERMAL_NUM_MAX):
thermal_val[i] = test_temp_list[i]
thermal_val[i] = thermal_val[i] + temp_test_data
fan_fail = 0
ori_state = fan_policy_state
current_state = fan_policy_state
if fan_dir == "INTAKE": # AFI
for i in range(THERMAL_NUM_MAX):
if ori_state == LEVEL_FAN_MID:
if thermal_val[i] >= fan_thermal_spec["mid_to_max_temp"][i]:
current_state = LEVEL_FAN_MAX
logging.debug("current_state=LEVEL_FAN_MAX")
break
else:
if (thermal_val[i] <= fan_thermal_spec["max_to_mid_temp"][i]):
max_to_mid = max_to_mid+1
if max_to_mid == THERMAL_NUM_MAX and fan_policy_state == LEVEL_FAN_MAX:
if fan_fail == 0:
current_state = LEVEL_FAN_MID
logging.debug("current_state=LEVEL_FAN_MID")
else: # AFO
psu_full_load = check_psu_loading()
for i in range(THERMAL_NUM_MAX):
if ori_state == LEVEL_FAN_MID:
if thermal_val[i] >= fan_thermal_spec["mid_to_max_temp"][i]:
current_state = LEVEL_FAN_MAX
break
else:
if psu_full_load != True and thermal_val[i] <= fan_thermal_spec["mid_to_min_temp"][i]:
mid_to_min = mid_to_min+1
elif ori_state == LEVEL_FAN_MIN:
if psu_full_load == True and fan_fail == 0:
current_state = LEVEL_FAN_MID
logging.warning("psu_full_load, set current_state=LEVEL_FAN_MID")
logging.debug("current_state=LEVEL_FAN_MID")
if thermal_val[i] >= fan_thermal_spec["min_to_mid_temp"][i] and fan_fail == 0:
current_state = LEVEL_FAN_MID
logging.debug("current_state=LEVEL_FAN_MID")
else:
if thermal_val[i] <= fan_thermal_spec["max_to_mid_temp"][i]:
max_to_mid = max_to_mid+1
if fan_policy_alarm == 0:
if thermal_val[i] >= fan_thermal_spec["max_to_yellow_alarm"][i]:
if send_yellow_alarm == 0:
logging.warning('Alarm-Yellow for temperature high is detected')
fan_policy_alarm = LEVEL_FAN_YELLOW_ALARM
send_yellow_alarm = 1
elif fan_policy_alarm == LEVEL_FAN_YELLOW_ALARM:
if thermal_val[i] >= fan_thermal_spec["yellow_to_red_alarm"][i]:
if send_red_alarm == 0:
logging.warning('Alarm-Red for temperature high is detected')
logging.warning('Alarm for temperature high is detected ')
fan_policy_alarm = LEVEL_FAN_RED_ALARM
send_red_alarm = 1
elif fan_policy_alarm == LEVEL_FAN_RED_ALARM:
if thermal_val[i] >= fan_thermal_spec["red_alarm_to_shutdown"][i]:
logging.critical('Alarm-Critical for temperature high is detected, shutdown DUT')
logging.critical('Alarm for temperature critical is detected ')
fan_policy_alarm = LEVEL_FAN_SHUTDOWN
time.sleep(2)
power_off_dut()
if max_to_mid == THERMAL_NUM_MAX and ori_state == LEVEL_FAN_MAX:
if fan_fail == 0:
current_state = LEVEL_FAN_MID
logging.debug("current_state=LEVEL_FAN_MID")
logging.debug("current_state=LEVEL_FAN_MID")
if fan_policy_alarm != 0:
logging.warning('Alarm for temperature high is cleared')
fan_policy_alarm = 0
send_yellow_alarm = 0
send_red_alarm = 0
test_temp_revert = 0
if mid_to_min == THERMAL_NUM_MAX and ori_state == LEVEL_FAN_MID:
if psu_full_load == 0:
current_state = LEVEL_FAN_MIN
logging.debug("current_state=LEVEL_FAN_MIN")
# Check Fan status
for i in range(FAN_TRAY_NUM * FAN_NUM):
if not platform_chassis.get_fan(i).get_status() or not platform_chassis.get_fan(i).get_speed_rpm():
new_duty_cycle = 100
logging.warning('fan_%d fail, set duty_cycle to 100', i+1)
if test_temp == 0:
fan_fail = 1
# 1.When insert/remove fan, fan speed/log still according to thermal policy.
# 2.If thermal policy state is bigger than LEVEL_FAN_MAX:
# Do not change back to LEVEL_FAN_MAX, beacuse still need to deal with LOG or shutdown case.
# 3.If thermal policy state is smaller than LEVEL_FAN_MAX, set state=MAX.
# When remove and insert back fan test, policy check temp and set to correct fan_speed.
#
if current_state < LEVEL_FAN_MAX:
current_state = LEVEL_FAN_MAX
logging.debug('fan_%d fail, current_state=LEVEL_FAN_MAX', i+1)
as9716_32d_set_fan_speed(new_duty_cycle)
break
else:
fan_fail = 0
if current_state != ori_state:
fan_policy_state = current_state
new_duty_cycle = fan_policy[current_state][0]
logging.debug("fan_policy_state=%d, new_duty_cycle=%d", fan_policy_state, new_duty_cycle)
if new_duty_cycle != ori_duty_cycle and fan_fail == 0:
as9716_32d_set_fan_speed(new_duty_cycle)
return True
if new_duty_cycle == 0 and fan_fail == 0:
as9716_32d_set_fan_speed(FAN_DUTY_CYCLE_MAX)
return True
def main(argv):
log_file = '%s.log' % FUNCTION_NAME
log_level = logging.INFO
global test_temp
if len(sys.argv) != 1:
try:
opts, args = getopt.getopt(argv, 'hdlt:', ['lfile='])
except getopt.GetoptError:
print("Usage: %s [-d] [-l <log_file>]" % sys.argv[0])
return 0
for opt, arg in opts:
if opt == '-h':
print("Usage: %s [-d] [-l <log_file>]" % sys.argv[0])
return 0
elif opt in ('-d', '--debug'):
log_level = logging.DEBUG
elif opt in ('-l', '--lfile'):
log_file = arg
if sys.argv[1] == '-t':
if len(sys.argv) != 10:
print("temp test, need input 8 temp")
return 0
i = 0
for x in range(2, 10):
test_temp_list[i] = int(sys.argv[x])*1000
i = i+1
test_temp = 1
log_level = logging.DEBUG
print(test_temp_list)
global platform_chassis
platform_chassis = platform.Platform().get_chassis()
status, output = subprocess.getstatusoutput('i2cset -f -y 17 0x66 0x33 0x0')
if status:
print("Warning: Fan speed watchdog could not be disabled")
as9716_32d_set_fan_speed(100)
monitor = device_monitor(log_file, log_level)
# Loop forever, doing something useful hopefully:
while True:
monitor.manage_fans()
time.sleep(10)
if __name__ == '__main__':
main(sys.argv[1:])

View File

@ -0,0 +1,23 @@
#!/bin/bash
dis_i2c_ir3570a()
{
local addr=$1
i2cset -y 0 $addr 0xE5 0x01 &>/dev/null
i2cset -y 0 $addr 0x12 0x02 &>/dev/null
}
ir3570_check()
{
dump=`i2cdump -y 0 0x42 s 0x9a |awk 'END {print $2}'`
if [ $dump -eq 24 ]; then
echo "Disabling i2c function of ir3570a"
dis_i2c_ir3570a 0x4
fi
}
ir3570_check
echo "AS9716 post PDDF driver install completed"

View File

@ -0,0 +1,82 @@
#!/usr/bin/env python
# Script to stop and start the respective platforms default services.
# This will be used while switching the pddf->non-pddf mode and vice versa
import commands
def check_pddf_support():
return True
def stop_platform_svc():
status, output = commands.getstatusoutput("systemctl stop as9716-32d-platform-monitor-fan.service")
if status:
print "Stop as9716-32d-platform-fan.service failed %d"%status
return False
status, output = commands.getstatusoutput("systemctl stop as9716-32d-platform-monitor-psu.service")
if status:
print "Stop as9716-32d-platform-psu.service failed %d"%status
return False
status, output = commands.getstatusoutput("systemctl stop as9716-32d-platform-monitor.service")
if status:
print "Stop as9716-32d-platform-init.service failed %d"%status
return False
status, output = commands.getstatusoutput("systemctl disable as9716-32d-platform-monitor.service")
if status:
print "Disable as9716-32d-platform-monitor.service failed %d"%status
return False
status, output = commands.getstatusoutput("/usr/local/bin/accton_as9716_32d_util.py clean")
if status:
print "accton_as9716_32d_util.py clean command failed %d"%status
return False
# HACK , stop the pddf-platform-init service if it is active
status, output = commands.getstatusoutput("systemctl stop pddf-platform-init.service")
if status:
print "Stop pddf-platform-init.service along with other platform serives failed %d"%status
return False
return True
def start_platform_svc():
status, output = commands.getstatusoutput("/usr/local/bin/accton_as9716_32d_util.py install")
if status:
print "accton_as9716_32d_util.py install command failed %d"%status
return False
status, output = commands.getstatusoutput("systemctl enable as9716-32d-platform-monitor.service")
if status:
print "Enable as9716-32d-platform-monitor.service failed %d"%status
return False
status, output = commands.getstatusoutput("systemctl start as9716-32d-platform-monitor-fan.service")
if status:
print "Start as9716-32d-platform-monitor-fan.service failed %d"%status
return False
status, output = commands.getstatusoutput("systemctl start as9716-32d-platform-monitor-psu.service")
if status:
print "Start as9716-32d-platform-monitor-psu.service failed %d"%status
return False
return True
def start_platform_pddf():
status, output = commands.getstatusoutput("systemctl start pddf-platform-init.service")
if status:
print "Start pddf-platform-init.service failed %d"%status
return False
return True
def stop_platform_pddf():
status, output = commands.getstatusoutput("systemctl stop pddf-platform-init.service")
if status:
print "Stop pddf-platform-init.service failed %d"%status
return False
return True

View File

@ -0,0 +1 @@
as9716-32d/sonic_platform-1.0-py3-none-any.whl usr/share/sonic/device/x86_64-accton_as9716_32d-r0/pddf

View File

@ -0,0 +1,27 @@
# Special arrangement to make PDDF mode default
# Disable monitor, monitor-fan, monitor-psu (not enabling them would imply they will be disabled by default)
# Enable pddf-platform-monitor
# Steps to check syseeprom i2c address
modprobe i2c-i801
modprobe i2c-dev
use_57_eeprom=false
(i2cget -y -f 0 0x57 0x0) > /dev/null 2>&1
if [ $? -eq 0 ]; then
use_57_eeprom=true
fi
if $use_57_eeprom ; then
echo "The board has system EEPROM at I2C address 0x57"
# syseeprom is at the i2c address 0x57. Change the PDDF JSON file
sed -i 's@"topo_info": {"parent_bus": "0x0", "dev_addr": "0x56", "dev_type": "24c02"},@\
"topo_info": {"parent_bus": "0x0", "dev_addr": "0x57", "dev_type": "24c02"},@g' \
/usr/share/sonic/device/x86_64-accton_as9716_32d-r0/pddf/pddf-device.json
sync
fi
depmod -a
systemctl enable pddf-platform-init.service
systemctl start pddf-platform-init.service
systemctl enable as9716-32d-pddf-platform-monitor.service
systemctl start as9716-32d-pddf-platform-monitor.service