mirror of
https://gitlab.com/asus-linux/asusctl.git
synced 2026-01-22 17:33:19 +01:00
137 lines
3.6 KiB
Rust
137 lines
3.6 KiB
Rust
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<Self> {
|
|
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"),
|
|
}
|
|
}
|
|
}
|