/* * Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors. * Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file. */ package test.unsigned import kotlin.math.* import kotlin.random.* import kotlin.test.* class UIntTest { private fun identity(u: UInt): UInt = (u.toInt() + 0).toUInt() val zero = 0u val one = 1u val max = UInt.MAX_VALUE @Test fun equality() { fun testEqual(uv1: UInt, uv2: UInt) { assertEquals(uv1, uv2, "Boxed values should be equal") assertTrue(uv1.equals(uv2), "Boxed values should be equal: $uv1, $uv2") assertTrue(uv1 == uv2, "Values should be equal: $uv1, $uv2") assertEquals(uv1.hashCode(), uv2.hashCode()) assertEquals((uv1 as Any).hashCode(), (uv2 as Any).hashCode()) assertEquals(uv1.toString(), uv2.toString()) assertEquals((uv1 as Any).toString(), (uv2 as Any).toString()) } testEqual(one, identity(one)) testEqual(max, identity(max)) fun testNotEqual(uv1: UInt, uv2: UInt) { assertNotEquals(uv1, uv2, "Boxed values should be equal") assertTrue(uv1 != uv2, "Values should be not equal: $uv1, $uv2") assertNotEquals(uv1.toString(), uv2.toString()) assertNotEquals((uv1 as Any).toString(), (uv2 as Any).toString()) } testNotEqual(one, zero) testNotEqual(max, zero) } @Test fun convertToString() { fun testToString(expected: String, u: UInt) { assertEquals(expected, u.toString()) assertEquals(expected, (u as Any).toString(), "Boxed toString") assertEquals(expected, "$u", "String template") } repeat(100) { val v = Random.nextBits(UInt.SIZE_BITS - 1) testToString(v.toString(), v.toUInt()) } repeat(100) { val v = Random.nextInt(294967295 + 1) testToString("4${v.toString().padStart(9, '0')}", (2_000_000_000.toUInt() * 2.toUInt() + v.toUInt())) } testToString("4294967295", UInt.MAX_VALUE) } @Test fun operations() { assertEquals(2_147_483_648u, Int.MAX_VALUE.toUInt() + identity(1u)) assertEquals(11u, UInt.MAX_VALUE + identity(12u)) assertEquals(UInt.MAX_VALUE - 99u, 45u - identity(145u)) assertEquals(UInt.MAX_VALUE - 1u, Int.MAX_VALUE.toUInt() * identity(2u)) assertEquals(2_147_483_645u, Int.MAX_VALUE.toUInt() * identity(3u)) testMulDivRem(125u, 3u, 41u, 2u) testMulDivRem(210u, 5u, 42u, 0u) testMulDivRem(UInt.MAX_VALUE, 256u, 16777215u, 255u) testMulDivRem(UInt.MAX_VALUE - 1u, UInt.MAX_VALUE, 0u, UInt.MAX_VALUE - 1u) testMulDivRem(UInt.MAX_VALUE, UInt.MAX_VALUE - 1u, 1u, 1u) testMulDivRem(UInt.MAX_VALUE, Int.MAX_VALUE.toUInt(), 2u, 1u) } private fun testMulDivRem(number: UInt, divisor: UInt, div: UInt, rem: UInt) { assertEquals(div, number / divisor) assertEquals(rem, number % divisor) assertEquals(div, number.floorDiv(divisor)) assertEquals(rem, number.mod(divisor)) assertEquals(number, div * divisor + rem) assertTrue(rem < divisor) assertTrue(div < number) } @Test fun divRem() = repeat(1000) { val number = Random.nextUInt() val divisor = Random.nextUInt(until = UInt.MAX_VALUE) + 1u testMulDivRem(number, divisor, number / divisor, number % divisor) } @Test fun comparisons() { fun compare(op1: Comparable, op2: T) = op1.compareTo(op2) fun testComparison(uv1: UInt, uv2: UInt, expected: Int) { val desc = "${uv1.toString()}, ${uv2.toString()}" assertEquals(expected, uv1.compareTo(uv2).sign, "compareTo: $desc") assertEquals(expected, (uv1 as Comparable).compareTo(uv2).sign, "Comparable.compareTo: $desc") assertEquals(expected, compare(uv1, uv2).sign, "Generic compareTo: $desc") assertEquals(expected < 0, uv1 < uv2) assertEquals(expected <= 0, uv1 <= uv2) assertEquals(expected > 0, uv1 > uv2) assertEquals(expected >= 0, uv1 >= uv2) } fun testEquals(uv1: UInt, uv2: UInt) = testComparison(uv1, uv2, 0) fun testCompare(uv1: UInt, uv2: UInt, expected12: Int) { testComparison(uv1, uv2, expected12) testComparison(uv2, uv1, -expected12) } testEquals(one, identity(one)) testEquals(max, identity(max)) testCompare(zero, one, -1) testCompare(Int.MAX_VALUE.toUInt(), zero, 1) testCompare(zero, UInt.MAX_VALUE, -1) testCompare((Int.MAX_VALUE).toUInt() + one, UInt.MAX_VALUE, -1) } @Test fun convertToFloat() { fun testEquals(v1: Float, v2: UInt) = assertEquals(v1, v2.toFloat()) testEquals(0.0f, zero) testEquals(1.0f, one) testEquals(0xFFFF_FFFF.toFloat(), max) repeat(100) { val long = Random.nextLong(0, 0xFFFF_FFFF) testEquals(long.toFloat(), long.toUInt()) } } @Test fun convertToDouble() { fun testEquals(v1: Double, v2: UInt) = assertEquals(v1, v2.toDouble()) testEquals(0.0, zero) testEquals(1.0, one) testEquals(max.toLong().toDouble(), max) repeat(100) { val long = Random.nextLong(0, max.toLong()) testEquals(long.toDouble(), long.toUInt()) } fun testRounding(from: UInt, count: UInt) { for (x in from..(from + count)) { val double = x.toDouble() val v = double.toUInt() val down = double.nextDown().toUInt() val up = double.nextUp().toUInt() assertTrue(down <= x && down <= v) assertTrue(up >= x && up >= v) if (v > x) { assertTrue(v - x <= x - down, "Expected $x being closer to $v than to $down") } else { assertTrue(x - v <= up - x, "Expected $x being closer to $v than to $up") } } } testRounding(0u, 100u) testRounding(Int.MAX_VALUE.toUInt() - 10u, 100u) testRounding(UInt.MAX_VALUE - 100u, 100u) } @Test fun convertDoubleToUInt() { fun testEquals(v1: Double, v2: UInt) = assertEquals(v1.toUInt(), v2) testEquals(0.0, zero) testEquals(-1.0, zero) testEquals(-2_000_000_000_000.0, zero) testEquals(-(2.0.pow(UInt.SIZE_BITS + 12)), zero) testEquals(Double.MIN_VALUE, zero) testEquals(Double.NEGATIVE_INFINITY, zero) testEquals(Double.NaN, zero) testEquals(1.0, one) testEquals(2_000_000_000_000.0, max) testEquals(max.toDouble(), max) testEquals(2.0.pow(UInt.SIZE_BITS), max) testEquals(2.0.pow(UInt.SIZE_BITS + 12), max) testEquals(Double.MAX_VALUE, max) testEquals(Double.POSITIVE_INFINITY, max) repeat(100) { val v = -Random.nextDouble(until = 2.0.pow(UInt.SIZE_BITS + 8)) testEquals(v, zero) } repeat(100) { val v = Random.nextDouble(from = max.toDouble(), until = 2.0.pow(UInt.SIZE_BITS + 8)) testEquals(v, max) } repeat(100) { val v = Random.nextDouble(until = max.toDouble()) testEquals(v, v.toLong().toUInt()) } } }