You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

syscall_illumos.go 4.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. // Copyright 2021 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.
  4. // illumos system calls not present on Solaris.
  5. //go:build amd64 && illumos
  6. // +build amd64,illumos
  7. package unix
  8. import (
  9. "fmt"
  10. "runtime"
  11. "unsafe"
  12. )
  13. func bytes2iovec(bs [][]byte) []Iovec {
  14. iovecs := make([]Iovec, len(bs))
  15. for i, b := range bs {
  16. iovecs[i].SetLen(len(b))
  17. if len(b) > 0 {
  18. iovecs[i].Base = &b[0]
  19. } else {
  20. iovecs[i].Base = (*byte)(unsafe.Pointer(&_zero))
  21. }
  22. }
  23. return iovecs
  24. }
  25. //sys readv(fd int, iovs []Iovec) (n int, err error)
  26. func Readv(fd int, iovs [][]byte) (n int, err error) {
  27. iovecs := bytes2iovec(iovs)
  28. n, err = readv(fd, iovecs)
  29. return n, err
  30. }
  31. //sys preadv(fd int, iovs []Iovec, off int64) (n int, err error)
  32. func Preadv(fd int, iovs [][]byte, off int64) (n int, err error) {
  33. iovecs := bytes2iovec(iovs)
  34. n, err = preadv(fd, iovecs, off)
  35. return n, err
  36. }
  37. //sys writev(fd int, iovs []Iovec) (n int, err error)
  38. func Writev(fd int, iovs [][]byte) (n int, err error) {
  39. iovecs := bytes2iovec(iovs)
  40. n, err = writev(fd, iovecs)
  41. return n, err
  42. }
  43. //sys pwritev(fd int, iovs []Iovec, off int64) (n int, err error)
  44. func Pwritev(fd int, iovs [][]byte, off int64) (n int, err error) {
  45. iovecs := bytes2iovec(iovs)
  46. n, err = pwritev(fd, iovecs, off)
  47. return n, err
  48. }
  49. //sys accept4(s int, rsa *RawSockaddrAny, addrlen *_Socklen, flags int) (fd int, err error) = libsocket.accept4
  50. func Accept4(fd int, flags int) (nfd int, sa Sockaddr, err error) {
  51. var rsa RawSockaddrAny
  52. var len _Socklen = SizeofSockaddrAny
  53. nfd, err = accept4(fd, &rsa, &len, flags)
  54. if err != nil {
  55. return
  56. }
  57. if len > SizeofSockaddrAny {
  58. panic("RawSockaddrAny too small")
  59. }
  60. sa, err = anyToSockaddr(fd, &rsa)
  61. if err != nil {
  62. Close(nfd)
  63. nfd = 0
  64. }
  65. return
  66. }
  67. //sys putmsg(fd int, clptr *strbuf, dataptr *strbuf, flags int) (err error)
  68. func Putmsg(fd int, cl []byte, data []byte, flags int) (err error) {
  69. var clp, datap *strbuf
  70. if len(cl) > 0 {
  71. clp = &strbuf{
  72. Len: int32(len(cl)),
  73. Buf: (*int8)(unsafe.Pointer(&cl[0])),
  74. }
  75. }
  76. if len(data) > 0 {
  77. datap = &strbuf{
  78. Len: int32(len(data)),
  79. Buf: (*int8)(unsafe.Pointer(&data[0])),
  80. }
  81. }
  82. return putmsg(fd, clp, datap, flags)
  83. }
  84. //sys getmsg(fd int, clptr *strbuf, dataptr *strbuf, flags *int) (err error)
  85. func Getmsg(fd int, cl []byte, data []byte) (retCl []byte, retData []byte, flags int, err error) {
  86. var clp, datap *strbuf
  87. if len(cl) > 0 {
  88. clp = &strbuf{
  89. Maxlen: int32(len(cl)),
  90. Buf: (*int8)(unsafe.Pointer(&cl[0])),
  91. }
  92. }
  93. if len(data) > 0 {
  94. datap = &strbuf{
  95. Maxlen: int32(len(data)),
  96. Buf: (*int8)(unsafe.Pointer(&data[0])),
  97. }
  98. }
  99. if err = getmsg(fd, clp, datap, &flags); err != nil {
  100. return nil, nil, 0, err
  101. }
  102. if len(cl) > 0 {
  103. retCl = cl[:clp.Len]
  104. }
  105. if len(data) > 0 {
  106. retData = data[:datap.Len]
  107. }
  108. return retCl, retData, flags, nil
  109. }
  110. func IoctlSetIntRetInt(fd int, req uint, arg int) (int, error) {
  111. return ioctlRet(fd, req, uintptr(arg))
  112. }
  113. func IoctlSetString(fd int, req uint, val string) error {
  114. bs := make([]byte, len(val)+1)
  115. copy(bs[:len(bs)-1], val)
  116. err := ioctl(fd, req, uintptr(unsafe.Pointer(&bs[0])))
  117. runtime.KeepAlive(&bs[0])
  118. return err
  119. }
  120. // Lifreq Helpers
  121. func (l *Lifreq) SetName(name string) error {
  122. if len(name) >= len(l.Name) {
  123. return fmt.Errorf("name cannot be more than %d characters", len(l.Name)-1)
  124. }
  125. for i := range name {
  126. l.Name[i] = int8(name[i])
  127. }
  128. return nil
  129. }
  130. func (l *Lifreq) SetLifruInt(d int) {
  131. *(*int)(unsafe.Pointer(&l.Lifru[0])) = d
  132. }
  133. func (l *Lifreq) GetLifruInt() int {
  134. return *(*int)(unsafe.Pointer(&l.Lifru[0]))
  135. }
  136. func (l *Lifreq) SetLifruUint(d uint) {
  137. *(*uint)(unsafe.Pointer(&l.Lifru[0])) = d
  138. }
  139. func (l *Lifreq) GetLifruUint() uint {
  140. return *(*uint)(unsafe.Pointer(&l.Lifru[0]))
  141. }
  142. func IoctlLifreq(fd int, req uint, l *Lifreq) error {
  143. return ioctl(fd, req, uintptr(unsafe.Pointer(l)))
  144. }
  145. // Strioctl Helpers
  146. func (s *Strioctl) SetInt(i int) {
  147. s.Len = int32(unsafe.Sizeof(i))
  148. s.Dp = (*int8)(unsafe.Pointer(&i))
  149. }
  150. func IoctlSetStrioctlRetInt(fd int, req uint, s *Strioctl) (int, error) {
  151. return ioctlRet(fd, req, uintptr(unsafe.Pointer(s)))
  152. }