sysconf_darwin.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. // Copyright 2018 Tobias Klauser. 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. package sysconf
  5. import (
  6. "strconv"
  7. "strings"
  8. "sync"
  9. "golang.org/x/sys/unix"
  10. )
  11. const (
  12. _HOST_NAME_MAX = _MAXHOSTNAMELEN - 1
  13. _LOGIN_NAME_MAX = _MAXLOGNAME
  14. _SYMLOOP_MAX = _MAXSYMLINKS
  15. )
  16. var uname struct {
  17. sync.Once
  18. macOSMajor int
  19. }
  20. // sysconf implements sysconf(4) as in the Darwin libc (derived from the FreeBSD
  21. // libc), version 1534.81.1.
  22. // See https://github.com/apple-oss-distributions/Libc/tree/Libc-1534.81.1.
  23. func sysconf(name int) (int64, error) {
  24. switch name {
  25. case SC_AIO_LISTIO_MAX:
  26. fallthrough
  27. case SC_AIO_MAX:
  28. return sysctl32("kern.aiomax"), nil
  29. case SC_AIO_PRIO_DELTA_MAX:
  30. return -1, nil
  31. case SC_ARG_MAX:
  32. return sysctl32("kern.argmax"), nil
  33. case SC_ATEXIT_MAX:
  34. return _INT_MAX, nil
  35. case SC_CHILD_MAX:
  36. var rlim unix.Rlimit
  37. if err := unix.Getrlimit(unix.RLIMIT_NPROC, &rlim); err == nil {
  38. if rlim.Cur != unix.RLIM_INFINITY {
  39. return int64(rlim.Cur), nil
  40. }
  41. }
  42. return -1, nil
  43. case SC_CLK_TCK:
  44. return _CLK_TCK, nil
  45. case SC_DELAYTIMER_MAX:
  46. return -1, nil
  47. case SC_GETGR_R_SIZE_MAX:
  48. return 4096, nil
  49. case SC_GETPW_R_SIZE_MAX:
  50. return 4096, nil
  51. case SC_IOV_MAX:
  52. return _IOV_MAX, nil
  53. case SC_MQ_OPEN_MAX:
  54. return -1, nil
  55. case SC_MQ_PRIO_MAX:
  56. return -1, nil
  57. case SC_NGROUPS_MAX:
  58. return sysctl32("kern.ngroups"), nil
  59. case SC_OPEN_MAX, SC_STREAM_MAX:
  60. var rlim unix.Rlimit
  61. if err := unix.Getrlimit(unix.RLIMIT_NOFILE, &rlim); err != nil {
  62. return -1, nil
  63. }
  64. if rlim.Cur > unix.RLIM_INFINITY {
  65. return -1, nil
  66. }
  67. if rlim.Cur > _LONG_MAX {
  68. return -1, unix.EOVERFLOW
  69. }
  70. return int64(rlim.Cur), nil
  71. case SC_RTSIG_MAX:
  72. return -1, nil
  73. case SC_SEM_NSEMS_MAX:
  74. return sysctl32("kern.sysv.semmns"), nil
  75. case SC_SEM_VALUE_MAX:
  76. return _POSIX_SEM_VALUE_MAX, nil
  77. case SC_SIGQUEUE_MAX:
  78. return -1, nil
  79. case SC_THREAD_DESTRUCTOR_ITERATIONS:
  80. return _PTHREAD_DESTRUCTOR_ITERATIONS, nil
  81. case SC_THREAD_KEYS_MAX:
  82. return _PTHREAD_KEYS_MAX, nil
  83. case SC_THREAD_PRIO_INHERIT:
  84. return _POSIX_THREAD_PRIO_INHERIT, nil
  85. case SC_THREAD_PRIO_PROTECT:
  86. return _POSIX_THREAD_PRIO_PROTECT, nil
  87. case SC_THREAD_STACK_MIN:
  88. return _PTHREAD_STACK_MIN, nil
  89. case SC_THREAD_THREADS_MAX:
  90. return -1, nil
  91. case SC_TIMER_MAX:
  92. return -1, nil
  93. case SC_TTY_NAME_MAX:
  94. // should be _PATH_DEV instead of "/"
  95. return pathconf("/", _PC_NAME_MAX), nil
  96. case SC_TZNAME_MAX:
  97. return pathconf(_PATH_ZONEINFO, _PC_NAME_MAX), nil
  98. case SC_IPV6:
  99. if _POSIX_IPV6 == 0 {
  100. fd, err := unix.Socket(unix.AF_INET6, unix.SOCK_DGRAM, 0)
  101. if err == nil && fd >= 0 {
  102. unix.Close(fd)
  103. return int64(200112), nil
  104. }
  105. return 0, nil
  106. }
  107. return _POSIX_IPV6, nil
  108. case SC_MESSAGE_PASSING:
  109. if _POSIX_MESSAGE_PASSING == 0 {
  110. return yesno(sysctl32("p1003_1b.message_passing")), nil
  111. }
  112. return _POSIX_MESSAGE_PASSING, nil
  113. case SC_PRIORITIZED_IO:
  114. if _POSIX_PRIORITIZED_IO == 0 {
  115. return yesno(sysctl32("p1003_1b.prioritized_io")), nil
  116. }
  117. return _POSIX_PRIORITIZED_IO, nil
  118. case SC_PRIORITY_SCHEDULING:
  119. if _POSIX_PRIORITY_SCHEDULING == 0 {
  120. return yesno(sysctl32("p1003_1b.priority_scheduling")), nil
  121. }
  122. return _POSIX_PRIORITY_SCHEDULING, nil
  123. case SC_REALTIME_SIGNALS:
  124. if _POSIX_REALTIME_SIGNALS == 0 {
  125. return yesno(sysctl32("p1003_1b.realtime_signals")), nil
  126. }
  127. return _POSIX_REALTIME_SIGNALS, nil
  128. case SC_SAVED_IDS:
  129. return yesno(sysctl32("kern.saved_ids")), nil
  130. case SC_SEMAPHORES:
  131. if _POSIX_SEMAPHORES == 0 {
  132. return yesno(sysctl32("p1003_1b.semaphores")), nil
  133. }
  134. return _POSIX_SEMAPHORES, nil
  135. case SC_SPAWN:
  136. uname.Once.Do(func() {
  137. var u unix.Utsname
  138. err := unix.Uname(&u)
  139. if err != nil {
  140. return
  141. }
  142. rel := unix.ByteSliceToString(u.Release[:])
  143. ver := strings.Split(rel, ".")
  144. maj, _ := strconv.Atoi(ver[0])
  145. uname.macOSMajor = maj
  146. })
  147. if uname.macOSMajor < 22 {
  148. return -1, nil
  149. }
  150. // macOS 13 (Ventura) and later
  151. return 200112, nil
  152. case SC_SPIN_LOCKS:
  153. return _POSIX_SPIN_LOCKS, nil
  154. case SC_SPORADIC_SERVER:
  155. return _POSIX_SPORADIC_SERVER, nil
  156. case SC_SS_REPL_MAX:
  157. return _POSIX_SS_REPL_MAX, nil
  158. case SC_SYNCHRONIZED_IO:
  159. if _POSIX_SYNCHRONIZED_IO == 0 {
  160. return yesno(sysctl32("p1003_1b.synchronized_io")), nil
  161. }
  162. return _POSIX_SYNCHRONIZED_IO, nil
  163. case SC_THREAD_ATTR_STACKADDR:
  164. return _POSIX_THREAD_ATTR_STACKADDR, nil
  165. case SC_THREAD_ATTR_STACKSIZE:
  166. return _POSIX_THREAD_ATTR_STACKSIZE, nil
  167. case SC_THREAD_CPUTIME:
  168. return _POSIX_THREAD_CPUTIME, nil
  169. case SC_THREAD_PRIORITY_SCHEDULING:
  170. return _POSIX_THREAD_PRIORITY_SCHEDULING, nil
  171. case SC_THREAD_PROCESS_SHARED:
  172. return _POSIX_THREAD_PROCESS_SHARED, nil
  173. case SC_THREAD_SAFE_FUNCTIONS:
  174. return _POSIX_THREAD_SAFE_FUNCTIONS, nil
  175. case SC_THREAD_SPORADIC_SERVER:
  176. return _POSIX_THREAD_SPORADIC_SERVER, nil
  177. case SC_TIMERS:
  178. if _POSIX_TIMERS == 0 {
  179. return yesno(sysctl32("p1003_1b.timers")), nil
  180. }
  181. return _POSIX_TIMERS, nil
  182. case SC_TRACE:
  183. return _POSIX_TRACE, nil
  184. case SC_TRACE_EVENT_FILTER:
  185. return _POSIX_TRACE_EVENT_FILTER, nil
  186. case SC_TRACE_EVENT_NAME_MAX:
  187. return _POSIX_TRACE_EVENT_NAME_MAX, nil
  188. case SC_TRACE_INHERIT:
  189. return _POSIX_TRACE_INHERIT, nil
  190. case SC_TRACE_LOG:
  191. return _POSIX_TRACE_LOG, nil
  192. case SC_TRACE_NAME_MAX:
  193. return _POSIX_TRACE_NAME_MAX, nil
  194. case SC_TRACE_SYS_MAX:
  195. return _POSIX_TRACE_SYS_MAX, nil
  196. case SC_TRACE_USER_EVENT_MAX:
  197. return _POSIX_TRACE_USER_EVENT_MAX, nil
  198. case SC_TYPED_MEMORY_OBJECTS:
  199. return _POSIX_TYPED_MEMORY_OBJECTS, nil
  200. case SC_VERSION:
  201. // TODO(tk): darwin libc uses sysctl(CTL_KERN, KERN_POSIX1)
  202. return _POSIX_VERSION, nil
  203. case SC_V6_ILP32_OFF32:
  204. if _V6_ILP32_OFF32 == 0 {
  205. if unix.SizeofInt*_CHAR_BIT == 32 &&
  206. unix.SizeofInt == unix.SizeofLong &&
  207. unix.SizeofLong == unix.SizeofPtr &&
  208. unix.SizeofPtr == sizeofOffT {
  209. return 1, nil
  210. }
  211. return -1, nil
  212. }
  213. return _V6_ILP32_OFF32, nil
  214. case SC_V6_ILP32_OFFBIG:
  215. if _V6_ILP32_OFFBIG == 0 {
  216. if unix.SizeofInt*_CHAR_BIT == 32 &&
  217. unix.SizeofInt == unix.SizeofLong &&
  218. unix.SizeofLong == unix.SizeofPtr &&
  219. sizeofOffT*_CHAR_BIT >= 64 {
  220. return 1, nil
  221. }
  222. return -1, nil
  223. }
  224. return _V6_ILP32_OFFBIG, nil
  225. case SC_V6_LP64_OFF64:
  226. if _V6_LP64_OFF64 == 0 {
  227. if unix.SizeofInt*_CHAR_BIT == 32 &&
  228. unix.SizeofLong*_CHAR_BIT == 64 &&
  229. unix.SizeofLong == unix.SizeofPtr &&
  230. unix.SizeofPtr == sizeofOffT {
  231. return 1, nil
  232. }
  233. return -1, nil
  234. }
  235. return _V6_LP64_OFF64, nil
  236. case SC_V6_LPBIG_OFFBIG:
  237. if _V6_LPBIG_OFFBIG == 0 {
  238. if unix.SizeofInt*_CHAR_BIT >= 32 &&
  239. unix.SizeofLong*_CHAR_BIT >= 64 &&
  240. unix.SizeofPtr*_CHAR_BIT >= 64 &&
  241. sizeofOffT*_CHAR_BIT >= 64 {
  242. return 1, nil
  243. }
  244. return -1, nil
  245. }
  246. return _V6_LPBIG_OFFBIG, nil
  247. case SC_2_CHAR_TERM:
  248. return _POSIX2_CHAR_TERM, nil
  249. case SC_2_PBS,
  250. SC_2_PBS_ACCOUNTING,
  251. SC_2_PBS_CHECKPOINT,
  252. SC_2_PBS_LOCATE,
  253. SC_2_PBS_MESSAGE,
  254. SC_2_PBS_TRACK:
  255. return _POSIX2_PBS, nil
  256. case SC_2_UPE:
  257. return _POSIX2_UPE, nil
  258. case SC_XOPEN_CRYPT:
  259. return _XOPEN_CRYPT, nil
  260. case SC_XOPEN_ENH_I18N:
  261. return _XOPEN_ENH_I18N, nil
  262. case SC_XOPEN_REALTIME:
  263. return _XOPEN_REALTIME, nil
  264. case SC_XOPEN_REALTIME_THREADS:
  265. return _XOPEN_REALTIME_THREADS, nil
  266. case SC_XOPEN_SHM:
  267. return _XOPEN_SHM, nil
  268. case SC_XOPEN_STREAMS:
  269. return -1, nil
  270. case SC_XOPEN_UNIX:
  271. return _XOPEN_UNIX, nil
  272. case SC_XOPEN_VERSION:
  273. return _XOPEN_VERSION, nil
  274. case SC_XOPEN_XCU_VERSION:
  275. return _XOPEN_XCU_VERSION, nil
  276. case SC_PHYS_PAGES:
  277. return sysctl64("hw.memsize") / int64(unix.Getpagesize()), nil
  278. case SC_NPROCESSORS_CONF:
  279. fallthrough
  280. case SC_NPROCESSORS_ONLN:
  281. return sysctl32("hw.ncpu"), nil
  282. }
  283. return sysconfGeneric(name)
  284. }