Skip to content

Commit 29fedc5

Browse files
committedFeb 12, 2025·
ebpf: implement PossibleCPU on Windows
Move the Linux implementation into internal/linux and add a wrapper around the Windows GetMaximumProcessorCount function. Signed-off-by: Lorenz Bauer <lmb@isovalent.com>
1 parent 70b1fec commit 29fedc5

File tree

6 files changed

+107
-74
lines changed

6 files changed

+107
-74
lines changed
 

‎cpu.go

-49
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,5 @@
11
package ebpf
22

3-
import (
4-
"fmt"
5-
"os"
6-
"strings"
7-
"sync"
8-
)
9-
10-
var possibleCPU = sync.OnceValues(func() (int, error) {
11-
return parseCPUsFromFile("/sys/devices/system/cpu/possible")
12-
})
13-
143
// PossibleCPU returns the max number of CPUs a system may possibly have
154
// Logical CPU numbers must be of the form 0-n
165
func PossibleCPU() (int, error) {
@@ -26,41 +15,3 @@ func MustPossibleCPU() int {
2615
}
2716
return cpus
2817
}
29-
30-
func parseCPUsFromFile(path string) (int, error) {
31-
spec, err := os.ReadFile(path)
32-
if err != nil {
33-
return 0, err
34-
}
35-
36-
n, err := parseCPUs(string(spec))
37-
if err != nil {
38-
return 0, fmt.Errorf("can't parse %s: %v", path, err)
39-
}
40-
41-
return n, nil
42-
}
43-
44-
// parseCPUs parses the number of cpus from a string produced
45-
// by bitmap_list_string() in the Linux kernel.
46-
// Multiple ranges are rejected, since they can't be unified
47-
// into a single number.
48-
// This is the format of /sys/devices/system/cpu/possible, it
49-
// is not suitable for /sys/devices/system/cpu/online, etc.
50-
func parseCPUs(spec string) (int, error) {
51-
if strings.Trim(spec, "\n") == "0" {
52-
return 1, nil
53-
}
54-
55-
var low, high int
56-
n, err := fmt.Sscanf(spec, "%d-%d\n", &low, &high)
57-
if n != 2 || err != nil {
58-
return 0, fmt.Errorf("invalid format: %s", spec)
59-
}
60-
if low != 0 {
61-
return 0, fmt.Errorf("CPU spec doesn't start at zero: %s", spec)
62-
}
63-
64-
// cpus is 0 indexed
65-
return high + 1, nil
66-
}

‎cpu_other.go

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
//go:build !windows
2+
3+
package ebpf
4+
5+
import (
6+
"sync"
7+
8+
"github.com/cilium/ebpf/internal/linux"
9+
)
10+
11+
var possibleCPU = sync.OnceValues(func() (int, error) {
12+
return linux.ParseCPUsFromFile("/sys/devices/system/cpu/possible")
13+
})

‎cpu_test.go

+6-25
Original file line numberDiff line numberDiff line change
@@ -2,31 +2,12 @@ package ebpf
22

33
import (
44
"testing"
5-
)
65

7-
func TestParseCPUs(t *testing.T) {
8-
for str, result := range map[string]int{
9-
"0-1": 2,
10-
"0-2\n": 3,
11-
"0": 1,
12-
} {
13-
n, err := parseCPUs(str)
14-
if err != nil {
15-
t.Errorf("Can't parse `%s`: %v", str, err)
16-
} else if n != result {
17-
t.Error("Parsing", str, "returns", n, "instead of", result)
18-
}
19-
}
6+
"github.com/go-quicktest/qt"
7+
)
208

21-
for _, str := range []string{
22-
"0,3-4",
23-
"0-",
24-
"1,",
25-
"",
26-
} {
27-
_, err := parseCPUs(str)
28-
if err == nil {
29-
t.Error("Parsed invalid format:", str)
30-
}
31-
}
9+
func TestPossibleCPU(t *testing.T) {
10+
num, err := PossibleCPU()
11+
qt.Assert(t, qt.IsNil(err))
12+
qt.Assert(t, qt.IsTrue(num > 0))
3213
}

‎cpu_windows.go

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
package ebpf
2+
3+
import (
4+
"sync"
5+
6+
"golang.org/x/sys/windows"
7+
)
8+
9+
var possibleCPU = sync.OnceValues(func() (int, error) {
10+
return int(windows.GetMaximumProcessorCount(windows.ALL_PROCESSOR_GROUPS)), nil
11+
})

‎internal/linux/cpu.go

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
package linux
2+
3+
import (
4+
"fmt"
5+
"os"
6+
"strings"
7+
)
8+
9+
func ParseCPUsFromFile(path string) (int, error) {
10+
spec, err := os.ReadFile(path)
11+
if err != nil {
12+
return 0, err
13+
}
14+
15+
n, err := parseCPUs(string(spec))
16+
if err != nil {
17+
return 0, fmt.Errorf("can't parse %s: %v", path, err)
18+
}
19+
20+
return n, nil
21+
}
22+
23+
// parseCPUs parses the number of cpus from a string produced
24+
// by bitmap_list_string() in the Linux kernel.
25+
// Multiple ranges are rejected, since they can't be unified
26+
// into a single number.
27+
// This is the format of /sys/devices/system/cpu/possible, it
28+
// is not suitable for /sys/devices/system/cpu/online, etc.
29+
func parseCPUs(spec string) (int, error) {
30+
if strings.Trim(spec, "\n") == "0" {
31+
return 1, nil
32+
}
33+
34+
var low, high int
35+
n, err := fmt.Sscanf(spec, "%d-%d\n", &low, &high)
36+
if n != 2 || err != nil {
37+
return 0, fmt.Errorf("invalid format: %s", spec)
38+
}
39+
if low != 0 {
40+
return 0, fmt.Errorf("CPU spec doesn't start at zero: %s", spec)
41+
}
42+
43+
// cpus is 0 indexed
44+
return high + 1, nil
45+
}

‎internal/linux/cpu_test.go

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
package linux
2+
3+
import (
4+
"testing"
5+
)
6+
7+
func TestParseCPUs(t *testing.T) {
8+
for str, result := range map[string]int{
9+
"0-1": 2,
10+
"0-2\n": 3,
11+
"0": 1,
12+
} {
13+
n, err := parseCPUs(str)
14+
if err != nil {
15+
t.Errorf("Can't parse `%s`: %v", str, err)
16+
} else if n != result {
17+
t.Error("Parsing", str, "returns", n, "instead of", result)
18+
}
19+
}
20+
21+
for _, str := range []string{
22+
"0,3-4",
23+
"0-",
24+
"1,",
25+
"",
26+
} {
27+
_, err := parseCPUs(str)
28+
if err == nil {
29+
t.Error("Parsed invalid format:", str)
30+
}
31+
}
32+
}

0 commit comments

Comments
 (0)
Please sign in to comment.