2019-01-12 04:58:27 +00:00
|
|
|
package hcs
|
|
|
|
|
|
|
|
import (
|
2019-09-30 23:25:17 +00:00
|
|
|
"context"
|
2019-01-12 04:58:27 +00:00
|
|
|
"encoding/json"
|
2019-09-30 23:25:17 +00:00
|
|
|
"errors"
|
2019-01-12 04:58:27 +00:00
|
|
|
"os"
|
|
|
|
"strconv"
|
2019-09-30 23:25:17 +00:00
|
|
|
"strings"
|
2019-01-12 04:58:27 +00:00
|
|
|
"sync"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
"github.com/Microsoft/hcsshim/internal/cow"
|
|
|
|
"github.com/Microsoft/hcsshim/internal/log"
|
|
|
|
"github.com/Microsoft/hcsshim/internal/oc"
|
2019-01-12 04:58:27 +00:00
|
|
|
"github.com/Microsoft/hcsshim/internal/schema1"
|
|
|
|
"github.com/Microsoft/hcsshim/internal/timeout"
|
2019-09-30 23:25:17 +00:00
|
|
|
"github.com/Microsoft/hcsshim/internal/vmcompute"
|
|
|
|
"go.opencensus.io/trace"
|
2019-01-12 04:58:27 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// currentContainerStarts is used to limit the number of concurrent container
|
|
|
|
// starts.
|
|
|
|
var currentContainerStarts containerStarts
|
|
|
|
|
|
|
|
type containerStarts struct {
|
|
|
|
maxParallel int
|
|
|
|
inProgress int
|
|
|
|
sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
mpsS := os.Getenv("HCSSHIM_MAX_PARALLEL_START")
|
|
|
|
if len(mpsS) > 0 {
|
|
|
|
mpsI, err := strconv.Atoi(mpsS)
|
|
|
|
if err != nil || mpsI < 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
currentContainerStarts.maxParallel = mpsI
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type System struct {
|
|
|
|
handleLock sync.RWMutex
|
2019-09-30 23:25:17 +00:00
|
|
|
handle vmcompute.HcsSystem
|
2019-01-12 04:58:27 +00:00
|
|
|
id string
|
|
|
|
callbackNumber uintptr
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
closedWaitOnce sync.Once
|
|
|
|
waitBlock chan struct{}
|
|
|
|
waitError error
|
|
|
|
exitError error
|
|
|
|
|
|
|
|
os, typ string
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func newSystem(id string) *System {
|
|
|
|
return &System{
|
2019-09-30 23:25:17 +00:00
|
|
|
id: id,
|
|
|
|
waitBlock: make(chan struct{}),
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateComputeSystem creates a new compute system with the given configuration but does not start it.
|
2019-09-30 23:25:17 +00:00
|
|
|
func CreateComputeSystem(ctx context.Context, id string, hcsDocumentInterface interface{}) (_ *System, err error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
operation := "hcsshim::CreateComputeSystem"
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// hcsCreateComputeSystemContext is an async operation. Start the outer span
|
|
|
|
// here to measure the full create time.
|
|
|
|
ctx, span := trace.StartSpan(ctx, operation)
|
|
|
|
defer span.End()
|
|
|
|
defer func() { oc.SetSpanStatus(span, err) }()
|
|
|
|
span.AddAttributes(trace.StringAttribute("cid", id))
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem := newSystem(id)
|
|
|
|
|
|
|
|
hcsDocumentB, err := json.Marshal(hcsDocumentInterface)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
hcsDocument := string(hcsDocumentB)
|
|
|
|
|
|
|
|
var (
|
2019-07-10 00:29:38 +00:00
|
|
|
identity syscall.Handle
|
2019-09-30 23:25:17 +00:00
|
|
|
resultJSON string
|
2019-07-10 00:29:38 +00:00
|
|
|
createError error
|
2019-01-12 04:58:27 +00:00
|
|
|
)
|
2019-09-30 23:25:17 +00:00
|
|
|
computeSystem.handle, resultJSON, createError = vmcompute.HcsCreateComputeSystem(ctx, id, hcsDocument, identity)
|
2019-01-12 04:58:27 +00:00
|
|
|
if createError == nil || IsPending(createError) {
|
2019-09-30 23:25:17 +00:00
|
|
|
defer func() {
|
|
|
|
if err != nil {
|
|
|
|
computeSystem.Close()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
if err = computeSystem.registerCallback(ctx); err != nil {
|
2019-01-12 04:58:27 +00:00
|
|
|
// Terminate the compute system if it still exists. We're okay to
|
|
|
|
// ignore a failure here.
|
2019-09-30 23:25:17 +00:00
|
|
|
computeSystem.Terminate(ctx)
|
2019-01-12 04:58:27 +00:00
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
events, err := processAsyncHcsResult(ctx, createError, resultJSON, computeSystem.callbackNumber, hcsNotificationSystemCreateCompleted, &timeout.SystemCreate)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
if err == ErrTimeout {
|
|
|
|
// Terminate the compute system if it still exists. We're okay to
|
|
|
|
// ignore a failure here.
|
2019-09-30 23:25:17 +00:00
|
|
|
computeSystem.Terminate(ctx)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
return nil, makeSystemError(computeSystem, operation, hcsDocument, err, events)
|
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
go computeSystem.waitBackground()
|
|
|
|
if err = computeSystem.getCachedProperties(ctx); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
return computeSystem, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// OpenComputeSystem opens an existing compute system by ID.
|
2019-09-30 23:25:17 +00:00
|
|
|
func OpenComputeSystem(ctx context.Context, id string) (*System, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
operation := "hcsshim::OpenComputeSystem"
|
|
|
|
|
|
|
|
computeSystem := newSystem(id)
|
2019-09-30 23:25:17 +00:00
|
|
|
handle, resultJSON, err := vmcompute.HcsOpenComputeSystem(ctx, id)
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
|
|
|
if err != nil {
|
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, events)
|
|
|
|
}
|
|
|
|
computeSystem.handle = handle
|
2019-07-10 00:29:38 +00:00
|
|
|
defer func() {
|
2019-09-30 23:25:17 +00:00
|
|
|
if err != nil {
|
|
|
|
computeSystem.Close()
|
2019-07-10 00:29:38 +00:00
|
|
|
}
|
|
|
|
}()
|
2019-09-30 23:25:17 +00:00
|
|
|
if err = computeSystem.registerCallback(ctx); err != nil {
|
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
|
|
|
}
|
|
|
|
go computeSystem.waitBackground()
|
|
|
|
if err = computeSystem.getCachedProperties(ctx); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return computeSystem, nil
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) getCachedProperties(ctx context.Context) error {
|
|
|
|
props, err := computeSystem.Properties(ctx)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return err
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
computeSystem.typ = strings.ToLower(props.SystemType)
|
|
|
|
computeSystem.os = strings.ToLower(props.RuntimeOSType)
|
|
|
|
if computeSystem.os == "" && computeSystem.typ == "container" {
|
|
|
|
// Pre-RS5 HCS did not return the OS, but it only supported containers
|
|
|
|
// that ran Windows.
|
|
|
|
computeSystem.os = "windows"
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// OS returns the operating system of the compute system, "linux" or "windows".
|
|
|
|
func (computeSystem *System) OS() string {
|
|
|
|
return computeSystem.os
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsOCI returns whether processes in the compute system should be created via
|
|
|
|
// OCI.
|
|
|
|
func (computeSystem *System) IsOCI() bool {
|
|
|
|
return computeSystem.os == "linux" && computeSystem.typ == "container"
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetComputeSystems gets a list of the compute systems on the system that match the query
|
2019-09-30 23:25:17 +00:00
|
|
|
func GetComputeSystems(ctx context.Context, q schema1.ComputeSystemQuery) ([]schema1.ContainerProperties, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
operation := "hcsshim::GetComputeSystems"
|
|
|
|
|
|
|
|
queryb, err := json.Marshal(q)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
computeSystemsJSON, resultJSON, err := vmcompute.HcsEnumerateComputeSystems(ctx, string(queryb))
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, &HcsError{Op: operation, Err: err, Events: events}
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
if computeSystemsJSON == "" {
|
2019-01-12 04:58:27 +00:00
|
|
|
return nil, ErrUnexpectedValue
|
|
|
|
}
|
|
|
|
computeSystems := []schema1.ContainerProperties{}
|
2019-09-30 23:25:17 +00:00
|
|
|
if err = json.Unmarshal([]byte(computeSystemsJSON), &computeSystems); err != nil {
|
2019-01-12 04:58:27 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return computeSystems, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start synchronously starts the computeSystem.
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) Start(ctx context.Context) (err error) {
|
|
|
|
operation := "hcsshim::System::Start"
|
|
|
|
|
|
|
|
// hcsStartComputeSystemContext is an async operation. Start the outer span
|
|
|
|
// here to measure the full start time.
|
|
|
|
ctx, span := trace.StartSpan(ctx, operation)
|
|
|
|
defer span.End()
|
|
|
|
defer func() { oc.SetSpanStatus(span, err) }()
|
|
|
|
span.AddAttributes(trace.StringAttribute("cid", computeSystem.id))
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", ErrAlreadyClosed, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is a very simple backoff-retry loop to limit the number
|
|
|
|
// of parallel container starts if environment variable
|
|
|
|
// HCSSHIM_MAX_PARALLEL_START is set to a positive integer.
|
|
|
|
// It should generally only be used as a workaround to various
|
|
|
|
// platform issues that exist between RS1 and RS4 as of Aug 2018
|
|
|
|
if currentContainerStarts.maxParallel > 0 {
|
|
|
|
for {
|
|
|
|
currentContainerStarts.Lock()
|
|
|
|
if currentContainerStarts.inProgress < currentContainerStarts.maxParallel {
|
|
|
|
currentContainerStarts.inProgress++
|
|
|
|
currentContainerStarts.Unlock()
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if currentContainerStarts.inProgress == currentContainerStarts.maxParallel {
|
|
|
|
currentContainerStarts.Unlock()
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Make sure we decrement the count when we are done.
|
|
|
|
defer func() {
|
|
|
|
currentContainerStarts.Lock()
|
|
|
|
currentContainerStarts.inProgress--
|
|
|
|
currentContainerStarts.Unlock()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
resultJSON, err := vmcompute.HcsStartComputeSystem(ctx, computeSystem.handle, "")
|
|
|
|
events, err := processAsyncHcsResult(ctx, err, resultJSON, computeSystem.callbackNumber, hcsNotificationSystemStartCompleted, &timeout.SystemStart)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ID returns the compute system's identifier.
|
|
|
|
func (computeSystem *System) ID() string {
|
|
|
|
return computeSystem.id
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// Shutdown requests a compute system shutdown.
|
|
|
|
func (computeSystem *System) Shutdown(ctx context.Context) error {
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
operation := "hcsshim::System::Shutdown"
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
resultJSON, err := vmcompute.HcsShutdownComputeSystem(ctx, computeSystem.handle, "")
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
|
|
|
switch err {
|
|
|
|
case nil, ErrVmcomputeAlreadyStopped, ErrComputeSystemDoesNotExist, ErrVmcomputeOperationPending:
|
|
|
|
default:
|
|
|
|
return makeSystemError(computeSystem, operation, "", err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// Terminate requests a compute system terminate.
|
|
|
|
func (computeSystem *System) Terminate(ctx context.Context) error {
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
operation := "hcsshim::System::Terminate"
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
resultJSON, err := vmcompute.HcsTerminateComputeSystem(ctx, computeSystem.handle, "")
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
|
|
|
switch err {
|
|
|
|
case nil, ErrVmcomputeAlreadyStopped, ErrComputeSystemDoesNotExist, ErrVmcomputeOperationPending:
|
|
|
|
default:
|
|
|
|
return makeSystemError(computeSystem, operation, "", err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// waitBackground waits for the compute system exit notification. Once received
|
|
|
|
// sets `computeSystem.waitError` (if any) and unblocks all `Wait` calls.
|
|
|
|
//
|
|
|
|
// This MUST be called exactly once per `computeSystem.handle` but `Wait` is
|
|
|
|
// safe to call multiple times.
|
|
|
|
func (computeSystem *System) waitBackground() {
|
|
|
|
operation := "hcsshim::System::waitBackground"
|
|
|
|
ctx, span := trace.StartSpan(context.Background(), operation)
|
|
|
|
defer span.End()
|
|
|
|
span.AddAttributes(trace.StringAttribute("cid", computeSystem.id))
|
|
|
|
|
|
|
|
err := waitForNotification(ctx, computeSystem.callbackNumber, hcsNotificationSystemExited, nil)
|
|
|
|
switch err {
|
|
|
|
case nil:
|
|
|
|
log.G(ctx).Debug("system exited")
|
|
|
|
case ErrVmcomputeUnexpectedExit:
|
|
|
|
log.G(ctx).Debug("unexpected system exit")
|
|
|
|
computeSystem.exitError = makeSystemError(computeSystem, operation, "", err, nil)
|
|
|
|
err = nil
|
|
|
|
default:
|
|
|
|
err = makeSystemError(computeSystem, operation, "", err, nil)
|
|
|
|
}
|
|
|
|
computeSystem.closedWaitOnce.Do(func() {
|
|
|
|
computeSystem.waitError = err
|
|
|
|
close(computeSystem.waitBlock)
|
|
|
|
})
|
|
|
|
oc.SetSpanStatus(span, err)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// Wait synchronously waits for the compute system to shutdown or terminate. If
|
|
|
|
// the compute system has already exited returns the previous error (if any).
|
|
|
|
func (computeSystem *System) Wait() error {
|
|
|
|
<-computeSystem.waitBlock
|
|
|
|
return computeSystem.waitError
|
2019-07-10 00:29:38 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// ExitError returns an error describing the reason the compute system terminated.
|
|
|
|
func (computeSystem *System) ExitError() error {
|
|
|
|
select {
|
|
|
|
case <-computeSystem.waitBlock:
|
|
|
|
if computeSystem.waitError != nil {
|
|
|
|
return computeSystem.waitError
|
|
|
|
}
|
|
|
|
return computeSystem.exitError
|
|
|
|
default:
|
|
|
|
return errors.New("container not exited")
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) Properties(ctx context.Context, types ...schema1.PropertyType) (*schema1.ContainerProperties, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
operation := "hcsshim::System::Properties"
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
queryBytes, err := json.Marshal(schema1.PropertyQuery{PropertyTypes: types})
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
propertiesJSON, resultJSON, err := vmcompute.HcsGetComputeSystemProperties(ctx, computeSystem.handle, string(queryBytes))
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
if propertiesJSON == "" {
|
2019-01-12 04:58:27 +00:00
|
|
|
return nil, ErrUnexpectedValue
|
|
|
|
}
|
|
|
|
properties := &schema1.ContainerProperties{}
|
2019-09-30 23:25:17 +00:00
|
|
|
if err := json.Unmarshal([]byte(propertiesJSON), properties); err != nil {
|
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return properties, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pause pauses the execution of the computeSystem. This feature is not enabled in TP5.
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) Pause(ctx context.Context) (err error) {
|
|
|
|
operation := "hcsshim::System::Pause"
|
|
|
|
|
|
|
|
// hcsPauseComputeSystemContext is an async peration. Start the outer span
|
|
|
|
// here to measure the full pause time.
|
|
|
|
ctx, span := trace.StartSpan(ctx, operation)
|
|
|
|
defer span.End()
|
|
|
|
defer func() { oc.SetSpanStatus(span, err) }()
|
|
|
|
span.AddAttributes(trace.StringAttribute("cid", computeSystem.id))
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", ErrAlreadyClosed, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
resultJSON, err := vmcompute.HcsPauseComputeSystem(ctx, computeSystem.handle, "")
|
|
|
|
events, err := processAsyncHcsResult(ctx, err, resultJSON, computeSystem.callbackNumber, hcsNotificationSystemPauseCompleted, &timeout.SystemPause)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resume resumes the execution of the computeSystem. This feature is not enabled in TP5.
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) Resume(ctx context.Context) (err error) {
|
|
|
|
operation := "hcsshim::System::Resume"
|
|
|
|
|
|
|
|
// hcsResumeComputeSystemContext is an async operation. Start the outer span
|
|
|
|
// here to measure the full restore time.
|
|
|
|
ctx, span := trace.StartSpan(ctx, operation)
|
|
|
|
defer span.End()
|
|
|
|
defer func() { oc.SetSpanStatus(span, err) }()
|
|
|
|
span.AddAttributes(trace.StringAttribute("cid", computeSystem.id))
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", ErrAlreadyClosed, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
resultJSON, err := vmcompute.HcsResumeComputeSystem(ctx, computeSystem.handle, "")
|
|
|
|
events, err := processAsyncHcsResult(ctx, err, resultJSON, computeSystem.callbackNumber, hcsNotificationSystemResumeCompleted, &timeout.SystemResume)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) createProcess(ctx context.Context, operation string, c interface{}) (*Process, *vmcompute.HcsProcessInformation, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil, nil, makeSystemError(computeSystem, operation, "", ErrAlreadyClosed, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
configurationb, err := json.Marshal(c)
|
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil, nil, makeSystemError(computeSystem, operation, "", err, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
configuration := string(configurationb)
|
2019-09-30 23:25:17 +00:00
|
|
|
processInfo, processHandle, resultJSON, err := vmcompute.HcsCreateProcess(ctx, computeSystem.handle, configuration)
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, makeSystemError(computeSystem, operation, configuration, err, events)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.G(ctx).WithField("pid", processInfo.ProcessId).Debug("created process pid")
|
|
|
|
return newProcess(processHandle, int(processInfo.ProcessId), computeSystem), &processInfo, nil
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// CreateProcessNoStdio launches a new process within the computeSystem. The
|
|
|
|
// Stdio handles are not cached on the process struct.
|
|
|
|
func (computeSystem *System) CreateProcessNoStdio(c interface{}) (_ cow.Process, err error) {
|
|
|
|
operation := "hcsshim::System::CreateProcessNoStdio"
|
|
|
|
ctx, span := trace.StartSpan(context.Background(), operation)
|
|
|
|
defer span.End()
|
|
|
|
defer func() { oc.SetSpanStatus(span, err) }()
|
|
|
|
span.AddAttributes(trace.StringAttribute("cid", computeSystem.id))
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
process, processInfo, err := computeSystem.createProcess(ctx, operation, c)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil, err
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
defer func() {
|
|
|
|
if err != nil {
|
|
|
|
process.Close()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// We don't do anything with these handles. Close them so they don't leak.
|
|
|
|
syscall.Close(processInfo.StdInput)
|
|
|
|
syscall.Close(processInfo.StdOutput)
|
|
|
|
syscall.Close(processInfo.StdError)
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
if err = process.registerCallback(ctx); err != nil {
|
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
|
|
|
}
|
|
|
|
go process.waitBackground()
|
|
|
|
|
|
|
|
return process, nil
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
// CreateProcess launches a new process within the computeSystem.
|
|
|
|
func (computeSystem *System) CreateProcess(ctx context.Context, c interface{}) (cow.Process, error) {
|
|
|
|
operation := "hcsshim::System::CreateProcess"
|
|
|
|
process, processInfo, err := computeSystem.createProcess(ctx, operation, c)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
defer func() {
|
|
|
|
if err != nil {
|
|
|
|
process.Close()
|
|
|
|
}
|
|
|
|
}()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
pipes, err := makeOpenFiles([]syscall.Handle{processInfo.StdInput, processInfo.StdOutput, processInfo.StdError})
|
|
|
|
if err != nil {
|
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
process.stdin = pipes[0]
|
|
|
|
process.stdout = pipes[1]
|
|
|
|
process.stderr = pipes[2]
|
|
|
|
|
|
|
|
if err = process.registerCallback(ctx); err != nil {
|
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
|
|
|
}
|
|
|
|
go process.waitBackground()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
return process, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// OpenProcess gets an interface to an existing process within the computeSystem.
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) OpenProcess(ctx context.Context, pid int) (*Process, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
operation := "hcsshim::System::OpenProcess"
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil, makeSystemError(computeSystem, operation, "", ErrAlreadyClosed, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
processHandle, resultJSON, err := vmcompute.HcsOpenProcess(ctx, computeSystem.handle, uint32(pid))
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
process := newProcess(processHandle, pid, computeSystem)
|
2019-09-30 23:25:17 +00:00
|
|
|
if err = process.registerCallback(ctx); err != nil {
|
|
|
|
return nil, makeSystemError(computeSystem, operation, "", err, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
go process.waitBackground()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
return process, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close cleans up any state associated with the compute system but does not terminate or wait for it.
|
|
|
|
func (computeSystem *System) Close() (err error) {
|
2019-09-30 23:25:17 +00:00
|
|
|
operation := "hcsshim::System::Close"
|
|
|
|
ctx, span := trace.StartSpan(context.Background(), operation)
|
|
|
|
defer span.End()
|
|
|
|
defer func() { oc.SetSpanStatus(span, err) }()
|
|
|
|
span.AddAttributes(trace.StringAttribute("cid", computeSystem.id))
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.Lock()
|
|
|
|
defer computeSystem.handleLock.Unlock()
|
|
|
|
|
|
|
|
// Don't double free this
|
|
|
|
if computeSystem.handle == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
if err = computeSystem.unregisterCallback(ctx); err != nil {
|
|
|
|
return makeSystemError(computeSystem, operation, "", err, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
err = vmcompute.HcsCloseComputeSystem(ctx, computeSystem.handle)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", err, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
computeSystem.handle = 0
|
2019-09-30 23:25:17 +00:00
|
|
|
computeSystem.closedWaitOnce.Do(func() {
|
|
|
|
computeSystem.waitError = ErrAlreadyClosed
|
|
|
|
close(computeSystem.waitBlock)
|
|
|
|
})
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) registerCallback(ctx context.Context) error {
|
|
|
|
callbackContext := ¬ifcationWatcherContext{
|
|
|
|
channels: newSystemChannels(),
|
|
|
|
systemID: computeSystem.id,
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
callbackMapLock.Lock()
|
|
|
|
callbackNumber := nextCallback
|
|
|
|
nextCallback++
|
2019-09-30 23:25:17 +00:00
|
|
|
callbackMap[callbackNumber] = callbackContext
|
2019-01-12 04:58:27 +00:00
|
|
|
callbackMapLock.Unlock()
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
callbackHandle, err := vmcompute.HcsRegisterComputeSystemCallback(ctx, computeSystem.handle, notificationWatcherCallback, callbackNumber)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-09-30 23:25:17 +00:00
|
|
|
callbackContext.handle = callbackHandle
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.callbackNumber = callbackNumber
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) unregisterCallback(ctx context.Context) error {
|
2019-01-12 04:58:27 +00:00
|
|
|
callbackNumber := computeSystem.callbackNumber
|
|
|
|
|
|
|
|
callbackMapLock.RLock()
|
2019-09-30 23:25:17 +00:00
|
|
|
callbackContext := callbackMap[callbackNumber]
|
2019-01-12 04:58:27 +00:00
|
|
|
callbackMapLock.RUnlock()
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
if callbackContext == nil {
|
2019-01-12 04:58:27 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
handle := callbackContext.handle
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
if handle == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// hcsUnregisterComputeSystemCallback has its own syncronization
|
|
|
|
// to wait for all callbacks to complete. We must NOT hold the callbackMapLock.
|
2019-09-30 23:25:17 +00:00
|
|
|
err := vmcompute.HcsUnregisterComputeSystemCallback(ctx, handle)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
closeChannels(callbackContext.channels)
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
callbackMapLock.Lock()
|
2019-09-30 23:25:17 +00:00
|
|
|
delete(callbackMap, callbackNumber)
|
2019-01-12 04:58:27 +00:00
|
|
|
callbackMapLock.Unlock()
|
|
|
|
|
|
|
|
handle = 0
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Modify the System by sending a request to HCS
|
2019-09-30 23:25:17 +00:00
|
|
|
func (computeSystem *System) Modify(ctx context.Context, config interface{}) error {
|
2019-01-12 04:58:27 +00:00
|
|
|
computeSystem.handleLock.RLock()
|
|
|
|
defer computeSystem.handleLock.RUnlock()
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
operation := "hcsshim::System::Modify"
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
if computeSystem.handle == 0 {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, "", ErrAlreadyClosed, nil)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
requestBytes, err := json.Marshal(config)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:25:17 +00:00
|
|
|
requestJSON := string(requestBytes)
|
|
|
|
resultJSON, err := vmcompute.HcsModifyComputeSystem(ctx, computeSystem.handle, requestJSON)
|
|
|
|
events := processHcsResult(ctx, resultJSON)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
2019-09-30 23:25:17 +00:00
|
|
|
return makeSystemError(computeSystem, operation, requestJSON, err, events)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|