1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
| //go:build linux
// +build linux
package main
import (
"flag"
"fmt"
v1 "hami_tool/v1"
"os"
"syscall"
)
type UsageInfo v1.Spec
type ContainerUsage struct {
PodUID string
ContainerName string
data []byte
Info v1.Spec
}
func readCacheFile(cacheFile string) (*ContainerUsage, error) {
// 1. 打开文件(只读模式)
f, err := os.OpenFile(cacheFile, os.O_RDONLY, 0666)
if err != nil {
return nil, fmt.Errorf("打开缓存文件失败: %w", err)
}
defer func() {
_ = f.Close()
fmt.Printf("已关闭文件:%s\n", cacheFile)
}()
// 2. 获取文件信息(大小)
info, err := f.Stat()
if err != nil {
return nil, fmt.Errorf("获取文件信息失败: %w", err)
}
fileSize := info.Size()
if fileSize == 0 {
return nil, fmt.Errorf("缓存文件为空:%s", cacheFile)
}
fmt.Printf("缓存文件大小:%d 字节\n", fileSize)
// 3. mmap映射文件(仅读取)
mmapData, err := syscall.Mmap(
int(f.Fd()),
0,
int(fileSize),
syscall.PROT_READ,
syscall.MAP_SHARED,
)
if err != nil {
return nil, fmt.Errorf("mmap映射文件失败: %w", err)
}
// 确保mmap内存最终会释放(无论解析是否成功)
defer func() {
if err := syscall.Munmap(mmapData); err != nil {
fmt.Printf("释放mmap内存失败: %v\n", err)
} else {
fmt.Printf("已释放mmap内存,长度:%d\n", len(mmapData))
}
}()
// 4. 深拷贝mmap数据到新的字节切片(核心修改)
dataCopy := make([]byte, len(mmapData))
copy(dataCopy, mmapData) // 将mmap内存的数据复制到新内存
// 5. 用拷贝后的数据解析(此时解析结果不再依赖mmap内存)
fmt.Printf("casting......v1\n")
usage := &ContainerUsage{}
usage.Info = v1.CastSpec(dataCopy)
// 6. 返回解析结果(仅保留Info,mmap内存会在defer中释放)
return usage, nil
}
func main() {
// 1. 解析命令行参数
var cacheFile string
flag.StringVar(&cacheFile, "f", "", "缓存文件路径(必填)")
flag.Parse()
// 2. 校验参数
if cacheFile == "" {
fmt.Printf("必须通过 -f 参数指定缓存文件路径")
flag.Usage() // 打印用法
os.Exit(1)
}
// 3. 核心逻辑:一次性读取并解析mmap文件
usage, err := readCacheFile(cacheFile)
if err != nil {
fmt.Printf("处理缓存文件失败: %v", err)
os.Exit(1)
}
for i, proc := range usage.Info.GetProcs() {
fmt.Printf(" %d proc hostpid [%d] pid [%d] util [%d] mem [%d]", i, proc.GetHostPid(), proc.GetPid(), proc.GetUtil(), proc.GetMemUtil())
}
}
|