mirror of
https://gitlab.com/asus-linux/asusctl.git
synced 2026-02-06 00:15:04 +01:00
- Create new rog-platform crate to manage all i/o in a universal way + kbd-led handling + platform handling (asus-nb-wmi) + hidraw + usbraw - Refactor how ROGCC handles IPC for background open, run-in-bg
599 lines
20 KiB
Rust
599 lines
20 KiB
Rust
use crate::{
|
|
error::RogError,
|
|
laptops::{LaptopLedData, ASUS_KEYBOARD_DEVICES},
|
|
CtrlTask,
|
|
};
|
|
use async_trait::async_trait;
|
|
use log::{error, info, warn};
|
|
use logind_zbus::manager::ManagerProxy;
|
|
use rog_aura::{
|
|
usb::{AuraDevice, LED_APPLY, LED_SET},
|
|
AuraEffect, LedBrightness, LED_MSG_LEN,
|
|
};
|
|
use rog_aura::{AuraZone, Direction, Speed, GRADIENT};
|
|
use rog_platform::{hid_raw::HidRaw, keyboard_led::KeyboardLed};
|
|
use rog_supported::LedSupportedFunctions;
|
|
use smol::{stream::StreamExt, Executor};
|
|
use std::collections::BTreeMap;
|
|
use std::sync::Arc;
|
|
use std::sync::Mutex;
|
|
use std::sync::MutexGuard;
|
|
use zbus::Connection;
|
|
|
|
use crate::GetSupported;
|
|
|
|
use super::config::{AuraConfig, AuraPowerConfig};
|
|
|
|
impl GetSupported for CtrlKbdLed {
|
|
type A = LedSupportedFunctions;
|
|
|
|
fn get_supported() -> Self::A {
|
|
// let mode = <&str>::from(&<AuraModes>::from(*mode));
|
|
let laptop = LaptopLedData::get_data();
|
|
let stock_led_modes = laptop.standard;
|
|
let multizone_led_mode = laptop.multizone;
|
|
let per_key_led_mode = laptop.per_key;
|
|
|
|
let mut prod_id = AuraDevice::Unknown;
|
|
for prod in ASUS_KEYBOARD_DEVICES.iter() {
|
|
if let Ok(_) = HidRaw::new(prod) {
|
|
prod_id = AuraDevice::from(*prod);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if let Ok(p) = KeyboardLed::new() {
|
|
if p.has_keyboard_rgb_mode() {
|
|
prod_id = AuraDevice::Tuf;
|
|
}
|
|
}
|
|
|
|
LedSupportedFunctions {
|
|
prod_id,
|
|
brightness_set: KeyboardLed::new().is_ok(),
|
|
stock_led_modes,
|
|
multizone_led_mode,
|
|
per_key_led_mode,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, PartialEq, PartialOrd)]
|
|
pub enum LEDNode {
|
|
KbdLed(KeyboardLed),
|
|
Rog(HidRaw),
|
|
None,
|
|
}
|
|
|
|
pub struct CtrlKbdLed {
|
|
// TODO: config stores the keyboard type as an AuraPower, use or update this
|
|
pub led_prod: Option<String>,
|
|
pub led_node: LEDNode,
|
|
pub kd_brightness: KeyboardLed,
|
|
pub supported_modes: LaptopLedData,
|
|
pub flip_effect_write: bool,
|
|
pub config: AuraConfig,
|
|
}
|
|
|
|
pub struct CtrlKbdLedTask {
|
|
inner: Arc<Mutex<CtrlKbdLed>>,
|
|
}
|
|
|
|
impl CtrlKbdLedTask {
|
|
pub fn new(inner: Arc<Mutex<CtrlKbdLed>>) -> Self {
|
|
Self { inner }
|
|
}
|
|
|
|
fn update_config(lock: &mut CtrlKbdLed) -> Result<(), RogError> {
|
|
let bright = lock.kd_brightness.get_brightness()?;
|
|
lock.config.read();
|
|
lock.config.brightness = (bright as u32).into();
|
|
lock.config.write();
|
|
return Ok(());
|
|
}
|
|
}
|
|
|
|
#[async_trait]
|
|
impl CtrlTask for CtrlKbdLedTask {
|
|
async fn create_tasks(&self, executor: &mut Executor) -> Result<(), RogError> {
|
|
let connection = Connection::system()
|
|
.await
|
|
.expect("CtrlKbdLedTask could not create dbus connection");
|
|
|
|
let manager = ManagerProxy::new(&connection)
|
|
.await
|
|
.expect("CtrlKbdLedTask could not create ManagerProxy");
|
|
|
|
let load_save = |start: bool, mut lock: MutexGuard<CtrlKbdLed>| {
|
|
// If waking up
|
|
if !start {
|
|
info!("CtrlKbdLedTask reloading brightness and modes");
|
|
lock.set_brightness(lock.config.brightness)
|
|
.map_err(|e| error!("CtrlKbdLedTask: {e}"))
|
|
.ok();
|
|
lock.write_current_config_mode()
|
|
.map_err(|e| error!("CtrlKbdLedTask: {e}"))
|
|
.ok();
|
|
} else if start {
|
|
info!("CtrlKbdLedTask saving last brightness");
|
|
Self::update_config(&mut lock)
|
|
.map_err(|e| error!("CtrlKbdLedTask: {e}"))
|
|
.ok();
|
|
}
|
|
};
|
|
|
|
let inner = self.inner.clone();
|
|
executor
|
|
.spawn(async move {
|
|
if let Ok(notif) = manager.receive_prepare_for_sleep().await {
|
|
notif
|
|
.for_each(|event| {
|
|
if let Ok(args) = event.args() {
|
|
loop {
|
|
// Loop so that we do aquire the lock but also don't block other
|
|
// threads (prevents potential deadlocks)
|
|
if let Ok(lock) = inner.clone().try_lock() {
|
|
load_save(args.start, lock);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
})
|
|
.await;
|
|
}
|
|
if let Ok(notif) = manager.receive_prepare_for_shutdown().await {
|
|
notif
|
|
.for_each(|event| {
|
|
if let Ok(args) = event.args() {
|
|
loop {
|
|
if let Ok(lock) = inner.clone().try_lock() {
|
|
load_save(args.start, lock);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
})
|
|
.await;
|
|
}
|
|
})
|
|
.detach();
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub struct CtrlKbdLedReloader(pub Arc<Mutex<CtrlKbdLed>>);
|
|
|
|
impl crate::Reloadable for CtrlKbdLedReloader {
|
|
fn reload(&mut self) -> Result<(), RogError> {
|
|
if let Ok(mut ctrl) = self.0.try_lock() {
|
|
ctrl.write_current_config_mode()?;
|
|
ctrl.set_power_states().map_err(|err| warn!("{err}")).ok();
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub struct CtrlKbdLedZbus(pub Arc<Mutex<CtrlKbdLed>>);
|
|
|
|
impl CtrlKbdLedZbus {
|
|
pub fn new(inner: Arc<Mutex<CtrlKbdLed>>) -> Self {
|
|
Self(inner)
|
|
}
|
|
}
|
|
|
|
impl CtrlKbdLed {
|
|
pub fn new(supported_modes: LaptopLedData, config: AuraConfig) -> Result<Self, RogError> {
|
|
let mut led_prod = None;
|
|
let mut led_node = None;
|
|
for prod in ASUS_KEYBOARD_DEVICES.iter() {
|
|
match HidRaw::new(prod) {
|
|
Ok(node) => {
|
|
led_prod = Some(prod.to_string());
|
|
led_node = Some(node);
|
|
info!("Looked for keyboard controller 0x{prod}: Found");
|
|
break;
|
|
}
|
|
Err(err) => info!("Looked for keyboard controller 0x{prod}: {err}"),
|
|
}
|
|
}
|
|
|
|
let bright_node = KeyboardLed::new();
|
|
let platform = KeyboardLed::new()?;
|
|
|
|
if led_node.is_none() && !platform.has_keyboard_rgb_mode() {
|
|
let dmi = sysfs_class::DmiId::default();
|
|
if let Ok(prod_family) = dmi.product_family() {
|
|
if prod_family.contains("TUF") {
|
|
warn!("A kernel patch is in progress for TUF RGB support");
|
|
}
|
|
}
|
|
return Err(RogError::NoAuraKeyboard);
|
|
}
|
|
|
|
let led_node = if let Some(rog) = led_node {
|
|
info!("Found ROG USB keyboard");
|
|
LEDNode::Rog(rog)
|
|
} else if platform.has_keyboard_rgb_mode() {
|
|
info!("Found TUF keyboard");
|
|
LEDNode::KbdLed(platform)
|
|
} else {
|
|
LEDNode::None
|
|
};
|
|
|
|
let ctrl = CtrlKbdLed {
|
|
led_prod,
|
|
led_node,
|
|
kd_brightness: bright_node?, // If was none then we already returned above
|
|
supported_modes,
|
|
flip_effect_write: false,
|
|
config,
|
|
};
|
|
Ok(ctrl)
|
|
}
|
|
|
|
pub(super) fn get_brightness(&self) -> Result<u8, RogError> {
|
|
self.kd_brightness
|
|
.get_brightness()
|
|
.map_err(|e| RogError::Platform(e))
|
|
}
|
|
|
|
pub(super) fn set_brightness(&self, brightness: LedBrightness) -> Result<(), RogError> {
|
|
self.kd_brightness
|
|
.set_brightness(brightness.as_char_code())
|
|
.map_err(|e| RogError::Platform(e))
|
|
}
|
|
|
|
pub fn next_brightness(&mut self) -> Result<(), RogError> {
|
|
let mut bright = (self.config.brightness as u32) + 1;
|
|
if bright > 3 {
|
|
bright = 0;
|
|
}
|
|
self.config.brightness = <LedBrightness>::from(bright);
|
|
self.config.write();
|
|
self.set_brightness(self.config.brightness)
|
|
}
|
|
|
|
pub fn prev_brightness(&mut self) -> Result<(), RogError> {
|
|
let mut bright = self.config.brightness as u32;
|
|
if bright == 0 {
|
|
bright = 3;
|
|
} else {
|
|
bright -= 1;
|
|
}
|
|
self.config.brightness = <LedBrightness>::from(bright);
|
|
self.config.write();
|
|
self.set_brightness(self.config.brightness)
|
|
}
|
|
|
|
/// Set combination state for boot animation/sleep animation/all leds/keys leds/side leds LED active
|
|
pub(super) fn set_power_states(&mut self) -> Result<(), RogError> {
|
|
if let LEDNode::KbdLed(platform) = &mut self.led_node {
|
|
if let Some(pwr) = AuraPowerConfig::to_tuf_bool_array(&self.config.enabled) {
|
|
let buf = [1, pwr[1] as u8, pwr[2] as u8, pwr[3] as u8, pwr[4] as u8];
|
|
platform.set_keyboard_rgb_state(&buf)?;
|
|
}
|
|
} else if let LEDNode::Rog(hid_raw) = &self.led_node {
|
|
let bytes = AuraPowerConfig::to_bytes(&self.config.enabled);
|
|
let message = [0x5d, 0xbd, 0x01, bytes[0], bytes[1], bytes[2]];
|
|
|
|
hid_raw.write_bytes(&message)?;
|
|
hid_raw.write_bytes(&LED_SET)?;
|
|
// Changes won't persist unless apply is set
|
|
hid_raw.write_bytes(&LED_APPLY)?;
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
/// Set an Aura effect if the effect mode or zone is supported.
|
|
///
|
|
/// On success the aura config file is read to refresh cached values, then the effect is
|
|
/// stored and config written to disk.
|
|
pub(crate) fn set_effect(&mut self, effect: AuraEffect) -> Result<(), RogError> {
|
|
if !self.supported_modes.standard.contains(&effect.mode) {
|
|
return Err(RogError::AuraEffectNotSupported);
|
|
} else if effect.zone != AuraZone::None
|
|
&& !self.supported_modes.multizone.contains(&effect.zone)
|
|
{
|
|
return Err(RogError::AuraEffectNotSupported);
|
|
}
|
|
|
|
self.write_mode(&effect)?;
|
|
self.config.read(); // refresh config if successful
|
|
self.config.set_builtin(effect);
|
|
self.config.write();
|
|
Ok(())
|
|
}
|
|
|
|
/// Write an effect block. This is for per-key
|
|
fn _write_effect(&mut self, effect: &[Vec<u8>]) -> Result<(), RogError> {
|
|
if let LEDNode::Rog(hid_raw) = &self.led_node {
|
|
if self.flip_effect_write {
|
|
for row in effect.iter().rev() {
|
|
hid_raw.write_bytes(row)?;
|
|
}
|
|
} else {
|
|
for row in effect.iter() {
|
|
hid_raw.write_bytes(row)?;
|
|
}
|
|
}
|
|
}
|
|
self.flip_effect_write = !self.flip_effect_write;
|
|
Ok(())
|
|
}
|
|
|
|
pub(super) fn toggle_mode(&mut self, reverse: bool) -> Result<(), RogError> {
|
|
let current = self.config.current_mode;
|
|
if let Some(idx) = self
|
|
.supported_modes
|
|
.standard
|
|
.iter()
|
|
.position(|v| *v == current)
|
|
{
|
|
let mut idx = idx;
|
|
// goes past end of array
|
|
if reverse {
|
|
if idx == 0 {
|
|
idx = self.supported_modes.standard.len() - 1;
|
|
} else {
|
|
idx -= 1;
|
|
}
|
|
} else {
|
|
idx += 1;
|
|
if idx == self.supported_modes.standard.len() {
|
|
idx = 0;
|
|
}
|
|
}
|
|
let next = self.supported_modes.standard[idx];
|
|
|
|
self.config.read();
|
|
// if self.config.builtins.contains_key(&next) {
|
|
self.config.current_mode = next;
|
|
self.write_current_config_mode()?;
|
|
// }
|
|
self.config.write();
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn write_mode(&self, mode: &AuraEffect) -> Result<(), RogError> {
|
|
if let LEDNode::KbdLed(platform) = &self.led_node {
|
|
let buf = [
|
|
1,
|
|
mode.mode as u8,
|
|
mode.colour1.0,
|
|
mode.colour1.1,
|
|
mode.colour1.2,
|
|
mode.speed as u8,
|
|
];
|
|
platform.set_keyboard_rgb_mode(&buf)?;
|
|
} else if let LEDNode::Rog(hid_raw) = &self.led_node {
|
|
let bytes: [u8; LED_MSG_LEN] = mode.into();
|
|
hid_raw.write_bytes(&bytes)?;
|
|
hid_raw.write_bytes(&LED_SET)?;
|
|
// Changes won't persist unless apply is set
|
|
hid_raw.write_bytes(&LED_APPLY)?;
|
|
} else {
|
|
return Err(RogError::NoAuraKeyboard);
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fn write_current_config_mode(&mut self) -> Result<(), RogError> {
|
|
if self.config.multizone_on {
|
|
let mode = self.config.current_mode;
|
|
let mut create = false;
|
|
// There is no multizone config for this mode so create one here
|
|
// using the colours of rainbow if it exists, or first available
|
|
// mode, or random
|
|
if self.config.multizone.is_none() {
|
|
create = true;
|
|
} else if let Some(multizones) = self.config.multizone.as_ref() {
|
|
if !multizones.contains_key(&mode) {
|
|
create = true;
|
|
}
|
|
}
|
|
if create {
|
|
info!("No user-set config for zone founding, attempting a default");
|
|
self.create_multizone_default()?;
|
|
}
|
|
|
|
if let Some(multizones) = self.config.multizone.as_ref() {
|
|
if let Some(set) = multizones.get(&mode) {
|
|
for mode in set {
|
|
self.write_mode(mode)?;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
let mode = self.config.current_mode;
|
|
if let Some(effect) = self.config.builtins.get(&mode) {
|
|
self.write_mode(effect)?;
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Create a default for the `current_mode` if multizone and no config exists.
|
|
fn create_multizone_default(&mut self) -> Result<(), RogError> {
|
|
let mut default = vec![];
|
|
for (i, tmp) in self.supported_modes.multizone.iter().enumerate() {
|
|
default.push(AuraEffect {
|
|
mode: self.config.current_mode,
|
|
zone: *tmp,
|
|
colour1: *GRADIENT.get(i).unwrap_or(&GRADIENT[0]),
|
|
colour2: *GRADIENT.get(GRADIENT.len() - i).unwrap_or(&GRADIENT[6]),
|
|
speed: Speed::Med,
|
|
direction: Direction::Left,
|
|
})
|
|
}
|
|
if default.is_empty() {
|
|
return Err(RogError::AuraEffectNotSupported);
|
|
}
|
|
|
|
if let Some(multizones) = self.config.multizone.as_mut() {
|
|
multizones.insert(self.config.current_mode, default);
|
|
} else {
|
|
let mut tmp = BTreeMap::new();
|
|
tmp.insert(self.config.current_mode, default);
|
|
self.config.multizone = Some(tmp);
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use rog_aura::{AuraEffect, AuraModeNum, AuraZone, Colour};
|
|
use rog_platform::keyboard_led::KeyboardLed;
|
|
|
|
use crate::{
|
|
ctrl_aura::{config::AuraConfig, controller::LEDNode},
|
|
laptops::LaptopLedData,
|
|
};
|
|
|
|
use super::CtrlKbdLed;
|
|
|
|
#[test]
|
|
// #[ignore = "Must be manually run due to detection stage"]
|
|
fn check_set_mode_errors() {
|
|
// Checking to ensure set_mode errors when unsupported modes are tried
|
|
let config = AuraConfig::default();
|
|
let supported_modes = LaptopLedData {
|
|
prod_family: "".into(),
|
|
board_names: vec![],
|
|
standard: vec![AuraModeNum::Static],
|
|
multizone: vec![],
|
|
per_key: false,
|
|
};
|
|
let mut controller = CtrlKbdLed {
|
|
led_prod: None,
|
|
led_node: LEDNode::None,
|
|
kd_brightness: KeyboardLed::default(),
|
|
supported_modes,
|
|
flip_effect_write: false,
|
|
config,
|
|
};
|
|
|
|
let mut effect = AuraEffect::default();
|
|
effect.colour1 = Colour(0xff, 0x00, 0xff);
|
|
effect.zone = AuraZone::None;
|
|
|
|
// This error comes from write_bytes because we don't have a keyboard node stored
|
|
assert_eq!(
|
|
controller
|
|
.set_effect(effect.clone())
|
|
.unwrap_err()
|
|
.to_string(),
|
|
"No supported Aura keyboard"
|
|
);
|
|
|
|
effect.mode = AuraModeNum::Laser;
|
|
assert_eq!(
|
|
controller
|
|
.set_effect(effect.clone())
|
|
.unwrap_err()
|
|
.to_string(),
|
|
"Aura effect not supported"
|
|
);
|
|
|
|
effect.mode = AuraModeNum::Static;
|
|
effect.zone = AuraZone::Key2;
|
|
assert_eq!(
|
|
controller
|
|
.set_effect(effect.clone())
|
|
.unwrap_err()
|
|
.to_string(),
|
|
"Aura effect not supported"
|
|
);
|
|
|
|
controller.supported_modes.multizone.push(AuraZone::Key2);
|
|
assert_eq!(
|
|
controller
|
|
.set_effect(effect.clone())
|
|
.unwrap_err()
|
|
.to_string(),
|
|
"No supported Aura keyboard"
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn create_multizone_if_no_config() {
|
|
// Checking to ensure set_mode errors when unsupported modes are tried
|
|
let config = AuraConfig::default();
|
|
let supported_modes = LaptopLedData {
|
|
prod_family: "".into(),
|
|
board_names: vec![],
|
|
standard: vec![AuraModeNum::Static],
|
|
multizone: vec![],
|
|
per_key: false,
|
|
};
|
|
let mut controller = CtrlKbdLed {
|
|
led_prod: None,
|
|
led_node: LEDNode::None,
|
|
kd_brightness: KeyboardLed::default(),
|
|
supported_modes,
|
|
flip_effect_write: false,
|
|
config,
|
|
};
|
|
|
|
assert!(controller.config.multizone.is_none());
|
|
assert!(controller.create_multizone_default().is_err());
|
|
assert!(controller.config.multizone.is_none());
|
|
|
|
controller.supported_modes.multizone.push(AuraZone::Key1);
|
|
controller.supported_modes.multizone.push(AuraZone::Key2);
|
|
assert!(controller.create_multizone_default().is_ok());
|
|
assert!(controller.config.multizone.is_some());
|
|
|
|
let m = controller.config.multizone.unwrap();
|
|
assert!(m.contains_key(&AuraModeNum::Static));
|
|
let e = m.get(&AuraModeNum::Static).unwrap();
|
|
assert_eq!(e.len(), 2);
|
|
assert_eq!(e[0].zone, AuraZone::Key1);
|
|
assert_eq!(e[1].zone, AuraZone::Key2);
|
|
}
|
|
|
|
#[test]
|
|
fn next_mode_create_multizone_if_no_config() {
|
|
// Checking to ensure set_mode errors when unsupported modes are tried
|
|
let config = AuraConfig::default();
|
|
let supported_modes = LaptopLedData {
|
|
prod_family: "".into(),
|
|
board_names: vec![],
|
|
standard: vec![AuraModeNum::Static],
|
|
multizone: vec![AuraZone::Key1, AuraZone::Key2],
|
|
per_key: false,
|
|
};
|
|
let mut controller = CtrlKbdLed {
|
|
led_prod: None,
|
|
led_node: LEDNode::None,
|
|
kd_brightness: KeyboardLed::default(),
|
|
supported_modes,
|
|
flip_effect_write: false,
|
|
config,
|
|
};
|
|
|
|
assert!(controller.config.multizone.is_none());
|
|
controller.config.multizone_on = true;
|
|
// This is called in toggle_mode. It will error here because we have no
|
|
// keyboard node in tests.
|
|
assert_eq!(
|
|
controller
|
|
.write_current_config_mode()
|
|
.unwrap_err()
|
|
.to_string(),
|
|
"No supported Aura keyboard"
|
|
);
|
|
assert!(controller.config.multizone.is_some());
|
|
|
|
let m = controller.config.multizone.unwrap();
|
|
assert!(m.contains_key(&AuraModeNum::Static));
|
|
let e = m.get(&AuraModeNum::Static).unwrap();
|
|
assert_eq!(e.len(), 2);
|
|
assert_eq!(e[0].zone, AuraZone::Key1);
|
|
assert_eq!(e[1].zone, AuraZone::Key2);
|
|
}
|
|
}
|