2018-01-15 17:11:14 +00:00
|
|
|
// Copyright (C) 2018 The Syncthing Authors.
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
|
2021-08-17 10:10:41 +02:00
|
|
|
//go:build !android
|
2018-03-27 22:03:09 +01:00
|
|
|
// +build !android
|
|
|
|
|
2018-01-15 17:11:14 +00:00
|
|
|
package osutil
|
|
|
|
|
|
|
|
import (
|
2022-08-16 10:01:49 +02:00
|
|
|
"fmt"
|
2018-01-17 12:32:11 +00:00
|
|
|
"os"
|
2018-01-15 17:11:14 +00:00
|
|
|
"syscall"
|
|
|
|
)
|
|
|
|
|
|
|
|
const ioprioClassShift = 13
|
|
|
|
|
|
|
|
type ioprioClass int
|
|
|
|
|
|
|
|
const (
|
|
|
|
ioprioClassRT ioprioClass = iota + 1
|
|
|
|
ioprioClassBE
|
|
|
|
ioprioClassIdle
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
ioprioWhoProcess = iota + 1
|
|
|
|
ioprioWhoPGRP
|
|
|
|
ioprioWhoUser
|
|
|
|
)
|
|
|
|
|
|
|
|
func ioprioSet(class ioprioClass, value int) error {
|
|
|
|
res, _, err := syscall.Syscall(syscall.SYS_IOPRIO_SET,
|
|
|
|
uintptr(ioprioWhoProcess), 0,
|
|
|
|
uintptr(class)<<ioprioClassShift|uintptr(value))
|
|
|
|
if res == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetLowPriority lowers the process CPU scheduling priority, and possibly
|
|
|
|
// I/O priority depending on the platform and OS.
|
|
|
|
func SetLowPriority() error {
|
2018-01-18 08:34:56 +00:00
|
|
|
// Process zero is "self", niceness value 9 is something between 0
|
|
|
|
// (default) and 19 (worst priority). But then, this is Linux, so of
|
|
|
|
// course we get this to take care of as well:
|
|
|
|
//
|
|
|
|
// "C library/kernel differences
|
|
|
|
//
|
|
|
|
// Within the kernel, nice values are actually represented using the
|
|
|
|
// range 40..1 (since negative numbers are error codes) and these are
|
|
|
|
// the values employed by the setpriority() and getpriority() system
|
|
|
|
// calls. The glibc wrapper functions for these system calls handle the
|
|
|
|
// translations between the user-land and kernel representations of the
|
|
|
|
// nice value according to the formula unice = 20 - knice. (Thus, the
|
|
|
|
// kernel's 40..1 range corresponds to the range -20..19 as seen by user
|
|
|
|
// space.)"
|
|
|
|
|
|
|
|
const (
|
|
|
|
pidSelf = 0
|
|
|
|
wantNiceLevel = 20 - 9
|
|
|
|
)
|
|
|
|
|
|
|
|
// Remember Linux kernel nice levels are upside down.
|
|
|
|
if cur, err := syscall.Getpriority(syscall.PRIO_PROCESS, 0); err == nil && cur <= wantNiceLevel {
|
|
|
|
// We're done here.
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-01-15 17:11:14 +00:00
|
|
|
// Move ourselves to a new process group so that we can use the process
|
|
|
|
// group variants of Setpriority etc to affect all of our threads in one
|
|
|
|
// go. If this fails, bail, so that we don't affect things we shouldn't.
|
2018-01-17 12:32:11 +00:00
|
|
|
// If we are already the leader of our own process group, do nothing.
|
2018-01-18 08:34:56 +00:00
|
|
|
//
|
|
|
|
// Oh and this is because Linux doesn't follow the POSIX threading model
|
|
|
|
// where setting the niceness of the process would actually set the
|
|
|
|
// niceness of the process, instead it just affects the current thread
|
|
|
|
// so we need this workaround...
|
|
|
|
if pgid, err := syscall.Getpgid(pidSelf); err != nil {
|
2018-01-17 12:32:11 +00:00
|
|
|
// This error really shouldn't happen
|
2022-08-16 10:01:49 +02:00
|
|
|
return fmt.Errorf("get process group: %w", err)
|
2018-01-17 12:32:11 +00:00
|
|
|
} else if pgid != os.Getpid() {
|
|
|
|
// We are not process group leader. Elevate!
|
2018-01-18 08:34:56 +00:00
|
|
|
if err := syscall.Setpgid(pidSelf, 0); err != nil {
|
2022-08-16 10:01:49 +02:00
|
|
|
return fmt.Errorf("set process group: %w", err)
|
2018-01-17 12:32:11 +00:00
|
|
|
}
|
2018-01-15 17:11:14 +00:00
|
|
|
}
|
|
|
|
|
2018-01-18 08:34:56 +00:00
|
|
|
if err := syscall.Setpriority(syscall.PRIO_PGRP, pidSelf, wantNiceLevel); err != nil {
|
2022-08-16 10:01:49 +02:00
|
|
|
return fmt.Errorf("set niceness: %w", err)
|
2018-01-15 17:11:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Best effort, somewhere to the end of the scale (0 through 7 being the
|
|
|
|
// range).
|
2022-08-16 10:01:49 +02:00
|
|
|
if err := ioprioSet(ioprioClassBE, 5); err != nil {
|
|
|
|
return fmt.Errorf("set I/O priority: %w", err)
|
|
|
|
}
|
|
|
|
return nil
|
2018-01-15 17:11:14 +00:00
|
|
|
}
|