Jump to content
Sign in to follow this  
ModemJunki

I fried a computer and made a temperature watchdog for the new one

Recommended Posts

Edit: I found out from looking into the post from coffeeturtle that not all motherboards support WMI temperature reading. You have to explore the WMI namespaces for your system!

I (stupidly) fried a computer. It was an old Zotac small form factor device that was junked at work, so I turned it into a home theater PC.

But I had put it in my "audio rack", which has glass doors and not very good ventilation. Normally I would open the doors while it was running and I use a media remote which would hibernate the system. At some point, it woke up by itself (probably Windows Update) and did not go to sleep again. Unfortunately this happened during the summer months while I was away on a small holiday and of course the glass doors were closed.

When I finally noticed the system was on, it was super hot and the fan was at full blast. Dead. No POST, no green light on the mainboard even with a new power supply, old mechanical laptop hard drive gives errors when connected to another device. My next home project will be to make some appropriate ventilation in the audio cabinet.

For the replacement I bought a used HP ultra small form factor machine and decided to start working on an app to monitor the ambient temperature sensor on the motherboard and shutdown the system if needed. I used some code for smooth label updates from here (needed updating to work with latest AutoIT) and temperature conversion from here (not really needed, only if you want temperatures in something other than C).

Maybe I will work on this some more and make it configurable with an .INI file or even storing the settings in registry, but since I probably won't I thought I'd put it up as an example of what a non-expert can do with an afternoon of coding and ideas from the community here.

This uses an ambient temperature sensor populated on HP machines in HP-specific WMI extensions but the WMI query can easily be changed to any available sensor including CPU or GPU.

The watchdog monitors temperature, warns with S.O.S beeping if the set point is exceeded, and either shuts down if a timeout is reached while the temperatures are high or goes back to monitoring if the temperature goes lower then the set point, and logs events to the Windows application event logs.

The GUI it puts up is very small at the top center of the screen (very small on a 4k display anyway).

#requireadmin because of the WMI query.

A timer for the WMI query because of prior experience using similar functions in WinPE to get model infos from HP WMI bios extensions.

If you play around with this, please post your fixes/improvements/changes back to this thread.

Have fun!

Edit: Updated to show "Unable to query temperature" if the WMI query returns a null

#AutoIt3Wrapper_Icon=temperature-2-multi-size.ico
#RequireAdmin
#NoTrayIcon
#include <Temperature.au3>
#include <FFLabels.au3>
#include <GUIConstantsEx.au3>
#include <StaticConstants.au3>
#include <WindowsConstants.au3>

Global $winLogger = "C:\WINDOWS\system32\eventcreate.exe" ; application to use for event logging
Global $MessageSource = "HP TempMon"
Global $width = 185
Global $height = 15
Global $FontSize = 8
Global $FontFamily = 'Microsoft Sans Serif'
Global $normClr = 0xFF000000
Global $warnClr = 0xFF0000
Global $WMInameSpace = "HP_BIOSNumericSensor"
;~ Global $WMIReadSensor = "Chassis Thermal Index" ; HP 8300
Global $WMIReadSensor = "System Ambient Temperature" ; HP Z400
Global $warnTmp = 46.1111111111 ; degrees C
;~ Global $warnTmp = 24 ; degrees C for testing
Global $ovrtmpTime = 5 ; in minutes
Global $hGUI = GUICreate("Board Temperature", $width, $height, -1, 0, BitOR($WS_SYSMENU, $WS_POPUP), BitOR($WS_EX_TOPMOST, $WS_EX_WINDOWEDGE, $WS_EX_TOOLWINDOW))
Global $lb1 = _GUICtrlFFLabel_Create($hGUI, "", -1, -1, $width, $height, 8, -1, 0, 1, $normClr)
GUISetState(@SW_SHOW)

_TempSenseLoop()

Func _TempSenseLoop()
    While 1
        Sleep(250)
        $nMsg = GUIGetMsg()
        Switch $nMsg
            Case $GUI_EVENT_CLOSE
                GUIDelete($hGUI)
                Return
            Case Else
                $s_ambTempC = _doQueryHPNumSens(5, $WMInameSpace, $WMIReadSensor, "100") ; + 100
                Select
                    Case $s_ambTempC = ""
                        _GUICtrlFFLabel_SetTextColor($lb1, $normClr)
                        _GUICtrlFFLabel_SetData($lb1, "Unable to query temperature")
                    Case $s_ambTempC < $warnTmp
                        _GUICtrlFFLabel_SetTextColor($lb1, $normClr)
                        _GUICtrlFFLabel_SetData($lb1, "Board temperature: " & $s_ambTempC & "C" & "/" & Round(_CelsiusToFahrenheit($s_ambTempC), 0) & "F")
                    Case Else
                        _GUICtrlFFLabel_SetTextColor($lb1, $warnClr)
                        _GUICtrlFFLabel_SetData($lb1, "Board temperature: " & $s_ambTempC & "C" & "/" & Round(_CelsiusToFahrenheit($s_ambTempC), 0) & "F")
                        _doLog("System ambient temperature has exceeded " & $warnTmp & "C.", "WARN", $MessageSource)
                        _AlarmMonitor($ovrtmpTime, $warnTmp)
                EndSelect
        EndSwitch
    WEnd
EndFunc   ;==>_TempSenseLoop

Func _AlarmMonitor($min, $tmp)
    Local $Shutdown = 1
    Local $s_ambTemp
    Local $freq = 3500
    Local $shrt = 200
    Local $long = 500
    Local $timer = TimerInit()
    Local $wait = 1000 * ($min * 60)
    While TimerDiff($timer) < $wait
        $s_ambTemp = _doQueryHPNumSens(5, $WMInameSpace, $WMIReadSensor, "100") ; + 100
        If $s_ambTemp >= $tmp Then
            _GUICtrlFFLabel_SetData($lb1, "WARNING! OVERTEMP!")
            Beep($freq, $shrt)
            Beep($freq, $shrt)
            Beep($freq, $shrt)
            Sleep(250)
            Beep($freq, $long)
            Beep($freq, $long)
            Beep($freq, $long)
            Sleep(250)
            Beep($freq, $shrt)
            Beep($freq, $shrt)
            Beep($freq, $shrt)
            _GUICtrlFFLabel_SetData($lb1, "Board temperature: " & $s_ambTemp & "C" & "/" & _CelsiusToFahrenheit($s_ambTemp) & "F")
            Sleep(2000)
        ElseIf $s_ambTemp < $tmp Then
            $Shutdown = 0
            ExitLoop
        EndIf
    WEnd
    If $Shutdown = 1 Then
        _doLog("The system is shutting down due to overtemperature conditions. The temperature at the time of shutdown was " & $s_ambTemp & "C" & "/" & _CelsiusToFahrenheit($s_ambTemp) & "F", "ERRR", $MessageSource)
        Shutdown(24)
    ElseIf $Shutdown = 0 Then
        _doLog("System ambient temperature has been reduced below " & $warnTmp & "C.", "INFO", $MessageSource)
    EndIf
EndFunc   ;==>_AlarmMonitor

Func _doQueryHPNumSens($min, $class, $name, $sleep) ; $sleep = milliseconds
    Local $s_tempReading = ""
    Local $s_HPBiosWMIService = "winmgmts:\\" & @ComputerName & "\Root\HP\InstrumentedBIOS"
    Local $objHPBiosWMIService = ObjGet($s_HPBiosWMIService)
    Local $s_biosQuery = $objHPBiosWMIService.ExecQuery("select * from " & $class & " where Name='" & $name & "'")

    $timer = TimerInit() ; we may need to wait for the HP WMI extensions to enumerate in WMI, in WinPE this takes some minutes.
    $wait = 1000 * ($min * 60)

    If Not IsObj($s_biosQuery) Then ; we do need to wait, put up a splash screen
        $s_WMISplash = 1
        SplashTextOn("WMI", "Probing WMI (up to 10 minutes)...", 300, 50, -1, -1, 1)
        Sleep(1000)
        While TimerDiff($timer) < $wait
            Sleep(500)
            $objHPBiosWMIService = ObjGet($s_HPBiosWMIService)
            If IsObj($objHPBiosWMIService) Then
                Sleep(500)
                $colProdName = $s_biosQuery
                If IsObj($colProdName) Then
                    For $oItem In $colProdName
                        $s_tempReading = $oItem.CurrentReading
                    Next
                EndIf
            EndIf
        WEnd
    ElseIf IsObj($s_biosQuery) Then
        ConsoleWrite($s_biosQuery & @CRLF)
        For $oItem In $s_biosQuery
            $s_tempReading = $oItem.CurrentReading
        Next
    Else
        $s_tempReading = 0
    EndIf
    Sleep($sleep)
    Return $s_tempReading
EndFunc   ;==>_doQueryHPNumSens

Func _doLog($message, $type, $msgsrc)
    Select
        Case $type = "INFO"
            $logTyp = "INFORMATION"
        Case $type = "WARN"
            $logTyp = "WARNING"
        Case $type = "ERRR"
            $logTyp = "ERROR"
    EndSelect

    If FileExists($winLogger) Then
        RunWait(@ComSpec & " /c " & $winLogger & " /L Application /T " & $logTyp & " /SO " & Chr(34) & $msgsrc & Chr(34) & " /ID 1000 /D " & Chr(34) & $message & Chr(34), "", @SW_HIDE)
    EndIf
EndFunc   ;==>_doLog

 

 

 

Temperature.au3

FFLabels.au3

HP_Temp.au3

Edited by ModemJunki
New information

Always carry a towel.

Share this post


Link to post
Share on other sites

 

3 hours ago, coffeeturtle said:

Very nice work!

 

I have an HP Pavilion - is the output in the screenshot correct?

 

Thanks!

temp.png

No, it means the system is not detecting temperature, something for which my simple code does not (yet) have the ability to report.

Did you modify the code to query a different sensor? At my house I have two HP systems, for example, and each uses a different sensor. One of them has a "Chassis Thermal Index" where the other one has a "System Ambient Temperature". At the moment I don't have access to any systems other than HP business class machines (my workplace junks the some of nicest old stuff, the Zotac was a weird find).

Scriptomatic is your friend when looking for WMI namespaces to query! At home I also use WMI Explorer http://www.ks-soft.net/hostmon.eng/downpage.htm#utils (Free for non-commercial use).

What version of Windows are you running?

Edit: Try WMI explorer and see if you can query Root\WMI -> MSAcpi_ThermalZoneTemperature

Maybe it's supported by your mainboard manufacturer.

To connect to the namespace in WMI explorer, select Action->Connect to host/namespace, then click the book icon next to Name space. After a few momente the namespaces load and you can browse for WMI (or, HP!)

Edited by ModemJunki

Always carry a towel.

Share this post


Link to post
Share on other sites

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!

Register a new account

Sign in

Already have an account? Sign in here.

Sign In Now
Sign in to follow this  

  • Recently Browsing   0 members

    No registered users viewing this page.

  • Similar Content

    • By shital
      #Region ;**** Directives created by AutoIt3Wrapper_GUI **** #AutoIt3Wrapper_UseX64=y #EndRegion ;**** Directives created by AutoIt3Wrapper_GUI **** RunAs(test", @ComputerName, "testinng",2,"wmic product where ""name= '%notepadexamples%'"" call uninstall", @SystemDir & "\wbem", @SW_MAXIMIZE)  it not working 
       
    • By Colduction
      Hi AutoIt Programmers!

      I've seen a code that gives Name of Group by writing it's group's SID, but this one response very slower than Windows Command-Line WMIC
       
      Func _GetLocalGroupName($sSID = 'S-1-5-18') $objWMIService = ObjGet ("winmgmts:\\" & @ComputerName & "\root\cimv2") $colItems = $objWMIService.ExecQuery('SELECT Name FROM Win32_Group where SID="' & $sSID & '"') For $GroupNames in $colItems MsgBox (0,"",$GroupNames.Name) ExitLoop Next EndFunc I don't want call and use any third-party programs even CMD, i just want use from Windows API, netapi32.dll or AutoIt Functions (Standalone).
      Do you have any idea to improve speed/performance of this code? I'll happy of your comments
       
       
       
      ====================== SOLOUTION by @Subz ======================
       
    • By jresine
      Hello, is it possible to know via a script or command, to have the percentage of disk usage of a process? thank you in advance.
      ps: see image

    • By Dwalfware
      HI
      Another head scratched
      Anyone solve how to get the IPV6 from WMI?
      I followed the usual SELECT * FROM Win32_NetworkAdapter WHERE NetConnectionID='" & GUICtrlRead($cboAdapters) & "'
      get the index 
      port it to 
      Local $query = $objWMIService.ExecQuery("SELECT * FROM Win32_NetworkAdapterConfiguration WHERE Index = " & _GetSelectedNIC(), "WQL", 0x30 )
      All i can see is IPv4 addresses
      My question is is there another place the IPv6 information is hiding? I configured my LAN with IPV6 and my wireless with IPv4 so I can see what shows.
      Thanks in Advance.
      PS. I did try google
       
    • By Dwalfware
      Hi Folks
      Can not get my head around this one.
      This function gets the DNS object s
      If you have two DNS entries fine. but if you have only one! Trouble
      $objItem.DNSServerSearchOrder[0] = Primary DNS
      $objItem.DNSServerSearchOrder[1] = Secondry DNS not always there. I need a way to return 0 if not there or the IP if its there.
      Or is there a way to tell how many objects there are in the "[?]"
      I'm terrible at WMI errors, thanks for help in advance.
      Func _GETcurrentNICinfo2() $oErrors = ObjEvent("AutoIt.Error", "Error_Handle") Local $DnsSecondry Local $objWMIService = ObjGet( "winmgmts:\\" & @ComputerName & "\root\CIMV2" ) Local $query = $objWMIService.ExecQuery("SELECT DNSServerSearchOrder FROM Win32_NetworkAdapterConfiguration WHERE Index = " & _GetSelectedNIC(), "WQL", 0x30 ) If @error Then Return 0 If NOT IsObj($query) Then Return 0 Else For $objItem In $query Return $objItem.DNSServerSearchOrder[1] next Endif EndFunc  
×
×
  • Create New...