Skip to content

Commit 53a8c94

Browse files
authored
Merge pull request #2538 from jterry75/runtime_v2_windows
Introduce containerd-shim-runhcs-v1 on Windows
2 parents 1a96db4 + 019b0c3 commit 53a8c94

File tree

101 files changed

+6743
-3657
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

101 files changed

+6743
-3657
lines changed

Makefile

+4
Original file line numberDiff line numberDiff line change
@@ -176,6 +176,10 @@ bin/containerd-shim-runc-v1: cmd/containerd-shim-runc-v1 FORCE # set !cgo and om
176176
@echo "$(WHALE) bin/containerd-shim-runc-v1"
177177
@CGO_ENABLED=0 go build ${GO_BUILD_FLAGS} -o bin/containerd-shim-runc-v1 ${SHIM_GO_LDFLAGS} ${GO_TAGS} ./cmd/containerd-shim-runc-v1
178178

179+
bin/containerd-shim-runhcs-v1: cmd/containerd-shim-runhcs-v1 FORCE # set !cgo and omit pie for a static shim build: https://github.com/golang/go/issues/17789#issuecomment-258542220
180+
@echo "$(WHALE) bin/containerd-shim-runhcs-v1${BINARY_SUFFIX}"
181+
@CGO_ENABLED=0 go build ${GO_BUILD_FLAGS} -o bin/containerd-shim-runhcs-v1${BINARY_SUFFIX} ${SHIM_GO_LDFLAGS} ${GO_TAGS} ./cmd/containerd-shim-runhcs-v1
182+
179183
binaries: $(BINARIES) ## build binaries
180184
@echo "$(WHALE) $@"
181185

Makefile.windows

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
#Windows specific settings.
1717
WHALE = "+"
1818
ONI = "-"
19+
COMMANDS += containerd-shim-runhcs-v1
1920

2021
BINARY_SUFFIX=".exe"
2122

cio/io_windows.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,7 @@ func copyIO(fifos *FIFOSet, ioset *Streams) (*cio, error) {
7474
if fifos.Stdout != "" {
7575
l, err := winio.ListenPipe(fifos.Stdout, nil)
7676
if err != nil {
77-
return nil, errors.Wrapf(err, "failed to create stdin pipe %s", fifos.Stdout)
77+
return nil, errors.Wrapf(err, "failed to create stdout pipe %s", fifos.Stdout)
7878
}
7979
defer func(l net.Listener) {
8080
if err != nil {

cmd/containerd-shim-runhcs-v1/main.go

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
// +build windows
2+
3+
/*
4+
Copyright The containerd Authors.
5+
6+
Licensed under the Apache License, Version 2.0 (the "License");
7+
you may not use this file except in compliance with the License.
8+
You may obtain a copy of the License at
9+
10+
http://www.apache.org/licenses/LICENSE-2.0
11+
12+
Unless required by applicable law or agreed to in writing, software
13+
distributed under the License is distributed on an "AS IS" BASIS,
14+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
See the License for the specific language governing permissions and
16+
limitations under the License.
17+
*/
18+
19+
package main
20+
21+
import (
22+
"github.com/containerd/containerd/runtime/v2/runhcs"
23+
"github.com/containerd/containerd/runtime/v2/shim"
24+
)
25+
26+
func main() {
27+
shim.Run("io.containerd.runhcs.v1", runhcs.New)
28+
}

runtime/v2/runhcs/cmdutil.go

+98
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,98 @@
1+
// +build windows
2+
3+
/*
4+
Copyright The containerd Authors.
5+
6+
Licensed under the Apache License, Version 2.0 (the "License");
7+
you may not use this file except in compliance with the License.
8+
You may obtain a copy of the License at
9+
10+
http://www.apache.org/licenses/LICENSE-2.0
11+
12+
Unless required by applicable law or agreed to in writing, software
13+
distributed under the License is distributed on an "AS IS" BASIS,
14+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
See the License for the specific language governing permissions and
16+
limitations under the License.
17+
*/
18+
19+
package runhcs
20+
21+
import (
22+
"bytes"
23+
"context"
24+
"os/exec"
25+
"sync"
26+
"syscall"
27+
"time"
28+
)
29+
30+
var (
31+
bytesBufferPool = sync.Pool{
32+
New: func() interface{} {
33+
return bytes.NewBuffer(nil)
34+
},
35+
}
36+
)
37+
38+
func getBuffer() *bytes.Buffer {
39+
return bytesBufferPool.Get().(*bytes.Buffer)
40+
}
41+
42+
func putBuffer(b *bytes.Buffer) {
43+
b.Reset()
44+
bytesBufferPool.Put(b)
45+
}
46+
47+
type processExit struct {
48+
pid uint32
49+
exitStatus uint32
50+
exitedAt time.Time
51+
exitErr error
52+
}
53+
54+
func runCmd(ctx context.Context, c *exec.Cmd) (*processExit, error) {
55+
ec, startErr := startCmd(ctx, c)
56+
if startErr != nil {
57+
return nil, startErr
58+
}
59+
er, cmdErr := waitCmd(ctx, ec)
60+
return er, cmdErr
61+
}
62+
63+
func startCmd(ctx context.Context, c *exec.Cmd) (<-chan *processExit, error) {
64+
if err := c.Start(); err != nil {
65+
return nil, err
66+
}
67+
ec := make(chan *processExit, 1)
68+
go func() {
69+
defer close(ec)
70+
71+
var status int
72+
eerr := c.Wait()
73+
if eerr != nil {
74+
status = 255
75+
if exitErr, ok := eerr.(*exec.ExitError); ok {
76+
if ws, ok := exitErr.Sys().(syscall.WaitStatus); ok {
77+
status = ws.ExitStatus()
78+
}
79+
}
80+
}
81+
ec <- &processExit{
82+
pid: uint32(c.Process.Pid),
83+
exitStatus: uint32(status),
84+
exitedAt: time.Now(),
85+
exitErr: eerr,
86+
}
87+
}()
88+
89+
return ec, nil
90+
}
91+
92+
func waitCmd(ctx context.Context, ec <-chan *processExit) (*processExit, error) {
93+
e := <-ec
94+
if e.exitStatus != 0 {
95+
return e, e.exitErr
96+
}
97+
return e, nil
98+
}

runtime/v2/runhcs/io.go

+159
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,159 @@
1+
// +build windows
2+
3+
/*
4+
Copyright The containerd Authors.
5+
6+
Licensed under the Apache License, Version 2.0 (the "License");
7+
you may not use this file except in compliance with the License.
8+
You may obtain a copy of the License at
9+
10+
http://www.apache.org/licenses/LICENSE-2.0
11+
12+
Unless required by applicable law or agreed to in writing, software
13+
distributed under the License is distributed on an "AS IS" BASIS,
14+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
See the License for the specific language governing permissions and
16+
limitations under the License.
17+
*/
18+
19+
package runhcs
20+
21+
import (
22+
"context"
23+
"io"
24+
"net"
25+
"sync"
26+
"time"
27+
28+
"github.com/Microsoft/go-winio"
29+
"github.com/containerd/containerd/log"
30+
runc "github.com/containerd/go-runc"
31+
"github.com/pkg/errors"
32+
"golang.org/x/sync/errgroup"
33+
)
34+
35+
type pipeSet struct {
36+
stdin net.Conn
37+
stdout net.Conn
38+
stderr net.Conn
39+
}
40+
41+
// newPipeSet connects to the provided pipe addresses
42+
func newPipeSet(ctx context.Context, stdin, stdout, stderr string, terminal bool) (*pipeSet, error) {
43+
var (
44+
err error
45+
set = &pipeSet{}
46+
)
47+
48+
defer func() {
49+
if err != nil {
50+
set.Close()
51+
}
52+
}()
53+
54+
g, _ := errgroup.WithContext(ctx)
55+
56+
dialfn := func(name string, conn *net.Conn) error {
57+
if name == "" {
58+
return nil
59+
}
60+
dialTimeout := 3 * time.Second
61+
c, err := winio.DialPipe(name, &dialTimeout)
62+
if err != nil {
63+
return errors.Wrapf(err, "failed to connect to %s", name)
64+
}
65+
*conn = c
66+
return nil
67+
}
68+
69+
g.Go(func() error {
70+
return dialfn(stdin, &set.stdin)
71+
})
72+
g.Go(func() error {
73+
return dialfn(stdout, &set.stdout)
74+
})
75+
g.Go(func() error {
76+
return dialfn(stderr, &set.stderr)
77+
})
78+
79+
err = g.Wait()
80+
if err != nil {
81+
return nil, err
82+
}
83+
return set, nil
84+
}
85+
86+
// Close terminates all successfully dialed IO connections
87+
func (p *pipeSet) Close() {
88+
for _, cn := range []net.Conn{p.stdin, p.stdout, p.stderr} {
89+
if cn != nil {
90+
cn.Close()
91+
}
92+
}
93+
}
94+
95+
type pipeRelay struct {
96+
ctx context.Context
97+
98+
ps *pipeSet
99+
io runc.IO
100+
101+
wg sync.WaitGroup
102+
once sync.Once
103+
}
104+
105+
func newPipeRelay(ctx context.Context, ps *pipeSet, downstream runc.IO) *pipeRelay {
106+
pr := &pipeRelay{
107+
ctx: ctx,
108+
ps: ps,
109+
io: downstream,
110+
}
111+
if ps.stdin != nil {
112+
go func() {
113+
if _, err := io.Copy(downstream.Stdin(), ps.stdin); err != nil {
114+
if err != winio.ErrFileClosed {
115+
log.G(ctx).WithError(err).Error("error copying stdin to pipe")
116+
}
117+
}
118+
}()
119+
}
120+
if ps.stdout != nil {
121+
pr.wg.Add(1)
122+
go func() {
123+
if _, err := io.Copy(ps.stdout, downstream.Stdout()); err != nil {
124+
log.G(ctx).WithError(err).Error("error copying stdout from pipe")
125+
}
126+
pr.wg.Done()
127+
}()
128+
}
129+
if ps.stderr != nil {
130+
pr.wg.Add(1)
131+
go func() {
132+
if _, err := io.Copy(ps.stderr, downstream.Stderr()); err != nil {
133+
log.G(pr.ctx).WithError(err).Error("error copying stderr from pipe")
134+
}
135+
pr.wg.Done()
136+
}()
137+
}
138+
return pr
139+
}
140+
141+
func (pr *pipeRelay) wait() {
142+
pr.wg.Wait()
143+
}
144+
145+
// closeIO closes stdin to unblock an waiters
146+
func (pr *pipeRelay) closeIO() {
147+
if pr.ps.stdin != nil {
148+
pr.ps.Close()
149+
pr.io.Stdin().Close()
150+
}
151+
}
152+
153+
// close closes all open pipes
154+
func (pr *pipeRelay) close() {
155+
pr.once.Do(func() {
156+
pr.io.Close()
157+
pr.ps.Close()
158+
})
159+
}

0 commit comments

Comments
 (0)