cpuset.go 1.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
  1. package systemd
  2. import (
  3. "errors"
  4. "math/big"
  5. "strconv"
  6. "strings"
  7. )
  8. // RangeToBits converts a text representation of a CPU mask (as written to
  9. // or read from cgroups' cpuset.* files, e.g. "1,3-5") to a slice of bytes
  10. // with the corresponding bits set (as consumed by systemd over dbus as
  11. // AllowedCPUs/AllowedMemoryNodes unit property value).
  12. func RangeToBits(str string) ([]byte, error) {
  13. bits := new(big.Int)
  14. for _, r := range strings.Split(str, ",") {
  15. // allow extra spaces around
  16. r = strings.TrimSpace(r)
  17. // allow empty elements (extra commas)
  18. if r == "" {
  19. continue
  20. }
  21. ranges := strings.SplitN(r, "-", 2)
  22. if len(ranges) > 1 {
  23. start, err := strconv.ParseUint(ranges[0], 10, 32)
  24. if err != nil {
  25. return nil, err
  26. }
  27. end, err := strconv.ParseUint(ranges[1], 10, 32)
  28. if err != nil {
  29. return nil, err
  30. }
  31. if start > end {
  32. return nil, errors.New("invalid range: " + r)
  33. }
  34. for i := start; i <= end; i++ {
  35. bits.SetBit(bits, int(i), 1)
  36. }
  37. } else {
  38. val, err := strconv.ParseUint(ranges[0], 10, 32)
  39. if err != nil {
  40. return nil, err
  41. }
  42. bits.SetBit(bits, int(val), 1)
  43. }
  44. }
  45. ret := bits.Bytes()
  46. if len(ret) == 0 {
  47. // do not allow empty values
  48. return nil, errors.New("empty value")
  49. }
  50. return ret, nil
  51. }