mirror of
https://github.com/mainnika/acpi-wakeup-fixxer.git
synced 2026-05-22 15:53:40 +00:00
initial setup of an app to toggle acpi wakeup devices
This commit is contained in:
@@ -0,0 +1,74 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"log/slog"
|
||||
"os"
|
||||
|
||||
"github.com/spf13/cobra"
|
||||
|
||||
"code.tokarch.uk/mainnika/acpi-wakeup-fixxer/pkg/wakeup"
|
||||
)
|
||||
|
||||
var (
|
||||
devicesToDisableFlag []string
|
||||
)
|
||||
|
||||
var rootCmd = &cobra.Command{
|
||||
Use: "acpi-wakeup-fixxer",
|
||||
Short: "A CLI tool to fix ACPI wakeup issues",
|
||||
Args: cobra.NoArgs,
|
||||
Long: "acpi-wakeup-fixxer is a command-line tool to fix ACPI wakeup issues on your system.",
|
||||
Run: rootCmdRun,
|
||||
}
|
||||
|
||||
func init() {
|
||||
rootCmd.PersistentFlags().StringArrayVar(&devicesToDisableFlag, "disable", []string{"LID0", "XHC1"}, "Devices to disable")
|
||||
}
|
||||
|
||||
func main() {
|
||||
if err := rootCmd.Execute(); err != nil {
|
||||
fmt.Println(err)
|
||||
os.Exit(1)
|
||||
}
|
||||
}
|
||||
|
||||
func rootCmdRun(cmd *cobra.Command, args []string) {
|
||||
wakeUpController := wakeup.NewWakeupController()
|
||||
|
||||
devices, err := wakeUpController.GetWakeupDevices(wakeup.StatusEnabled)
|
||||
if err != nil {
|
||||
slog.Error("failed to get wakeup devices", "error", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
|
||||
slog.Info("Found devices", "devices", devices)
|
||||
slog.Info("Disabling devices", "devices", devicesToDisableFlag)
|
||||
|
||||
var disabledDevices []string
|
||||
for _, device := range devices {
|
||||
needDisable := false
|
||||
for _, deviceToDisable := range devicesToDisableFlag {
|
||||
if device == deviceToDisable {
|
||||
needDisable = true
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
if needDisable {
|
||||
slog.Info("Disabling device", "device", device)
|
||||
if err := wakeUpController.ToggleWakeupDevice(device); err != nil {
|
||||
slog.Error("failed to disable device", "device", device, "error", err)
|
||||
continue
|
||||
}
|
||||
|
||||
disabledDevices = append(disabledDevices, device)
|
||||
}
|
||||
}
|
||||
|
||||
if len(disabledDevices) > 0 {
|
||||
slog.Info("Disabled devices", "devices", disabledDevices)
|
||||
} else {
|
||||
slog.Info("No devices were disabled")
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,16 @@
|
||||
module code.tokarch.uk/mainnika/acpi-wakeup-fixxer
|
||||
|
||||
go 1.22.2
|
||||
|
||||
require (
|
||||
github.com/spf13/cobra v1.8.0
|
||||
github.com/stretchr/testify v1.9.0
|
||||
)
|
||||
|
||||
require (
|
||||
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||
github.com/spf13/pflag v1.0.5 // indirect
|
||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||
)
|
||||
@@ -0,0 +1,18 @@
|
||||
github.com/cpuguy83/go-md2man/v2 v2.0.3/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
|
||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
|
||||
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
|
||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
||||
github.com/spf13/cobra v1.8.0 h1:7aJaZx1B85qltLMc546zn58BxxfZdR/W22ej9CFoEf0=
|
||||
github.com/spf13/cobra v1.8.0/go.mod h1:WXLWApfZ71AjXPya3WOlMsY9yMs7YeiHhFVlvLyhcho=
|
||||
github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
|
||||
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
||||
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||
@@ -0,0 +1,36 @@
|
||||
package procfs
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
)
|
||||
|
||||
const (
|
||||
acpiWakeupDefaultPath = "/proc/acpi/wakeup"
|
||||
)
|
||||
|
||||
type Procfs interface {
|
||||
ACPIWakeup() (io.ReadCloser, error)
|
||||
ACPIWakeupWrite() (io.WriteCloser, error)
|
||||
}
|
||||
|
||||
type ProcfsDefaultPath struct{}
|
||||
|
||||
func (p *ProcfsDefaultPath) ACPIWakeup() (io.ReadCloser, error) {
|
||||
f, err := os.Open(acpiWakeupDefaultPath)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to open %s: %w", acpiWakeupDefaultPath, err)
|
||||
}
|
||||
|
||||
return f, nil
|
||||
}
|
||||
|
||||
func (p *ProcfsDefaultPath) ACPIWakeupWrite() (io.WriteCloser, error) {
|
||||
f, err := os.OpenFile(acpiWakeupDefaultPath, os.O_WRONLY, 0)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to open for writing %s: %w", acpiWakeupDefaultPath, err)
|
||||
}
|
||||
|
||||
return f, nil
|
||||
}
|
||||
@@ -0,0 +1,88 @@
|
||||
package wakeup
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"fmt"
|
||||
"log/slog"
|
||||
"strings"
|
||||
|
||||
"code.tokarch.uk/mainnika/acpi-wakeup-fixxer/pkg/procfs"
|
||||
)
|
||||
|
||||
const (
|
||||
deviceHeader = "Device"
|
||||
columnDeviceIndex = 0
|
||||
columnStatusIndex = 2
|
||||
columnIndexMax = 3
|
||||
)
|
||||
|
||||
type Status string
|
||||
|
||||
const StatusAll Status = ""
|
||||
const StatusEnabled Status = "*enabled"
|
||||
const StatusDisabled Status = "*disabled"
|
||||
|
||||
type WakeupController struct {
|
||||
ProcfsProvider procfs.Procfs
|
||||
}
|
||||
|
||||
func NewWakeupController() *WakeupController {
|
||||
return &WakeupController{ProcfsProvider: &procfs.ProcfsDefaultPath{}}
|
||||
}
|
||||
|
||||
func (w *WakeupController) GetWakeupDevices(withStatus Status) ([]string, error) {
|
||||
wakeupFile, err := w.ProcfsProvider.ACPIWakeup()
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to get wakeup file: %w", err)
|
||||
}
|
||||
defer func() { _ = wakeupFile.Close() }()
|
||||
|
||||
scanner := bufio.NewScanner(wakeupFile)
|
||||
var devices []string
|
||||
|
||||
for scanner.Scan() {
|
||||
line := scanner.Text()
|
||||
fields := strings.Fields(line)
|
||||
if len(fields) < columnIndexMax {
|
||||
slog.Warn("unexpected number of columns", "line", line)
|
||||
continue
|
||||
}
|
||||
|
||||
device := fields[columnDeviceIndex]
|
||||
if device == deviceHeader {
|
||||
continue
|
||||
}
|
||||
|
||||
status := Status(fields[columnStatusIndex])
|
||||
if withStatus != StatusAll && status != withStatus {
|
||||
continue
|
||||
}
|
||||
|
||||
devices = append(devices, device)
|
||||
}
|
||||
if err := scanner.Err(); err != nil {
|
||||
return nil, fmt.Errorf("failed to scan wakeup file: %w", err)
|
||||
}
|
||||
|
||||
return devices, nil
|
||||
}
|
||||
|
||||
func (w *WakeupController) ToggleWakeupDevice(device string) error {
|
||||
wakeupFile, err := w.ProcfsProvider.ACPIWakeupWrite()
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to open wakeup file for writing: %w", err)
|
||||
}
|
||||
|
||||
_, err = fmt.Fprintf(wakeupFile, "%s\n", device)
|
||||
if err != nil {
|
||||
_ = wakeupFile.Close()
|
||||
return fmt.Errorf("failed to write to wakeup file: %w", err)
|
||||
}
|
||||
|
||||
err = wakeupFile.Close()
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to close wakeup file: %w", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
@@ -0,0 +1,105 @@
|
||||
package wakeup
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
)
|
||||
|
||||
type procfsMock struct{}
|
||||
|
||||
func (p *procfsMock) ACPIWakeup() (io.ReadCloser, error) {
|
||||
const testFileRaw = `Device S-state Status Sysfs node
|
||||
PEG0 S3 *disabled
|
||||
EC S4 *disabled platform:PNP0C09:00
|
||||
HDEF S3 *disabled pci:0000:00:1b.0
|
||||
RP01 S3 *enabled pci:0000:00:1c.0
|
||||
RP02 S3 *enabled pci:0000:00:1c.1
|
||||
RP03 S3 *enabled pci:0000:00:1c.2
|
||||
ARPT S4 *enabled pci:0000:03:00.0
|
||||
RP05 S3 *enabled pci:0000:00:1c.4
|
||||
RP06 S3 *enabled pci:0000:00:1c.5
|
||||
SPIT S3 *disabled spi:spi-APP000D:00
|
||||
XHC1 S3 *disabled pci:0000:00:14.0
|
||||
ADP1 S4 *disabled platform:ACPI0003:00
|
||||
LID0 S4 *disabled platform:PNP0C0D:00
|
||||
`
|
||||
|
||||
return io.NopCloser(strings.NewReader(testFileRaw)), nil
|
||||
}
|
||||
|
||||
func (p *procfsMock) ACPIWakeupWrite() (io.WriteCloser, error) {
|
||||
return nil, fmt.Errorf("not implemented")
|
||||
}
|
||||
|
||||
func TestSomething(t *testing.T) {
|
||||
|
||||
tt := []struct {
|
||||
name string
|
||||
wantStatus Status
|
||||
wantDevices []string
|
||||
}{
|
||||
{
|
||||
name: "Read and parse successfully",
|
||||
wantStatus: StatusEnabled,
|
||||
wantDevices: []string{
|
||||
"RP01",
|
||||
"RP02",
|
||||
"RP03",
|
||||
"RP05",
|
||||
"RP06",
|
||||
"ARPT",
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "Read and parse disabled successfully",
|
||||
wantStatus: StatusDisabled,
|
||||
wantDevices: []string{
|
||||
"PEG0",
|
||||
"EC",
|
||||
"HDEF",
|
||||
"SPIT",
|
||||
"XHC1",
|
||||
"ADP1",
|
||||
"LID0",
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "Read and parse all successfully",
|
||||
wantStatus: StatusAll,
|
||||
wantDevices: []string{
|
||||
"RP01",
|
||||
"RP02",
|
||||
"RP03",
|
||||
"RP05",
|
||||
"RP06",
|
||||
"ARPT",
|
||||
"PEG0",
|
||||
"EC",
|
||||
"HDEF",
|
||||
"SPIT",
|
||||
"XHC1",
|
||||
"ADP1",
|
||||
"LID0",
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
for _, tc := range tt {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
c := NewWakeupController()
|
||||
c.ProcfsProvider = &procfsMock{}
|
||||
|
||||
devices, err := c.GetWakeupDevices(tc.wantStatus)
|
||||
assert.NoError(t, err)
|
||||
|
||||
assert.Len(t, devices, len(tc.wantDevices))
|
||||
for i, wantDevice := range tc.wantDevices {
|
||||
assert.Contains(t, devices, wantDevice, "device %d not found", i)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user