mirror of
https://github.com/kuhyx/testsAndMisc.git
synced 2026-07-04 15:03:01 +02:00
Resource-usage report showed ~29 cores of average load coming from i3blocks helper scripts forking awk/tr/grep/bc/sensors/nvidia-smi every tick. Rewrite all five hot-path scripts to eliminate forks: - volume.sh: persist mode, blocks on 'pactl subscribe' event stream. No polling, no sleep, no fork per tick. - gpu_monitor.sh: persist mode, single long-lived 'nvidia-smi --loop=5' feeds a bash 'while read' loop. Falls back to /sys for amdgpu. - battery_status.sh: reads /sys/class/power_supply/BAT*/ directly. Zero forks; replaces 'acpi | awk' pipeline. - cpu_monitor.sh: reads /proc/loadavg and k10temp/coretemp /sys/class/hwmon. Zero forks; replaces 'sensors | awk | tr' + bc arithmetic. - motherboard_temp.sh: reads nct*/it*/f71* Super-I/O hwmon node directly. Zero forks. Configure volume + gpu_monitor with interval=persist so i3blocks keeps one long-lived producer each instead of forking per tick. Also add: - kill_stale_recorders.sh -- kill stray ffmpeg x11grab / dotnet-trace / dotnet-monitor processes left running after sessions. - monitors.slice -- resource-capped user slice (CPUQuota=50%, MemoryMax=512M, MemorySwapMax=0 for zram safety, TasksMax=256) to bound future monitoring regressions. - efficient-polling-scripts SKILL -- rules for writing status-bar and polling scripts without forks; fork-pipeline to bash-builtin translation table; verification checklist. Verified live: strace -c on cpu_monitor.sh shows 1 execve / 0 clones; persist producers (pactl subscribe, nvidia-smi --loop) show 0 CPU ticks over a 3s idle sample. Per-invocation timing 1.6-1.9 ms (was 30-80 ms).
76 lines
2.0 KiB
Bash
Executable File
76 lines
2.0 KiB
Bash
Executable File
#!/bin/bash
|
|
# i3blocks GPU monitor, persist mode.
|
|
#
|
|
# Keeps a single long-lived `nvidia-smi --loop=5` (or reads amdgpu sysfs
|
|
# in a blocking-read loop) instead of forking nvidia-smi/lspci/awk/tr/bc
|
|
# every interval. No sleep, no polling loop in bash — nvidia-smi's own
|
|
# periodic emitter drives updates and we block on `read`.
|
|
#
|
|
# Configure with `interval=persist` in the i3blocks config.
|
|
|
|
set -u
|
|
|
|
emit() {
|
|
local temp=$1 load=$2 color
|
|
if [[ $load == 'N/A' ]]; then
|
|
color='#FFFFFF'
|
|
elif ((load < 50)); then
|
|
color='#50FA7B'
|
|
elif ((load < 75)); then
|
|
color='#F1FA8C'
|
|
else
|
|
color='#FF5555'
|
|
fi
|
|
printf '<span color="%s"> %s°C, %s%%</span>\n\n%s\n' \
|
|
"$color" "$temp" "$load" "$color"
|
|
}
|
|
|
|
# Prefer NVIDIA if present (persist via --loop).
|
|
if command -v nvidia-smi > /dev/null 2>&1; then
|
|
# One child process for the lifetime of i3blocks; emits CSV every 5s.
|
|
nvidia-smi \
|
|
--query-gpu=temperature.gpu,utilization.gpu \
|
|
--format=csv,noheader,nounits \
|
|
--loop=5 2> /dev/null |
|
|
while IFS=',' read -r temp load; do
|
|
# Strip leading/trailing whitespace using parameter expansion.
|
|
temp=${temp## }
|
|
temp=${temp%% }
|
|
load=${load## }
|
|
load=${load%% }
|
|
[[ -z $temp || -z $load ]] && continue
|
|
emit "$temp" "$load"
|
|
done
|
|
exit 0
|
|
fi
|
|
|
|
# AMD fallback: read sysfs directly; emit once (i3blocks restarts on exit).
|
|
amdgpu=''
|
|
for d in /sys/class/hwmon/hwmon*/; do
|
|
[[ -r ${d}name ]] || continue
|
|
read -r n < "${d}name"
|
|
[[ $n == amdgpu ]] && {
|
|
amdgpu=$d
|
|
break
|
|
}
|
|
done
|
|
if [[ -n $amdgpu ]]; then
|
|
temp='N/A'
|
|
if [[ -r ${amdgpu}temp1_input ]]; then
|
|
read -r milli < "${amdgpu}temp1_input"
|
|
temp=$((milli / 1000))
|
|
fi
|
|
load='N/A'
|
|
# drm card matching the amdgpu hwmon exposes gpu_busy_percent.
|
|
for card in /sys/class/drm/card*/device/gpu_busy_percent; do
|
|
[[ -r $card ]] && {
|
|
read -r load < "$card"
|
|
break
|
|
}
|
|
done
|
|
emit "$temp" "$load"
|
|
exit 0
|
|
fi
|
|
|
|
printf 'No supported GPU\n\n#FF5555\n'
|