2019-09-30 23:25:17 +00:00
package vmcompute
import (
gcontext "context"
"syscall"
"time"
"github.com/Microsoft/hcsshim/internal/interop"
"github.com/Microsoft/hcsshim/internal/log"
"github.com/Microsoft/hcsshim/internal/logfields"
"github.com/Microsoft/hcsshim/internal/oc"
"github.com/Microsoft/hcsshim/internal/timeout"
"go.opencensus.io/trace"
)
//go:generate go run ../../mksyscall_windows.go -output zsyscall_windows.go vmcompute.go
//sys hcsEnumerateComputeSystems(query string, computeSystems **uint16, result **uint16) (hr error) = vmcompute.HcsEnumerateComputeSystems?
//sys hcsCreateComputeSystem(id string, configuration string, identity syscall.Handle, computeSystem *HcsSystem, result **uint16) (hr error) = vmcompute.HcsCreateComputeSystem?
//sys hcsOpenComputeSystem(id string, computeSystem *HcsSystem, result **uint16) (hr error) = vmcompute.HcsOpenComputeSystem?
//sys hcsCloseComputeSystem(computeSystem HcsSystem) (hr error) = vmcompute.HcsCloseComputeSystem?
//sys hcsStartComputeSystem(computeSystem HcsSystem, options string, result **uint16) (hr error) = vmcompute.HcsStartComputeSystem?
//sys hcsShutdownComputeSystem(computeSystem HcsSystem, options string, result **uint16) (hr error) = vmcompute.HcsShutdownComputeSystem?
//sys hcsTerminateComputeSystem(computeSystem HcsSystem, options string, result **uint16) (hr error) = vmcompute.HcsTerminateComputeSystem?
//sys hcsPauseComputeSystem(computeSystem HcsSystem, options string, result **uint16) (hr error) = vmcompute.HcsPauseComputeSystem?
//sys hcsResumeComputeSystem(computeSystem HcsSystem, options string, result **uint16) (hr error) = vmcompute.HcsResumeComputeSystem?
//sys hcsGetComputeSystemProperties(computeSystem HcsSystem, propertyQuery string, properties **uint16, result **uint16) (hr error) = vmcompute.HcsGetComputeSystemProperties?
//sys hcsModifyComputeSystem(computeSystem HcsSystem, configuration string, result **uint16) (hr error) = vmcompute.HcsModifyComputeSystem?
//sys hcsRegisterComputeSystemCallback(computeSystem HcsSystem, callback uintptr, context uintptr, callbackHandle *HcsCallback) (hr error) = vmcompute.HcsRegisterComputeSystemCallback?
//sys hcsUnregisterComputeSystemCallback(callbackHandle HcsCallback) (hr error) = vmcompute.HcsUnregisterComputeSystemCallback?
//sys hcsCreateProcess(computeSystem HcsSystem, processParameters string, processInformation *HcsProcessInformation, process *HcsProcess, result **uint16) (hr error) = vmcompute.HcsCreateProcess?
//sys hcsOpenProcess(computeSystem HcsSystem, pid uint32, process *HcsProcess, result **uint16) (hr error) = vmcompute.HcsOpenProcess?
//sys hcsCloseProcess(process HcsProcess) (hr error) = vmcompute.HcsCloseProcess?
//sys hcsTerminateProcess(process HcsProcess, result **uint16) (hr error) = vmcompute.HcsTerminateProcess?
//sys hcsSignalProcess(process HcsProcess, options string, result **uint16) (hr error) = vmcompute.HcsSignalProcess?
//sys hcsGetProcessInfo(process HcsProcess, processInformation *HcsProcessInformation, result **uint16) (hr error) = vmcompute.HcsGetProcessInfo?
//sys hcsGetProcessProperties(process HcsProcess, processProperties **uint16, result **uint16) (hr error) = vmcompute.HcsGetProcessProperties?
//sys hcsModifyProcess(process HcsProcess, settings string, result **uint16) (hr error) = vmcompute.HcsModifyProcess?
//sys hcsGetServiceProperties(propertyQuery string, properties **uint16, result **uint16) (hr error) = vmcompute.HcsGetServiceProperties?
//sys hcsRegisterProcessCallback(process HcsProcess, callback uintptr, context uintptr, callbackHandle *HcsCallback) (hr error) = vmcompute.HcsRegisterProcessCallback?
//sys hcsUnregisterProcessCallback(callbackHandle HcsCallback) (hr error) = vmcompute.HcsUnregisterProcessCallback?
// errVmcomputeOperationPending is an error encountered when the operation is being completed asynchronously
const errVmcomputeOperationPending = syscall . Errno ( 0xC0370103 )
// HcsSystem is the handle associated with a created compute system.
type HcsSystem syscall . Handle
// HcsProcess is the handle associated with a created process in a compute
// system.
type HcsProcess syscall . Handle
// HcsCallback is the handle associated with the function to call when events
// occur.
type HcsCallback syscall . Handle
// HcsProcessInformation is the structure used when creating or getting process
// info.
type HcsProcessInformation struct {
// ProcessId is the pid of the created process.
ProcessId uint32
reserved uint32
// StdInput is the handle associated with the stdin of the process.
StdInput syscall . Handle
// StdOutput is the handle associated with the stdout of the process.
StdOutput syscall . Handle
// StdError is the handle associated with the stderr of the process.
StdError syscall . Handle
}
func execute ( ctx gcontext . Context , timeout time . Duration , f func ( ) error ) error {
if timeout > 0 {
var cancel gcontext . CancelFunc
ctx , cancel = gcontext . WithTimeout ( ctx , timeout )
defer cancel ( )
}
done := make ( chan error , 1 )
go func ( ) {
done <- f ( )
} ( )
select {
case <- ctx . Done ( ) :
if ctx . Err ( ) == gcontext . DeadlineExceeded {
log . G ( ctx ) . WithField ( logfields . Timeout , timeout ) .
Warning ( "Syscall did not complete within operation timeout. This may indicate a platform issue. If it appears to be making no forward progress, obtain the stacks and see if there is a syscall stuck in the platform API for a significant length of time." )
}
return ctx . Err ( )
case err := <- done :
return err
}
}
func HcsEnumerateComputeSystems ( ctx gcontext . Context , query string ) ( computeSystems , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsEnumerateComputeSystems" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . StringAttribute ( "query" , query ) )
return computeSystems , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var (
computeSystemsp * uint16
resultp * uint16
)
err := hcsEnumerateComputeSystems ( query , & computeSystemsp , & resultp )
if computeSystemsp != nil {
computeSystems = interop . ConvertAndFreeCoTaskMemString ( computeSystemsp )
}
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsCreateComputeSystem ( ctx gcontext . Context , id string , configuration string , identity syscall . Handle ) ( computeSystem HcsSystem , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsCreateComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
if hr != errVmcomputeOperationPending {
oc . SetSpanStatus ( span , hr )
}
} ( )
span . AddAttributes (
trace . StringAttribute ( "id" , id ) ,
trace . StringAttribute ( "configuration" , configuration ) )
return computeSystem , result , execute ( ctx , timeout . SystemCreate , func ( ) error {
var resultp * uint16
err := hcsCreateComputeSystem ( id , configuration , identity , & computeSystem , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsOpenComputeSystem ( ctx gcontext . Context , id string ) ( computeSystem HcsSystem , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsOpenComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
return computeSystem , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsOpenComputeSystem ( id , & computeSystem , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsCloseComputeSystem ( ctx gcontext . Context , computeSystem HcsSystem ) ( hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsCloseComputeSystem" )
defer span . End ( )
defer func ( ) { oc . SetSpanStatus ( span , hr ) } ( )
return execute ( ctx , timeout . SyscallWatcher , func ( ) error {
return hcsCloseComputeSystem ( computeSystem )
} )
}
func HcsStartComputeSystem ( ctx gcontext . Context , computeSystem HcsSystem , options string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsStartComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
if hr != errVmcomputeOperationPending {
oc . SetSpanStatus ( span , hr )
}
} ( )
span . AddAttributes ( trace . StringAttribute ( "options" , options ) )
return result , execute ( ctx , timeout . SystemStart , func ( ) error {
var resultp * uint16
err := hcsStartComputeSystem ( computeSystem , options , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsShutdownComputeSystem ( ctx gcontext . Context , computeSystem HcsSystem , options string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsShutdownComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
2020-08-10 17:43:49 +00:00
if hr != errVmcomputeOperationPending {
oc . SetSpanStatus ( span , hr )
}
2019-09-30 23:25:17 +00:00
} ( )
span . AddAttributes ( trace . StringAttribute ( "options" , options ) )
return result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsShutdownComputeSystem ( computeSystem , options , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsTerminateComputeSystem ( ctx gcontext . Context , computeSystem HcsSystem , options string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsTerminateComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
if hr != errVmcomputeOperationPending {
oc . SetSpanStatus ( span , hr )
}
} ( )
span . AddAttributes ( trace . StringAttribute ( "options" , options ) )
return result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsTerminateComputeSystem ( computeSystem , options , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsPauseComputeSystem ( ctx gcontext . Context , computeSystem HcsSystem , options string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsPauseComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
if hr != errVmcomputeOperationPending {
oc . SetSpanStatus ( span , hr )
}
} ( )
span . AddAttributes ( trace . StringAttribute ( "options" , options ) )
return result , execute ( ctx , timeout . SystemPause , func ( ) error {
var resultp * uint16
err := hcsPauseComputeSystem ( computeSystem , options , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsResumeComputeSystem ( ctx gcontext . Context , computeSystem HcsSystem , options string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsResumeComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
if hr != errVmcomputeOperationPending {
oc . SetSpanStatus ( span , hr )
}
} ( )
span . AddAttributes ( trace . StringAttribute ( "options" , options ) )
return result , execute ( ctx , timeout . SystemResume , func ( ) error {
var resultp * uint16
err := hcsResumeComputeSystem ( computeSystem , options , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsGetComputeSystemProperties ( ctx gcontext . Context , computeSystem HcsSystem , propertyQuery string ) ( properties , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsGetComputeSystemProperties" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . StringAttribute ( "propertyQuery" , propertyQuery ) )
return properties , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var (
propertiesp * uint16
resultp * uint16
)
err := hcsGetComputeSystemProperties ( computeSystem , propertyQuery , & propertiesp , & resultp )
if propertiesp != nil {
properties = interop . ConvertAndFreeCoTaskMemString ( propertiesp )
}
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsModifyComputeSystem ( ctx gcontext . Context , computeSystem HcsSystem , configuration string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsModifyComputeSystem" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . StringAttribute ( "configuration" , configuration ) )
return result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsModifyComputeSystem ( computeSystem , configuration , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsRegisterComputeSystemCallback ( ctx gcontext . Context , computeSystem HcsSystem , callback uintptr , context uintptr ) ( callbackHandle HcsCallback , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsRegisterComputeSystemCallback" )
defer span . End ( )
defer func ( ) { oc . SetSpanStatus ( span , hr ) } ( )
return callbackHandle , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
return hcsRegisterComputeSystemCallback ( computeSystem , callback , context , & callbackHandle )
} )
}
func HcsUnregisterComputeSystemCallback ( ctx gcontext . Context , callbackHandle HcsCallback ) ( hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsUnregisterComputeSystemCallback" )
defer span . End ( )
defer func ( ) { oc . SetSpanStatus ( span , hr ) } ( )
return execute ( ctx , timeout . SyscallWatcher , func ( ) error {
return hcsUnregisterComputeSystemCallback ( callbackHandle )
} )
}
func HcsCreateProcess ( ctx gcontext . Context , computeSystem HcsSystem , processParameters string ) ( processInformation HcsProcessInformation , process HcsProcess , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsCreateProcess" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . StringAttribute ( "processParameters" , processParameters ) )
return processInformation , process , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsCreateProcess ( computeSystem , processParameters , & processInformation , & process , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsOpenProcess ( ctx gcontext . Context , computeSystem HcsSystem , pid uint32 ) ( process HcsProcess , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsOpenProcess" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . Int64Attribute ( "pid" , int64 ( pid ) ) )
return process , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsOpenProcess ( computeSystem , pid , & process , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsCloseProcess ( ctx gcontext . Context , process HcsProcess ) ( hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsCloseProcess" )
defer span . End ( )
defer func ( ) { oc . SetSpanStatus ( span , hr ) } ( )
return execute ( ctx , timeout . SyscallWatcher , func ( ) error {
return hcsCloseProcess ( process )
} )
}
func HcsTerminateProcess ( ctx gcontext . Context , process HcsProcess ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsTerminateProcess" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
return result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsTerminateProcess ( process , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsSignalProcess ( ctx gcontext . Context , process HcsProcess , options string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsSignalProcess" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . StringAttribute ( "options" , options ) )
return result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsSignalProcess ( process , options , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsGetProcessInfo ( ctx gcontext . Context , process HcsProcess ) ( processInformation HcsProcessInformation , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsGetProcessInfo" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
return processInformation , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsGetProcessInfo ( process , & processInformation , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsGetProcessProperties ( ctx gcontext . Context , process HcsProcess ) ( processProperties , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsGetProcessProperties" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
return processProperties , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var (
processPropertiesp * uint16
resultp * uint16
)
err := hcsGetProcessProperties ( process , & processPropertiesp , & resultp )
if processPropertiesp != nil {
processProperties = interop . ConvertAndFreeCoTaskMemString ( processPropertiesp )
}
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsModifyProcess ( ctx gcontext . Context , process HcsProcess , settings string ) ( result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsModifyProcess" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . StringAttribute ( "settings" , settings ) )
return result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var resultp * uint16
err := hcsModifyProcess ( process , settings , & resultp )
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsGetServiceProperties ( ctx gcontext . Context , propertyQuery string ) ( properties , result string , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsGetServiceProperties" )
defer span . End ( )
defer func ( ) {
if result != "" {
span . AddAttributes ( trace . StringAttribute ( "result" , result ) )
}
oc . SetSpanStatus ( span , hr )
} ( )
span . AddAttributes ( trace . StringAttribute ( "propertyQuery" , propertyQuery ) )
return properties , result , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
var (
propertiesp * uint16
resultp * uint16
)
err := hcsGetServiceProperties ( propertyQuery , & propertiesp , & resultp )
if propertiesp != nil {
properties = interop . ConvertAndFreeCoTaskMemString ( propertiesp )
}
if resultp != nil {
result = interop . ConvertAndFreeCoTaskMemString ( resultp )
}
return err
} )
}
func HcsRegisterProcessCallback ( ctx gcontext . Context , process HcsProcess , callback uintptr , context uintptr ) ( callbackHandle HcsCallback , hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsRegisterProcessCallback" )
defer span . End ( )
defer func ( ) { oc . SetSpanStatus ( span , hr ) } ( )
return callbackHandle , execute ( ctx , timeout . SyscallWatcher , func ( ) error {
return hcsRegisterProcessCallback ( process , callback , context , & callbackHandle )
} )
}
func HcsUnregisterProcessCallback ( ctx gcontext . Context , callbackHandle HcsCallback ) ( hr error ) {
ctx , span := trace . StartSpan ( ctx , "HcsUnregisterProcessCallback" )
defer span . End ( )
defer func ( ) { oc . SetSpanStatus ( span , hr ) } ( )
return execute ( ctx , timeout . SyscallWatcher , func ( ) error {
return hcsUnregisterProcessCallback ( callbackHandle )
} )
}