process_freebsd.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. // +build freebsd
  2. package process
  3. import (
  4. "bytes"
  5. "context"
  6. "encoding/binary"
  7. "strings"
  8. cpu "github.com/shirou/gopsutil/cpu"
  9. "github.com/shirou/gopsutil/internal/common"
  10. net "github.com/shirou/gopsutil/net"
  11. "golang.org/x/sys/unix"
  12. )
  13. // MemoryInfoExStat is different between OSes
  14. type MemoryInfoExStat struct {
  15. }
  16. type MemoryMapsStat struct {
  17. }
  18. func Pids() ([]int32, error) {
  19. return PidsWithContext(context.Background())
  20. }
  21. func PidsWithContext(ctx context.Context) ([]int32, error) {
  22. var ret []int32
  23. procs, err := Processes()
  24. if err != nil {
  25. return ret, nil
  26. }
  27. for _, p := range procs {
  28. ret = append(ret, p.Pid)
  29. }
  30. return ret, nil
  31. }
  32. func (p *Process) Ppid() (int32, error) {
  33. return p.PpidWithContext(context.Background())
  34. }
  35. func (p *Process) PpidWithContext(ctx context.Context) (int32, error) {
  36. k, err := p.getKProc()
  37. if err != nil {
  38. return 0, err
  39. }
  40. return k.Ppid, nil
  41. }
  42. func (p *Process) Name() (string, error) {
  43. return p.NameWithContext(context.Background())
  44. }
  45. func (p *Process) NameWithContext(ctx context.Context) (string, error) {
  46. k, err := p.getKProc()
  47. if err != nil {
  48. return "", err
  49. }
  50. return common.IntToString(k.Comm[:]), nil
  51. }
  52. func (p *Process) Tgid() (int32, error) {
  53. return 0, common.ErrNotImplementedError
  54. }
  55. func (p *Process) Exe() (string, error) {
  56. return p.ExeWithContext(context.Background())
  57. }
  58. func (p *Process) ExeWithContext(ctx context.Context) (string, error) {
  59. return "", common.ErrNotImplementedError
  60. }
  61. func (p *Process) Cmdline() (string, error) {
  62. return p.CmdlineWithContext(context.Background())
  63. }
  64. func (p *Process) CmdlineWithContext(ctx context.Context) (string, error) {
  65. mib := []int32{CTLKern, KernProc, KernProcArgs, p.Pid}
  66. buf, _, err := common.CallSyscall(mib)
  67. if err != nil {
  68. return "", err
  69. }
  70. ret := strings.FieldsFunc(string(buf), func(r rune) bool {
  71. if r == '\u0000' {
  72. return true
  73. }
  74. return false
  75. })
  76. return strings.Join(ret, " "), nil
  77. }
  78. func (p *Process) CmdlineSlice() ([]string, error) {
  79. return p.CmdlineSliceWithContext(context.Background())
  80. }
  81. func (p *Process) CmdlineSliceWithContext(ctx context.Context) ([]string, error) {
  82. mib := []int32{CTLKern, KernProc, KernProcArgs, p.Pid}
  83. buf, _, err := common.CallSyscall(mib)
  84. if err != nil {
  85. return nil, err
  86. }
  87. if len(buf) == 0 {
  88. return nil, nil
  89. }
  90. if buf[len(buf)-1] == 0 {
  91. buf = buf[:len(buf)-1]
  92. }
  93. parts := bytes.Split(buf, []byte{0})
  94. var strParts []string
  95. for _, p := range parts {
  96. strParts = append(strParts, string(p))
  97. }
  98. return strParts, nil
  99. }
  100. func (p *Process) CreateTime() (int64, error) {
  101. return p.CreateTimeWithContext(context.Background())
  102. }
  103. func (p *Process) CreateTimeWithContext(ctx context.Context) (int64, error) {
  104. return 0, common.ErrNotImplementedError
  105. }
  106. func (p *Process) Cwd() (string, error) {
  107. return p.CwdWithContext(context.Background())
  108. }
  109. func (p *Process) CwdWithContext(ctx context.Context) (string, error) {
  110. return "", common.ErrNotImplementedError
  111. }
  112. func (p *Process) Parent() (*Process, error) {
  113. return p.ParentWithContext(context.Background())
  114. }
  115. func (p *Process) ParentWithContext(ctx context.Context) (*Process, error) {
  116. return p, common.ErrNotImplementedError
  117. }
  118. func (p *Process) Status() (string, error) {
  119. return p.StatusWithContext(context.Background())
  120. }
  121. func (p *Process) StatusWithContext(ctx context.Context) (string, error) {
  122. k, err := p.getKProc()
  123. if err != nil {
  124. return "", err
  125. }
  126. var s string
  127. switch k.Stat {
  128. case SIDL:
  129. s = "I"
  130. case SRUN:
  131. s = "R"
  132. case SSLEEP:
  133. s = "S"
  134. case SSTOP:
  135. s = "T"
  136. case SZOMB:
  137. s = "Z"
  138. case SWAIT:
  139. s = "W"
  140. case SLOCK:
  141. s = "L"
  142. }
  143. return s, nil
  144. }
  145. func (p *Process) Uids() ([]int32, error) {
  146. return p.UidsWithContext(context.Background())
  147. }
  148. func (p *Process) UidsWithContext(ctx context.Context) ([]int32, error) {
  149. k, err := p.getKProc()
  150. if err != nil {
  151. return nil, err
  152. }
  153. uids := make([]int32, 0, 3)
  154. uids = append(uids, int32(k.Ruid), int32(k.Uid), int32(k.Svuid))
  155. return uids, nil
  156. }
  157. func (p *Process) Gids() ([]int32, error) {
  158. return p.GidsWithContext(context.Background())
  159. }
  160. func (p *Process) GidsWithContext(ctx context.Context) ([]int32, error) {
  161. k, err := p.getKProc()
  162. if err != nil {
  163. return nil, err
  164. }
  165. gids := make([]int32, 0, 3)
  166. gids = append(gids, int32(k.Rgid), int32(k.Ngroups), int32(k.Svgid))
  167. return gids, nil
  168. }
  169. func (p *Process) Terminal() (string, error) {
  170. return p.TerminalWithContext(context.Background())
  171. }
  172. func (p *Process) TerminalWithContext(ctx context.Context) (string, error) {
  173. k, err := p.getKProc()
  174. if err != nil {
  175. return "", err
  176. }
  177. ttyNr := uint64(k.Tdev)
  178. termmap, err := getTerminalMap()
  179. if err != nil {
  180. return "", err
  181. }
  182. return termmap[ttyNr], nil
  183. }
  184. func (p *Process) Nice() (int32, error) {
  185. return p.NiceWithContext(context.Background())
  186. }
  187. func (p *Process) NiceWithContext(ctx context.Context) (int32, error) {
  188. k, err := p.getKProc()
  189. if err != nil {
  190. return 0, err
  191. }
  192. return int32(k.Nice), nil
  193. }
  194. func (p *Process) IOnice() (int32, error) {
  195. return p.IOniceWithContext(context.Background())
  196. }
  197. func (p *Process) IOniceWithContext(ctx context.Context) (int32, error) {
  198. return 0, common.ErrNotImplementedError
  199. }
  200. func (p *Process) Rlimit() ([]RlimitStat, error) {
  201. return p.RlimitWithContext(context.Background())
  202. }
  203. func (p *Process) RlimitWithContext(ctx context.Context) ([]RlimitStat, error) {
  204. var rlimit []RlimitStat
  205. return rlimit, common.ErrNotImplementedError
  206. }
  207. func (p *Process) RlimitUsage(gatherUsed bool) ([]RlimitStat, error) {
  208. return p.RlimitUsageWithContext(context.Background(), gatherUsed)
  209. }
  210. func (p *Process) RlimitUsageWithContext(ctx context.Context, gatherUsed bool) ([]RlimitStat, error) {
  211. var rlimit []RlimitStat
  212. return rlimit, common.ErrNotImplementedError
  213. }
  214. func (p *Process) IOCounters() (*IOCountersStat, error) {
  215. return p.IOCountersWithContext(context.Background())
  216. }
  217. func (p *Process) IOCountersWithContext(ctx context.Context) (*IOCountersStat, error) {
  218. k, err := p.getKProc()
  219. if err != nil {
  220. return nil, err
  221. }
  222. return &IOCountersStat{
  223. ReadCount: uint64(k.Rusage.Inblock),
  224. WriteCount: uint64(k.Rusage.Oublock),
  225. }, nil
  226. }
  227. func (p *Process) NumCtxSwitches() (*NumCtxSwitchesStat, error) {
  228. return p.NumCtxSwitchesWithContext(context.Background())
  229. }
  230. func (p *Process) NumCtxSwitchesWithContext(ctx context.Context) (*NumCtxSwitchesStat, error) {
  231. return nil, common.ErrNotImplementedError
  232. }
  233. func (p *Process) NumFDs() (int32, error) {
  234. return p.NumFDsWithContext(context.Background())
  235. }
  236. func (p *Process) NumFDsWithContext(ctx context.Context) (int32, error) {
  237. return 0, common.ErrNotImplementedError
  238. }
  239. func (p *Process) NumThreads() (int32, error) {
  240. return p.NumThreadsWithContext(context.Background())
  241. }
  242. func (p *Process) NumThreadsWithContext(ctx context.Context) (int32, error) {
  243. k, err := p.getKProc()
  244. if err != nil {
  245. return 0, err
  246. }
  247. return k.Numthreads, nil
  248. }
  249. func (p *Process) Threads() (map[int32]*cpu.TimesStat, error) {
  250. return p.ThreadsWithContext(context.Background())
  251. }
  252. func (p *Process) ThreadsWithContext(ctx context.Context) (map[int32]*cpu.TimesStat, error) {
  253. ret := make(map[int32]*cpu.TimesStat)
  254. return ret, common.ErrNotImplementedError
  255. }
  256. func (p *Process) Times() (*cpu.TimesStat, error) {
  257. return p.TimesWithContext(context.Background())
  258. }
  259. func (p *Process) TimesWithContext(ctx context.Context) (*cpu.TimesStat, error) {
  260. k, err := p.getKProc()
  261. if err != nil {
  262. return nil, err
  263. }
  264. return &cpu.TimesStat{
  265. CPU: "cpu",
  266. User: float64(k.Rusage.Utime.Sec) + float64(k.Rusage.Utime.Usec)/1000000,
  267. System: float64(k.Rusage.Stime.Sec) + float64(k.Rusage.Stime.Usec)/1000000,
  268. }, nil
  269. }
  270. func (p *Process) CPUAffinity() ([]int32, error) {
  271. return p.CPUAffinityWithContext(context.Background())
  272. }
  273. func (p *Process) CPUAffinityWithContext(ctx context.Context) ([]int32, error) {
  274. return nil, common.ErrNotImplementedError
  275. }
  276. func (p *Process) MemoryInfo() (*MemoryInfoStat, error) {
  277. return p.MemoryInfoWithContext(context.Background())
  278. }
  279. func (p *Process) MemoryInfoWithContext(ctx context.Context) (*MemoryInfoStat, error) {
  280. k, err := p.getKProc()
  281. if err != nil {
  282. return nil, err
  283. }
  284. v, err := unix.Sysctl("vm.stats.vm.v_page_size")
  285. if err != nil {
  286. return nil, err
  287. }
  288. pageSize := common.LittleEndian.Uint16([]byte(v))
  289. return &MemoryInfoStat{
  290. RSS: uint64(k.Rssize) * uint64(pageSize),
  291. VMS: uint64(k.Size),
  292. }, nil
  293. }
  294. func (p *Process) MemoryInfoEx() (*MemoryInfoExStat, error) {
  295. return p.MemoryInfoExWithContext(context.Background())
  296. }
  297. func (p *Process) MemoryInfoExWithContext(ctx context.Context) (*MemoryInfoExStat, error) {
  298. return nil, common.ErrNotImplementedError
  299. }
  300. func (p *Process) Children() ([]*Process, error) {
  301. return p.ChildrenWithContext(context.Background())
  302. }
  303. func (p *Process) ChildrenWithContext(ctx context.Context) ([]*Process, error) {
  304. pids, err := common.CallPgrepWithContext(ctx, invoke, p.Pid)
  305. if err != nil {
  306. return nil, err
  307. }
  308. ret := make([]*Process, 0, len(pids))
  309. for _, pid := range pids {
  310. np, err := NewProcess(pid)
  311. if err != nil {
  312. return nil, err
  313. }
  314. ret = append(ret, np)
  315. }
  316. return ret, nil
  317. }
  318. func (p *Process) OpenFiles() ([]OpenFilesStat, error) {
  319. return p.OpenFilesWithContext(context.Background())
  320. }
  321. func (p *Process) OpenFilesWithContext(ctx context.Context) ([]OpenFilesStat, error) {
  322. return nil, common.ErrNotImplementedError
  323. }
  324. func (p *Process) Connections() ([]net.ConnectionStat, error) {
  325. return p.ConnectionsWithContext(context.Background())
  326. }
  327. func (p *Process) ConnectionsWithContext(ctx context.Context) ([]net.ConnectionStat, error) {
  328. return nil, common.ErrNotImplementedError
  329. }
  330. func (p *Process) NetIOCounters(pernic bool) ([]net.IOCountersStat, error) {
  331. return p.NetIOCountersWithContext(context.Background(), pernic)
  332. }
  333. func (p *Process) NetIOCountersWithContext(ctx context.Context, pernic bool) ([]net.IOCountersStat, error) {
  334. return nil, common.ErrNotImplementedError
  335. }
  336. func (p *Process) IsRunning() (bool, error) {
  337. return p.IsRunningWithContext(context.Background())
  338. }
  339. func (p *Process) IsRunningWithContext(ctx context.Context) (bool, error) {
  340. return true, common.ErrNotImplementedError
  341. }
  342. func (p *Process) MemoryMaps(grouped bool) (*[]MemoryMapsStat, error) {
  343. return p.MemoryMapsWithContext(context.Background(), grouped)
  344. }
  345. func (p *Process) MemoryMapsWithContext(ctx context.Context, grouped bool) (*[]MemoryMapsStat, error) {
  346. var ret []MemoryMapsStat
  347. return &ret, common.ErrNotImplementedError
  348. }
  349. func Processes() ([]*Process, error) {
  350. return ProcessesWithContext(context.Background())
  351. }
  352. func ProcessesWithContext(ctx context.Context) ([]*Process, error) {
  353. results := []*Process{}
  354. mib := []int32{CTLKern, KernProc, KernProcProc, 0}
  355. buf, length, err := common.CallSyscall(mib)
  356. if err != nil {
  357. return results, err
  358. }
  359. // get kinfo_proc size
  360. count := int(length / uint64(sizeOfKinfoProc))
  361. // parse buf to procs
  362. for i := 0; i < count; i++ {
  363. b := buf[i*sizeOfKinfoProc : (i+1)*sizeOfKinfoProc]
  364. k, err := parseKinfoProc(b)
  365. if err != nil {
  366. continue
  367. }
  368. p, err := NewProcess(int32(k.Pid))
  369. if err != nil {
  370. continue
  371. }
  372. results = append(results, p)
  373. }
  374. return results, nil
  375. }
  376. func parseKinfoProc(buf []byte) (KinfoProc, error) {
  377. var k KinfoProc
  378. br := bytes.NewReader(buf)
  379. err := common.Read(br, binary.LittleEndian, &k)
  380. return k, err
  381. }
  382. func (p *Process) getKProc() (*KinfoProc, error) {
  383. return p.getKProcWithContext(context.Background())
  384. }
  385. func (p *Process) getKProcWithContext(ctx context.Context) (*KinfoProc, error) {
  386. mib := []int32{CTLKern, KernProc, KernProcPID, p.Pid}
  387. buf, length, err := common.CallSyscall(mib)
  388. if err != nil {
  389. return nil, err
  390. }
  391. if length != sizeOfKinfoProc {
  392. return nil, err
  393. }
  394. k, err := parseKinfoProc(buf)
  395. if err != nil {
  396. return nil, err
  397. }
  398. return &k, nil
  399. }
  400. func NewProcess(pid int32) (*Process, error) {
  401. p := &Process{Pid: pid}
  402. return p, nil
  403. }