blob: bf704e3ad1f0c9a1284ca578b910431de10cc0f4 [file] [log] [blame]
// Copyright 2017 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef V8_UTILS_BOXED_FLOAT_H_
#define V8_UTILS_BOXED_FLOAT_H_
#include <cmath>
#include "src/base/hashing.h"
#include "src/base/macros.h"
#include "src/base/numbers/double.h"
#include "src/common/globals.h"
namespace v8 {
namespace internal {
// TODO(ahaas): Make these classes with the one in double.h
// Safety wrapper for a 32-bit floating-point value to make sure we don't lose
// the exact bit pattern during deoptimization when passing this value.
class Float32 {
public:
Float32() = default;
// This constructor does not guarantee that bit pattern of the input value
// is preserved if the input is a NaN.
explicit Float32(float value)
: bit_pattern_(base::bit_cast<uint32_t>(value)) {
// Check that the provided value is not a NaN, because the bit pattern of a
// NaN may be changed by a base::bit_cast, e.g. for signalling NaNs on
// ia32.
DCHECK(!std::isnan(value));
}
uint32_t get_bits() const { return bit_pattern_; }
float get_scalar() const { return base::bit_cast<float>(bit_pattern_); }
bool is_nan() const {
// Even though {get_scalar()} might flip the quiet NaN bit, it's ok here,
// because this does not change the is_nan property.
return std::isnan(get_scalar());
}
// Return a pointer to the field storing the bit pattern. Used in code
// generation tests to store generated values there directly.
uint32_t* get_bits_address() { return &bit_pattern_; }
static constexpr Float32 FromBits(uint32_t bits) { return Float32(bits); }
private:
uint32_t bit_pattern_ = 0;
explicit constexpr Float32(uint32_t bit_pattern)
: bit_pattern_(bit_pattern) {}
};
ASSERT_TRIVIALLY_COPYABLE(Float32);
// Safety wrapper for a 64-bit floating-point value to make sure we don't lose
// the exact bit pattern during deoptimization when passing this value.
// TODO(ahaas): Unify this class with Double in double.h
class Float64 {
public:
Float64() = default;
// This constructor does not guarantee that bit pattern of the input value
// is preserved if the input is a NaN.
explicit Float64(double value)
: bit_pattern_(base::bit_cast<uint64_t>(value)) {
// Check that the provided value is not a NaN, because the bit pattern of a
// NaN may be changed by a base::bit_cast, e.g. for signalling NaNs on
// ia32.
DCHECK(!std::isnan(value));
}
explicit Float64(base::Double value) : bit_pattern_(value.AsUint64()) {}
uint64_t get_bits() const { return bit_pattern_; }
double get_scalar() const { return base::bit_cast<double>(bit_pattern_); }
bool is_hole_nan() const { return bit_pattern_ == kHoleNanInt64; }
bool is_nan() const {
// Even though {get_scalar()} might flip the quiet NaN bit, it's ok here,
// because this does not change the is_nan property.
return std::isnan(get_scalar());
}
// Return a pointer to the field storing the bit pattern. Used in code
// generation tests to store generated values there directly.
uint64_t* get_bits_address() { return &bit_pattern_; }
static constexpr Float64 FromBits(uint64_t bits) { return Float64(bits); }
// Unlike doubles, equality is defined as equally behaving as far as the
// optimizers are concerned. I.e., two NaN's are equal as long as they are
// both the hole nor not.
bool operator==(const Float64& other) const {
if (is_nan() && other.is_nan()) {
return is_hole_nan() == other.is_hole_nan();
}
return get_scalar() == other.get_scalar();
}
friend size_t hash_value(internal::Float64 f64) { return f64.bit_pattern_; }
private:
uint64_t bit_pattern_ = 0;
explicit constexpr Float64(uint64_t bit_pattern)
: bit_pattern_(bit_pattern) {}
};
ASSERT_TRIVIALLY_COPYABLE(Float64);
} // namespace internal
namespace base {
inline size_t hash_value(const i::Float64& f64) {
return f64.is_nan() ? hash_value(f64.is_hole_nan())
: hash_value(f64.get_bits());
}
} // namespace base
} // namespace v8
#endif // V8_UTILS_BOXED_FLOAT_H_