lenovo_fix.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. #!/usr/bin/env python3
  2. import configparser
  3. import dbus
  4. import glob
  5. import os
  6. import psutil
  7. import struct
  8. import subprocess
  9. from collections import defaultdict
  10. from dbus.mainloop.glib import DBusGMainLoop
  11. from mmio import MMIO
  12. from multiprocessing import cpu_count
  13. from threading import Event, Thread
  14. try:
  15. from gi.repository import GObject
  16. except ImportError:
  17. import gobject as GObject
  18. SYSFS_POWER_PATH = '/sys/class/power_supply/AC/online'
  19. CONFIG_PATH = '/etc/lenovo_fix.conf'
  20. VOLTAGE_PLANES = {
  21. 'CORE': 0,
  22. 'GPU': 1,
  23. 'CACHE': 2,
  24. 'UNCORE': 3,
  25. 'ANALOGIO': 4,
  26. }
  27. TRIP_TEMP_RANGE = (40, 97)
  28. power = {'source': None, 'method': 'polling'}
  29. def writemsr(msr, val):
  30. n = glob.glob('/dev/cpu/[0-9]*/msr')
  31. for c in n:
  32. f = os.open(c, os.O_WRONLY)
  33. os.lseek(f, msr, os.SEEK_SET)
  34. os.write(f, struct.pack('Q', val))
  35. os.close(f)
  36. if not n:
  37. try:
  38. subprocess.check_call(('modprobe', 'msr'))
  39. except subprocess.CalledProcessError:
  40. raise OSError("Unable to load msr module.")
  41. def is_on_battery():
  42. with open(SYSFS_POWER_PATH) as f:
  43. return not bool(int(f.read()))
  44. def calc_time_window_vars(t):
  45. for Y in range(2**5):
  46. for Z in range(2**2):
  47. if t <= (2**Y) * (1. + Z / 4.) * 0.000977:
  48. return (Y, Z)
  49. raise Exception('Unable to find a good combination!')
  50. def undervolt(config):
  51. for plane in VOLTAGE_PLANES:
  52. writemsr(0x150, calc_undervolt_msr(plane, config.getfloat('UNDERVOLT', plane)))
  53. def calc_undervolt_msr(plane, offset):
  54. assert offset <= 0
  55. assert plane in VOLTAGE_PLANES
  56. offset = int(round(offset * 1.024))
  57. offset = 0xFFE00000 & ((offset & 0xFFF) << 21)
  58. return 0x8000001100000000 | (VOLTAGE_PLANES[plane] << 40) | offset
  59. def load_config():
  60. config = configparser.ConfigParser()
  61. config.read(CONFIG_PATH)
  62. # config values sanity check
  63. for power_source in ('AC', 'BATTERY'):
  64. for option in (
  65. 'Update_Rate_s',
  66. 'PL1_Tdp_W',
  67. 'PL1_Duration_s',
  68. 'PL2_Tdp_W',
  69. 'PL2_Duration_S',
  70. ):
  71. config.set(power_source, option, str(max(0.1, config.getfloat(power_source, option))))
  72. trip_temp = config.getfloat(power_source, 'Trip_Temp_C')
  73. valid_trip_temp = min(TRIP_TEMP_RANGE[1], max(TRIP_TEMP_RANGE[0], trip_temp))
  74. if trip_temp != valid_trip_temp:
  75. config.set(power_source, 'Trip_Temp_C', str(valid_trip_temp))
  76. print('[!] Overriding invalid "Trip_Temp_C" value in "{:s}": {:.1f} -> {:.1f}'.format(
  77. power_source, trip_temp, valid_trip_temp))
  78. for plane in VOLTAGE_PLANES:
  79. value = config.getfloat('UNDERVOLT', plane)
  80. valid_value = min(0, value)
  81. if value != valid_value:
  82. config.set('UNDERVOLT', plane, str(valid_value))
  83. print('[!] Overriding invalid "UNDERVOLT" value in "{:s}" voltage plane: {:.0f} -> {:.0f}'.format(
  84. plane, value, valid_value))
  85. return config
  86. def calc_reg_values(config):
  87. regs = defaultdict(dict)
  88. for power_source in ('AC', 'BATTERY'):
  89. # the critical temperature for this CPU is 100 C
  90. trip_offset = int(round(100 - config.getfloat(power_source, 'Trip_Temp_C')))
  91. regs[power_source]['MSR_TEMPERATURE_TARGET'] = trip_offset << 24
  92. # 0.125 is the power unit of this CPU
  93. PL1 = int(round(config.getfloat(power_source, 'PL1_Tdp_W') / 0.125))
  94. Y, Z = calc_time_window_vars(config.getfloat(power_source, 'PL1_Duration_s'))
  95. TW1 = Y | (Z << 5)
  96. PL2 = int(round(config.getfloat(power_source, 'PL2_Tdp_W') / 0.125))
  97. Y, Z = calc_time_window_vars(config.getfloat(power_source, 'PL2_Duration_s'))
  98. TW2 = Y | (Z << 5)
  99. regs[power_source]['MSR_PKG_POWER_LIMIT'] = PL1 | (1 << 15) | (TW1 << 17) | (PL2 << 32) | (1 << 47) | (
  100. TW2 << 49)
  101. return regs
  102. def set_hwp(pref):
  103. # set HWP energy performance hints
  104. assert pref in ('performance', 'balance_performance', 'default', 'balance_power', 'power')
  105. n = glob.glob('/sys/devices/system/cpu/cpu[0-9]*/cpufreq/energy_performance_preference')
  106. for c in n:
  107. with open(c, 'wb') as f:
  108. f.write(pref.encode())
  109. def power_thread(config, regs, exit_event):
  110. mchbar_mmio = MMIO(0xfed159a0, 8)
  111. while not exit_event.is_set():
  112. #
  113. if power['method'] == 'polling':
  114. power['source'] = 'BATTERY' if is_on_battery() else 'AC'
  115. # set temperature trip point
  116. writemsr(0x1a2, regs[power['source']]['MSR_TEMPERATURE_TARGET'])
  117. # set PL1/2 on MSR
  118. writemsr(0x610, regs[power['source']]['MSR_PKG_POWER_LIMIT'])
  119. # set MCHBAR register to the same PL1/2 values
  120. mchbar_mmio.write32(0, regs[power['source']]['MSR_PKG_POWER_LIMIT'] & 0xffffffff)
  121. mchbar_mmio.write32(4, regs[power['source']]['MSR_PKG_POWER_LIMIT'] >> 32)
  122. wait_t = config.getfloat(power['source'], 'Update_Rate_s')
  123. enable_hwp_mode = config.getboolean('AC', 'HWP_Mode', fallback=False)
  124. if power['source'] == 'AC' and enable_hwp_mode:
  125. cpu_usage = float(psutil.cpu_percent(interval=wait_t))
  126. # set full performance mode only when load is greater than this threshold (~ at least 1 core full speed)
  127. performance_mode = cpu_usage > 100. / (cpu_count() * 1.25)
  128. # check again if we are on AC, since in the meantime we might have switched to BATTERY
  129. if not is_on_battery():
  130. set_hwp('performance' if performance_mode else 'balance_performance')
  131. else:
  132. exit_event.wait(wait_t)
  133. def main():
  134. power['source'] = 'BATTERY' if is_on_battery() else 'AC'
  135. config = load_config()
  136. regs = calc_reg_values(config)
  137. if not config.getboolean('GENERAL', 'Enabled'):
  138. return
  139. exit_event = Event()
  140. t = Thread(target=power_thread, args=(config, regs, exit_event))
  141. t.daemon = True
  142. t.start()
  143. undervolt(config)
  144. # handle dbus events for applying undervolt on resume from sleep/hybernate
  145. def handle_sleep_callback(sleeping):
  146. if not sleeping:
  147. undervolt(config)
  148. def handle_ac_callback(*args):
  149. try:
  150. power['source'] = 'BATTERY' if args[1]['Online'] == 0 else 'AC'
  151. power['method'] = 'dbus'
  152. except:
  153. power['method'] = 'polling'
  154. DBusGMainLoop(set_as_default=True)
  155. bus = dbus.SystemBus()
  156. # add dbus receiver only if undervolt is enabled in config
  157. if any(config.getfloat('UNDERVOLT', plane) != 0 for plane in VOLTAGE_PLANES):
  158. bus.add_signal_receiver(handle_sleep_callback, 'PrepareForSleep', 'org.freedesktop.login1.Manager',
  159. 'org.freedesktop.login1')
  160. bus.add_signal_receiver(
  161. handle_ac_callback,
  162. signal_name="PropertiesChanged",
  163. dbus_interface="org.freedesktop.DBus.Properties",
  164. path="/org/freedesktop/UPower/devices/line_power_AC")
  165. try:
  166. GObject.threads_init()
  167. loop = GObject.MainLoop()
  168. loop.run()
  169. except (KeyboardInterrupt, SystemExit):
  170. pass
  171. exit_event.set()
  172. loop.quit()
  173. t.join(timeout=1)
  174. if __name__ == '__main__':
  175. main()