Source file
src/syscall/syscall_bsd.go
Documentation: syscall
1
2
3
4
5
6
7
8
9
10
11
12
13 package syscall
14
15 import (
16 "runtime"
17 "unsafe"
18 )
19
20 const ImplementsGetwd = true
21
22 func Getwd() (string, error) {
23 var buf [pathMax]byte
24 _, err := getcwd(buf[:])
25 if err != nil {
26 return "", err
27 }
28 n := clen(buf[:])
29 if n < 1 {
30 return "", EINVAL
31 }
32 return string(buf[:n]), nil
33 }
34
35
38
39
40
41
42 func Getgroups() (gids []int, err error) {
43 n, err := getgroups(0, nil)
44 if err != nil {
45 return nil, err
46 }
47 if n == 0 {
48 return nil, nil
49 }
50
51
52 if n < 0 || n > 1000 {
53 return nil, EINVAL
54 }
55
56 a := make([]_Gid_t, n)
57 n, err = getgroups(n, &a[0])
58 if err != nil {
59 return nil, err
60 }
61 gids = make([]int, n)
62 for i, v := range a[0:n] {
63 gids[i] = int(v)
64 }
65 return
66 }
67
68 func Setgroups(gids []int) (err error) {
69 if len(gids) == 0 {
70 return setgroups(0, nil)
71 }
72
73 a := make([]_Gid_t, len(gids))
74 for i, v := range gids {
75 a[i] = _Gid_t(v)
76 }
77 return setgroups(len(a), &a[0])
78 }
79
80 func ReadDirent(fd int, buf []byte) (n int, err error) {
81
82
83
84
85 var base = (*uintptr)(unsafe.Pointer(new(uint64)))
86 return Getdirentries(fd, buf, base)
87 }
88
89
90
91
92
93
94
95 type WaitStatus uint32
96
97 const (
98 mask = 0x7F
99 core = 0x80
100 shift = 8
101
102 exited = 0
103 stopped = 0x7F
104 )
105
106 func (w WaitStatus) Exited() bool { return w&mask == exited }
107
108 func (w WaitStatus) ExitStatus() int {
109 if w&mask != exited {
110 return -1
111 }
112 return int(w >> shift)
113 }
114
115 func (w WaitStatus) Signaled() bool { return w&mask != stopped && w&mask != 0 }
116
117 func (w WaitStatus) Signal() Signal {
118 sig := Signal(w & mask)
119 if sig == stopped || sig == 0 {
120 return -1
121 }
122 return sig
123 }
124
125 func (w WaitStatus) CoreDump() bool { return w.Signaled() && w&core != 0 }
126
127 func (w WaitStatus) Stopped() bool { return w&mask == stopped && Signal(w>>shift) != SIGSTOP }
128
129 func (w WaitStatus) Continued() bool { return w&mask == stopped && Signal(w>>shift) == SIGSTOP }
130
131 func (w WaitStatus) StopSignal() Signal {
132 if !w.Stopped() {
133 return -1
134 }
135 return Signal(w>>shift) & 0xFF
136 }
137
138 func (w WaitStatus) TrapCause() int { return -1 }
139
140
141
142 func Wait4(pid int, wstatus *WaitStatus, options int, rusage *Rusage) (wpid int, err error) {
143 var status _C_int
144 wpid, err = wait4(pid, &status, options, rusage)
145 if wstatus != nil {
146 *wstatus = WaitStatus(status)
147 }
148 return
149 }
150
151
152
153
154
155
156
157
158
159
160
161 func (sa *SockaddrInet4) sockaddr() (unsafe.Pointer, _Socklen, error) {
162 if sa.Port < 0 || sa.Port > 0xFFFF {
163 return nil, 0, EINVAL
164 }
165 sa.raw.Len = SizeofSockaddrInet4
166 sa.raw.Family = AF_INET
167 p := (*[2]byte)(unsafe.Pointer(&sa.raw.Port))
168 p[0] = byte(sa.Port >> 8)
169 p[1] = byte(sa.Port)
170 sa.raw.Addr = sa.Addr
171 return unsafe.Pointer(&sa.raw), _Socklen(sa.raw.Len), nil
172 }
173
174 func (sa *SockaddrInet6) sockaddr() (unsafe.Pointer, _Socklen, error) {
175 if sa.Port < 0 || sa.Port > 0xFFFF {
176 return nil, 0, EINVAL
177 }
178 sa.raw.Len = SizeofSockaddrInet6
179 sa.raw.Family = AF_INET6
180 p := (*[2]byte)(unsafe.Pointer(&sa.raw.Port))
181 p[0] = byte(sa.Port >> 8)
182 p[1] = byte(sa.Port)
183 sa.raw.Scope_id = sa.ZoneId
184 sa.raw.Addr = sa.Addr
185 return unsafe.Pointer(&sa.raw), _Socklen(sa.raw.Len), nil
186 }
187
188 func (sa *SockaddrUnix) sockaddr() (unsafe.Pointer, _Socklen, error) {
189 name := sa.Name
190 n := len(name)
191 if n >= len(sa.raw.Path) || n == 0 {
192 return nil, 0, EINVAL
193 }
194 sa.raw.Len = byte(3 + n)
195 sa.raw.Family = AF_UNIX
196 for i := 0; i < n; i++ {
197 sa.raw.Path[i] = int8(name[i])
198 }
199 return unsafe.Pointer(&sa.raw), _Socklen(sa.raw.Len), nil
200 }
201
202 func (sa *SockaddrDatalink) sockaddr() (unsafe.Pointer, _Socklen, error) {
203 if sa.Index == 0 {
204 return nil, 0, EINVAL
205 }
206 sa.raw.Len = sa.Len
207 sa.raw.Family = AF_LINK
208 sa.raw.Index = sa.Index
209 sa.raw.Type = sa.Type
210 sa.raw.Nlen = sa.Nlen
211 sa.raw.Alen = sa.Alen
212 sa.raw.Slen = sa.Slen
213 sa.raw.Data = sa.Data
214 return unsafe.Pointer(&sa.raw), SizeofSockaddrDatalink, nil
215 }
216
217 func anyToSockaddr(rsa *RawSockaddrAny) (Sockaddr, error) {
218 switch rsa.Addr.Family {
219 case AF_LINK:
220 pp := (*RawSockaddrDatalink)(unsafe.Pointer(rsa))
221 sa := new(SockaddrDatalink)
222 sa.Len = pp.Len
223 sa.Family = pp.Family
224 sa.Index = pp.Index
225 sa.Type = pp.Type
226 sa.Nlen = pp.Nlen
227 sa.Alen = pp.Alen
228 sa.Slen = pp.Slen
229 sa.Data = pp.Data
230 return sa, nil
231
232 case AF_UNIX:
233 pp := (*RawSockaddrUnix)(unsafe.Pointer(rsa))
234 if pp.Len < 2 || pp.Len > SizeofSockaddrUnix {
235 return nil, EINVAL
236 }
237 sa := new(SockaddrUnix)
238
239
240
241
242
243 n := int(pp.Len) - 2
244 for i := 0; i < n; i++ {
245 if pp.Path[i] == 0 {
246
247
248 n = i
249 break
250 }
251 }
252 sa.Name = string(unsafe.Slice((*byte)(unsafe.Pointer(&pp.Path[0])), n))
253 return sa, nil
254
255 case AF_INET:
256 pp := (*RawSockaddrInet4)(unsafe.Pointer(rsa))
257 sa := new(SockaddrInet4)
258 p := (*[2]byte)(unsafe.Pointer(&pp.Port))
259 sa.Port = int(p[0])<<8 + int(p[1])
260 sa.Addr = pp.Addr
261 return sa, nil
262
263 case AF_INET6:
264 pp := (*RawSockaddrInet6)(unsafe.Pointer(rsa))
265 sa := new(SockaddrInet6)
266 p := (*[2]byte)(unsafe.Pointer(&pp.Port))
267 sa.Port = int(p[0])<<8 + int(p[1])
268 sa.ZoneId = pp.Scope_id
269 sa.Addr = pp.Addr
270 return sa, nil
271 }
272 return nil, EAFNOSUPPORT
273 }
274
275 func Accept(fd int) (nfd int, sa Sockaddr, err error) {
276 var rsa RawSockaddrAny
277 var len _Socklen = SizeofSockaddrAny
278 nfd, err = accept(fd, &rsa, &len)
279 if err != nil {
280 return
281 }
282 if (runtime.GOOS == "darwin" || runtime.GOOS == "ios") && len == 0 {
283
284
285
286
287 Close(nfd)
288 return 0, nil, ECONNABORTED
289 }
290 sa, err = anyToSockaddr(&rsa)
291 if err != nil {
292 Close(nfd)
293 nfd = 0
294 }
295 return
296 }
297
298 func Getsockname(fd int) (sa Sockaddr, err error) {
299 var rsa RawSockaddrAny
300 var len _Socklen = SizeofSockaddrAny
301 if err = getsockname(fd, &rsa, &len); err != nil {
302 return
303 }
304
305
306 if runtime.GOOS == "dragonfly" && rsa.Addr.Family == AF_UNSPEC && rsa.Addr.Len == 0 {
307 rsa.Addr.Family = AF_UNIX
308 rsa.Addr.Len = SizeofSockaddrUnix
309 }
310 return anyToSockaddr(&rsa)
311 }
312
313
314
315 func GetsockoptByte(fd, level, opt int) (value byte, err error) {
316 var n byte
317 vallen := _Socklen(1)
318 err = getsockopt(fd, level, opt, unsafe.Pointer(&n), &vallen)
319 return n, err
320 }
321
322 func GetsockoptInet4Addr(fd, level, opt int) (value [4]byte, err error) {
323 vallen := _Socklen(4)
324 err = getsockopt(fd, level, opt, unsafe.Pointer(&value[0]), &vallen)
325 return value, err
326 }
327
328 func GetsockoptIPMreq(fd, level, opt int) (*IPMreq, error) {
329 var value IPMreq
330 vallen := _Socklen(SizeofIPMreq)
331 err := getsockopt(fd, level, opt, unsafe.Pointer(&value), &vallen)
332 return &value, err
333 }
334
335 func GetsockoptIPv6Mreq(fd, level, opt int) (*IPv6Mreq, error) {
336 var value IPv6Mreq
337 vallen := _Socklen(SizeofIPv6Mreq)
338 err := getsockopt(fd, level, opt, unsafe.Pointer(&value), &vallen)
339 return &value, err
340 }
341
342 func GetsockoptIPv6MTUInfo(fd, level, opt int) (*IPv6MTUInfo, error) {
343 var value IPv6MTUInfo
344 vallen := _Socklen(SizeofIPv6MTUInfo)
345 err := getsockopt(fd, level, opt, unsafe.Pointer(&value), &vallen)
346 return &value, err
347 }
348
349 func GetsockoptICMPv6Filter(fd, level, opt int) (*ICMPv6Filter, error) {
350 var value ICMPv6Filter
351 vallen := _Socklen(SizeofICMPv6Filter)
352 err := getsockopt(fd, level, opt, unsafe.Pointer(&value), &vallen)
353 return &value, err
354 }
355
356
357
358
359
360 func recvmsgRaw(fd int, p, oob []byte, flags int, rsa *RawSockaddrAny) (n, oobn int, recvflags int, err error) {
361 var msg Msghdr
362 msg.Name = (*byte)(unsafe.Pointer(rsa))
363 msg.Namelen = uint32(SizeofSockaddrAny)
364 var iov Iovec
365 if len(p) > 0 {
366 iov.Base = &p[0]
367 iov.SetLen(len(p))
368 }
369 var dummy byte
370 if len(oob) > 0 {
371
372 if len(p) == 0 {
373 iov.Base = &dummy
374 iov.SetLen(1)
375 }
376 msg.Control = &oob[0]
377 msg.SetControllen(len(oob))
378 }
379 msg.Iov = &iov
380 msg.Iovlen = 1
381 if n, err = recvmsg(fd, &msg, flags); err != nil {
382 return
383 }
384 oobn = int(msg.Controllen)
385 recvflags = int(msg.Flags)
386 return
387 }
388
389
390
391 func sendmsgN(fd int, p, oob []byte, ptr unsafe.Pointer, salen _Socklen, flags int) (n int, err error) {
392 var msg Msghdr
393 msg.Name = (*byte)(ptr)
394 msg.Namelen = uint32(salen)
395 var iov Iovec
396 if len(p) > 0 {
397 iov.Base = &p[0]
398 iov.SetLen(len(p))
399 }
400 var dummy byte
401 if len(oob) > 0 {
402
403 if len(p) == 0 {
404 iov.Base = &dummy
405 iov.SetLen(1)
406 }
407 msg.Control = &oob[0]
408 msg.SetControllen(len(oob))
409 }
410 msg.Iov = &iov
411 msg.Iovlen = 1
412 if n, err = sendmsg(fd, &msg, flags); err != nil {
413 return 0, err
414 }
415 if len(oob) > 0 && len(p) == 0 {
416 n = 0
417 }
418 return n, nil
419 }
420
421
422
423 func Kevent(kq int, changes, events []Kevent_t, timeout *Timespec) (n int, err error) {
424 var change, event unsafe.Pointer
425 if len(changes) > 0 {
426 change = unsafe.Pointer(&changes[0])
427 }
428 if len(events) > 0 {
429 event = unsafe.Pointer(&events[0])
430 }
431 return kevent(kq, change, len(changes), event, len(events), timeout)
432 }
433
434 func Sysctl(name string) (value string, err error) {
435
436 mib, err := nametomib(name)
437 if err != nil {
438 return "", err
439 }
440
441
442 n := uintptr(0)
443 if err = sysctl(mib, nil, &n, nil, 0); err != nil {
444 return "", err
445 }
446 if n == 0 {
447 return "", nil
448 }
449
450
451 buf := make([]byte, n)
452 if err = sysctl(mib, &buf[0], &n, nil, 0); err != nil {
453 return "", err
454 }
455
456
457 if n > 0 && buf[n-1] == '\x00' {
458 n--
459 }
460 return string(buf[0:n]), nil
461 }
462
463 func SysctlUint32(name string) (value uint32, err error) {
464
465 mib, err := nametomib(name)
466 if err != nil {
467 return 0, err
468 }
469
470
471 n := uintptr(4)
472 buf := make([]byte, 4)
473 if err = sysctl(mib, &buf[0], &n, nil, 0); err != nil {
474 return 0, err
475 }
476 if n != 4 {
477 return 0, EIO
478 }
479 return *(*uint32)(unsafe.Pointer(&buf[0])), nil
480 }
481
482
483
484 func Utimes(path string, tv []Timeval) (err error) {
485 if len(tv) != 2 {
486 return EINVAL
487 }
488 return utimes(path, (*[2]Timeval)(unsafe.Pointer(&tv[0])))
489 }
490
491 func UtimesNano(path string, ts []Timespec) error {
492 if len(ts) != 2 {
493 return EINVAL
494 }
495 err := utimensat(_AT_FDCWD, path, (*[2]Timespec)(unsafe.Pointer(&ts[0])), 0)
496 if err != ENOSYS {
497 return err
498 }
499
500
501 tv := [2]Timeval{
502 NsecToTimeval(TimespecToNsec(ts[0])),
503 NsecToTimeval(TimespecToNsec(ts[1])),
504 }
505 return utimes(path, (*[2]Timeval)(unsafe.Pointer(&tv[0])))
506 }
507
508
509
510 func Futimes(fd int, tv []Timeval) (err error) {
511 if len(tv) != 2 {
512 return EINVAL
513 }
514 return futimes(fd, (*[2]Timeval)(unsafe.Pointer(&tv[0])))
515 }
516
517
518
519
520
521
522 var mapper = &mmapper{
523 active: make(map[*byte][]byte),
524 mmap: mmap,
525 munmap: munmap,
526 }
527
528 func Mmap(fd int, offset int64, length int, prot int, flags int) (data []byte, err error) {
529 return mapper.Mmap(fd, offset, length, prot, flags)
530 }
531
532 func Munmap(b []byte) (err error) {
533 return mapper.Munmap(b)
534 }
535
View as plain text