blob: 58e5b2c00bad002f600fb1388f1614a2b567f26f [file] [log] [blame]
// Copyright 2013 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'dart:math' as math;
import 'dart:typed_data';
// This is a small library to handle stability for floating point operations.
//
// Since we are representing an infinite number of real numbers in finite
// number of bits, when we perform comparisons of coordinates for paths for
// example, we want to make sure that line and curve sections that are too
// close to each other (number of floating point numbers
// representable in bits between two numbers) are handled correctly and
// don't cause algorithms to fail when we perform operations such as
// subtraction or between checks.
//
// Small introduction into floating point comparison:
//
// For some good articles on the topic, see
// https://randomascii.wordpress.com/category/floating-point/page/2/
// Port based on:
// https://github.com/google/skia/blob/master/include/private/SkFloatBits.h
//
// Here is the 32 bit IEEE representation:
// uint32_t mantissa : 23;
// uint32_t exponent : 8;
// uint32_t sign : 1;
// As you can see it was carefully designed to be reinterpreted as an integer.
//
// Ulps stands for unit in the last place. ulp(x) is the gap between two
// floating point numbers nearest x.
/// Converts a sign-bit int (float interpreted as int) into a 2s complement
/// int. Also converts 0x80000000 to 0. Allows result to be compared using
/// int comparison.
int signBitTo2sCompliment(int x) =>
(x & 0x80000000) != 0 ? (-(x & 0x7fffffff)) : x;
/// Convert a 2s complement int to a sign-bit (i.e. int interpreted as float).
int twosComplimentToSignBit(int x) {
if ((x & 0x80000000) == 0) {
return x;
}
x = ~x + 1;
x |= 0x80000000;
return x;
}
class _FloatBitConverter {
final Float32List float32List;
final Int32List int32List;
factory _FloatBitConverter() {
final Float32List float32List = Float32List(1);
return _FloatBitConverter._(
float32List, float32List.buffer.asInt32List(0, 1));
}
_FloatBitConverter._(this.float32List, this.int32List);
int toInt(Float32List source, int index) {
float32List[0] = source[index];
return int32List[0];
}
int toBits(double x) {
float32List[0] = x;
return int32List[0];
}
double toDouble(int bits) {
int32List[0] = bits;
return float32List[0];
}
}
// Singleton bit converter to prevent typed array allocations.
final _FloatBitConverter _floatBitConverter = _FloatBitConverter();
// Converts float to bits.
int float2Bits(Float32List source, int index) {
return _floatBitConverter.toInt(source, index);
}
// Converts bits to float.
double bitsToFloat(int bits) {
return _floatBitConverter.toDouble(bits);
}
const int floatBitsExponentMask = 0x7F800000;
const int floatBitsMatissaMask = 0x007FFFFF;
/// Returns a float as 2s complement int to be able to compare floats to each
/// other.
int floatFromListAs2sCompliment(Float32List source, int index) =>
signBitTo2sCompliment(float2Bits(source, index));
int floatAs2sCompliment(double x) =>
signBitTo2sCompliment(_floatBitConverter.toBits(x));
double twosComplimentAsFloat(int x) => bitsToFloat(twosComplimentToSignBit(x));
bool _argumentsDenormalized(double a, double b, int epsilon) {
final double denormalizedCheck = kFltEpsilon * epsilon / 2;
return a.abs() <= denormalizedCheck && b.abs() <= denormalizedCheck;
}
bool equalUlps(double a, double b, int epsilon, int depsilon) {
if (_argumentsDenormalized(a, b, depsilon)) {
return true;
}
final int aBits = floatAs2sCompliment(a);
final int bBits = floatAs2sCompliment(b);
// Find the difference in ULPs.
return aBits < bBits + epsilon && bBits < aBits + epsilon;
}
/// General equality check that covers between, product and division by using
/// ulps epsilon 16.
bool almostEqualUlps(double a, double b) {
const int kUlpsEpsilon = 16;
return equalUlps(a, b, kUlpsEpsilon, kUlpsEpsilon);
}
/// Equality using the same error term for between comparison.
bool almostBequalUlps(double a, double b) {
const int kUlpsEpsilon = 2;
return equalUlps(a, b, kUlpsEpsilon, kUlpsEpsilon);
}
/// Equality check for product.
bool almostPequalUlps(double a, double b) {
const int kUlpsEpsilon = 8;
return equalUlps(a, b, kUlpsEpsilon, kUlpsEpsilon);
}
/// Equality check for division.
bool almostDequalUlps(double a, double b) {
const int kUlpsEpsilon = 16;
return equalUlps(a, b, kUlpsEpsilon, kUlpsEpsilon);
}
/// Checks if 2 points are roughly equal (ulp 256) to each other.
bool approximatelyEqual(double ax, double ay, double bx, double by) {
if (approximatelyEqualT(ax, bx) && approximatelyEqualT(ay, by)) {
return true;
}
if (!roughlyEqualUlps(ax, bx) || !roughlyEqualUlps(ay, by)) {
return false;
}
final double dx = ax - bx;
final double dy = ay - by;
final double dist = math.sqrt(dx * dx + dy * dy);
final double tiniest = math.min(math.min(math.min(ax, bx), ay), by);
double largest = math.max(math.max(math.max(ax, bx), ay), by);
largest = math.max(largest, -tiniest);
return almostDequalUlps(largest, largest + dist);
}
/// Equality check for comparing curve T values in the range of 0 to 1.
///
/// For general numbers (larger and smaller) use
/// AlmostEqualUlps instead.
bool approximatelyEqualT(double t1, double t2) {
return approximatelyZero(t1 - t2);
}
bool approximatelyZero(double value) => value.abs() < kFltEpsilon;
bool roughlyEqualUlps(double a, double b) {
const int kUlpsEpsilon = 256;
const int kDUlpsEpsilon = 1024;
return equalUlps(a, b, kUlpsEpsilon, kDUlpsEpsilon);
}
bool dEqualUlpsEpsilon(double a, double b, int epsilon) {
final int aBits = floatAs2sCompliment(a);
final int bBits = floatAs2sCompliment(b);
// Find the difference in ULPs.
return aBits < bBits + epsilon && bBits < aBits + epsilon;
}
// Checks equality for division.
bool almostDequalUlpsDouble(double a, double b) {
final double absA = a.abs();
final double absB = b.abs();
if (absA < kScalarMax && absB < kScalarMax) {
return almostDequalUlps(a, b);
}
return (a - b).abs() / math.max(absA, absB) < kDblEpsilonSubdivideErr;
}
const double kFltEpsilon = 1.19209290E-07; // == 1 / (2 ^ 23)
const double kDblEpsilon = 2.22045e-16;
const double kFltEpsilonCubed = kFltEpsilon * kFltEpsilon * kFltEpsilon;
const double kFltEpsilonHalf = kFltEpsilon / 2;
const double kFltEpsilonDouble = kFltEpsilon * 2;
// Epsilon to use when ordering vectors.
const double kFltEpsilonOrderableErr = kFltEpsilon * 16;
const double kFltEpsilonSquared = kFltEpsilon * kFltEpsilon;
// Use a compile-time constant for FLT_EPSILON_SQRT to avoid initializers.
// A 17 digit constant guarantees exact results.
const double kFltEpsilonSqrt = 0.00034526697709225118; // sqrt(kFltEpsilon);
const double kFltEpsilonInverse = 1 / kFltEpsilon;
const double kDblEpsilonErr = kDblEpsilon * 4;
const double kDblEpsilonSubdivideErr = kDblEpsilon * 16;
const double kRoughEpsilon = kFltEpsilon * 64;
const double kMoreRoughEpsilon = kFltEpsilon * 256;
const double kWayRoughEpsilon = kFltEpsilon * 2048;
const double kBumpEpsilon = kFltEpsilon * 4096;
// Scalar max is based on 32 bit float since [PathRef] stores values in
// Float32List.
const double kScalarMax = 3.402823466e+38;
const double kScalarMin = -kScalarMax;