-
Notifications
You must be signed in to change notification settings - Fork 30
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
42a4a91
commit 16a27d4
Showing
4 changed files
with
4,097 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,103 @@ | ||
import cpuinfo | ||
from carbontracker.components.handler import Handler | ||
from typing import List, Optional | ||
import csv | ||
import os | ||
from carbontracker.loggerutil import Logger | ||
import statistics | ||
|
||
logger = Logger() | ||
|
||
class GenericCPU(Handler): | ||
def __init__(self, pids: List[int], devices_by_pid: bool): | ||
super().__init__(pids, devices_by_pid) | ||
self.cpu_brand = self.get_cpu_brand() | ||
self.tdp = None | ||
self.cpu_power_data = self.load_cpu_power_data() | ||
self.average_tdp = self.calculate_average_tdp() | ||
|
||
def get_cpu_brand(self) -> str: | ||
try: | ||
info = cpuinfo.get_cpu_info() | ||
cpu_brand = info.get('brand_raw', '') | ||
logger.info(f"Detected CPU: {cpu_brand}") | ||
return cpu_brand | ||
except Exception as e: | ||
logger.err_warn(f"Failed to get CPU info: {e}") | ||
return '' | ||
|
||
def load_cpu_power_data(self): | ||
csv_path = os.path.join(os.path.dirname(__file__), '..', '..', 'data', 'cpu_power.csv') | ||
cpu_power_data = {} | ||
try: | ||
with open(csv_path, 'r') as f: | ||
reader = csv.DictReader(f) | ||
for row in reader: | ||
try: | ||
tdp_value = row['TDP'] | ||
if '.' in tdp_value: | ||
# Handle range case | ||
tdp_parts = tdp_value.split('.') | ||
if len(tdp_parts) == 3: # e.g., "33.34.8" | ||
lower = float(f"{tdp_parts[0]}.{tdp_parts[1]}") | ||
upper = float(f"{tdp_parts[0]}.{tdp_parts[2]}") | ||
tdp = (lower + upper) / 2 | ||
else: | ||
tdp = float(tdp_value) | ||
else: | ||
tdp = float(tdp_value) | ||
cpu_power_data[row['Name']] = tdp | ||
except ValueError: | ||
logger.err_warn(f"Invalid TDP value for CPU {row['Name']}: {row['TDP']}") | ||
return cpu_power_data | ||
except Exception as e: | ||
logger.err_warn(f"Failed to load CPU power data: {e}") | ||
return {} | ||
|
||
def calculate_average_tdp(self) -> float: | ||
if not self.cpu_power_data: | ||
return 0.0 | ||
return statistics.mean(self.cpu_power_data.values()) / 2 # 50% utilization | ||
|
||
def init(self): | ||
logger.info("[setup] CPU Tracking...") | ||
if not self.cpu_brand: | ||
logger.err_warn("Failed to detect CPU. Falling back to generic CPU handler.") | ||
self.cpu_brand = "Unknown CPU" | ||
else: | ||
logger.info(f"CPU Model: {self.cpu_brand}") | ||
|
||
self.tdp = self.find_matching_tdp() | ||
|
||
if self.tdp is None: | ||
self.tdp = self.average_tdp | ||
logger.err_warn(f"No matching TDP found for CPU: {self.cpu_brand}. Using average TDP of {self.tdp:.2f}W as fallback.") | ||
else: | ||
self.tdp = self.tdp / 2 # 50% utilization | ||
logger.info(f"Using TDP of {self.tdp:.2f}W for {self.cpu_brand}") | ||
|
||
def find_matching_tdp(self) -> Optional[float]: | ||
# Try direct match | ||
if self.cpu_brand in self.cpu_power_data: | ||
return self.cpu_power_data[self.cpu_brand] | ||
|
||
# Try matching without frequency | ||
cpu_name_without_freq = self.cpu_brand.split('@')[0].strip() | ||
for cpu_name, tdp in self.cpu_power_data.items(): | ||
if cpu_name_without_freq in cpu_name: | ||
logger.info(f"Matched CPU {self.cpu_brand} to {cpu_name} with TDP {tdp}W") | ||
return tdp | ||
|
||
return None | ||
|
||
def devices(self) -> List[str]: | ||
return [self.cpu_brand] | ||
|
||
def available(self) -> bool: | ||
return True | ||
|
||
def power_usage(self) -> List[float]: | ||
return [self.tdp] # Already adjusted for 50% utilization | ||
|
||
def shutdown(self): | ||
pass |
Oops, something went wrong.