Use __builtin_memmove for trivially copyable types
[official-gcc.git] / libgo / go / syscall / exec_bsd.go
blob9115cf0c9035886147c6b3958d7abf76e9335abc
1 // Copyright 2011 The Go Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
5 // +build aix darwin dragonfly freebsd netbsd openbsd solaris
7 package syscall
9 import (
10 "unsafe"
13 type SysProcAttr struct {
14 Chroot string // Chroot.
15 Credential *Credential // Credential.
16 Ptrace bool // Enable tracing.
17 Setsid bool // Create session.
18 Setpgid bool // Set process group ID to Pgid, or, if Pgid == 0, to new pid.
19 Setctty bool // Set controlling terminal to fd Ctty
20 Noctty bool // Detach fd 0 from controlling terminal
21 Ctty int // Controlling TTY fd
22 Foreground bool // Place child's process group in foreground. (Implies Setpgid. Uses Ctty as fd of controlling TTY)
23 Pgid int // Child's process group ID if Setpgid.
26 // Implemented in runtime package.
27 func runtime_BeforeFork()
28 func runtime_AfterFork()
29 func runtime_AfterForkInChild()
31 // Fork, dup fd onto 0..len(fd), and exec(argv0, argvv, envv) in child.
32 // If a dup or exec fails, write the errno error to pipe.
33 // (Pipe is close-on-exec so if exec succeeds, it will be closed.)
34 // In the child, this function must not acquire any locks, because
35 // they might have been locked at the time of the fork. This means
36 // no rescheduling, no malloc calls, and no new stack segments.
37 // For the same reason compiler does not race instrument it.
38 // The calls to RawSyscall are okay because they are assembly
39 // functions that do not grow the stack.
40 //go:norace
41 func forkAndExecInChild(argv0 *byte, argv, envv []*byte, chroot, dir *byte, attr *ProcAttr, sys *SysProcAttr, pipe int) (pid int, err Errno) {
42 // Declare all variables at top in case any
43 // declarations require heap allocation (e.g., err1).
44 var (
45 r1 Pid_t
46 err1 Errno
47 nextfd int
48 i int
51 // guard against side effects of shuffling fds below.
52 // Make sure that nextfd is beyond any currently open files so
53 // that we can't run the risk of overwriting any of them.
54 fd := make([]int, len(attr.Files))
55 nextfd = len(attr.Files)
56 for i, ufd := range attr.Files {
57 if nextfd < int(ufd) {
58 nextfd = int(ufd)
60 fd[i] = int(ufd)
62 nextfd++
64 // About to call fork.
65 // No more allocation or calls of non-assembly functions.
66 runtime_BeforeFork()
67 r1, err1 = raw_fork()
68 if err1 != 0 {
69 runtime_AfterFork()
70 return 0, err1
73 if r1 != 0 {
74 // parent; return PID
75 runtime_AfterFork()
76 return int(r1), 0
79 // Fork succeeded, now in child.
81 runtime_AfterForkInChild()
83 // Enable tracing if requested.
84 if sys.Ptrace {
85 err1 = raw_ptrace(_PTRACE_TRACEME, 0, nil, nil)
86 if err1 != 0 {
87 goto childerror
91 // Session ID
92 if sys.Setsid {
93 err1 = raw_setsid()
94 if err1 != 0 {
95 goto childerror
99 // Set process group
100 if sys.Setpgid || sys.Foreground {
101 // Place child in process group.
102 err1 = raw_setpgid(0, sys.Pgid)
103 if err1 != 0 {
104 goto childerror
108 if sys.Foreground {
109 pgrp := Pid_t(sys.Pgid)
110 if pgrp == 0 {
111 pgrp = raw_getpid()
114 // Place process group in foreground.
115 _, err1 = raw_ioctl_ptr(sys.Ctty, TIOCSPGRP, unsafe.Pointer(&pgrp))
116 if err1 != 0 {
117 goto childerror
121 // Chroot
122 if chroot != nil {
123 err1 = raw_chroot(chroot)
124 if err1 != 0 {
125 goto childerror
129 // User and groups
130 if cred := sys.Credential; cred != nil {
131 ngroups := len(cred.Groups)
132 var groups *Gid_t
133 if ngroups > 0 {
134 gids := make([]Gid_t, ngroups)
135 for i, v := range cred.Groups {
136 gids[i] = Gid_t(v)
138 groups = &gids[0]
140 if !cred.NoSetGroups {
141 err2 := setgroups(ngroups, groups)
142 if err2 == nil {
143 err1 = 0
144 } else {
145 err1 = err2.(Errno)
147 if err1 != 0 {
148 goto childerror
151 err2 := Setgid(int(cred.Gid))
152 if err2 != nil {
153 err1 = err2.(Errno)
154 goto childerror
156 err2 = Setuid(int(cred.Uid))
157 if err2 != nil {
158 err1 = err2.(Errno)
159 goto childerror
163 // Chdir
164 if dir != nil {
165 err1 = raw_chdir(dir)
166 if err1 != 0 {
167 goto childerror
171 // Pass 1: look for fd[i] < i and move those up above len(fd)
172 // so that pass 2 won't stomp on an fd it needs later.
173 if pipe < nextfd {
174 err1 = raw_dup2(pipe, nextfd)
175 if err1 != 0 {
176 goto childerror
178 raw_fcntl(nextfd, F_SETFD, FD_CLOEXEC)
179 pipe = nextfd
180 nextfd++
182 for i = 0; i < len(fd); i++ {
183 if fd[i] >= 0 && fd[i] < int(i) {
184 if nextfd == pipe { // don't stomp on pipe
185 nextfd++
187 err1 = raw_dup2(fd[i], nextfd)
188 if err1 != 0 {
189 goto childerror
191 raw_fcntl(nextfd, F_SETFD, FD_CLOEXEC)
192 fd[i] = nextfd
193 nextfd++
197 // Pass 2: dup fd[i] down onto i.
198 for i = 0; i < len(fd); i++ {
199 if fd[i] == -1 {
200 raw_close(i)
201 continue
203 if fd[i] == int(i) {
204 // dup2(i, i) won't clear close-on-exec flag on Linux,
205 // probably not elsewhere either.
206 _, err1 = raw_fcntl(fd[i], F_SETFD, 0)
207 if err1 != 0 {
208 goto childerror
210 continue
212 // The new fd is created NOT close-on-exec,
213 // which is exactly what we want.
214 err1 = raw_dup2(fd[i], i)
215 if err1 != 0 {
216 goto childerror
220 // By convention, we don't close-on-exec the fds we are
221 // started with, so if len(fd) < 3, close 0, 1, 2 as needed.
222 // Programs that know they inherit fds >= 3 will need
223 // to set them close-on-exec.
224 for i = len(fd); i < 3; i++ {
225 raw_close(i)
228 // Detach fd 0 from tty
229 if sys.Noctty {
230 _, err1 = raw_ioctl(0, TIOCNOTTY, 0)
231 if err1 != 0 {
232 goto childerror
236 // Set the controlling TTY to Ctty
237 if sys.Setctty {
238 if TIOCSCTTY == 0 {
239 err1 = ENOSYS
240 goto childerror
242 _, err1 = raw_ioctl(sys.Ctty, TIOCSCTTY, 0)
243 if err1 != 0 {
244 goto childerror
248 // Time to exec.
249 err1 = raw_execve(argv0, &argv[0], &envv[0])
251 childerror:
252 // send error code on pipe
253 raw_write(pipe, (*byte)(unsafe.Pointer(&err1)), int(unsafe.Sizeof(err1)))
254 for {
255 raw_exit(253)