process_linux.go 32 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262
  1. // +build linux
  2. package process
  3. import (
  4. "bufio"
  5. "bytes"
  6. "context"
  7. "encoding/json"
  8. "fmt"
  9. "io/ioutil"
  10. "math"
  11. "os"
  12. "path/filepath"
  13. "strconv"
  14. "strings"
  15. "github.com/shirou/gopsutil/cpu"
  16. "github.com/shirou/gopsutil/host"
  17. "github.com/shirou/gopsutil/internal/common"
  18. "github.com/shirou/gopsutil/net"
  19. "golang.org/x/sys/unix"
  20. )
  21. var PageSize = uint64(os.Getpagesize())
  22. const (
  23. PrioProcess = 0 // linux/resource.h
  24. ClockTicks = 100 // C.sysconf(C._SC_CLK_TCK)
  25. )
  26. // MemoryInfoExStat is different between OSes
  27. type MemoryInfoExStat struct {
  28. RSS uint64 `json:"rss"` // bytes
  29. VMS uint64 `json:"vms"` // bytes
  30. Shared uint64 `json:"shared"` // bytes
  31. Text uint64 `json:"text"` // bytes
  32. Lib uint64 `json:"lib"` // bytes
  33. Data uint64 `json:"data"` // bytes
  34. Dirty uint64 `json:"dirty"` // bytes
  35. }
  36. func (m MemoryInfoExStat) String() string {
  37. s, _ := json.Marshal(m)
  38. return string(s)
  39. }
  40. type MemoryMapsStat struct {
  41. Path string `json:"path"`
  42. Rss uint64 `json:"rss"`
  43. Size uint64 `json:"size"`
  44. Pss uint64 `json:"pss"`
  45. SharedClean uint64 `json:"sharedClean"`
  46. SharedDirty uint64 `json:"sharedDirty"`
  47. PrivateClean uint64 `json:"privateClean"`
  48. PrivateDirty uint64 `json:"privateDirty"`
  49. Referenced uint64 `json:"referenced"`
  50. Anonymous uint64 `json:"anonymous"`
  51. Swap uint64 `json:"swap"`
  52. }
  53. // String returns JSON value of the process.
  54. func (m MemoryMapsStat) String() string {
  55. s, _ := json.Marshal(m)
  56. return string(s)
  57. }
  58. // NewProcess creates a new Process instance, it only stores the pid and
  59. // checks that the process exists. Other method on Process can be used
  60. // to get more information about the process. An error will be returned
  61. // if the process does not exist.
  62. func NewProcess(pid int32) (*Process, error) {
  63. p := &Process{
  64. Pid: int32(pid),
  65. }
  66. file, err := os.Open(common.HostProc(strconv.Itoa(int(p.Pid))))
  67. defer file.Close()
  68. return p, err
  69. }
  70. // Ppid returns Parent Process ID of the process.
  71. func (p *Process) Ppid() (int32, error) {
  72. return p.PpidWithContext(context.Background())
  73. }
  74. func (p *Process) PpidWithContext(ctx context.Context) (int32, error) {
  75. _, ppid, _, _, _, _, err := p.fillFromStat()
  76. if err != nil {
  77. return -1, err
  78. }
  79. return ppid, nil
  80. }
  81. // Name returns name of the process.
  82. func (p *Process) Name() (string, error) {
  83. return p.NameWithContext(context.Background())
  84. }
  85. func (p *Process) NameWithContext(ctx context.Context) (string, error) {
  86. if p.name == "" {
  87. if err := p.fillFromStatus(); err != nil {
  88. return "", err
  89. }
  90. }
  91. return p.name, nil
  92. }
  93. // Tgid returns tgid, a Linux-synonym for user-space Pid
  94. func (p *Process) Tgid() (int32, error) {
  95. if p.tgid == 0 {
  96. if err := p.fillFromStatus(); err != nil {
  97. return 0, err
  98. }
  99. }
  100. return p.tgid, nil
  101. }
  102. // Exe returns executable path of the process.
  103. func (p *Process) Exe() (string, error) {
  104. return p.ExeWithContext(context.Background())
  105. }
  106. func (p *Process) ExeWithContext(ctx context.Context) (string, error) {
  107. return p.fillFromExe()
  108. }
  109. // Cmdline returns the command line arguments of the process as a string with
  110. // each argument separated by 0x20 ascii character.
  111. func (p *Process) Cmdline() (string, error) {
  112. return p.CmdlineWithContext(context.Background())
  113. }
  114. func (p *Process) CmdlineWithContext(ctx context.Context) (string, error) {
  115. return p.fillFromCmdline()
  116. }
  117. // CmdlineSlice returns the command line arguments of the process as a slice with each
  118. // element being an argument.
  119. func (p *Process) CmdlineSlice() ([]string, error) {
  120. return p.CmdlineSliceWithContext(context.Background())
  121. }
  122. func (p *Process) CmdlineSliceWithContext(ctx context.Context) ([]string, error) {
  123. return p.fillSliceFromCmdline()
  124. }
  125. // CreateTime returns created time of the process in milliseconds since the epoch, in UTC.
  126. func (p *Process) CreateTime() (int64, error) {
  127. return p.CreateTimeWithContext(context.Background())
  128. }
  129. func (p *Process) CreateTimeWithContext(ctx context.Context) (int64, error) {
  130. _, _, _, createTime, _, _, err := p.fillFromStat()
  131. if err != nil {
  132. return 0, err
  133. }
  134. return createTime, nil
  135. }
  136. // Cwd returns current working directory of the process.
  137. func (p *Process) Cwd() (string, error) {
  138. return p.CwdWithContext(context.Background())
  139. }
  140. func (p *Process) CwdWithContext(ctx context.Context) (string, error) {
  141. return p.fillFromCwd()
  142. }
  143. // Parent returns parent Process of the process.
  144. func (p *Process) Parent() (*Process, error) {
  145. return p.ParentWithContext(context.Background())
  146. }
  147. func (p *Process) ParentWithContext(ctx context.Context) (*Process, error) {
  148. err := p.fillFromStatus()
  149. if err != nil {
  150. return nil, err
  151. }
  152. if p.parent == 0 {
  153. return nil, fmt.Errorf("wrong number of parents")
  154. }
  155. return NewProcess(p.parent)
  156. }
  157. // Status returns the process status.
  158. // Return value could be one of these.
  159. // R: Running S: Sleep T: Stop I: Idle
  160. // Z: Zombie W: Wait L: Lock
  161. // The charactor is same within all supported platforms.
  162. func (p *Process) Status() (string, error) {
  163. return p.StatusWithContext(context.Background())
  164. }
  165. func (p *Process) StatusWithContext(ctx context.Context) (string, error) {
  166. err := p.fillFromStatus()
  167. if err != nil {
  168. return "", err
  169. }
  170. return p.status, nil
  171. }
  172. // Uids returns user ids of the process as a slice of the int
  173. func (p *Process) Uids() ([]int32, error) {
  174. return p.UidsWithContext(context.Background())
  175. }
  176. func (p *Process) UidsWithContext(ctx context.Context) ([]int32, error) {
  177. err := p.fillFromStatus()
  178. if err != nil {
  179. return []int32{}, err
  180. }
  181. return p.uids, nil
  182. }
  183. // Gids returns group ids of the process as a slice of the int
  184. func (p *Process) Gids() ([]int32, error) {
  185. return p.GidsWithContext(context.Background())
  186. }
  187. func (p *Process) GidsWithContext(ctx context.Context) ([]int32, error) {
  188. err := p.fillFromStatus()
  189. if err != nil {
  190. return []int32{}, err
  191. }
  192. return p.gids, nil
  193. }
  194. // Terminal returns a terminal which is associated with the process.
  195. func (p *Process) Terminal() (string, error) {
  196. return p.TerminalWithContext(context.Background())
  197. }
  198. func (p *Process) TerminalWithContext(ctx context.Context) (string, error) {
  199. t, _, _, _, _, _, err := p.fillFromStat()
  200. if err != nil {
  201. return "", err
  202. }
  203. termmap, err := getTerminalMap()
  204. if err != nil {
  205. return "", err
  206. }
  207. terminal := termmap[t]
  208. return terminal, nil
  209. }
  210. // Nice returns a nice value (priority).
  211. // Notice: gopsutil can not set nice value.
  212. func (p *Process) Nice() (int32, error) {
  213. return p.NiceWithContext(context.Background())
  214. }
  215. func (p *Process) NiceWithContext(ctx context.Context) (int32, error) {
  216. _, _, _, _, _, nice, err := p.fillFromStat()
  217. if err != nil {
  218. return 0, err
  219. }
  220. return nice, nil
  221. }
  222. // IOnice returns process I/O nice value (priority).
  223. func (p *Process) IOnice() (int32, error) {
  224. return p.IOniceWithContext(context.Background())
  225. }
  226. func (p *Process) IOniceWithContext(ctx context.Context) (int32, error) {
  227. return 0, common.ErrNotImplementedError
  228. }
  229. // Rlimit returns Resource Limits.
  230. func (p *Process) Rlimit() ([]RlimitStat, error) {
  231. return p.RlimitWithContext(context.Background())
  232. }
  233. func (p *Process) RlimitWithContext(ctx context.Context) ([]RlimitStat, error) {
  234. return p.RlimitUsage(false)
  235. }
  236. // RlimitUsage returns Resource Limits.
  237. // If gatherUsed is true, the currently used value will be gathered and added
  238. // to the resulting RlimitStat.
  239. func (p *Process) RlimitUsage(gatherUsed bool) ([]RlimitStat, error) {
  240. return p.RlimitUsageWithContext(context.Background(), gatherUsed)
  241. }
  242. func (p *Process) RlimitUsageWithContext(ctx context.Context, gatherUsed bool) ([]RlimitStat, error) {
  243. rlimits, err := p.fillFromLimits()
  244. if !gatherUsed || err != nil {
  245. return rlimits, err
  246. }
  247. _, _, _, _, rtprio, nice, err := p.fillFromStat()
  248. if err != nil {
  249. return nil, err
  250. }
  251. if err := p.fillFromStatus(); err != nil {
  252. return nil, err
  253. }
  254. for i := range rlimits {
  255. rs := &rlimits[i]
  256. switch rs.Resource {
  257. case RLIMIT_CPU:
  258. times, err := p.Times()
  259. if err != nil {
  260. return nil, err
  261. }
  262. rs.Used = uint64(times.User + times.System)
  263. case RLIMIT_DATA:
  264. rs.Used = uint64(p.memInfo.Data)
  265. case RLIMIT_STACK:
  266. rs.Used = uint64(p.memInfo.Stack)
  267. case RLIMIT_RSS:
  268. rs.Used = uint64(p.memInfo.RSS)
  269. case RLIMIT_NOFILE:
  270. n, err := p.NumFDs()
  271. if err != nil {
  272. return nil, err
  273. }
  274. rs.Used = uint64(n)
  275. case RLIMIT_MEMLOCK:
  276. rs.Used = uint64(p.memInfo.Locked)
  277. case RLIMIT_AS:
  278. rs.Used = uint64(p.memInfo.VMS)
  279. case RLIMIT_LOCKS:
  280. //TODO we can get the used value from /proc/$pid/locks. But linux doesn't enforce it, so not a high priority.
  281. case RLIMIT_SIGPENDING:
  282. rs.Used = p.sigInfo.PendingProcess
  283. case RLIMIT_NICE:
  284. // The rlimit for nice is a little unusual, in that 0 means the niceness cannot be decreased beyond the current value, but it can be increased.
  285. // So effectively: if rs.Soft == 0 { rs.Soft = rs.Used }
  286. rs.Used = uint64(nice)
  287. case RLIMIT_RTPRIO:
  288. rs.Used = uint64(rtprio)
  289. }
  290. }
  291. return rlimits, err
  292. }
  293. // IOCounters returns IO Counters.
  294. func (p *Process) IOCounters() (*IOCountersStat, error) {
  295. return p.IOCountersWithContext(context.Background())
  296. }
  297. func (p *Process) IOCountersWithContext(ctx context.Context) (*IOCountersStat, error) {
  298. return p.fillFromIO()
  299. }
  300. // NumCtxSwitches returns the number of the context switches of the process.
  301. func (p *Process) NumCtxSwitches() (*NumCtxSwitchesStat, error) {
  302. return p.NumCtxSwitchesWithContext(context.Background())
  303. }
  304. func (p *Process) NumCtxSwitchesWithContext(ctx context.Context) (*NumCtxSwitchesStat, error) {
  305. err := p.fillFromStatus()
  306. if err != nil {
  307. return nil, err
  308. }
  309. return p.numCtxSwitches, nil
  310. }
  311. // NumFDs returns the number of File Descriptors used by the process.
  312. func (p *Process) NumFDs() (int32, error) {
  313. return p.NumFDsWithContext(context.Background())
  314. }
  315. func (p *Process) NumFDsWithContext(ctx context.Context) (int32, error) {
  316. _, fnames, err := p.fillFromfdList()
  317. return int32(len(fnames)), err
  318. }
  319. // NumThreads returns the number of threads used by the process.
  320. func (p *Process) NumThreads() (int32, error) {
  321. return p.NumThreadsWithContext(context.Background())
  322. }
  323. func (p *Process) NumThreadsWithContext(ctx context.Context) (int32, error) {
  324. err := p.fillFromStatus()
  325. if err != nil {
  326. return 0, err
  327. }
  328. return p.numThreads, nil
  329. }
  330. func (p *Process) Threads() (map[int32]*cpu.TimesStat, error) {
  331. return p.ThreadsWithContext(context.Background())
  332. }
  333. func (p *Process) ThreadsWithContext(ctx context.Context) (map[int32]*cpu.TimesStat, error) {
  334. ret := make(map[int32]*cpu.TimesStat)
  335. taskPath := common.HostProc(strconv.Itoa(int(p.Pid)), "task")
  336. tids, err := readPidsFromDir(taskPath)
  337. if err != nil {
  338. return nil, err
  339. }
  340. for _, tid := range tids {
  341. _, _, cpuTimes, _, _, _, err := p.fillFromTIDStat(tid)
  342. if err != nil {
  343. return nil, err
  344. }
  345. ret[tid] = cpuTimes
  346. }
  347. return ret, nil
  348. }
  349. // Times returns CPU times of the process.
  350. func (p *Process) Times() (*cpu.TimesStat, error) {
  351. return p.TimesWithContext(context.Background())
  352. }
  353. func (p *Process) TimesWithContext(ctx context.Context) (*cpu.TimesStat, error) {
  354. _, _, cpuTimes, _, _, _, err := p.fillFromStat()
  355. if err != nil {
  356. return nil, err
  357. }
  358. return cpuTimes, nil
  359. }
  360. // CPUAffinity returns CPU affinity of the process.
  361. //
  362. // Notice: Not implemented yet.
  363. func (p *Process) CPUAffinity() ([]int32, error) {
  364. return p.CPUAffinityWithContext(context.Background())
  365. }
  366. func (p *Process) CPUAffinityWithContext(ctx context.Context) ([]int32, error) {
  367. return nil, common.ErrNotImplementedError
  368. }
  369. // MemoryInfo returns platform in-dependend memory information, such as RSS, VMS and Swap
  370. func (p *Process) MemoryInfo() (*MemoryInfoStat, error) {
  371. return p.MemoryInfoWithContext(context.Background())
  372. }
  373. func (p *Process) MemoryInfoWithContext(ctx context.Context) (*MemoryInfoStat, error) {
  374. meminfo, _, err := p.fillFromStatm()
  375. if err != nil {
  376. return nil, err
  377. }
  378. return meminfo, nil
  379. }
  380. // MemoryInfoEx returns platform dependend memory information.
  381. func (p *Process) MemoryInfoEx() (*MemoryInfoExStat, error) {
  382. return p.MemoryInfoExWithContext(context.Background())
  383. }
  384. func (p *Process) MemoryInfoExWithContext(ctx context.Context) (*MemoryInfoExStat, error) {
  385. _, memInfoEx, err := p.fillFromStatm()
  386. if err != nil {
  387. return nil, err
  388. }
  389. return memInfoEx, nil
  390. }
  391. // Children returns a slice of Process of the process.
  392. func (p *Process) Children() ([]*Process, error) {
  393. return p.ChildrenWithContext(context.Background())
  394. }
  395. func (p *Process) ChildrenWithContext(ctx context.Context) ([]*Process, error) {
  396. pids, err := common.CallPgrepWithContext(ctx, invoke, p.Pid)
  397. if err != nil {
  398. if pids == nil || len(pids) == 0 {
  399. return nil, ErrorNoChildren
  400. }
  401. return nil, err
  402. }
  403. ret := make([]*Process, 0, len(pids))
  404. for _, pid := range pids {
  405. np, err := NewProcess(pid)
  406. if err != nil {
  407. return nil, err
  408. }
  409. ret = append(ret, np)
  410. }
  411. return ret, nil
  412. }
  413. // OpenFiles returns a slice of OpenFilesStat opend by the process.
  414. // OpenFilesStat includes a file path and file descriptor.
  415. func (p *Process) OpenFiles() ([]OpenFilesStat, error) {
  416. return p.OpenFilesWithContext(context.Background())
  417. }
  418. func (p *Process) OpenFilesWithContext(ctx context.Context) ([]OpenFilesStat, error) {
  419. _, ofs, err := p.fillFromfd()
  420. if err != nil {
  421. return nil, err
  422. }
  423. ret := make([]OpenFilesStat, len(ofs))
  424. for i, o := range ofs {
  425. ret[i] = *o
  426. }
  427. return ret, nil
  428. }
  429. // Connections returns a slice of net.ConnectionStat used by the process.
  430. // This returns all kind of the connection. This measn TCP, UDP or UNIX.
  431. func (p *Process) Connections() ([]net.ConnectionStat, error) {
  432. return p.ConnectionsWithContext(context.Background())
  433. }
  434. func (p *Process) ConnectionsWithContext(ctx context.Context) ([]net.ConnectionStat, error) {
  435. return net.ConnectionsPid("all", p.Pid)
  436. }
  437. // NetIOCounters returns NetIOCounters of the process.
  438. func (p *Process) NetIOCounters(pernic bool) ([]net.IOCountersStat, error) {
  439. return p.NetIOCountersWithContext(context.Background(), pernic)
  440. }
  441. func (p *Process) NetIOCountersWithContext(ctx context.Context, pernic bool) ([]net.IOCountersStat, error) {
  442. filename := common.HostProc(strconv.Itoa(int(p.Pid)), "net/dev")
  443. return net.IOCountersByFile(pernic, filename)
  444. }
  445. // IsRunning returns whether the process is running or not.
  446. // Not implemented yet.
  447. func (p *Process) IsRunning() (bool, error) {
  448. return p.IsRunningWithContext(context.Background())
  449. }
  450. func (p *Process) IsRunningWithContext(ctx context.Context) (bool, error) {
  451. return true, common.ErrNotImplementedError
  452. }
  453. // MemoryMaps get memory maps from /proc/(pid)/smaps
  454. func (p *Process) MemoryMaps(grouped bool) (*[]MemoryMapsStat, error) {
  455. return p.MemoryMapsWithContext(context.Background(), grouped)
  456. }
  457. func (p *Process) MemoryMapsWithContext(ctx context.Context, grouped bool) (*[]MemoryMapsStat, error) {
  458. pid := p.Pid
  459. var ret []MemoryMapsStat
  460. smapsPath := common.HostProc(strconv.Itoa(int(pid)), "smaps")
  461. contents, err := ioutil.ReadFile(smapsPath)
  462. if err != nil {
  463. return nil, err
  464. }
  465. lines := strings.Split(string(contents), "\n")
  466. // function of parsing a block
  467. getBlock := func(first_line []string, block []string) (MemoryMapsStat, error) {
  468. m := MemoryMapsStat{}
  469. m.Path = first_line[len(first_line)-1]
  470. for _, line := range block {
  471. if strings.Contains(line, "VmFlags") {
  472. continue
  473. }
  474. field := strings.Split(line, ":")
  475. if len(field) < 2 {
  476. continue
  477. }
  478. v := strings.Trim(field[1], " kB") // remove last "kB"
  479. t, err := strconv.ParseUint(v, 10, 64)
  480. if err != nil {
  481. return m, err
  482. }
  483. switch field[0] {
  484. case "Size":
  485. m.Size = t
  486. case "Rss":
  487. m.Rss = t
  488. case "Pss":
  489. m.Pss = t
  490. case "Shared_Clean":
  491. m.SharedClean = t
  492. case "Shared_Dirty":
  493. m.SharedDirty = t
  494. case "Private_Clean":
  495. m.PrivateClean = t
  496. case "Private_Dirty":
  497. m.PrivateDirty = t
  498. case "Referenced":
  499. m.Referenced = t
  500. case "Anonymous":
  501. m.Anonymous = t
  502. case "Swap":
  503. m.Swap = t
  504. }
  505. }
  506. return m, nil
  507. }
  508. blocks := make([]string, 16)
  509. for _, line := range lines {
  510. field := strings.Split(line, " ")
  511. if strings.HasSuffix(field[0], ":") == false {
  512. // new block section
  513. if len(blocks) > 0 {
  514. g, err := getBlock(field, blocks)
  515. if err != nil {
  516. return &ret, err
  517. }
  518. ret = append(ret, g)
  519. }
  520. // starts new block
  521. blocks = make([]string, 16)
  522. } else {
  523. blocks = append(blocks, line)
  524. }
  525. }
  526. return &ret, nil
  527. }
  528. /**
  529. ** Internal functions
  530. **/
  531. func limitToInt(val string) (int32, error) {
  532. if val == "unlimited" {
  533. return math.MaxInt32, nil
  534. } else {
  535. res, err := strconv.ParseInt(val, 10, 32)
  536. if err != nil {
  537. return 0, err
  538. }
  539. return int32(res), nil
  540. }
  541. }
  542. // Get num_fds from /proc/(pid)/limits
  543. func (p *Process) fillFromLimits() ([]RlimitStat, error) {
  544. return p.fillFromLimitsWithContext(context.Background())
  545. }
  546. func (p *Process) fillFromLimitsWithContext(ctx context.Context) ([]RlimitStat, error) {
  547. pid := p.Pid
  548. limitsFile := common.HostProc(strconv.Itoa(int(pid)), "limits")
  549. d, err := os.Open(limitsFile)
  550. if err != nil {
  551. return nil, err
  552. }
  553. defer d.Close()
  554. var limitStats []RlimitStat
  555. limitsScanner := bufio.NewScanner(d)
  556. for limitsScanner.Scan() {
  557. var statItem RlimitStat
  558. str := strings.Fields(limitsScanner.Text())
  559. // Remove the header line
  560. if strings.Contains(str[len(str)-1], "Units") {
  561. continue
  562. }
  563. // Assert that last item is a Hard limit
  564. statItem.Hard, err = limitToInt(str[len(str)-1])
  565. if err != nil {
  566. // On error remove last item an try once again since it can be unit or header line
  567. str = str[:len(str)-1]
  568. statItem.Hard, err = limitToInt(str[len(str)-1])
  569. if err != nil {
  570. return nil, err
  571. }
  572. }
  573. // Remove last item from string
  574. str = str[:len(str)-1]
  575. //Now last item is a Soft limit
  576. statItem.Soft, err = limitToInt(str[len(str)-1])
  577. if err != nil {
  578. return nil, err
  579. }
  580. // Remove last item from string
  581. str = str[:len(str)-1]
  582. //The rest is a stats name
  583. resourceName := strings.Join(str, " ")
  584. switch resourceName {
  585. case "Max cpu time":
  586. statItem.Resource = RLIMIT_CPU
  587. case "Max file size":
  588. statItem.Resource = RLIMIT_FSIZE
  589. case "Max data size":
  590. statItem.Resource = RLIMIT_DATA
  591. case "Max stack size":
  592. statItem.Resource = RLIMIT_STACK
  593. case "Max core file size":
  594. statItem.Resource = RLIMIT_CORE
  595. case "Max resident set":
  596. statItem.Resource = RLIMIT_RSS
  597. case "Max processes":
  598. statItem.Resource = RLIMIT_NPROC
  599. case "Max open files":
  600. statItem.Resource = RLIMIT_NOFILE
  601. case "Max locked memory":
  602. statItem.Resource = RLIMIT_MEMLOCK
  603. case "Max address space":
  604. statItem.Resource = RLIMIT_AS
  605. case "Max file locks":
  606. statItem.Resource = RLIMIT_LOCKS
  607. case "Max pending signals":
  608. statItem.Resource = RLIMIT_SIGPENDING
  609. case "Max msgqueue size":
  610. statItem.Resource = RLIMIT_MSGQUEUE
  611. case "Max nice priority":
  612. statItem.Resource = RLIMIT_NICE
  613. case "Max realtime priority":
  614. statItem.Resource = RLIMIT_RTPRIO
  615. case "Max realtime timeout":
  616. statItem.Resource = RLIMIT_RTTIME
  617. default:
  618. continue
  619. }
  620. limitStats = append(limitStats, statItem)
  621. }
  622. if err := limitsScanner.Err(); err != nil {
  623. return nil, err
  624. }
  625. return limitStats, nil
  626. }
  627. // Get list of /proc/(pid)/fd files
  628. func (p *Process) fillFromfdList() (string, []string, error) {
  629. return p.fillFromfdListWithContext(context.Background())
  630. }
  631. func (p *Process) fillFromfdListWithContext(ctx context.Context) (string, []string, error) {
  632. pid := p.Pid
  633. statPath := common.HostProc(strconv.Itoa(int(pid)), "fd")
  634. d, err := os.Open(statPath)
  635. if err != nil {
  636. return statPath, []string{}, err
  637. }
  638. defer d.Close()
  639. fnames, err := d.Readdirnames(-1)
  640. return statPath, fnames, err
  641. }
  642. // Get num_fds from /proc/(pid)/fd
  643. func (p *Process) fillFromfd() (int32, []*OpenFilesStat, error) {
  644. return p.fillFromfdWithContext(context.Background())
  645. }
  646. func (p *Process) fillFromfdWithContext(ctx context.Context) (int32, []*OpenFilesStat, error) {
  647. statPath, fnames, err := p.fillFromfdList()
  648. if err != nil {
  649. return 0, nil, err
  650. }
  651. numFDs := int32(len(fnames))
  652. var openfiles []*OpenFilesStat
  653. for _, fd := range fnames {
  654. fpath := filepath.Join(statPath, fd)
  655. filepath, err := os.Readlink(fpath)
  656. if err != nil {
  657. continue
  658. }
  659. t, err := strconv.ParseUint(fd, 10, 64)
  660. if err != nil {
  661. return numFDs, openfiles, err
  662. }
  663. o := &OpenFilesStat{
  664. Path: filepath,
  665. Fd: t,
  666. }
  667. openfiles = append(openfiles, o)
  668. }
  669. return numFDs, openfiles, nil
  670. }
  671. // Get cwd from /proc/(pid)/cwd
  672. func (p *Process) fillFromCwd() (string, error) {
  673. return p.fillFromCwdWithContext(context.Background())
  674. }
  675. func (p *Process) fillFromCwdWithContext(ctx context.Context) (string, error) {
  676. pid := p.Pid
  677. cwdPath := common.HostProc(strconv.Itoa(int(pid)), "cwd")
  678. cwd, err := os.Readlink(cwdPath)
  679. if err != nil {
  680. return "", err
  681. }
  682. return string(cwd), nil
  683. }
  684. // Get exe from /proc/(pid)/exe
  685. func (p *Process) fillFromExe() (string, error) {
  686. return p.fillFromExeWithContext(context.Background())
  687. }
  688. func (p *Process) fillFromExeWithContext(ctx context.Context) (string, error) {
  689. pid := p.Pid
  690. exePath := common.HostProc(strconv.Itoa(int(pid)), "exe")
  691. exe, err := os.Readlink(exePath)
  692. if err != nil {
  693. return "", err
  694. }
  695. return string(exe), nil
  696. }
  697. // Get cmdline from /proc/(pid)/cmdline
  698. func (p *Process) fillFromCmdline() (string, error) {
  699. return p.fillFromCmdlineWithContext(context.Background())
  700. }
  701. func (p *Process) fillFromCmdlineWithContext(ctx context.Context) (string, error) {
  702. pid := p.Pid
  703. cmdPath := common.HostProc(strconv.Itoa(int(pid)), "cmdline")
  704. cmdline, err := ioutil.ReadFile(cmdPath)
  705. if err != nil {
  706. return "", err
  707. }
  708. ret := strings.FieldsFunc(string(cmdline), func(r rune) bool {
  709. if r == '\u0000' {
  710. return true
  711. }
  712. return false
  713. })
  714. return strings.Join(ret, " "), nil
  715. }
  716. func (p *Process) fillSliceFromCmdline() ([]string, error) {
  717. return p.fillSliceFromCmdlineWithContext(context.Background())
  718. }
  719. func (p *Process) fillSliceFromCmdlineWithContext(ctx context.Context) ([]string, error) {
  720. pid := p.Pid
  721. cmdPath := common.HostProc(strconv.Itoa(int(pid)), "cmdline")
  722. cmdline, err := ioutil.ReadFile(cmdPath)
  723. if err != nil {
  724. return nil, err
  725. }
  726. if len(cmdline) == 0 {
  727. return nil, nil
  728. }
  729. if cmdline[len(cmdline)-1] == 0 {
  730. cmdline = cmdline[:len(cmdline)-1]
  731. }
  732. parts := bytes.Split(cmdline, []byte{0})
  733. var strParts []string
  734. for _, p := range parts {
  735. strParts = append(strParts, string(p))
  736. }
  737. return strParts, nil
  738. }
  739. // Get IO status from /proc/(pid)/io
  740. func (p *Process) fillFromIO() (*IOCountersStat, error) {
  741. return p.fillFromIOWithContext(context.Background())
  742. }
  743. func (p *Process) fillFromIOWithContext(ctx context.Context) (*IOCountersStat, error) {
  744. pid := p.Pid
  745. ioPath := common.HostProc(strconv.Itoa(int(pid)), "io")
  746. ioline, err := ioutil.ReadFile(ioPath)
  747. if err != nil {
  748. return nil, err
  749. }
  750. lines := strings.Split(string(ioline), "\n")
  751. ret := &IOCountersStat{}
  752. for _, line := range lines {
  753. field := strings.Fields(line)
  754. if len(field) < 2 {
  755. continue
  756. }
  757. t, err := strconv.ParseUint(field[1], 10, 64)
  758. if err != nil {
  759. return nil, err
  760. }
  761. param := field[0]
  762. if strings.HasSuffix(param, ":") {
  763. param = param[:len(param)-1]
  764. }
  765. switch param {
  766. case "syscr":
  767. ret.ReadCount = t
  768. case "syscw":
  769. ret.WriteCount = t
  770. case "read_bytes":
  771. ret.ReadBytes = t
  772. case "write_bytes":
  773. ret.WriteBytes = t
  774. }
  775. }
  776. return ret, nil
  777. }
  778. // Get memory info from /proc/(pid)/statm
  779. func (p *Process) fillFromStatm() (*MemoryInfoStat, *MemoryInfoExStat, error) {
  780. return p.fillFromStatmWithContext(context.Background())
  781. }
  782. func (p *Process) fillFromStatmWithContext(ctx context.Context) (*MemoryInfoStat, *MemoryInfoExStat, error) {
  783. pid := p.Pid
  784. memPath := common.HostProc(strconv.Itoa(int(pid)), "statm")
  785. contents, err := ioutil.ReadFile(memPath)
  786. if err != nil {
  787. return nil, nil, err
  788. }
  789. fields := strings.Split(string(contents), " ")
  790. vms, err := strconv.ParseUint(fields[0], 10, 64)
  791. if err != nil {
  792. return nil, nil, err
  793. }
  794. rss, err := strconv.ParseUint(fields[1], 10, 64)
  795. if err != nil {
  796. return nil, nil, err
  797. }
  798. memInfo := &MemoryInfoStat{
  799. RSS: rss * PageSize,
  800. VMS: vms * PageSize,
  801. }
  802. shared, err := strconv.ParseUint(fields[2], 10, 64)
  803. if err != nil {
  804. return nil, nil, err
  805. }
  806. text, err := strconv.ParseUint(fields[3], 10, 64)
  807. if err != nil {
  808. return nil, nil, err
  809. }
  810. lib, err := strconv.ParseUint(fields[4], 10, 64)
  811. if err != nil {
  812. return nil, nil, err
  813. }
  814. dirty, err := strconv.ParseUint(fields[5], 10, 64)
  815. if err != nil {
  816. return nil, nil, err
  817. }
  818. memInfoEx := &MemoryInfoExStat{
  819. RSS: rss * PageSize,
  820. VMS: vms * PageSize,
  821. Shared: shared * PageSize,
  822. Text: text * PageSize,
  823. Lib: lib * PageSize,
  824. Dirty: dirty * PageSize,
  825. }
  826. return memInfo, memInfoEx, nil
  827. }
  828. // Get various status from /proc/(pid)/status
  829. func (p *Process) fillFromStatus() error {
  830. return p.fillFromStatusWithContext(context.Background())
  831. }
  832. func (p *Process) fillFromStatusWithContext(ctx context.Context) error {
  833. pid := p.Pid
  834. statPath := common.HostProc(strconv.Itoa(int(pid)), "status")
  835. contents, err := ioutil.ReadFile(statPath)
  836. if err != nil {
  837. return err
  838. }
  839. lines := strings.Split(string(contents), "\n")
  840. p.numCtxSwitches = &NumCtxSwitchesStat{}
  841. p.memInfo = &MemoryInfoStat{}
  842. p.sigInfo = &SignalInfoStat{}
  843. for _, line := range lines {
  844. tabParts := strings.SplitN(line, "\t", 2)
  845. if len(tabParts) < 2 {
  846. continue
  847. }
  848. value := tabParts[1]
  849. switch strings.TrimRight(tabParts[0], ":") {
  850. case "Name":
  851. p.name = strings.Trim(value, " \t")
  852. if len(p.name) >= 15 {
  853. cmdlineSlice, err := p.CmdlineSlice()
  854. if err != nil {
  855. return err
  856. }
  857. if len(cmdlineSlice) > 0 {
  858. extendedName := filepath.Base(cmdlineSlice[0])
  859. if strings.HasPrefix(extendedName, p.name) {
  860. p.name = extendedName
  861. } else {
  862. p.name = cmdlineSlice[0]
  863. }
  864. }
  865. }
  866. case "State":
  867. p.status = value[0:1]
  868. case "PPid", "Ppid":
  869. pval, err := strconv.ParseInt(value, 10, 32)
  870. if err != nil {
  871. return err
  872. }
  873. p.parent = int32(pval)
  874. case "Tgid":
  875. pval, err := strconv.ParseInt(value, 10, 32)
  876. if err != nil {
  877. return err
  878. }
  879. p.tgid = int32(pval)
  880. case "Uid":
  881. p.uids = make([]int32, 0, 4)
  882. for _, i := range strings.Split(value, "\t") {
  883. v, err := strconv.ParseInt(i, 10, 32)
  884. if err != nil {
  885. return err
  886. }
  887. p.uids = append(p.uids, int32(v))
  888. }
  889. case "Gid":
  890. p.gids = make([]int32, 0, 4)
  891. for _, i := range strings.Split(value, "\t") {
  892. v, err := strconv.ParseInt(i, 10, 32)
  893. if err != nil {
  894. return err
  895. }
  896. p.gids = append(p.gids, int32(v))
  897. }
  898. case "Threads":
  899. v, err := strconv.ParseInt(value, 10, 32)
  900. if err != nil {
  901. return err
  902. }
  903. p.numThreads = int32(v)
  904. case "voluntary_ctxt_switches":
  905. v, err := strconv.ParseInt(value, 10, 64)
  906. if err != nil {
  907. return err
  908. }
  909. p.numCtxSwitches.Voluntary = v
  910. case "nonvoluntary_ctxt_switches":
  911. v, err := strconv.ParseInt(value, 10, 64)
  912. if err != nil {
  913. return err
  914. }
  915. p.numCtxSwitches.Involuntary = v
  916. case "VmRSS":
  917. value := strings.Trim(value, " kB") // remove last "kB"
  918. v, err := strconv.ParseUint(value, 10, 64)
  919. if err != nil {
  920. return err
  921. }
  922. p.memInfo.RSS = v * 1024
  923. case "VmSize":
  924. value := strings.Trim(value, " kB") // remove last "kB"
  925. v, err := strconv.ParseUint(value, 10, 64)
  926. if err != nil {
  927. return err
  928. }
  929. p.memInfo.VMS = v * 1024
  930. case "VmSwap":
  931. value := strings.Trim(value, " kB") // remove last "kB"
  932. v, err := strconv.ParseUint(value, 10, 64)
  933. if err != nil {
  934. return err
  935. }
  936. p.memInfo.Swap = v * 1024
  937. case "VmData":
  938. value := strings.Trim(value, " kB") // remove last "kB"
  939. v, err := strconv.ParseUint(value, 10, 64)
  940. if err != nil {
  941. return err
  942. }
  943. p.memInfo.Data = v * 1024
  944. case "VmStk":
  945. value := strings.Trim(value, " kB") // remove last "kB"
  946. v, err := strconv.ParseUint(value, 10, 64)
  947. if err != nil {
  948. return err
  949. }
  950. p.memInfo.Stack = v * 1024
  951. case "VmLck":
  952. value := strings.Trim(value, " kB") // remove last "kB"
  953. v, err := strconv.ParseUint(value, 10, 64)
  954. if err != nil {
  955. return err
  956. }
  957. p.memInfo.Locked = v * 1024
  958. case "SigPnd":
  959. v, err := strconv.ParseUint(value, 16, 64)
  960. if err != nil {
  961. return err
  962. }
  963. p.sigInfo.PendingThread = v
  964. case "ShdPnd":
  965. v, err := strconv.ParseUint(value, 16, 64)
  966. if err != nil {
  967. return err
  968. }
  969. p.sigInfo.PendingProcess = v
  970. case "SigBlk":
  971. v, err := strconv.ParseUint(value, 16, 64)
  972. if err != nil {
  973. return err
  974. }
  975. p.sigInfo.Blocked = v
  976. case "SigIgn":
  977. v, err := strconv.ParseUint(value, 16, 64)
  978. if err != nil {
  979. return err
  980. }
  981. p.sigInfo.Ignored = v
  982. case "SigCgt":
  983. v, err := strconv.ParseUint(value, 16, 64)
  984. if err != nil {
  985. return err
  986. }
  987. p.sigInfo.Caught = v
  988. }
  989. }
  990. return nil
  991. }
  992. func (p *Process) fillFromTIDStat(tid int32) (uint64, int32, *cpu.TimesStat, int64, uint32, int32, error) {
  993. return p.fillFromTIDStatWithContext(context.Background(), tid)
  994. }
  995. func (p *Process) fillFromTIDStatWithContext(ctx context.Context, tid int32) (uint64, int32, *cpu.TimesStat, int64, uint32, int32, error) {
  996. pid := p.Pid
  997. var statPath string
  998. if tid == -1 {
  999. statPath = common.HostProc(strconv.Itoa(int(pid)), "stat")
  1000. } else {
  1001. statPath = common.HostProc(strconv.Itoa(int(pid)), "task", strconv.Itoa(int(tid)), "stat")
  1002. }
  1003. contents, err := ioutil.ReadFile(statPath)
  1004. if err != nil {
  1005. return 0, 0, nil, 0, 0, 0, err
  1006. }
  1007. fields := strings.Fields(string(contents))
  1008. i := 1
  1009. for !strings.HasSuffix(fields[i], ")") {
  1010. i++
  1011. }
  1012. terminal, err := strconv.ParseUint(fields[i+5], 10, 64)
  1013. if err != nil {
  1014. return 0, 0, nil, 0, 0, 0, err
  1015. }
  1016. ppid, err := strconv.ParseInt(fields[i+2], 10, 32)
  1017. if err != nil {
  1018. return 0, 0, nil, 0, 0, 0, err
  1019. }
  1020. utime, err := strconv.ParseFloat(fields[i+12], 64)
  1021. if err != nil {
  1022. return 0, 0, nil, 0, 0, 0, err
  1023. }
  1024. stime, err := strconv.ParseFloat(fields[i+13], 64)
  1025. if err != nil {
  1026. return 0, 0, nil, 0, 0, 0, err
  1027. }
  1028. cpuTimes := &cpu.TimesStat{
  1029. CPU: "cpu",
  1030. User: float64(utime / ClockTicks),
  1031. System: float64(stime / ClockTicks),
  1032. }
  1033. bootTime, _ := host.BootTime()
  1034. t, err := strconv.ParseUint(fields[i+20], 10, 64)
  1035. if err != nil {
  1036. return 0, 0, nil, 0, 0, 0, err
  1037. }
  1038. ctime := (t / uint64(ClockTicks)) + uint64(bootTime)
  1039. createTime := int64(ctime * 1000)
  1040. rtpriority, err := strconv.ParseInt(fields[i+16], 10, 32)
  1041. if err != nil {
  1042. return 0, 0, nil, 0, 0, 0, err
  1043. }
  1044. if rtpriority < 0 {
  1045. rtpriority = rtpriority*-1 - 1
  1046. } else {
  1047. rtpriority = 0
  1048. }
  1049. // p.Nice = mustParseInt32(fields[18])
  1050. // use syscall instead of parse Stat file
  1051. snice, _ := unix.Getpriority(PrioProcess, int(pid))
  1052. nice := int32(snice) // FIXME: is this true?
  1053. return terminal, int32(ppid), cpuTimes, createTime, uint32(rtpriority), nice, nil
  1054. }
  1055. func (p *Process) fillFromStat() (uint64, int32, *cpu.TimesStat, int64, uint32, int32, error) {
  1056. return p.fillFromStatWithContext(context.Background())
  1057. }
  1058. func (p *Process) fillFromStatWithContext(ctx context.Context) (uint64, int32, *cpu.TimesStat, int64, uint32, int32, error) {
  1059. return p.fillFromTIDStat(-1)
  1060. }
  1061. // Pids returns a slice of process ID list which are running now.
  1062. func Pids() ([]int32, error) {
  1063. return PidsWithContext(context.Background())
  1064. }
  1065. func PidsWithContext(ctx context.Context) ([]int32, error) {
  1066. return readPidsFromDir(common.HostProc())
  1067. }
  1068. // Process returns a slice of pointers to Process structs for all
  1069. // currently running processes.
  1070. func Processes() ([]*Process, error) {
  1071. return ProcessesWithContext(context.Background())
  1072. }
  1073. func ProcessesWithContext(ctx context.Context) ([]*Process, error) {
  1074. out := []*Process{}
  1075. pids, err := Pids()
  1076. if err != nil {
  1077. return out, err
  1078. }
  1079. for _, pid := range pids {
  1080. p, err := NewProcess(pid)
  1081. if err != nil {
  1082. continue
  1083. }
  1084. out = append(out, p)
  1085. }
  1086. return out, nil
  1087. }
  1088. func readPidsFromDir(path string) ([]int32, error) {
  1089. var ret []int32
  1090. d, err := os.Open(path)
  1091. if err != nil {
  1092. return nil, err
  1093. }
  1094. defer d.Close()
  1095. fnames, err := d.Readdirnames(-1)
  1096. if err != nil {
  1097. return nil, err
  1098. }
  1099. for _, fname := range fnames {
  1100. pid, err := strconv.ParseInt(fname, 10, 32)
  1101. if err != nil {
  1102. // if not numeric name, just skip
  1103. continue
  1104. }
  1105. ret = append(ret, int32(pid))
  1106. }
  1107. return ret, nil
  1108. }