2011-07-05 11:54:11 +00:00
|
|
|
// Copyright 2011 the V8 project authors. All rights reserved.
|
2014-04-29 06:42:26 +00:00
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
2010-03-11 14:49:35 +00:00
|
|
|
|
|
|
|
#ifndef V8_DIY_FP_H_
|
|
|
|
#define V8_DIY_FP_H_
|
|
|
|
|
2015-08-14 09:41:32 +00:00
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "src/base/logging.h"
|
|
|
|
|
2010-03-11 14:49:35 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
// This "Do It Yourself Floating Point" class implements a floating-point number
|
|
|
|
// with a uint64 significand and an int exponent. Normalized DiyFp numbers will
|
|
|
|
// have the most significant bit of the significand set.
|
|
|
|
// Multiplication and Subtraction do not normalize their results.
|
|
|
|
// DiyFp are not designed to contain special doubles (NaN and Infinity).
|
|
|
|
class DiyFp {
|
|
|
|
public:
|
|
|
|
static const int kSignificandSize = 64;
|
|
|
|
|
|
|
|
DiyFp() : f_(0), e_(0) {}
|
|
|
|
DiyFp(uint64_t f, int e) : f_(f), e_(e) {}
|
|
|
|
|
|
|
|
// this = this - other.
|
|
|
|
// The exponents of both numbers must be the same and the significand of this
|
|
|
|
// must be bigger than the significand of other.
|
|
|
|
// The result will not be normalized.
|
|
|
|
void Subtract(const DiyFp& other) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(e_ == other.e_);
|
|
|
|
DCHECK(f_ >= other.f_);
|
2010-03-11 14:49:35 +00:00
|
|
|
f_ -= other.f_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a - b.
|
|
|
|
// The exponents of both numbers must be the same and this must be bigger
|
|
|
|
// than other. The result will not be normalized.
|
|
|
|
static DiyFp Minus(const DiyFp& a, const DiyFp& b) {
|
|
|
|
DiyFp result = a;
|
|
|
|
result.Subtract(b);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// this = this * other.
|
2010-03-17 15:57:26 +00:00
|
|
|
void Multiply(const DiyFp& other);
|
2010-03-11 14:49:35 +00:00
|
|
|
|
|
|
|
// returns a * b;
|
|
|
|
static DiyFp Times(const DiyFp& a, const DiyFp& b) {
|
|
|
|
DiyFp result = a;
|
|
|
|
result.Multiply(b);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Normalize() {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(f_ != 0);
|
2010-03-11 14:49:35 +00:00
|
|
|
uint64_t f = f_;
|
|
|
|
int e = e_;
|
|
|
|
|
|
|
|
// This method is mainly called for normalizing boundaries. In general
|
|
|
|
// boundaries need to be shifted by 10 bits. We thus optimize for this case.
|
2011-07-05 11:54:11 +00:00
|
|
|
const uint64_t k10MSBits = static_cast<uint64_t>(0x3FF) << 54;
|
2010-03-11 14:49:35 +00:00
|
|
|
while ((f & k10MSBits) == 0) {
|
|
|
|
f <<= 10;
|
|
|
|
e -= 10;
|
|
|
|
}
|
|
|
|
while ((f & kUint64MSB) == 0) {
|
|
|
|
f <<= 1;
|
|
|
|
e--;
|
|
|
|
}
|
|
|
|
f_ = f;
|
|
|
|
e_ = e;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DiyFp Normalize(const DiyFp& a) {
|
|
|
|
DiyFp result = a;
|
|
|
|
result.Normalize();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t f() const { return f_; }
|
|
|
|
int e() const { return e_; }
|
|
|
|
|
|
|
|
void set_f(uint64_t new_value) { f_ = new_value; }
|
|
|
|
void set_e(int new_value) { e_ = new_value; }
|
|
|
|
|
|
|
|
private:
|
2011-07-05 11:54:11 +00:00
|
|
|
static const uint64_t kUint64MSB = static_cast<uint64_t>(1) << 63;
|
2010-03-11 14:49:35 +00:00
|
|
|
|
|
|
|
uint64_t f_;
|
|
|
|
int e_;
|
|
|
|
};
|
|
|
|
|
|
|
|
} } // namespace v8::internal
|
|
|
|
|
|
|
|
#endif // V8_DIY_FP_H_
|