blob: cdf6a9681305e44bda1f97091c2943c1d36f558e [file] [log] [blame]
/*
* Copyright 2010-2018 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.numbers
import test.isFloat32RangeEnforced
import kotlin.test.*
object NumbersTestConstants {
public const val byteMinSucc: Byte = (Byte.MIN_VALUE + 1).toByte()
public const val byteMaxPred: Byte = (Byte.MAX_VALUE - 1).toByte()
public const val shortMinSucc: Short = (Short.MIN_VALUE + 1).toShort()
public const val shortMaxPred: Short = (Short.MAX_VALUE - 1).toShort()
public const val intMinSucc: Int = Int.MIN_VALUE + 1
public const val intMaxPred: Int = Int.MAX_VALUE - 1
public const val longMinSucc: Long = Long.MIN_VALUE + 1L
public const val longMaxPred: Long = Long.MAX_VALUE - 1L
public const val doubleMaxHalf: Double = Double.MAX_VALUE / 2
public const val doubleMinTwice: Double = Double.MIN_VALUE * 2
public const val floatMaxHalf: Float = Float.MAX_VALUE / 2
public const val floatMinTwice: Float = Float.MIN_VALUE * 2
}
class NumbersTest {
var one: Int = 1
var oneS: Short = 1
var oneB: Byte = 1
var two: Int = 2
@Test fun intMinMaxValues() {
assertTrue(Int.MIN_VALUE < 0)
assertTrue(Int.MAX_VALUE > 0)
assertEquals(NumbersTestConstants.intMinSucc, Int.MIN_VALUE + one)
assertEquals(NumbersTestConstants.intMaxPred, Int.MAX_VALUE - one)
// overflow behavior
expect(Int.MIN_VALUE) { Int.MAX_VALUE + one }
expect(Int.MAX_VALUE) { Int.MIN_VALUE - one }
}
@Test fun longMinMaxValues() {
assertTrue(Long.MIN_VALUE < 0)
assertTrue(Long.MAX_VALUE > 0)
assertEquals(NumbersTestConstants.longMinSucc, Long.MIN_VALUE + one)
assertEquals(NumbersTestConstants.longMaxPred, Long.MAX_VALUE - one)
// overflow behavior
expect(Long.MIN_VALUE) { Long.MAX_VALUE + one }
expect(Long.MAX_VALUE) { Long.MIN_VALUE - one }
}
@Test fun shortMinMaxValues() {
assertTrue(Short.MIN_VALUE < 0)
assertTrue(Short.MAX_VALUE > 0)
assertEquals(NumbersTestConstants.shortMinSucc, Short.MIN_VALUE.inc())
assertEquals(NumbersTestConstants.shortMaxPred, Short.MAX_VALUE.dec())
// overflow behavior
expect(Short.MIN_VALUE) { (Short.MAX_VALUE + oneS).toShort() }
expect(Short.MAX_VALUE) { (Short.MIN_VALUE - oneS).toShort() }
}
@Test fun byteMinMaxValues() {
assertTrue(Byte.MIN_VALUE < 0)
assertTrue(Byte.MAX_VALUE > 0)
assertEquals(NumbersTestConstants.byteMinSucc, Byte.MIN_VALUE.inc())
assertEquals(NumbersTestConstants.byteMaxPred, Byte.MAX_VALUE.dec())
// overflow behavior
expect(Byte.MIN_VALUE) { (Byte.MAX_VALUE + oneB).toByte() }
expect(Byte.MAX_VALUE) { (Byte.MIN_VALUE - oneB).toByte() }
}
@Test fun doubleMinMaxValues() {
assertTrue(Double.MIN_VALUE > 0)
assertTrue(Double.MAX_VALUE > 0)
assertEquals(NumbersTestConstants.doubleMaxHalf, Double.MAX_VALUE / two)
assertEquals(NumbersTestConstants.doubleMinTwice, Double.MIN_VALUE * two)
// overflow behavior
expect(Double.POSITIVE_INFINITY) { Double.MAX_VALUE * 2 }
expect(Double.NEGATIVE_INFINITY) {-Double.MAX_VALUE * 2 }
expect(0.0) { Double.MIN_VALUE / 2 }
}
@Test fun floatMinMaxValues() {
assertTrue(Float.MIN_VALUE > 0)
assertTrue(Float.MAX_VALUE > 0)
if (isFloat32RangeEnforced) {
assertEquals(NumbersTestConstants.floatMaxHalf, Float.MAX_VALUE / two)
assertEquals(NumbersTestConstants.floatMinTwice, Float.MIN_VALUE * two)
} else {
assertAlmostEquals(NumbersTestConstants.floatMaxHalf, Float.MAX_VALUE / two, 0.0000001 * NumbersTestConstants.floatMaxHalf)
assertAlmostEquals(NumbersTestConstants.floatMinTwice, Float.MIN_VALUE * two, 0.0000001 * NumbersTestConstants.floatMinTwice)
}
// overflow behavior
if (isFloat32RangeEnforced) {
expect(Float.POSITIVE_INFINITY) { Float.MAX_VALUE * 2 }
expect(Float.NEGATIVE_INFINITY) { -Float.MAX_VALUE * 2 }
expect(0.0F) { Float.MIN_VALUE / 2.0F }
}
}
@Test fun charMinMaxValues() {
assertTrue(Char.MIN_VALUE.toInt() == 0)
assertTrue(Char.MAX_VALUE.toInt() > 0)
// overflow behavior
expect(Char.MIN_VALUE) { Char.MAX_VALUE + one }
expect(Char.MAX_VALUE) { Char.MIN_VALUE - one }
}
@Test fun doubleProperties() {
for (value in listOf(1.0, 0.0, Double.MIN_VALUE, Double.MAX_VALUE))
doTestNumber(value)
for (value in listOf(Double.POSITIVE_INFINITY, Double.NEGATIVE_INFINITY))
doTestNumber(value, isInfinite = true)
doTestNumber(Double.NaN, isNaN = true)
}
@Test fun floatProperties() {
for (value in listOf(1.0F, 0.0F, Float.MAX_VALUE, Float.MIN_VALUE))
doTestNumber(value)
for (value in listOf(Float.POSITIVE_INFINITY, Float.NEGATIVE_INFINITY))
doTestNumber(value, isInfinite = true)
doTestNumber(Float.NaN, isNaN = true)
}
@Test fun floatFitsInFloatArray() {
val values = listOf(1.0F, 0.0F, Float.MAX_VALUE, Float.MIN_VALUE, Float.POSITIVE_INFINITY, Float.NEGATIVE_INFINITY, Float.NaN)
val valuesArray = values.toFloatArray()
for (index in values.indices) {
val value = values[index]
val tolerance = if (value == Float.MIN_VALUE) 0.001 * value else 0.0000001 * value
assertAlmostEquals(value, valuesArray[index], tolerance)
}
}
private fun doTestNumber(value: Double, isNaN: Boolean = false, isInfinite: Boolean = false) {
assertEquals(isNaN, value.isNaN(), "Expected $value to have isNaN: $isNaN")
assertEquals(isInfinite, value.isInfinite(), "Expected $value to have isInfinite: $isInfinite")
assertEquals(!isNaN && !isInfinite, value.isFinite())
}
private fun doTestNumber(value: Float, isNaN: Boolean = false, isInfinite: Boolean = false) {
assertEquals(isNaN, value.isNaN(), "Expected $value to have isNaN: $isNaN")
assertEquals(isInfinite, value.isInfinite(), "Expected $value to have isInfinite: $isInfinite")
assertEquals(!isNaN && !isInfinite, value.isFinite())
}
@Test fun doubleToBits() {
assertEquals(0x400921fb54442d18L, kotlin.math.PI.toBits())
assertEquals(0x400921fb54442d18L, kotlin.math.PI.toRawBits())
assertEquals(kotlin.math.PI, Double.fromBits(0x400921fb54442d18L))
for (value in listOf(Double.NEGATIVE_INFINITY, -Double.MAX_VALUE, -1.0, -Double.MIN_VALUE, -0.0, 0.0, Double.POSITIVE_INFINITY, Double.MAX_VALUE, 1.0, Double.MIN_VALUE)) {
assertEquals(value, Double.fromBits(value.toBits()))
assertEquals(value, Double.fromBits(value.toRawBits()))
}
assertTrue(Double.NaN.toBits().let(Double.Companion::fromBits).isNaN())
assertTrue(Double.NaN.toRawBits().let { Double.fromBits(it) }.isNaN())
assertEquals(0x7FF00000L shl 32, Double.POSITIVE_INFINITY.toBits())
assertEquals(0xFFF00000L shl 32, Double.NEGATIVE_INFINITY.toBits())
assertEquals(0x7FF80000_00000000L, Double.NaN.toBits())
assertEquals(0x7FF80000_00000000L, Double.NaN.toRawBits())
val bitsNaN = Double.NaN.toBits()
for (bitsDenormNaN in listOf(0xFFF80000L shl 32, bitsNaN or 1)) {
assertTrue(Double.fromBits(bitsDenormNaN).isNaN(), "expected $bitsDenormNaN represent NaN")
assertEquals(bitsNaN, Double.fromBits(bitsDenormNaN).toBits())
}
}
@Test fun floatToBits() {
val PI_F = kotlin.math.PI.toFloat()
assertEquals(0x40490fdb, PI_F.toBits())
if (isFloat32RangeEnforced) {
assertEquals(PI_F, Float.fromBits(0x40490fdb))
} else {
assertAlmostEquals(PI_F, Float.fromBits(0x40490fdb)) // PI_F is actually Double in JS
}
for (value in listOf(Float.NEGATIVE_INFINITY, -1.0F, -0.0F, 0.0F, Float.POSITIVE_INFINITY, 1.0F)) {
assertEquals(value, Float.fromBits(value.toBits()))
assertEquals(value, Float.fromBits(value.toRawBits()))
}
for (value in listOf(-Float.MAX_VALUE, Float.MAX_VALUE, -Float.MIN_VALUE, Float.MIN_VALUE)) {
if (isFloat32RangeEnforced) {
assertEquals(value, Float.fromBits(value.toBits()))
assertEquals(value, Float.fromBits(value.toRawBits()))
} else {
val tolerance = if (kotlin.math.abs(value) == Float.MIN_VALUE) 0.001 * value else 0.0000001 * value
assertAlmostEquals(value, Float.fromBits(value.toBits()), tolerance)
assertAlmostEquals(value, Float.fromBits(value.toRawBits()), tolerance)
}
}
assertTrue(Float.NaN.toBits().let(Float.Companion::fromBits).isNaN())
assertTrue(Float.NaN.toRawBits().let { Float.fromBits(it) }.isNaN())
assertEquals(0xbf800000.toInt(), (-1.0F).toBits())
assertEquals(0x7fc00000, Float.NaN.toBits())
assertEquals(0x7fc00000, Float.NaN.toRawBits())
val bitsNaN = Float.NaN.toBits()
for (bitsDenormNaN in listOf(0xFFFC0000.toInt(), bitsNaN or 1)) {
assertTrue(Float.fromBits(bitsDenormNaN).isNaN(), "expected $bitsDenormNaN represent NaN")
assertEquals(bitsNaN, Float.fromBits(bitsDenormNaN).toBits())
}
}
@Test fun sizeInBitsAndBytes() {
fun testSizes(companion: Any, sizeBytes: Int, sizeBits: Int, expectedSizeBytes: Int) {
assertEquals(expectedSizeBytes, sizeBytes, companion.toString())
assertEquals(expectedSizeBytes * 8, sizeBits, companion.toString())
}
testSizes(Char, Char.SIZE_BYTES, Char.SIZE_BITS, 2)
testSizes(Byte, Byte.SIZE_BYTES, Byte.SIZE_BITS, 1)
testSizes(Short, Short.SIZE_BYTES, Short.SIZE_BITS, 2)
testSizes(Int, Int.SIZE_BYTES, Int.SIZE_BITS, 4)
testSizes(Long, Long.SIZE_BYTES, Long.SIZE_BITS, 8)
testSizes(Float, Float.SIZE_BYTES, Float.SIZE_BITS, 4)
testSizes(Double, Double.SIZE_BYTES, Double.SIZE_BITS, 8)
testSizes(UByte, UByte.SIZE_BYTES, UByte.SIZE_BITS, 1)
testSizes(UShort, UShort.SIZE_BYTES, UShort.SIZE_BITS, 2)
testSizes(UInt, UInt.SIZE_BYTES, UInt.SIZE_BITS, 4)
testSizes(ULong, ULong.SIZE_BYTES, ULong.SIZE_BITS, 8)
}
}