aboutsummaryrefslogtreecommitdiff
path: root/lib/std/math.zig
diff options
context:
space:
mode:
authorKate Tsuyu <kate@kxt.io>2020-08-29 00:21:17 -0400
committerKate Tsuyu <kate@kxt.io>2020-08-29 00:21:17 -0400
commit933bfcff6483897a6fcf24f38e0571d2ae960441 (patch)
tree8bb66f5ed01256d0a6433327c2af41e619de58a6 /lib/std/math.zig
parentf577d12fbc6a5136b238b4d8e7e361578a66ddd6 (diff)
downloadzig-933bfcff6483897a6fcf24f38e0571d2ae960441.tar.gz
zig-933bfcff6483897a6fcf24f38e0571d2ae960441.zip
std.math.divCeil: support comptime_int/float
Diffstat (limited to 'lib/std/math.zig')
-rw-r--r--lib/std/math.zig34
1 files changed, 26 insertions, 8 deletions
diff --git a/lib/std/math.zig b/lib/std/math.zig
index 03a945163a..f5aa9e4b36 100644
--- a/lib/std/math.zig
+++ b/lib/std/math.zig
@@ -624,15 +624,21 @@ fn testDivFloor() void {
pub fn divCeil(comptime T: type, numerator: T, denominator: T) !T {
@setRuntimeSafety(false);
if (denominator == 0) return error.DivisionByZero;
- if (@typeInfo(T) == .Float) return @ceil(numerator / denominator);
- if (T.is_signed and numerator < 0 and denominator < 0) {
- if (numerator == minInt(T) and denominator == -1)
- return error.Overflow;
- return @divFloor(numerator + 1, denominator) + 1;
+ const info = @typeInfo(T);
+ switch (info) {
+ .ComptimeFloat, .Float => return @ceil(numerator / denominator),
+ .ComptimeInt, .Int => {
+ if (numerator < 0 and denominator < 0) {
+ if (info == .Int and numerator == minInt(T) and denominator == -1)
+ return error.Overflow;
+ return @divFloor(numerator + 1, denominator) + 1;
+ }
+ if (numerator > 0 and denominator > 0)
+ return @divFloor(numerator - 1, denominator) + 1;
+ return @divTrunc(numerator, denominator);
+ },
+ else => @compileError("divCeil unsupported on " ++ @typeName(T)),
}
- if (numerator > 0 and denominator > 0)
- return @divFloor(numerator - 1, denominator) + 1;
- return @divTrunc(numerator, denominator);
}
test "math.divCeil" {
@@ -654,6 +660,18 @@ fn testDivCeil() void {
testing.expect((divCeil(f32, -5.0, 3.0) catch unreachable) == -1.0);
testing.expect((divCeil(f32, 5.0, -3.0) catch unreachable) == -1.0);
testing.expect((divCeil(f32, -5.0, -3.0) catch unreachable) == 2.0);
+
+ testing.expect((divCeil(comptime_int, 23, 4) catch unreachable) == 6);
+ testing.expect((divCeil(comptime_int, -23, 4) catch unreachable) == -5);
+ testing.expect((divCeil(comptime_int, 23, -4) catch unreachable) == -5);
+ testing.expect((divCeil(comptime_int, -23, -4) catch unreachable) == 6);
+ testing.expectError(error.DivisionByZero, divCeil(comptime_int, 23, 0));
+
+ testing.expect((divCeil(comptime_float, 23.0, 4.0) catch unreachable) == 6.0);
+ testing.expect((divCeil(comptime_float, -23.0, 4.0) catch unreachable) == -5.0);
+ testing.expect((divCeil(comptime_float, 23.0, -4.0) catch unreachable) == -5.0);
+ testing.expect((divCeil(comptime_float, -23.0, -4.0) catch unreachable) == 6.0);
+ testing.expectError(error.DivisionByZero, divCeil(comptime_float, 23.0, 0.0));
}
pub fn divExact(comptime T: type, numerator: T, denominator: T) !T {