This file is indexed.

/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)