112 lines
3.0 KiB
Go
112 lines
3.0 KiB
Go
|
// Copyright 2018 Google Inc.
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
// you may not use this file except in compliance with the License.
|
||
|
// You may obtain a copy of the License at
|
||
|
//
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
// See the License for the specific language governing permissions and
|
||
|
// limitations under the License.
|
||
|
|
||
|
package kernel
|
||
|
|
||
|
// Accounting, limits, timers.
|
||
|
|
||
|
import (
|
||
|
"gvisor.googlesource.com/gvisor/pkg/abi/linux"
|
||
|
ktime "gvisor.googlesource.com/gvisor/pkg/sentry/kernel/time"
|
||
|
"gvisor.googlesource.com/gvisor/pkg/sentry/limits"
|
||
|
"gvisor.googlesource.com/gvisor/pkg/sentry/usage"
|
||
|
)
|
||
|
|
||
|
// IOUsage returns the io usage of the thread.
|
||
|
func (t *Task) IOUsage() *usage.IO {
|
||
|
return t.ioUsage
|
||
|
}
|
||
|
|
||
|
// IOUsage returns the total io usage of all dead and live threads in the group.
|
||
|
func (tg *ThreadGroup) IOUsage() *usage.IO {
|
||
|
tg.pidns.owner.mu.RLock()
|
||
|
defer tg.pidns.owner.mu.RUnlock()
|
||
|
|
||
|
io := *tg.ioUsage
|
||
|
// Account for active tasks.
|
||
|
for t := tg.tasks.Front(); t != nil; t = t.Next() {
|
||
|
io.Accumulate(t.IOUsage())
|
||
|
}
|
||
|
return &io
|
||
|
}
|
||
|
|
||
|
// Name returns t's name.
|
||
|
func (t *Task) Name() string {
|
||
|
t.mu.Lock()
|
||
|
defer t.mu.Unlock()
|
||
|
return t.tc.Name
|
||
|
}
|
||
|
|
||
|
// SetName changes t's name.
|
||
|
func (t *Task) SetName(name string) {
|
||
|
t.mu.Lock()
|
||
|
defer t.mu.Unlock()
|
||
|
t.tc.Name = name
|
||
|
t.Debugf("Set thread name to %q", name)
|
||
|
}
|
||
|
|
||
|
// SetCPUTimer is used by setrlimit(RLIMIT_CPU) to enforce the hard and soft
|
||
|
// limits on CPU time used by this process.
|
||
|
func (tg *ThreadGroup) SetCPUTimer(l *limits.Limit) {
|
||
|
tg.Timer().applyCPULimits(*l)
|
||
|
}
|
||
|
|
||
|
// Limits implements context.Context.Limits.
|
||
|
func (t *Task) Limits() *limits.LimitSet {
|
||
|
return t.ThreadGroup().Limits()
|
||
|
}
|
||
|
|
||
|
// StartTime returns t's start time.
|
||
|
func (t *Task) StartTime() ktime.Time {
|
||
|
t.mu.Lock()
|
||
|
defer t.mu.Unlock()
|
||
|
return t.startTime
|
||
|
}
|
||
|
|
||
|
// MaxRSS returns the maximum resident set size of the task in bytes. which
|
||
|
// should be one of RUSAGE_SELF, RUSAGE_CHILDREN, RUSAGE_THREAD, or
|
||
|
// RUSAGE_BOTH. See getrusage(2) for documentation on the behavior of these
|
||
|
// flags.
|
||
|
func (t *Task) MaxRSS(which int32) uint64 {
|
||
|
t.tg.pidns.owner.mu.RLock()
|
||
|
defer t.tg.pidns.owner.mu.RUnlock()
|
||
|
|
||
|
switch which {
|
||
|
case linux.RUSAGE_SELF, linux.RUSAGE_THREAD:
|
||
|
// If there's an active mm we can use its value.
|
||
|
if mm := t.MemoryManager(); mm != nil {
|
||
|
if mmMaxRSS := mm.MaxResidentSetSize(); mmMaxRSS > t.tg.maxRSS {
|
||
|
return mmMaxRSS
|
||
|
}
|
||
|
}
|
||
|
return t.tg.maxRSS
|
||
|
case linux.RUSAGE_CHILDREN:
|
||
|
return t.tg.childMaxRSS
|
||
|
case linux.RUSAGE_BOTH:
|
||
|
maxRSS := t.tg.maxRSS
|
||
|
if maxRSS < t.tg.childMaxRSS {
|
||
|
maxRSS = t.tg.childMaxRSS
|
||
|
}
|
||
|
if mm := t.MemoryManager(); mm != nil {
|
||
|
if mmMaxRSS := mm.MaxResidentSetSize(); mmMaxRSS > maxRSS {
|
||
|
return mmMaxRSS
|
||
|
}
|
||
|
}
|
||
|
return maxRSS
|
||
|
default:
|
||
|
// We'll only get here if which is invalid.
|
||
|
return 0
|
||
|
}
|
||
|
}
|