diff options
| author | Robin Voetter <robin@voetter.nl> | 2021-10-16 00:13:33 +0200 |
|---|---|---|
| committer | Robin Voetter <robin@voetter.nl> | 2021-10-16 11:32:05 +0200 |
| commit | f6bf24b2f3e1d65ce66625c4466aa2af9edff41c (patch) | |
| tree | f4534875df693da5f00a91725bf7fbe9f8600dd5 /src/value.zig | |
| parent | efa4f76c8bb9cbeacc1bcda89d95db4120c821d5 (diff) | |
| download | zig-f6bf24b2f3e1d65ce66625c4466aa2af9edff41c.tar.gz zig-f6bf24b2f3e1d65ce66625c4466aa2af9edff41c.zip | |
stage2: comptime saturating shl
Diffstat (limited to 'src/value.zig')
| -rw-r--r-- | src/value.zig | 33 |
1 files changed, 33 insertions, 0 deletions
diff --git a/src/value.zig b/src/value.zig index 264525a204..127c6c27b4 100644 --- a/src/value.zig +++ b/src/value.zig @@ -2404,6 +2404,39 @@ pub const Value = extern union { } } + pub fn shlSat( + lhs: Value, + rhs: Value, + ty: Type, + arena: *Allocator, + target: Target, + ) !Value { + // TODO is this a performance issue? maybe we should try the operation without + // resorting to BigInt first. + const info = ty.intInfo(target); + + var lhs_space: Value.BigIntSpace = undefined; + const lhs_bigint = lhs.toBigInt(&lhs_space); + const shift = rhs.toUnsignedInt(); + const limbs = try arena.alloc( + std.math.big.Limb, + std.math.big.int.calcTwosCompLimbCount(info.bits), + ); + var result_bigint = BigIntMutable{ + .limbs = limbs, + .positive = undefined, + .len = undefined, + }; + result_bigint.shiftLeftSat(lhs_bigint, shift, info.signedness, info.bits); + const result_limbs = result_bigint.limbs[0..result_bigint.len]; + + if (result_bigint.positive) { + return Value.Tag.int_big_positive.create(arena, result_limbs); + } else { + return Value.Tag.int_big_negative.create(arena, result_limbs); + } + } + pub fn shr(lhs: Value, rhs: Value, allocator: *Allocator) !Value { // TODO is this a performance issue? maybe we should try the operation without // resorting to BigInt first. |
