lenovo_fix.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684
  1. #!/usr/bin/env python3
  2. from __future__ import print_function
  3. import argparse
  4. import configparser
  5. import dbus
  6. import glob
  7. import gzip
  8. import os
  9. import re
  10. import struct
  11. import subprocess
  12. import sys
  13. from collections import defaultdict
  14. from dbus.mainloop.glib import DBusGMainLoop
  15. from errno import EACCES, EPERM
  16. from gi.repository import GLib
  17. from mmio import MMIO, MMIOError
  18. from multiprocessing import cpu_count
  19. from platform import uname
  20. from threading import Event, Thread
  21. from time import time
  22. DEFAULT_SYSFS_POWER_PATH = '/sys/class/power_supply/AC*/online'
  23. VOLTAGE_PLANES = {'CORE': 0, 'GPU': 1, 'CACHE': 2, 'UNCORE': 3, 'ANALOGIO': 4}
  24. TRIP_TEMP_RANGE = [40, 97]
  25. UNDERVOLT_KEYS = ('UNDERVOLT', 'UNDERVOLT.AC', 'UNDERVOLT.BATTERY')
  26. power = {'source': None, 'method': 'polling'}
  27. platform_info_bits = {
  28. 'maximum_non_turbo_ratio': [8, 15],
  29. 'maximum_efficiency_ratio': [40, 47],
  30. 'minimum_operating_ratio': [48, 55],
  31. 'feature_ppin_cap': [23, 23],
  32. 'feature_programmable_turbo_ratio': [28, 28],
  33. 'feature_programmable_tdp_limit': [29, 29],
  34. 'number_of_additional_tdp_profiles': [33, 34],
  35. 'feature_programmable_temperature_target': [30, 30],
  36. 'feature_low_power_mode': [32, 32],
  37. }
  38. thermal_status_bits = {
  39. 'thermal_limit_status': [0, 0],
  40. 'thermal_limit_log': [1, 1],
  41. 'prochot_or_forcepr_status': [2, 2],
  42. 'prochot_or_forcepr_log': [3, 3],
  43. 'crit_temp_status': [4, 4],
  44. 'crit_temp_log': [5, 5],
  45. 'thermal_threshold1_status': [6, 6],
  46. 'thermal_threshold1_log': [7, 7],
  47. 'thermal_threshold2_status': [8, 8],
  48. 'thermal_threshold2_log': [9, 9],
  49. 'power_limit_status': [10, 10],
  50. 'power_limit_log': [11, 11],
  51. 'current_limit_status': [12, 12],
  52. 'current_limit_log': [13, 13],
  53. 'cross_domain_limit_status': [14, 14],
  54. 'cross_domain_limit_log': [15, 15],
  55. 'cpu_temp': [16, 22],
  56. 'temp_resolution': [27, 30],
  57. 'reading_valid': [31, 31],
  58. }
  59. supported_cpus = {
  60. 'Haswell': (0x3C, 0x3F, 0x45, 0x46),
  61. 'Broadwell': (0x3D, 0x47, 0x4F, 0x56),
  62. 'Skylake': (0x4E, 0x55),
  63. 'Skylake-S': (0x5E,),
  64. 'Ice Lake': (0x7E,),
  65. 'Kaby Lake (R)': (0x8E, 0x9E),
  66. 'Coffee Lake': (0x9E,),
  67. 'Cannon Lake': (0x66,),
  68. }
  69. class bcolors:
  70. YELLOW = '\033[93m'
  71. GREEN = '\033[92m'
  72. RED = '\033[91m'
  73. RESET = '\033[0m'
  74. BOLD = '\033[1m'
  75. OK = bcolors.GREEN + bcolors.BOLD + 'OK' + bcolors.RESET
  76. ERR = bcolors.RED + bcolors.BOLD + 'ERR' + bcolors.RESET
  77. LIM = bcolors.YELLOW + bcolors.BOLD + 'LIM' + bcolors.RESET
  78. def fatal(msg, code=1):
  79. print('[E] {:s}'.format(msg))
  80. sys.exit(code)
  81. def warning(msg):
  82. print('[W] {:s}'.format(msg))
  83. def writemsr(msr, val):
  84. msr_list = ['/dev/cpu/{:d}/msr'.format(x) for x in range(cpu_count())]
  85. if not os.path.exists(msr_list[0]):
  86. try:
  87. subprocess.check_call(('modprobe', 'msr'))
  88. except subprocess.CalledProcessError:
  89. fatal('Unable to load the msr module.')
  90. try:
  91. for addr in msr_list:
  92. f = os.open(addr, os.O_WRONLY)
  93. os.lseek(f, msr, os.SEEK_SET)
  94. os.write(f, struct.pack('Q', val))
  95. os.close(f)
  96. except (IOError, OSError) as e:
  97. if e.errno == EPERM or e.errno == EACCES:
  98. fatal(
  99. 'Unable to write to MSR. Try to disable Secure Boot '
  100. 'and check if your kernel does not restrict access to MSR.'
  101. )
  102. else:
  103. raise e
  104. # returns the value between from_bit and to_bit as unsigned long
  105. def readmsr(msr, from_bit=0, to_bit=63, cpu=None, flatten=False):
  106. assert cpu is None or cpu in range(cpu_count())
  107. if from_bit > to_bit:
  108. fatal('Wrong readmsr bit params')
  109. msr_list = ['/dev/cpu/{:d}/msr'.format(x) for x in range(cpu_count())]
  110. if not os.path.exists(msr_list[0]):
  111. try:
  112. subprocess.check_call(('modprobe', 'msr'))
  113. except subprocess.CalledProcessError:
  114. fatal('Unable to load the msr module.')
  115. try:
  116. output = []
  117. for addr in msr_list:
  118. f = os.open(addr, os.O_RDONLY)
  119. os.lseek(f, msr, os.SEEK_SET)
  120. val = struct.unpack('Q', os.read(f, 8))[0]
  121. os.close(f)
  122. output.append(get_value_for_bits(val, from_bit, to_bit))
  123. if flatten:
  124. return output[0] if len(set(output)) == 1 else output
  125. return output[cpu] if cpu is not None else output
  126. except (IOError, OSError) as e:
  127. if e.errno == EPERM or e.errno == EACCES:
  128. fatal('Unable to read from MSR. Try to disable Secure Boot.')
  129. else:
  130. raise e
  131. def cpu_usage_pct(exit_event, interval=1.0):
  132. last_idle = last_total = 0
  133. for i in range(2):
  134. with open('/proc/stat') as f:
  135. fields = [float(column) for column in f.readline().strip().split()[1:]]
  136. idle, total = fields[3], sum(fields)
  137. idle_delta, total_delta = idle - last_idle, total - last_total
  138. last_idle, last_total = idle, total
  139. if i == 0:
  140. exit_event.wait(interval)
  141. return 100.0 * (1.0 - idle_delta / total_delta)
  142. def get_value_for_bits(val, from_bit=0, to_bit=63):
  143. mask = sum(2 ** x for x in range(from_bit, to_bit + 1))
  144. return (val & mask) >> from_bit
  145. def is_on_battery(config):
  146. try:
  147. for path in glob.glob(config.get('GENERAL', 'Sysfs_Power_Path', fallback=DEFAULT_SYSFS_POWER_PATH)):
  148. with open(path) as f:
  149. return not bool(int(f.read()))
  150. except:
  151. pass
  152. fatal('[E] No valid Sysfs_Power_Path found!')
  153. def get_cpu_platform_info():
  154. features_msr_value = readmsr(0xCE, cpu=0)
  155. cpu_platform_info = {}
  156. for key, value in platform_info_bits.items():
  157. cpu_platform_info[key] = int(get_value_for_bits(features_msr_value, value[0], value[1]))
  158. return cpu_platform_info
  159. def get_reset_thermal_status():
  160. # read thermal status
  161. thermal_status_msr_value = readmsr(0x19C)
  162. thermal_status = []
  163. for core in range(cpu_count()):
  164. thermal_status_core = {}
  165. for key, value in thermal_status_bits.items():
  166. thermal_status_core[key] = int(get_value_for_bits(thermal_status_msr_value[core], value[0], value[1]))
  167. thermal_status.append(thermal_status_core)
  168. # reset log bits
  169. writemsr(0x19C, 0)
  170. return thermal_status
  171. def get_time_unit():
  172. # 0.000977 is the time unit of my CPU
  173. # TODO formula might be different for other CPUs
  174. return 1.0 / 2 ** readmsr(0x606, 16, 19, cpu=0)
  175. def get_power_unit():
  176. # 0.125 is the power unit of my CPU
  177. # TODO formula might be different for other CPUs
  178. return 1.0 / 2 ** readmsr(0x606, 0, 3, cpu=0)
  179. def get_critical_temp():
  180. # the critical temperature for my CPU is 100 'C
  181. return readmsr(0x1A2, 16, 23, cpu=0)
  182. def get_cur_pkg_power_limits():
  183. value = readmsr(0x610, 0, 55, flatten=True)
  184. return {
  185. 'PL1': get_value_for_bits(value, 0, 14),
  186. 'TW1': get_value_for_bits(value, 17, 23),
  187. 'PL2': get_value_for_bits(value, 32, 46),
  188. 'TW2': get_value_for_bits(value, 49, 55),
  189. }
  190. def calc_time_window_vars(t):
  191. time_unit = get_time_unit()
  192. for Y in range(2 ** 5):
  193. for Z in range(2 ** 2):
  194. if t <= (2 ** Y) * (1.0 + Z / 4.0) * time_unit:
  195. return (Y, Z)
  196. raise ValueError('Unable to find a good combination!')
  197. def calc_undervolt_msr(plane, offset):
  198. """Return the value to be written in the MSR 150h for setting the given
  199. offset voltage (in mV) to the given voltage plane.
  200. """
  201. assert offset <= 0
  202. assert plane in VOLTAGE_PLANES
  203. offset = int(round(offset * 1.024))
  204. offset = 0xFFE00000 & ((offset & 0xFFF) << 21)
  205. return 0x8000001100000000 | (VOLTAGE_PLANES[plane] << 40) | offset
  206. def calc_undervolt_mv(msr_value):
  207. """Return the offset voltage (in mV) from the given raw MSR 150h value.
  208. """
  209. offset = (msr_value & 0xFFE00000) >> 21
  210. offset = offset if offset <= 0x400 else -(0x800 - offset)
  211. return int(round(offset / 1.024))
  212. def undervolt(config):
  213. for plane in VOLTAGE_PLANES:
  214. write_offset_mv = config.getfloat(
  215. 'UNDERVOLT.{:s}'.format(power['source']), plane, fallback=config.getfloat('UNDERVOLT', plane, fallback=0.0)
  216. )
  217. write_value = calc_undervolt_msr(plane, write_offset_mv)
  218. writemsr(0x150, write_value)
  219. if args.debug:
  220. write_value &= 0xFFFFFFFF
  221. writemsr(0x150, 0x8000001000000000 | (VOLTAGE_PLANES[plane] << 40))
  222. read_value = readmsr(0x150, flatten=True)
  223. read_offset_mv = calc_undervolt_mv(read_value)
  224. match = OK if write_value == read_value else ERR
  225. print(
  226. '[D] Undervolt plane {:s} - write {:.0f} mV ({:#x}) - read {:.0f} mV ({:#x}) - match {}'.format(
  227. plane, write_offset_mv, write_value, read_offset_mv, read_value, match
  228. )
  229. )
  230. def load_config():
  231. config = configparser.ConfigParser()
  232. config.read(args.config)
  233. # config values sanity check
  234. for power_source in ('AC', 'BATTERY'):
  235. for option in ('Update_Rate_s', 'PL1_Tdp_W', 'PL1_Duration_s', 'PL2_Tdp_W', 'PL2_Duration_S'):
  236. value = config.getfloat(power_source, option, fallback=None)
  237. if value is not None:
  238. value = config.set(power_source, option, str(max(0.1, value)))
  239. elif option == 'Update_Rate_s':
  240. fatal('The mandatory "Update_Rate_s" parameter is missing.')
  241. trip_temp = config.getfloat(power_source, 'Trip_Temp_C', fallback=None)
  242. if trip_temp is not None:
  243. valid_trip_temp = min(TRIP_TEMP_RANGE[1], max(TRIP_TEMP_RANGE[0], trip_temp))
  244. if trip_temp != valid_trip_temp:
  245. config.set(power_source, 'Trip_Temp_C', str(valid_trip_temp))
  246. print(
  247. '[!] Overriding invalid "Trip_Temp_C" value in "{:s}": {:.1f} -> {:.1f}'.format(
  248. power_source, trip_temp, valid_trip_temp
  249. )
  250. )
  251. # fix any invalid value (ie. < 0) in the undervolt settings
  252. for key in UNDERVOLT_KEYS:
  253. for plane in VOLTAGE_PLANES:
  254. if key in config:
  255. value = config.getfloat(key, plane)
  256. valid_value = min(0, value)
  257. if value != valid_value:
  258. config.set(key, plane, str(valid_value))
  259. print(
  260. '[!] Overriding invalid "{:s}" value in "{:s}" voltage plane: {:.0f} -> {:.0f}'.format(
  261. key, plane, value, valid_value
  262. )
  263. )
  264. # handle the case where only one of UNDERVOLT.AC, UNDERVOLT.BATTERY keys exists
  265. # by forcing the other key to all zeros (ie. no undervolt)
  266. if any(key in config for key in UNDERVOLT_KEYS[1:]):
  267. for key in UNDERVOLT_KEYS[1:]:
  268. if key not in config:
  269. config.add_section(key)
  270. for plane in VOLTAGE_PLANES:
  271. value = config.getfloat(key, plane, fallback=0.0)
  272. config.set(key, plane, str(value))
  273. return config
  274. def calc_reg_values(platform_info, config):
  275. regs = defaultdict(dict)
  276. for power_source in ('AC', 'BATTERY'):
  277. if platform_info['feature_programmable_temperature_target'] != 1:
  278. warning("Setting temperature target is not supported by this CPU")
  279. else:
  280. # the critical temperature for my CPU is 100 'C
  281. critical_temp = get_critical_temp()
  282. # update the allowed temp range to keep at least 3 'C from the CPU critical temperature
  283. global TRIP_TEMP_RANGE
  284. TRIP_TEMP_RANGE[1] = min(TRIP_TEMP_RANGE[1], critical_temp - 3)
  285. Trip_Temp_C = config.getfloat(power_source, 'Trip_Temp_C', fallback=None)
  286. if Trip_Temp_C is not None:
  287. trip_offset = int(round(critical_temp - Trip_Temp_C))
  288. regs[power_source]['MSR_TEMPERATURE_TARGET'] = trip_offset << 24
  289. else:
  290. print('[I] {:s} trip temperature is disabled in config.'.format(power_source))
  291. power_unit = get_power_unit()
  292. PL1_Tdp_W = config.getfloat(power_source, 'PL1_Tdp_W', fallback=None)
  293. PL1_Duration_s = config.getfloat(power_source, 'PL1_Duration_s', fallback=None)
  294. PL2_Tdp_W = config.getfloat(power_source, 'PL2_Tdp_W', fallback=None)
  295. PL2_Duration_s = config.getfloat(power_source, 'PL2_Duration_s', fallback=None)
  296. if (PL1_Tdp_W, PL1_Duration_s, PL2_Tdp_W, PL2_Duration_s).count(None) < 4:
  297. cur_pkg_power_limits = get_cur_pkg_power_limits()
  298. if PL1_Tdp_W is None:
  299. PL1 = cur_pkg_power_limits['PL1']
  300. print('[I] {:s} PL1_Tdp_W disabled in config.'.format(power_source))
  301. else:
  302. PL1 = int(round(PL1_Tdp_W / power_unit))
  303. if PL1_Duration_s is None:
  304. TW1 = cur_pkg_power_limits['TW1']
  305. print('[I] {:s} PL1_Duration_s disabled in config.'.format(power_source))
  306. else:
  307. Y, Z = calc_time_window_vars(PL1_Duration_s)
  308. TW1 = Y | (Z << 5)
  309. if PL2_Tdp_W is None:
  310. PL2 = cur_pkg_power_limits['PL2']
  311. print('[I] {:s} PL2_Tdp_W disabled in config.'.format(power_source))
  312. else:
  313. PL2 = int(round(PL2_Tdp_W / power_unit))
  314. if PL2_Duration_s is None:
  315. TW2 = cur_pkg_power_limits['TW2']
  316. print('[I] {:s} PL2_Duration_s disabled in config.'.format(power_source))
  317. else:
  318. Y, Z = calc_time_window_vars(PL2_Duration_s)
  319. TW2 = Y | (Z << 5)
  320. regs[power_source]['MSR_PKG_POWER_LIMIT'] = (
  321. PL1 | (1 << 15) | (TW1 << 17) | (PL2 << 32) | (1 << 47) | (TW2 << 49)
  322. )
  323. else:
  324. print('[I] {:s} package power limits are disabled in config.'.format(power_source))
  325. # cTDP
  326. c_tdp_target_value = config.getint(power_source, 'cTDP', fallback=None)
  327. if c_tdp_target_value is not None:
  328. if platform_info['feature_programmable_tdp_limit'] != 1:
  329. print("[W] cTDP setting not supported by this CPU")
  330. elif platform_info['number_of_additional_tdp_profiles'] < c_tdp_target_value:
  331. print("[W] the configured cTDP profile is not supported by this CPU")
  332. else:
  333. valid_c_tdp_target_value = max(0, c_tdp_target_value)
  334. regs[power_source]['MSR_CONFIG_TDP_CONTROL'] = valid_c_tdp_target_value
  335. return regs
  336. def set_hwp(pref):
  337. # set HWP energy performance hints
  338. assert pref in ('performance', 'balance_performance', 'default', 'balance_power', 'power')
  339. CPUs = [
  340. '/sys/devices/system/cpu/cpu{:d}/cpufreq/energy_performance_preference'.format(x) for x in range(cpu_count())
  341. ]
  342. for i, c in enumerate(CPUs):
  343. with open(c, 'wb') as f:
  344. f.write(pref.encode())
  345. if args.debug:
  346. with open(c) as f:
  347. read_value = f.read().strip()
  348. match = OK if pref == read_value else ERR
  349. print('[D] HWP for cpu{:d} - write "{:s}" - read "{:s}" - match {}'.format(i, pref, read_value, match))
  350. def power_thread(config, regs, exit_event):
  351. try:
  352. mchbar_mmio = MMIO(0xFED159A0, 8)
  353. except MMIOError:
  354. fatal('Unable to open /dev/mem. Try to disable Secure Boot.')
  355. while not exit_event.is_set():
  356. # print thermal status
  357. if args.debug:
  358. thermal_status = get_reset_thermal_status()
  359. for index, core_thermal_status in enumerate(thermal_status):
  360. for key, value in core_thermal_status.items():
  361. print('[D] core {} thermal status: {} = {}'.format(index, key.replace("_", " "), value))
  362. # switch back to sysfs polling
  363. if power['method'] == 'polling':
  364. power['source'] = 'BATTERY' if is_on_battery(config) else 'AC'
  365. # set temperature trip point
  366. if 'MSR_TEMPERATURE_TARGET' in regs[power['source']]:
  367. write_value = regs[power['source']]['MSR_TEMPERATURE_TARGET']
  368. writemsr(0x1A2, write_value)
  369. if args.debug:
  370. read_value = readmsr(0x1A2, 24, 29, flatten=True)
  371. match = OK if write_value >> 24 == read_value else ERR
  372. print(
  373. '[D] TEMPERATURE_TARGET - write {:#x} - read {:#x} - match {}'.format(
  374. write_value >> 24, read_value, match
  375. )
  376. )
  377. # set cTDP
  378. if 'MSR_CONFIG_TDP_CONTROL' in regs[power['source']]:
  379. write_value = regs[power['source']]['MSR_CONFIG_TDP_CONTROL']
  380. writemsr(0x64B, write_value)
  381. if args.debug:
  382. read_value = readmsr(0x64B, 0, 1, flatten=True)
  383. match = OK if write_value == read_value else ERR
  384. print(
  385. '[D] CONFIG_TDP_CONTROL - write {:#x} - read {:#x} - match {}'.format(
  386. write_value, read_value, match
  387. )
  388. )
  389. # set PL1/2 on MSR
  390. write_value = regs[power['source']]['MSR_PKG_POWER_LIMIT']
  391. writemsr(0x610, write_value)
  392. if args.debug:
  393. read_value = readmsr(0x610, 0, 55, flatten=True)
  394. match = OK if write_value == read_value else ERR
  395. print(
  396. '[D] MSR PACKAGE_POWER_LIMIT - write {:#x} - read {:#x} - match {}'.format(
  397. write_value, read_value, match
  398. )
  399. )
  400. # set MCHBAR register to the same PL1/2 values
  401. mchbar_mmio.write32(0, write_value & 0xFFFFFFFF)
  402. mchbar_mmio.write32(4, write_value >> 32)
  403. if args.debug:
  404. read_value = mchbar_mmio.read32(0) | (mchbar_mmio.read32(4) << 32)
  405. match = OK if write_value == read_value else ERR
  406. print(
  407. '[D] MCHBAR PACKAGE_POWER_LIMIT - write {:#x} - read {:#x} - match {}'.format(
  408. write_value, read_value, match
  409. )
  410. )
  411. wait_t = config.getfloat(power['source'], 'Update_Rate_s')
  412. enable_hwp_mode = config.getboolean('AC', 'HWP_Mode', fallback=False)
  413. if power['source'] == 'AC' and enable_hwp_mode:
  414. cpu_usage = cpu_usage_pct(exit_event, interval=wait_t)
  415. # set full performance mode only when load is greater than this threshold (~ at least 1 core full speed)
  416. performance_mode = cpu_usage > 100.0 / (cpu_count() * 1.25)
  417. # check again if we are on AC, since in the meantime we might have switched to BATTERY
  418. if not is_on_battery(config):
  419. set_hwp('performance' if performance_mode else 'balance_performance')
  420. else:
  421. exit_event.wait(wait_t)
  422. def check_kernel():
  423. if os.geteuid() != 0:
  424. fatal('No root no party. Try again with sudo.')
  425. kernel_config = None
  426. try:
  427. with open(os.path.join('/boot', 'config-{:s}'.format(uname()[2]))) as f:
  428. kernel_config = f.read()
  429. except IOError:
  430. config_gz_path = os.path.join('/proc', 'config.gz')
  431. try:
  432. if not os.path.isfile(config_gz_path):
  433. subprocess.check_call(('modprobe', 'configs'))
  434. with gzip.open(config_gz_path) as f:
  435. kernel_config = f.read().decode()
  436. except (subprocess.CalledProcessError, IOError):
  437. pass
  438. if kernel_config is None:
  439. print('[W] Unable to obtain and validate kernel config.')
  440. elif not re.search('CONFIG_DEVMEM=y', kernel_config):
  441. fatal('Bad kernel config: you need CONFIG_DEVMEM=y.')
  442. elif not re.search('CONFIG_X86_MSR=(y|m)', kernel_config):
  443. fatal('Bad kernel config: you need CONFIG_X86_MSR builtin or as module.')
  444. def check_cpu():
  445. try:
  446. with open('/proc/cpuinfo') as f:
  447. cpuinfo = {}
  448. for row in f.readlines():
  449. try:
  450. key, value = map(lambda x: x.strip(), row.split(':'))
  451. if key == 'processor' and value == '1':
  452. break
  453. try:
  454. cpuinfo[key] = int(value, 0)
  455. except ValueError:
  456. cpuinfo[key] = value
  457. except ValueError:
  458. pass
  459. if cpuinfo['vendor_id'] != 'GenuineIntel':
  460. fatal('This tool is designed for Intel CPUs only.')
  461. cpu_model = None
  462. for model in supported_cpus:
  463. if cpuinfo['model'] in supported_cpus[model]:
  464. cpu_model = model
  465. break
  466. if cpuinfo['cpu family'] != 6 or cpu_model is None:
  467. fatal('Your CPU model is not supported.')
  468. print('[I] Detected CPU architecture: Intel {:s}'.format(cpu_model))
  469. except:
  470. fatal('Unable to identify CPU model.')
  471. def monitor(exit_event, wait):
  472. IA32_THERM_STATUS = 0x19C
  473. IA32_PERF_STATUS = 0x198
  474. MSR_RAPL_POWER_UNIT = 0x606
  475. MSR_INTEL_PKG_ENERGY_STATUS = 0x611
  476. MSR_PP1_ENERGY_STATUS = 0x641
  477. MSR_DRAM_ENERGY_STATUS = 0x619
  478. wait = max(0.1, wait)
  479. rapl_power_unit = 0.5 ** readmsr(MSR_RAPL_POWER_UNIT, from_bit=8, to_bit=12, cpu=0)
  480. power_plane_msr = {
  481. 'Package': MSR_INTEL_PKG_ENERGY_STATUS,
  482. 'Graphics': MSR_PP1_ENERGY_STATUS,
  483. 'DRAM': MSR_DRAM_ENERGY_STATUS,
  484. }
  485. prev_energy = {
  486. 'Package': (readmsr(MSR_INTEL_PKG_ENERGY_STATUS, cpu=0) * rapl_power_unit, time()),
  487. 'Graphics': (readmsr(MSR_PP1_ENERGY_STATUS, cpu=0) * rapl_power_unit, time()),
  488. 'DRAM': (readmsr(MSR_DRAM_ENERGY_STATUS, cpu=0) * rapl_power_unit, time()),
  489. }
  490. print('Realtime monitoring of throttling causes:\n')
  491. while not exit_event.is_set():
  492. value = readmsr(IA32_THERM_STATUS, from_bit=0, to_bit=15, cpu=0)
  493. offsets = {'Thermal': 0, 'Power': 10, 'Current': 12, 'Cross-comain (e.g. GPU)': 14}
  494. output = ('{:s}: {:s}'.format(cause, LIM if bool((value >> offsets[cause]) & 1) else OK) for cause in offsets)
  495. # ugly code, just testing...
  496. vcore = readmsr(IA32_PERF_STATUS, from_bit=32, to_bit=47, cpu=0) / (2.0 ** 13) * 1000
  497. stats2 = {'VCore': '{:.0f} mV'.format(vcore)}
  498. for power_plane in ('Package', 'Graphics', 'DRAM'):
  499. energy_j = readmsr(power_plane_msr[power_plane], cpu=0) * rapl_power_unit
  500. now = time()
  501. prev_energy[power_plane], energy_w = (
  502. (energy_j, now),
  503. (energy_j - prev_energy[power_plane][0]) / (now - prev_energy[power_plane][1]),
  504. )
  505. stats2[power_plane] = '{:.1f} W'.format(energy_w)
  506. output2 = ('{:s}: {:s}'.format(label, stats2[label]) for label in stats2)
  507. print('[{}] {} || {}{}'.format(power['source'], ' - '.join(output), ' - '.join(output2), ' ' * 10), end='\r')
  508. exit_event.wait(wait)
  509. def main():
  510. global args
  511. parser = argparse.ArgumentParser()
  512. exclusive_group = parser.add_mutually_exclusive_group()
  513. exclusive_group.add_argument('--debug', action='store_true', help='add some debug info and additional checks')
  514. exclusive_group.add_argument(
  515. '--monitor',
  516. metavar='update_rate',
  517. const=1.0,
  518. type=float,
  519. nargs='?',
  520. help='realtime monitoring of throttling causes (default 1s)',
  521. )
  522. parser.add_argument('--config', default='/etc/lenovo_fix.conf', help='override default config file path')
  523. parser.add_argument('--force', action='store_true', help='bypass compatibility checks (EXPERTS only)')
  524. args = parser.parse_args()
  525. if not args.force:
  526. check_kernel()
  527. check_cpu()
  528. config = load_config()
  529. power['source'] = 'BATTERY' if is_on_battery(config) else 'AC'
  530. platform_info = get_cpu_platform_info()
  531. if args.debug:
  532. for key, value in platform_info.items():
  533. print('[D] cpu platform info: {} = {}'.format(key.replace("_", " "), value))
  534. regs = calc_reg_values(platform_info, config)
  535. if not config.getboolean('GENERAL', 'Enabled'):
  536. return
  537. exit_event = Event()
  538. thread = Thread(target=power_thread, args=(config, regs, exit_event))
  539. thread.daemon = True
  540. thread.start()
  541. undervolt(config)
  542. # handle dbus events for applying undervolt on resume from sleep/hybernate
  543. def handle_sleep_callback(sleeping):
  544. if not sleeping:
  545. undervolt(config)
  546. def handle_ac_callback(*args):
  547. try:
  548. power['source'] = 'BATTERY' if args[1]['Online'] == 0 else 'AC'
  549. power['method'] = 'dbus'
  550. except:
  551. power['method'] = 'polling'
  552. DBusGMainLoop(set_as_default=True)
  553. bus = dbus.SystemBus()
  554. # add dbus receiver only if undervolt is enabled in config
  555. if any(config.getfloat(key, plane, fallback=0) != 0 for plane in VOLTAGE_PLANES for key in UNDERVOLT_KEYS):
  556. bus.add_signal_receiver(
  557. handle_sleep_callback, 'PrepareForSleep', 'org.freedesktop.login1.Manager', 'org.freedesktop.login1'
  558. )
  559. bus.add_signal_receiver(
  560. handle_ac_callback,
  561. signal_name="PropertiesChanged",
  562. dbus_interface="org.freedesktop.DBus.Properties",
  563. path="/org/freedesktop/UPower/devices/line_power_AC",
  564. )
  565. if args.monitor is not None:
  566. monitor_thread = Thread(target=monitor, args=(exit_event, args.monitor))
  567. monitor_thread.daemon = True
  568. monitor_thread.start()
  569. try:
  570. loop = GLib.MainLoop()
  571. loop.run()
  572. except (KeyboardInterrupt, SystemExit):
  573. pass
  574. exit_event.set()
  575. loop.quit()
  576. thread.join(timeout=1)
  577. if args.monitor is not None:
  578. monitor_thread.join(timeout=0.1)
  579. if __name__ == '__main__':
  580. main()