/usr/sbin/smbios-thermal-ctl is in smbios-utils 2.4.1-1.
This file is owned by root:root, with mode 0o755.
The actual contents of the file can be viewed below.
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 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 | #!/usr/bin/python3
# vim:expandtab:autoindent:tabstop=4:shiftwidth=4:filetype=python:tw=0
#############################################################################
#
# Copyright (c) 2015 Dell Computer Corporation
# by Srinivas Gowda <srinivas_g_gowda@dell.com>
# Dual Licenced under GNU GPL and OSL
#
#############################################################################
"""smbios-thermal-ctl"""
# import arranged alphabetically
import gettext
import locale
import os
import sys
# the following vars are all substituted on install
# this bin isnt byte-compiled, so this is ok
pythondir="/usr/lib/python3/dist-packages"
clidir="/usr/share/smbios-utils"
# end vars
# import all local modules after this.
sys.path.insert(0,pythondir)
sys.path.insert(0,clidir)
import cli
from libsmbios_c import smbios_token, smbios, smi, system_info as sysinfo, localedir, GETTEXT_PACKAGE
from libsmbios_c.trace_decorator import traceLog, getLog
__VERSION__="2.4.1"
locale.setlocale(locale.LC_ALL, '')
gettext.install(GETTEXT_PACKAGE, localedir)
moduleLog = getLog()
verboseLog = getLog(prefix="verbose.")
class RunTimeThermalSensorErr(Exception): pass
def command_parse():
parser = cli.OptionParser(usage=__doc__, version=__VERSION__)
parser.add_option("--info", "-i", action="store_true", default=False, help=_("This will Display the Supported Features of USTT and AAC"))
parser.add_option("--get-thermal-info", "-g", action="store_true", default=False, help=_("This will Display the Thermal Information of a system"))
parser.add_option("--set-thermal-mode", action="store", dest="thermal_mode", help=_("Option to set Thermal Mode; balanced, cool-bottom, quiet, performance"))
cli.addStdOptions(parser, passwordOpts=True, securityKeyOpt=True)
if len(sys.argv) == 1:
parser.print_help()
sys.exit()
return parser.parse_args()
def is_set(num, pos):
mask = 1 << pos
return True if (num & mask) else False
# Return a the byte specified by byte_pos from a double word num
def get_byte(num, byte_pos):
try :
if byte_pos < 0 or byte_pos > 3:
raise RunTimeThermalSensorErr( "Internal Error: Byte position out of bound - ", byte_pos )
except RunTimeThermalSensorErr as e:
print("\t%s" % e)
return ((num >> (byte_pos*8)) & 0xFF)
def PrintSupportedThermModes(flag='default'):
print(_("\n Print all the Available Thermal Information of your system: "))
print(_("-------------------------------------------------------------------"))
try:
res = smi.simple_ci_smi( 17, 19, 0 )
verboseLog.info( _("Get Thermal Information: res[smi.cbRES1]=0x%X,res[smi.cbRES2]=0x%X, res[smi.cbRES3]=0x%X") %
(res[smi.cbRES1],res[smi.cbRES2],res[smi.cbRES3]) )
if res[smi.cbRES1] != 0:
raise RunTimeThermalSensorErr( _(" Info: Unable to Get Thermal Information on this system\n") )
if flag == 'thermal-mode': return (res[smi.cbRES2] & 0xF )
print(_(" \nSupported Thermal Modes: "))
if is_set (res[smi.cbRES2], 0): print("\t Balanced")
if is_set (res[smi.cbRES2], 1): print("\t Cool Bottom")
if is_set (res[smi.cbRES2], 2): print("\t Quiet")
if is_set (res[smi.cbRES2], 3): print("\t Performance")
print(_(" \nSupported Active Acoustic Controller (AAC) modes: "))
if is_set ((get_byte(res[smi.cbRES2], 1)), 0): print("\t AAC (Balanced)")
if is_set ((get_byte(res[smi.cbRES2], 1)), 1): print("\t AAC (Cool Bottom)")
if is_set ((get_byte(res[smi.cbRES2], 1)), 2): print("\t ACC (Quiet)")
if is_set ((get_byte(res[smi.cbRES2], 1)), 3): print("\t ACC (Performance)")
if flag == 'acc-config' : return get_byte(res[smi.cbRES3], 1)
print(_(" \nSupported AAC Configuration type: "))
if (get_byte(res[smi.cbRES3], 1)) == 0 : print("\tGlobal (AAC enable/disable applies to all supported USTT modes)")
if (get_byte(res[smi.cbRES3], 1)) == 1 : print("\tUser Selectable Thermal Table(USTT) mode specific")
except RunTimeThermalSensorErr as e:
print("\t%s" % e)
def GetCurrentThermalMode(flag='default'):
try:
res = smi.simple_ci_smi( 17, 19, 0 )
verboseLog.info( _("Get Thermal Information: res[smi.cbRES1]=0x%X,res[smi.cbRES2]=0x%X,res[smi.cbRES3]=0x%X") %
(res[smi.cbRES1],res[smi.cbRES2],res[smi.cbRES3]) )
if res[smi.cbRES1] != 0:
raise RunTimeThermalSensorErr( _(" Info: Unable to Get Thermal Information on this system\n") )
if flag == 'thermal-mode': return res[smi.cbRES3]
if flag == 'acc-mode' : return (get_byte(res[smi.cbRES3], 2))
print(_("\n Print Current Status of Thermal Information: "))
print(_("-------------------------------------------------------------------"))
print(_(" \nCurrent Thermal Modes: "))
if is_set (res[smi.cbRES3], 0): print("\t Balanced")
if is_set (res[smi.cbRES3], 1): print("\t Cool Bottom")
if is_set (res[smi.cbRES3], 2): print("\t Quiet")
if is_set (res[smi.cbRES3], 3): print("\t Performance")
print(_(" \nCurrent Active Acoustic Controller (AAC) Mode: "))
"If AAC Configuration Type is Global,"
if (get_byte(res[smi.cbRES3], 2))== 0 : print("\t AAC mode Disabled")
if (get_byte(res[smi.cbRES3], 2))== 1 : print("\t AAC mode Enabled ")
print(_(" \nCurrent Active Acoustic Controller (AAC) Mode: "))
if (get_byte(res[smi.cbRES3], 1)) == 0:
if (get_byte(res[smi.cbRES3], 2)) == 0 : print("\tGlobal (AAC enable/disable applies to all supported USTT modes)")
if (get_byte(res[smi.cbRES3], 2)) == 1 : print("\tUSTT mode specific")
elif (get_byte(res[smi.cbRES3], 1)) == 1:
if is_set ((get_byte(res[smi.cbRES3], 2)), 0): print("\t AAC (Balanced)")
if is_set ((get_byte(res[smi.cbRES3], 2)), 1): print("\t AAC (Cool Bottom)")
if is_set ((get_byte(res[smi.cbRES3], 2)), 2): print("\t ACC (Quiet)")
if is_set ((get_byte(res[smi.cbRES3], 2)), 3): print("\t ACC (Performance)")
print(_(" \nCurrent Fan Failure Mode: "))
if is_set ((get_byte(res[smi.cbRES3], 3)), 0): print("\tMinimal Fan Failure (at least one fan has failed, one fan working)")
if is_set ((get_byte(res[smi.cbRES3], 3)), 1): print("\tCatastrophic Fan Failure (all fans have failed)")
except RunTimeThermalSensorErr as e:
print("\t%s" % e)
def SetThermalMode(thermal_mode):
verboseLog.info( _("Selected Thermal Mode = %s") % (thermal_mode))
try :
if thermal_mode.lower() == 'balanced' : var_thermal_mode = 1
elif thermal_mode.lower() == 'cool-bottom': var_thermal_mode = 1 << 1
elif thermal_mode.lower() == 'quiet' : var_thermal_mode = 1 << 2
elif thermal_mode.lower() == 'performance': var_thermal_mode = 1 << 3
else :
raise RunTimeThermalSensorErr( _("Invalid thermal mode option specified: %s, \
Supported modes : balanced, cool-bottom, quiet, performance") % thermal_mode)
#Set ACC settings as current, since here we only modify the thermal mode
acc_mode = GetCurrentThermalMode('acc-mode')
cbArg2 = acc_mode << 8 | var_thermal_mode
res = smi.simple_ci_smi( 17, 19, 1, cbArg2 )
verboseLog.info( _("Set Thermal Information: res[smi.cbRES1]=0x%X,res[smi.cbRES2]=0x%X,res[smi.cbRES3]=0x%X") %
(res[smi.cbRES1],res[smi.cbRES2],res[smi.cbRES3]))
if res[smi.cbRES1] != 0:
raise RunTimeThermalSensorErr( _("Set Thermal Information failed: arg2- %x cbRES1: 0x%X\
cbRES2: 0x%X cbRES3: 0x%X \n") % (cbArg2, res[smi.cbRES1], res[smi.cbRES3], res[smi.cbRES4]))
print(( _("Thermal Information Set successfully to: %s") % thermal_mode))
except RunTimeThermalSensorErr as e:
print("\t %s" % e)
print("\t Set Thermal Information Failed")
def main():
exit_code = 0
(options, args) = command_parse()
cli.setup_std_options(options)
try:
if options.info:
print(_("Libsmbios version : %s") % sysinfo.get_library_version_string())
print(_("smbios-thermal-ctl version : %s") % __VERSION__)
verboseLog.info( _(" You can use smbios-thermal-ctl utlity to control the fan speed"))
# Info will show the all the supported attributes of User selectable Thermal settings
PrintSupportedThermModes()
if options.get_thermal_info:
print(_("Helper function to Get current Thermal Mode settings"))
GetCurrentThermalMode()
if options.thermal_mode:
print(_("Helper function to Set Thermal Mode"))
SetThermalMode(options.thermal_mode)
except (smi.SMIExecutionError, ) as e:
exit_code=3
moduleLog.info( _("ERROR: Could not execute SMI.") )
verboseLog.info( _("The smi library returned this error:") )
verboseLog.info( str(e) )
moduleLog.info( cli.standardFailMessage )
except (smbios.TableParseError, token.TokenTableParseError) as e:
exit_code=3
moduleLog.info( _("ERROR: Could not parse system SMBIOS table.") )
verboseLog.info( _("The smbios library returned this error:") )
verboseLog.info( str(e) )
moduleLog.info( cli.standardFailMessage )
except (token.TokenManipulationFailure,) as e:
exit_code=4
moduleLog.info( _("ERROR: Could not manipulate system token.") )
verboseLog.info( _("The token library returned this error:") )
verboseLog.info( str(e) )
moduleLog.info( cli.standardFailMessage )
return exit_code
if __name__ == "__main__":
sys.exit( main() )
# cbClass 17
# cbSelect 19
# User Selectable Thermal Tables(USTT)
# cbArg1 determines the function to be performed
# cbArg1 0x0 = Get Thermal Information
# cbRES1 Standard return codes (0, -1, -2)
# cbRES2, byte 0 Bitmap of supported thermal modes. A mode is supported if its bit is set to 1
# Bit 0 Balanced
# Bit 1 Cool Bottom
# Bit 2 Quiet
# Bit 3 Performance
# cbRES2, byte 1 Bitmap of supported Active Acoustic Controller (AAC) modes. Each mode
# corresponds to the supported thermal modes in byte 0. A mode is supported if
# its bit is set to 1.
# Bit 0 AAC (Balanced)
# Bit 1 AAC (Cool Bottom
# Bit 2 AAC (Quiet)
# Bit 3 AAC (Performance)
# cbRes3, byte 0 Current Thermal Mode
# Bit 0 Balanced
# Bit 1 Cool Bottom
# Bit 2 Quiet
# Bit 3 Performanc
# cbRes3, byte 1 AAC Configuration type
# 0 Global (AAC enable/disable applies to all supported USTT modes)
# 1 USTT mode specific
# cbRes3, byte 2 Current Active Acoustic Controller (AAC) Mode
# If AAC Configuration Type is Global,
# 0 AAC mode disabled
# 1 AAC mode enabled
# If AAC Configuration Type is USTT mode specific (multiple bits may be set),
# Bit 0 AAC (Balanced)
# Bit 1 AAC (Cool Bottom
# Bit 2 AAC (Quiet)
# Bit 3 AAC (Performance)
# cbRes3, byte 3 Current Fan Failure Mode
# Bit 0 Minimal Fan Failure (at least one fan has failed, one fan working)
# Bit 1 Catastrophic Fan Failure (all fans have failed)
# cbArg1 0x1 (Set Thermal Information), both desired thermal mode and
# desired AAC mode shall be applied
# cbArg2, byte 0 Desired Thermal Mode to set (only one bit may be set for this parameter)
# Bit 0 Balanced
# Bit 1 Cool Bottom
# Bit 2 Quiet
# Bit 3 Performance
# cbArg2, byte 1 Desired Active Acoustic Controller (AAC) Mode to set
# If AAC Configuration Type is Global,
# 0 AAC mode disabled
# 1 AAC mode enabled
#
# If AAC Configuration Type is USTT mode specific (multiple bits may be set for this parameter),
# Bit 0 AAC (Balanced)
# Bit 1 AAC (Cool Bottom
# Bit 2 AAC (Quiet)
# Bit 3 AAC (Performance)
|