use std::{fmt::Display, path::PathBuf, str::FromStr}; use log::{info, warn}; use serde::{Deserialize, Serialize}; use zvariant::Type; use crate::{ attr_bool, attr_u8, error::{PlatformError, Result}, to_device, }; /// The "platform" device provides access to things like: /// - dgpu_disable /// - egpu_enable /// - panel_od /// - gpu_mux /// - keyboard_mode, set keyboard RGB mode and speed /// - keyboard_state, set keyboard power states #[derive(Debug, PartialEq, Eq, PartialOrd, Clone)] pub struct AsusPlatform { path: PathBuf, pp_path: PathBuf, } impl AsusPlatform { pub fn new() -> Result { let mut enumerator = udev::Enumerator::new().map_err(|err| { warn!("{}", err); PlatformError::Udev("enumerator failed".into(), err) })?; enumerator.match_subsystem("platform").map_err(|err| { warn!("{}", err); PlatformError::Udev("match_subsystem failed".into(), err) })?; enumerator.match_sysname("asus-nb-wmi").map_err(|err| { warn!("{}", err); PlatformError::Udev("match_subsystem failed".into(), err) })?; if let Some(device) = (enumerator.scan_devices().map_err(|err| { warn!("{}", err); PlatformError::Udev("scan_devices failed".into(), err) })?) .next() { info!("Found platform support at {:?}", device.sysname()); return Ok(Self { path: device.syspath().to_owned(), pp_path: PathBuf::from_str("/sys/firmware/acpi").unwrap(), }); } Err(PlatformError::MissingFunction( "asus-nb-wmi not found".into(), )) } attr_bool!("dgpu_disable", path); attr_bool!("egpu_enable", path); attr_bool!("panel_od", path); attr_bool!("gpu_mux_mode", path); // This is technically the same as `platform_profile` since both are tied in-kernel attr_u8!("throttle_thermal_policy", path); // The acpi platform_profile support attr_u8!("platform_profile", pp_path); } #[derive(Serialize, Deserialize, Type, Debug, PartialEq, Eq, Clone, Copy)] pub enum GpuMode { Discrete, Optimus, Integrated, Egpu, Error, NotSupported, } impl GpuMode { /// For writing to `gpu_mux_mode` attribute pub fn to_mux_attr(&self) -> bool { if *self == Self::Discrete { return false; } true } pub fn to_dgpu_attr(&self) -> u8 { if *self == Self::Integrated { return 1; } 0 } pub fn to_egpu_attr(&self) -> u8 { if *self == Self::Egpu { return 1; } 0 } pub fn from_mux(num: u8) -> Self { if num == 0 { return Self::Discrete; } Self::Optimus } pub fn from_dgpu(num: u8) -> Self { if num == 1 { return Self::Integrated; } Self::Optimus } // `from_dgpu()` should be called also, and should take precedence if result // are not equal. pub fn from_egpu(num: u8) -> Self { if num == 1 { return Self::Egpu; } Self::Optimus } } impl Display for GpuMode { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { GpuMode::Discrete => write!(f, "Discrete"), GpuMode::Optimus => write!(f, "Optimus"), GpuMode::Integrated => write!(f, "Integrated"), GpuMode::Egpu => write!(f, "eGPU"), GpuMode::Error => write!(f, "Error"), GpuMode::NotSupported => write!(f, "Not Supported"), } } }