aboutsummaryrefslogtreecommitdiff
path: root/src/main.zig
diff options
context:
space:
mode:
authorJakub Konka <kubkon@jakubkonka.com>2020-11-03 10:07:39 +0100
committerGitHub <noreply@github.com>2020-11-03 10:07:39 +0100
commit50604971745c0ddd2373bba933c30b59db624d2b (patch)
tree6bbf752e96849ad05ac269ba769daebc8bb5937e /src/main.zig
parentb30a765b95cf666409c1e5324718a7c8c60af4af (diff)
parente023a5fe5d8e0c2849c160ffab993ec1e378681c (diff)
downloadzig-50604971745c0ddd2373bba933c30b59db624d2b.tar.gz
zig-50604971745c0ddd2373bba933c30b59db624d2b.zip
Merge pull request #6921 from xackus/gimmeMoreOfThoseSweetSweetFileDescriptors
stage2: ask for more file descriptors
Diffstat (limited to 'src/main.zig')
-rw-r--r--src/main.zig52
1 files changed, 27 insertions, 25 deletions
diff --git a/src/main.zig b/src/main.zig
index 45fcb5ce61..7df8cb1eda 100644
--- a/src/main.zig
+++ b/src/main.zig
@@ -2982,35 +2982,37 @@ fn parseCodeModel(arg: []const u8) std.builtin.CodeModel {
/// garbage collector to run concurrently to zig processes, and to allow multiple
/// zig processes to run concurrently with each other, without clobbering each other.
fn gimmeMoreOfThoseSweetSweetFileDescriptors() void {
- switch (std.Target.current.os.tag) {
- .windows, .wasi, .uefi, .other, .freestanding => return,
- // std lib is missing getrlimit/setrlimit.
- // https://github.com/ziglang/zig/issues/6361
- //else => {},
- else => return,
- }
+ if (!@hasDecl(std.os, "rlimit")) return;
const posix = std.os;
- var lim = posix.getrlimit(posix.RLIMIT_NOFILE, &lim) catch return; // Oh well; we tried.
+
+ var lim = posix.getrlimit(.NOFILE) catch return; // Oh well; we tried.
+ if (comptime std.Target.current.isDarwin()) {
+ // On Darwin, `NOFILE` is bounded by a hardcoded value `OPEN_MAX`.
+ // According to the man pages for setrlimit():
+ // setrlimit() now returns with errno set to EINVAL in places that historically succeeded.
+ // It no longer accepts "rlim_cur = RLIM_INFINITY" for RLIM_NOFILE.
+ // Use "rlim_cur = min(OPEN_MAX, rlim_max)".
+ lim.max = std.math.min(std.os.darwin.OPEN_MAX, lim.max);
+ }
if (lim.cur == lim.max) return;
+
+ // Do a binary search for the limit.
+ var min: posix.rlim_t = lim.cur;
+ var max: posix.rlim_t = 1 << 20;
+ // But if there's a defined upper bound, don't search, just set it.
+ if (lim.max != posix.RLIM_INFINITY) {
+ min = lim.max;
+ max = lim.max;
+ }
+
while (true) {
- // Do a binary search for the limit.
- var min: posix.rlim_t = lim.cur;
- var max: posix.rlim_t = 1 << 20;
- // But if there's a defined upper bound, don't search, just set it.
- if (lim.max != posix.RLIM_INFINITY) {
- min = lim.max;
- max = lim.max;
- }
- while (true) {
- lim.cur = min + (max - min) / 2;
- if (posix.setrlimit(posix.RLIMIT_NOFILE, lim)) |_| {
- min = lim.cur;
- } else |_| {
- max = lim.cur;
- }
- if (min + 1 < max) continue;
- return;
+ lim.cur = min + @divTrunc(max - min, 2); // on freebsd rlim_t is signed
+ if (posix.setrlimit(.NOFILE, lim)) |_| {
+ min = lim.cur;
+ } else |_| {
+ max = lim.cur;
}
+ if (min + 1 >= max) break;
}
}