#!/usr/bin/env python3 # # auto-cpufreq - core functionality import os import platform as pl import re import shutil import sys import time import warnings from math import isclose from pathlib import Path from pprint import pformat from subprocess import getoutput, call, run import psutil import distro import click warnings.filterwarnings("ignore") # ToDo: # - re-enable CPU fan speed display and make more generic and not only for thinkpad # - replace get system/CPU load from: psutil.getloadavg() | available in 5.6.2) SCRIPTS_DIR = Path("/usr/local/share/auto-cpufreq/scripts/") # from the highest performance to the lowest ALL_GOVERNORS = ("performance", "ondemand", "conservative", "schedutil", "userspace", "powersave") CPUS = os.cpu_count() def turbo(value: bool = None): """ Get and set turbo mode """ p_state = Path("/sys/devices/system/cpu/intel_pstate/no_turbo") cpufreq = Path("/sys/devices/system/cpu/cpufreq/boost") if p_state.exists(): inverse = True f = p_state elif cpufreq.exists(): f = cpufreq inverse = False else: print("Warning: CPU turbo is not available") return False if value is not None: if inverse: value = not value try: f.write_text(str(int(value)) + "\n") except PermissionError: print("Warning: Changing CPU turbo is not supported. Skipping.") return False value = bool(int(f.read_text().strip())) if inverse: value = not value return value def charging(): """ get charge state: is battery charging or discharging """ bat_info = psutil.sensors_battery() if bat_info is None: state = True else: state = bat_info.power_plugged return state def get_avail_gov(): f = Path("/sys/devices/system/cpu/cpu0/cpufreq/scaling_available_governors") return f.read_text().strip().split(" ") def get_avail_powersave(): """ Iterate over ALL_GOVERNORS in reverse order: from powersave to performance :return: """ for g in ALL_GOVERNORS[::-1]: if g in get_avail_gov(): return g def get_avail_performance(): for g in ALL_GOVERNORS: if g in get_avail_gov(): return g def get_current_gov(): return getoutput("cpufreqctl --governor").strip().split(" ")[0] # auto-cpufreq log file auto_cpufreq_log_file = Path("/var/log/auto-cpufreq.log") auto_cpufreq_log_file_snap = Path("/var/snap/auto-cpufreq/current/auto-cpufreq.log") # daemon check dcheck = getoutput("snapctl get daemon") def cpufreqctl(): """ deploy cpufreqctl script """ # detect if running on a SNAP if os.getenv('PKG_MARKER') == "SNAP": pass else: # deploy cpufreqctl script (if missing) if os.path.isfile("/usr/bin/cpufreqctl"): shutil.copy("/usr/bin/cpufreqctl", "/usr/bin/cpufreqctl.auto-cpufreq.bak") shutil.copy(SCRIPTS_DIR / "cpufreqctl.sh", "/usr/bin/cpufreqctl") else: shutil.copy(SCRIPTS_DIR / "cpufreqctl.sh", "/usr/bin/cpufreqctl") def cpufreqctl_restore(): """ restore original cpufreqctl script """ # detect if running on a SNAP if os.getenv('PKG_MARKER') == "SNAP": pass else: # restore original cpufreqctl script if os.path.isfile("/usr/bin/cpufreqctl.auto-cpufreq.bak"): os.system("cp /usr/bin/cpufreqctl.auto-cpufreq.bak /usr/bin/cpufreqctl") os.remove("/usr/bin/cpufreqctl.auto-cpufreq.bak") # ToDo: implement mechanism to make sure cpufreqctl (auto-cpufreq) file is # restored if overwritten by system. But during tool removal to also remove it # in def cpufreqctl def footer(l=79): print("\n" + "-" * l + "\n") def deploy_complete_msg(): print("\n" + "-" * 17 + " auto-cpufreq daemon installed and running " + "-" * 17 + "\n") print("To view live log, run:\nauto-cpufreq --log") print("\nTo disable and remove auto-cpufreq daemon, run:\nsudo auto-cpufreq --remove") footer() def remove_complete_msg(): print("\n" + "-" * 25 + " auto-cpufreq daemon removed " + "-" * 25 + "\n") print("auto-cpufreq successfully removed.") footer() def deploy_daemon(): print("\n" + "-" * 21 + " Deploying auto-cpufreq as a daemon " + "-" * 22 + "\n") # deploy cpufreqctl script func call cpufreqctl() print("* Turn off bluetooth on boot") btconf = Path("/etc/bluetooth/main.conf") try: orig_set = "AutoEnable=true" change_set = "AutoEnable=false" with btconf.open(mode="r+") as f: content = f.read() f.seek(0) f.truncate() f.write(content.replace(orig_set, change_set)) except: print("\nERROR:\nWas unable to turn off bluetooth on boot") auto_cpufreq_log_file.touch(exist_ok=True) print("\n* Deploy auto-cpufreq install script") shutil.copy(SCRIPTS_DIR / "auto-cpufreq-install.sh", "/usr/bin/auto-cpufreq-install") print("\n* Deploy auto-cpufreq remove script") shutil.copy(SCRIPTS_DIR / "auto-cpufreq-remove.sh", "/usr/bin/auto-cpufreq-remove") call("/usr/bin/auto-cpufreq-install", shell=True) # remove auto-cpufreq daemon def remove(): print("\n" + "-" * 21 + " Removing auto-cpufreq daemon " + "-" * 22 + "\n") print("* Turn on bluetooth on boot") btconf = "/etc/bluetooth/main.conf" try: orig_set = "AutoEnable=true" change_set = "AutoEnable=false" with open(btconf, "r+") as f: content = f.read() f.seek(0) f.truncate() f.write(content.replace(change_set, orig_set)) except: print("\nERROR:\nWas unable to turn on bluetooth on boot") # run auto-cpufreq daemon install script call("/usr/bin/auto-cpufreq-remove", shell=True) # remove auto-cpufreq-remove os.remove("/usr/bin/auto-cpufreq-remove") # delete log file if auto_cpufreq_log_file.exists(): auto_cpufreq_log_file.unlink() # restore original cpufrectl script cpufreqctl_restore() def gov_check(): for gov in get_avail_gov(): if gov not in ALL_GOVERNORS: print("\n" + "-" * 18 + " Checking for necessary scaling governors " + "-" * 19 + "\n") sys.exit("ERROR:\n\nCouldn't find any of the necessary scaling governors.\n") # root check func def root_check(): if not os.geteuid() == 0: print("\n" + "-" * 33 + " Root check " + "-" * 34 + "\n") print("ERROR:\n\nMust be run root for this functionality to work, i.e: \nsudo auto-cpufreq") footer() exit(1) # refresh countdown def countdown(s): # Fix for wrong log output and "TERM environment variable not set" os.environ['TERM'] = 'xterm' for remaining in range(s, 0, -1): sys.stdout.write("\r") sys.stdout.write("\t\t\t\"auto-cpufreq\" refresh in:{:2d}".format(remaining)) sys.stdout.flush() time.sleep(1) # set powersave and enable turbo def set_powersave(): print(f"Setting to use: \"{get_avail_powersave()}\" governor") run(f"cpufreqctl --governor --set={get_avail_powersave()}", shell=True) if Path("/sys/devices/system/cpu/cpu0/cpufreq/energy_performance_preference").exists(): run("cpufreqctl --epp --set=balance_power", shell=True) print("Setting to use: \"balance_power\" EPP") # get system/CPU load load1m, _, _ = os.getloadavg() # get CPU utilization as a percentage cpuload = psutil.cpu_percent(interval=1) print("\nTotal CPU usage:", cpuload, "%") print("Total system load:", load1m, "\n") # conditions for setting turbo in powersave if load1m > (15*CPUS)/100: print("High load, setting turbo boost: on") turbo(True) elif psutil.cpu_percent(percpu=False, interval=0.01) >= 25.0 or isclose(max(psutil.cpu_percent(percpu=True, interval=0.01)), 100): print("High CPU load, setting turbo boost: on") turbo(True) else: print("Load optimal, setting turbo boost: off") turbo(False) footer() # make turbo suggestions in powersave def mon_powersave(): # get system/CPU load load1m, _, _ = os.getloadavg() # get CPU utilization as a percentage cpuload = psutil.cpu_percent(interval=1) print("\nTotal CPU usage:", cpuload, "%") print("Total system load:", load1m, "\n") if load1m > (15*CPUS)/100: print("High load, suggesting to set turbo boost: on") if turbo(): print("Currently turbo boost is: on") else: print("Currently turbo boost is: off") footer() elif psutil.cpu_percent(percpu=False, interval=0.01) >= 25.0 or isclose(max(psutil.cpu_percent(percpu=True, interval=0.01)), 100): print("High CPU load, suggesting to set turbo boost: on") if turbo(): print("Currently turbo boost is: on") else: print("Currently turbo boost is: off") footer() else: print("Load optimal, suggesting to set turbo boost: off") if turbo(): print("Currently turbo boost is: on") else: print("Currently turbo boost is: off") footer() # set performance and enable turbo def set_performance(): print(f"Setting to use: \"{get_avail_performance()}\" governor") run(f"cpufreqctl --governor --set={get_avail_performance()}", shell=True) if os.path.exists("/sys/devices/system/cpu/cpu0/cpufreq/energy_performance_preference"): run("cpufreqctl --epp --set=balance_performance", shell=True) print("Setting to use: \"balance_performance\" EPP") load1m, _, _ = os.getloadavg() cpuload = psutil.cpu_percent(interval=1) print("\nTotal CPU usage:", cpuload, "%") print("Total system load:", load1m, "\n") if load1m >= (10*CPUS)/100: print("High load, setting turbo boost: on") turbo(True) elif psutil.cpu_percent(percpu=False, interval=0.01) >= 15.0 or isclose(max(psutil.cpu_percent(percpu=True, interval=0.01)), 100): print("High CPU load, setting turbo boost: on") turbo(True) else: print("Load optimal, setting turbo boost: off") turbo(False) footer() # make turbo suggestions in performance def mon_performance(): # get system/CPU load load1m, _, _ = os.getloadavg() # get CPU utilization as a percentage cpuload = psutil.cpu_percent(interval=1) print("\nTotal CPU usage:", cpuload, "%") print("Total system load:", load1m, "\n") if turbo(): print("Currently turbo boost is: on") print("Suggesting to set turbo boost: on") else: print("Currently turbo boost is: off") print("Suggesting to set turbo boost: on") footer() def set_autofreq(): """ set cpufreq governor based if device is charging """ print("\n" + "-" * 28 + " CPU frequency scaling " + "-" * 28 + "\n") # determine which governor should be used if charging(): print("Battery is: charging") set_performance() else: print("Battery is: discharging") set_powersave() def mon_autofreq(): """ make cpufreq suggestions :return: """ print("\n" + "-" * 28 + " CPU frequency scaling " + "-" * 28 + "\n") # determine which governor should be used if charging(): print("Battery is: charging") print(f"Suggesting use of \"{get_avail_performance()}\" governor\nCurrently using:", get_current_gov()) mon_performance() else: print("Battery is: discharging") print(f"Suggesting use of \"{get_avail_powersave()}\" governor\nCurrently using:", get_current_gov()) mon_powersave() def python_info(): print("Python:", pl.python_version()) print("psutil package:", psutil.__version__) print("platform package:", pl.__version__) print("click package:", click.__version__) # workaround: Module 'distro' has no '__version__' member () (https://github.com/nir0s/distro/issues/265) #print("distro:", distro.__version__) run("echo \"distro package\" $(pip3 show distro | sed -n -e 's/^.*Version: //p')", shell=True) def distro_info(): dist = "UNKNOWN distro" version = "UNKNOWN version" # get distro information in snap env. if os.getenv("PKG_MARKER") == "SNAP": try: with open("/var/lib/snapd/hostfs/etc/os-release", "r") as searchfile: for line in searchfile: if line.startswith('NAME='): dist = line[5:line.find('$')].strip("\"") continue elif line.startswith('VERSION='): version = line[8:line.find('$')].strip("\"") continue except PermissionError: pass dist = f"{dist} {version}" else: # get distro information fdist = distro.linux_distribution() dist = " ".join(x for x in fdist) print("Linux distro: " + dist) print("Linux kernel: " + pl.release()) def sysinfo(): """ get system information """ # processor_info with open("/proc/cpuinfo", "r") as f: line = f.readline() while line: if "model name" in line: print("Processor:" + line.split(':')[1].rstrip()) break line = f.readline() # get cores count cpu_count = psutil.cpu_count() print("Cores:", cpu_count) # get architecture cpu_arch = pl.machine() print("Architecture:", cpu_arch) # get driver driver = getoutput("cpufreqctl --driver") print("Driver: " + driver) print("\n" + "-" * 30 + " Current CPU states " + "-" * 30 + "\n") print(f"CPU max frequency: {psutil.cpu_freq().max:.0f}MHz") print(f"CPU min frequency: {psutil.cpu_freq().min:.0f}MHz") core_usage = psutil.cpu_freq(percpu=True) print("\nCPU frequency for each core:\n") core_num = 0 while core_num < cpu_count: print(f"CPU{core_num}: {core_usage[core_num].current:.0f} MHz") core_num += 1 # get number of core temp sensors core_temp_num = psutil.cpu_count(logical=False) # get hardware temperatures core_temp = psutil.sensors_temperatures() print("\nTemperature for each physical core:\n") core_num = 0 while core_num < core_temp_num: temp = float("nan") try: if "coretemp" in core_temp: temp = core_temp['coretemp'][core_num].current elif "k10temp" in core_temp: # https://www.kernel.org/doc/Documentation/hwmon/k10temp temp = core_temp['k10temp'][0].current elif "acpitz" in core_temp: temp = core_temp['acpitz'][0].current except: pass print(f"CPU{core_num} temp: {temp:.0f}°C") core_num += 1 # print current fan speed | temporarily commented # current_fans = psutil.sensors_fans()['thinkpad'][0].current # print("\nCPU fan speed:", current_fans, "RPM") # read log func def read_log(): if os.getenv("PKG_MARKER") == "SNAP": call(["tail", "-n 50", "-f", str(auto_cpufreq_log_file_snap)]) elif os.path.isfile(auto_cpufreq_log_file): call(["tail", "-n 50", "-f", str(auto_cpufreq_log_file)]) else: print("\n" + "-" * 30 + " auto-cpufreq log " + "-" * 31 + "\n") print("ERROR: auto-cpufreq log is missing.\n\nMake sure to run: \"auto-cpufreq --install\" first") footer() # check if program (argument) is running def is_running(program, argument): # iterate over all process id's found by psutil for pid in psutil.pids(): try: # requests the process information corresponding to each process id proc = psutil.Process(pid) # check if value of program-variable that was used to call the function # matches the name field of the plutil.Process(pid) output if program in proc.name(): # check output of p.name(), output name of program # p.cmdline() - echo the exact command line via which p was called. for arg in proc.cmdline(): if argument in str(arg): return True except: continue # check if auto-cpufreq --daemon is running def running_daemon(): if is_running('auto-cpufreq', '--daemon'): deploy_complete_msg() exit(1) elif os.getenv("PKG_MARKER") == "SNAP" and dcheck == "enabled": deploy_complete_msg() exit(1)