mirror of
https://github.com/corda/corda.git
synced 2025-01-23 04:48:09 +00:00
Merge pull request #142 from joshuawarner32/compiler-boilerplate
Remove boilerplate in Compiler interface
This commit is contained in:
commit
be549ecae3
@ -124,73 +124,20 @@ class Compiler {
|
|||||||
virtual Operand* loadz(unsigned size, unsigned srcSelectSize, Operand* src,
|
virtual Operand* loadz(unsigned size, unsigned srcSelectSize, Operand* src,
|
||||||
unsigned dstSize) = 0;
|
unsigned dstSize) = 0;
|
||||||
|
|
||||||
virtual void jumpIfEqual
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfNotEqual
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfLess
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfGreater
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfLessOrEqual
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfGreaterOrEqual
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
|
|
||||||
virtual void jumpIfFloatEqual
|
virtual void condJump(lir::TernaryOperation type, unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatNotEqual
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatLess
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatGreater
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatLessOrEqual
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatGreaterOrEqual
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatLessOrUnordered
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatGreaterOrUnordered
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatLessOrEqualOrUnordered
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
virtual void jumpIfFloatGreaterOrEqualOrUnordered
|
|
||||||
(unsigned size, Operand* a, Operand* b, Operand* address) = 0;
|
|
||||||
|
|
||||||
virtual void jmp(Operand* address) = 0;
|
virtual void jmp(Operand* address) = 0;
|
||||||
virtual void exit(Operand* address) = 0;
|
virtual void exit(Operand* address) = 0;
|
||||||
virtual Operand* add(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* sub(unsigned size, Operand* a, Operand* b) = 0;
|
virtual Operand* binaryOp(lir::TernaryOperation type, unsigned size, Operand* a, Operand* b) = 0;
|
||||||
virtual Operand* mul(unsigned size, Operand* a, Operand* b) = 0;
|
virtual Operand* unaryOp(lir::BinaryOperation type, unsigned size, Operand* a) = 0;
|
||||||
virtual Operand* div(unsigned size, Operand* a, Operand* b) = 0;
|
virtual void nullaryOp(lir::Operation type) = 0;
|
||||||
virtual Operand* rem(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* fadd(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* fsub(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* fmul(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* fdiv(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* frem(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* shl(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* shr(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* ushr(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* and_(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* or_(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* xor_(unsigned size, Operand* a, Operand* b) = 0;
|
|
||||||
virtual Operand* neg(unsigned size, Operand* a) = 0;
|
|
||||||
virtual Operand* fneg(unsigned size, Operand* a) = 0;
|
|
||||||
virtual Operand* abs(unsigned size, Operand* a) = 0;
|
|
||||||
virtual Operand* fabs(unsigned size, Operand* a) = 0;
|
|
||||||
virtual Operand* fsqrt(unsigned size, Operand* a) = 0;
|
|
||||||
virtual Operand* f2f(unsigned aSize, unsigned resSize, Operand* a) = 0;
|
virtual Operand* f2f(unsigned aSize, unsigned resSize, Operand* a) = 0;
|
||||||
virtual Operand* f2i(unsigned aSize, unsigned resSize, Operand* a) = 0;
|
virtual Operand* f2i(unsigned aSize, unsigned resSize, Operand* a) = 0;
|
||||||
virtual Operand* i2f(unsigned aSize, unsigned resSize, Operand* a) = 0;
|
virtual Operand* i2f(unsigned aSize, unsigned resSize, Operand* a) = 0;
|
||||||
|
|
||||||
virtual void trap() = 0;
|
|
||||||
|
|
||||||
virtual void loadBarrier() = 0;
|
|
||||||
virtual void storeStoreBarrier() = 0;
|
|
||||||
virtual void storeLoadBarrier() = 0;
|
|
||||||
|
|
||||||
virtual void compile(uintptr_t stackOverflowHandler,
|
virtual void compile(uintptr_t stackOverflowHandler,
|
||||||
unsigned stackLimitOffset) = 0;
|
unsigned stackLimitOffset) = 0;
|
||||||
virtual unsigned resolve(uint8_t* dst) = 0;
|
virtual unsigned resolve(uint8_t* dst) = 0;
|
||||||
|
@ -99,7 +99,6 @@ const unsigned OperandTypeCount = MemoryOperand + 1;
|
|||||||
|
|
||||||
const int NoRegister = -1;
|
const int NoRegister = -1;
|
||||||
|
|
||||||
|
|
||||||
inline bool isBranch(lir::TernaryOperation op) {
|
inline bool isBranch(lir::TernaryOperation op) {
|
||||||
return op > FloatMin;
|
return op > FloatMin;
|
||||||
}
|
}
|
||||||
@ -108,6 +107,26 @@ inline bool isFloatBranch(lir::TernaryOperation op) {
|
|||||||
return op > JumpIfNotEqual;
|
return op > JumpIfNotEqual;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline bool isGeneralBranch(lir::TernaryOperation op) {
|
||||||
|
return isBranch(op) && !isFloatBranch(op);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool isGeneralBinaryOp(lir::TernaryOperation op) {
|
||||||
|
return op < FloatAdd;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool isFloatBinaryOp(lir::TernaryOperation op) {
|
||||||
|
return op >= FloatAdd && op <= FloatMin;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool isGeneralUnaryOp(lir::BinaryOperation op) {
|
||||||
|
return op == Negate || op == Absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool isFloatUnaryOp(lir::BinaryOperation op) {
|
||||||
|
return op == FloatNegate || op == FloatSquareRoot || op == FloatAbsolute;
|
||||||
|
}
|
||||||
|
|
||||||
class Operand { };
|
class Operand { };
|
||||||
|
|
||||||
class Constant: public Operand {
|
class Constant: public Operand {
|
||||||
|
54
include/avian/vm/codegen/runtime.h
Normal file
54
include/avian/vm/codegen/runtime.h
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
/* Copyright (c) 2008-2013, Avian Contributors
|
||||||
|
|
||||||
|
Permission to use, copy, modify, and/or distribute this software
|
||||||
|
for any purpose with or without fee is hereby granted, provided
|
||||||
|
that the above copyright notice and this permission notice appear
|
||||||
|
in all copies.
|
||||||
|
|
||||||
|
There is NO WARRANTY for this software. See license.txt for
|
||||||
|
details. */
|
||||||
|
|
||||||
|
#ifndef AVIAN_CODEGEN_RUNTIME_H
|
||||||
|
#define AVIAN_CODEGEN_RUNTIME_H
|
||||||
|
|
||||||
|
namespace avian {
|
||||||
|
namespace codegen {
|
||||||
|
namespace runtime {
|
||||||
|
|
||||||
|
int64_t compareDoublesG(uint64_t bi, uint64_t ai);
|
||||||
|
int64_t compareDoublesL(uint64_t bi, uint64_t ai);
|
||||||
|
int64_t compareFloatsG(uint32_t bi, uint32_t ai);
|
||||||
|
int64_t compareFloatsL(uint32_t bi, uint32_t ai);
|
||||||
|
int64_t compareLongs(uint64_t b, uint64_t a);
|
||||||
|
uint64_t addDouble(uint64_t b, uint64_t a);
|
||||||
|
uint64_t subtractDouble(uint64_t b, uint64_t a);
|
||||||
|
uint64_t multiplyDouble(uint64_t b, uint64_t a);
|
||||||
|
uint64_t divideDouble(uint64_t b, uint64_t a);
|
||||||
|
uint64_t moduloDouble(uint64_t b, uint64_t a);
|
||||||
|
uint64_t negateDouble(uint64_t a);
|
||||||
|
uint64_t squareRootDouble(uint64_t a);
|
||||||
|
uint64_t doubleToFloat(int64_t a);
|
||||||
|
int64_t doubleToInt(int64_t a);
|
||||||
|
int64_t doubleToLong(int64_t a);
|
||||||
|
uint64_t addFloat(uint32_t b, uint32_t a);
|
||||||
|
uint64_t subtractFloat(uint32_t b, uint32_t a);
|
||||||
|
uint64_t multiplyFloat(uint32_t b, uint32_t a);
|
||||||
|
uint64_t divideFloat(uint32_t b, uint32_t a);
|
||||||
|
uint64_t moduloFloat(uint32_t b, uint32_t a);
|
||||||
|
uint64_t negateFloat(uint32_t a);
|
||||||
|
uint64_t absoluteFloat(uint32_t a);
|
||||||
|
int64_t absoluteLong(int64_t a);
|
||||||
|
int64_t absoluteInt(int32_t a);
|
||||||
|
uint64_t floatToDouble(int32_t a);
|
||||||
|
int64_t floatToInt(int32_t a);
|
||||||
|
int64_t floatToLong(int32_t a);
|
||||||
|
uint64_t intToDouble(int32_t a);
|
||||||
|
uint64_t intToFloat(int32_t a);
|
||||||
|
uint64_t longToDouble(int64_t a);
|
||||||
|
uint64_t longToFloat(int64_t a);
|
||||||
|
|
||||||
|
} // namespace runtime
|
||||||
|
} // namespace codegen
|
||||||
|
} // namespace avian
|
||||||
|
|
||||||
|
#endif // AVIAN_CODEGEN_RUNTIME_H
|
1
makefile
1
makefile
@ -1132,6 +1132,7 @@ compiler-sources = \
|
|||||||
$(src)/codegen/compiler.cpp \
|
$(src)/codegen/compiler.cpp \
|
||||||
$(wildcard $(src)/codegen/compiler/*.cpp) \
|
$(wildcard $(src)/codegen/compiler/*.cpp) \
|
||||||
$(src)/codegen/registers.cpp \
|
$(src)/codegen/registers.cpp \
|
||||||
|
$(src)/codegen/runtime.cpp \
|
||||||
$(src)/codegen/targets.cpp
|
$(src)/codegen/targets.cpp
|
||||||
|
|
||||||
x86-assembler-sources = $(wildcard $(src)/codegen/target/x86/*.cpp)
|
x86-assembler-sources = $(wildcard $(src)/codegen/target/x86/*.cpp)
|
||||||
|
@ -2592,170 +2592,17 @@ class MyCompiler: public Compiler {
|
|||||||
return dst;
|
return dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void jumpIfEqual(unsigned size, Operand* a, Operand* b,
|
virtual void condJump(lir::TernaryOperation type, unsigned size, Operand* a, Operand* b,
|
||||||
Operand* address)
|
Operand* address)
|
||||||
{
|
{
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
assert(&c,
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
(isGeneralBranch(type) and isGeneralValue(a) and isGeneralValue(b))
|
||||||
|
or (isFloatBranch(type) and isFloatValue(a) and isFloatValue(b)));
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfEqual, size, static_cast<Value*>(a),
|
appendBranch(&c, type, size, static_cast<Value*>(a),
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
static_cast<Value*>(b), static_cast<Value*>(address));
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void jumpIfNotEqual(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfNotEqual, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfLess(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfLess, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfGreater(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfGreater, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfLessOrEqual(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfLessOrEqual, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfGreaterOrEqual(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfGreaterOrEqual, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatEqual(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatEqual, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatNotEqual(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatNotEqual, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatLess(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatLess, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatGreater(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatGreater, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatLessOrEqual(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatLessOrEqual, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatGreaterOrEqual(unsigned size, Operand* a, Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatGreaterOrEqual, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatLessOrUnordered(unsigned size, Operand* a,
|
|
||||||
Operand* b, Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatLessOrUnordered, size, static_cast<Value*>(a),
|
|
||||||
static_cast<Value*>(b), static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatGreaterOrUnordered(unsigned size, Operand* a,
|
|
||||||
Operand* b, Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatGreaterOrUnordered, size,
|
|
||||||
static_cast<Value*>(a), static_cast<Value*>(b),
|
|
||||||
static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatLessOrEqualOrUnordered(unsigned size, Operand* a,
|
|
||||||
Operand* b, Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatLessOrEqualOrUnordered, size,
|
|
||||||
static_cast<Value*>(a), static_cast<Value*>(b),
|
|
||||||
static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jumpIfFloatGreaterOrEqualOrUnordered(unsigned size, Operand* a,
|
|
||||||
Operand* b,
|
|
||||||
Operand* address)
|
|
||||||
{
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
|
|
||||||
appendBranch(&c, lir::JumpIfFloatGreaterOrEqualOrUnordered, size,
|
|
||||||
static_cast<Value*>(a), static_cast<Value*>(b),
|
|
||||||
static_cast<Value*>(address));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void jmp(Operand* address) {
|
virtual void jmp(Operand* address) {
|
||||||
appendJump(&c, lir::Jump, static_cast<Value*>(address));
|
appendJump(&c, lir::Jump, static_cast<Value*>(address));
|
||||||
}
|
}
|
||||||
@ -2764,182 +2611,23 @@ class MyCompiler: public Compiler {
|
|||||||
appendJump(&c, lir::Jump, static_cast<Value*>(address), true);
|
appendJump(&c, lir::Jump, static_cast<Value*>(address), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual Operand* add(unsigned size, Operand* a, Operand* b) {
|
virtual Operand* binaryOp(lir::TernaryOperation type, unsigned size, Operand* a, Operand* b) {
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
assert(&c,
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
(isGeneralBinaryOp(type) and isGeneralValue(a) and isGeneralValue(b))
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
or (isFloatBinaryOp(type) and isFloatValue(a) and isFloatValue(b)));
|
||||||
appendCombine(&c, lir::Add, size, static_cast<Value*>(a),
|
|
||||||
|
Value* result = value(&c, static_cast<Value*>(a)->type);
|
||||||
|
|
||||||
|
appendCombine(&c, type, size, static_cast<Value*>(a),
|
||||||
size, static_cast<Value*>(b), size, result);
|
size, static_cast<Value*>(b), size, result);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual Operand* sub(unsigned size, Operand* a, Operand* b) {
|
virtual Operand* unaryOp(lir::BinaryOperation type, unsigned size, Operand* a) {
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
assert(&c, (isGeneralUnaryOp(type) and isGeneralValue(a))or(
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
isFloatUnaryOp(type) and isFloatValue(a)));
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
Value* result = value(&c, static_cast<Value*>(a)->type);
|
||||||
appendCombine(&c, lir::Subtract, size, static_cast<Value*>(a),
|
appendTranslate(&c, type, size, static_cast<Value*>(a), size, result);
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* mul(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::Multiply, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* div(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::Divide, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* rem(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::Remainder, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* fadd(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
static_cast<Value*>(a)->type = static_cast<Value*>(b)->type = lir::ValueFloat;
|
|
||||||
appendCombine(&c, lir::FloatAdd, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* fsub(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
static_cast<Value*>(a)->type = static_cast<Value*>(b)->type = lir::ValueFloat;
|
|
||||||
appendCombine(&c, lir::FloatSubtract, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* fmul(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
static_cast<Value*>(a)->type = static_cast<Value*>(b)->type = lir::ValueFloat;
|
|
||||||
appendCombine(&c, lir::FloatMultiply, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* fdiv(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
appendCombine(&c, lir::FloatDivide, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* frem(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat
|
|
||||||
and static_cast<Value*>(b)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
appendCombine(&c, lir::FloatRemainder, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* shl(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::ShiftLeft, TargetBytesPerWord, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* shr(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::ShiftRight, TargetBytesPerWord, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* ushr(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine
|
|
||||||
(&c, lir::UnsignedShiftRight, TargetBytesPerWord, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* and_(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::And, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* or_(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::Or, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* xor_(unsigned size, Operand* a, Operand* b) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendCombine(&c, lir::Xor, size, static_cast<Value*>(a),
|
|
||||||
size, static_cast<Value*>(b), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* neg(unsigned size, Operand* a) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendTranslate(&c, lir::Negate, size, static_cast<Value*>(a), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* fneg(unsigned size, Operand* a) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
appendTranslate(&c, lir::FloatNegate, size, static_cast<Value*>(a), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* abs(unsigned size, Operand* a) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueGeneral);
|
|
||||||
Value* result = value(&c, lir::ValueGeneral);
|
|
||||||
appendTranslate(&c, lir::Absolute, size, static_cast<Value*>(a), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* fabs(unsigned size, Operand* a) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
appendTranslate
|
|
||||||
(&c, lir::FloatAbsolute, size, static_cast<Value*>(a), size, result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual Operand* fsqrt(unsigned size, Operand* a) {
|
|
||||||
assert(&c, static_cast<Value*>(a)->type == lir::ValueFloat);
|
|
||||||
Value* result = value(&c, lir::ValueFloat);
|
|
||||||
appendTranslate
|
|
||||||
(&c, lir::FloatSquareRoot, size, static_cast<Value*>(a), size, result);
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2967,20 +2655,8 @@ class MyCompiler: public Compiler {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void trap() {
|
virtual void nullaryOp(lir::Operation type) {
|
||||||
appendOperation(&c, lir::Trap);
|
appendOperation(&c, type);
|
||||||
}
|
|
||||||
|
|
||||||
virtual void loadBarrier() {
|
|
||||||
appendOperation(&c, lir::LoadBarrier);
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void storeStoreBarrier() {
|
|
||||||
appendOperation(&c, lir::StoreStoreBarrier);
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void storeLoadBarrier() {
|
|
||||||
appendOperation(&c, lir::StoreLoadBarrier);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void compile(uintptr_t stackOverflowHandler,
|
virtual void compile(uintptr_t stackOverflowHandler,
|
||||||
|
@ -67,6 +67,13 @@ class Value: public Compiler::Operand {
|
|||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
inline bool isGeneralValue(Compiler::Operand* a) {
|
||||||
|
return static_cast<Value*>(a)->type == lir::ValueGeneral;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool isFloatValue(Compiler::Operand* a) {
|
||||||
|
return static_cast<Value*>(a)->type == lir::ValueFloat;
|
||||||
|
}
|
||||||
|
|
||||||
Value* value(Context* c, lir::ValueType type, Site* site = 0, Site* target = 0);
|
Value* value(Context* c, lir::ValueType type, Site* site = 0, Site* target = 0);
|
||||||
|
|
||||||
|
282
src/codegen/runtime.cpp
Normal file
282
src/codegen/runtime.cpp
Normal file
@ -0,0 +1,282 @@
|
|||||||
|
/* Copyright (c) 2008-2013, Avian Contributors
|
||||||
|
|
||||||
|
Permission to use, copy, modify, and/or distribute this software
|
||||||
|
for any purpose with or without fee is hereby granted, provided
|
||||||
|
that the above copyright notice and this permission notice appear
|
||||||
|
in all copies.
|
||||||
|
|
||||||
|
There is NO WARRANTY for this software. See license.txt for
|
||||||
|
details. */
|
||||||
|
|
||||||
|
#include <avian/common.h>
|
||||||
|
|
||||||
|
namespace avian {
|
||||||
|
namespace codegen {
|
||||||
|
namespace runtime {
|
||||||
|
|
||||||
|
static bool isNaN(double v)
|
||||||
|
{
|
||||||
|
return fpclassify(v) == FP_NAN;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool isNaN(float v)
|
||||||
|
{
|
||||||
|
return fpclassify(v) == FP_NAN;
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t compareDoublesG(uint64_t bi, uint64_t ai)
|
||||||
|
{
|
||||||
|
double a = vm::bitsToDouble(ai);
|
||||||
|
double b = vm::bitsToDouble(bi);
|
||||||
|
|
||||||
|
if (isNaN(a) or isNaN(b)) {
|
||||||
|
return 1;
|
||||||
|
} else if (a < b) {
|
||||||
|
return -1;
|
||||||
|
} else if (a > b) {
|
||||||
|
return 1;
|
||||||
|
} else if (a == b) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t compareDoublesL(uint64_t bi, uint64_t ai)
|
||||||
|
{
|
||||||
|
double a = vm::bitsToDouble(ai);
|
||||||
|
double b = vm::bitsToDouble(bi);
|
||||||
|
|
||||||
|
if (isNaN(a) or isNaN(b)) {
|
||||||
|
return -1;
|
||||||
|
} else if (a < b) {
|
||||||
|
return -1;
|
||||||
|
} else if (a > b) {
|
||||||
|
return 1;
|
||||||
|
} else if (a == b) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t compareFloatsG(uint32_t bi, uint32_t ai)
|
||||||
|
{
|
||||||
|
float a = vm::bitsToFloat(ai);
|
||||||
|
float b = vm::bitsToFloat(bi);
|
||||||
|
|
||||||
|
if (isNaN(a) or isNaN(b)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (a < b) {
|
||||||
|
return -1;
|
||||||
|
} else if (a > b) {
|
||||||
|
return 1;
|
||||||
|
} else if (a == b) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t compareFloatsL(uint32_t bi, uint32_t ai)
|
||||||
|
{
|
||||||
|
float a = vm::bitsToFloat(ai);
|
||||||
|
float b = vm::bitsToFloat(bi);
|
||||||
|
|
||||||
|
if (isNaN(a) or isNaN(b)) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (a < b) {
|
||||||
|
return -1;
|
||||||
|
} else if (a > b) {
|
||||||
|
return 1;
|
||||||
|
} else if (a == b) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t compareLongs(uint64_t b, uint64_t a)
|
||||||
|
{
|
||||||
|
if (a < b) {
|
||||||
|
return -1;
|
||||||
|
} else if (a > b) {
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t addDouble(uint64_t b, uint64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(vm::bitsToDouble(a) + vm::bitsToDouble(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t subtractDouble(uint64_t b, uint64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(vm::bitsToDouble(a) - vm::bitsToDouble(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t multiplyDouble(uint64_t b, uint64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(vm::bitsToDouble(a) * vm::bitsToDouble(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t divideDouble(uint64_t b, uint64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(vm::bitsToDouble(a) / vm::bitsToDouble(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t moduloDouble(uint64_t b, uint64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(fmod(vm::bitsToDouble(a), vm::bitsToDouble(b)));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t negateDouble(uint64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(-vm::bitsToDouble(a));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t squareRootDouble(uint64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(sqrt(vm::bitsToDouble(a)));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t doubleToFloat(int64_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(static_cast<float>(vm::bitsToDouble(a)));
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t doubleToInt(int64_t a)
|
||||||
|
{
|
||||||
|
double f = vm::bitsToDouble(a);
|
||||||
|
switch (fpclassify(f)) {
|
||||||
|
case FP_NAN:
|
||||||
|
return 0;
|
||||||
|
case FP_INFINITE:
|
||||||
|
return signbit(f) ? INT32_MIN : INT32_MAX;
|
||||||
|
default:
|
||||||
|
return f >= INT32_MAX
|
||||||
|
? INT32_MAX
|
||||||
|
: (f <= INT32_MIN ? INT32_MIN : static_cast<int32_t>(f));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t doubleToLong(int64_t a)
|
||||||
|
{
|
||||||
|
double f = vm::bitsToDouble(a);
|
||||||
|
switch (fpclassify(f)) {
|
||||||
|
case FP_NAN:
|
||||||
|
return 0;
|
||||||
|
case FP_INFINITE:
|
||||||
|
return signbit(f) ? INT64_MIN : INT64_MAX;
|
||||||
|
default:
|
||||||
|
return f >= INT64_MAX
|
||||||
|
? INT64_MAX
|
||||||
|
: (f <= INT64_MIN ? INT64_MIN : static_cast<int64_t>(f));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t addFloat(uint32_t b, uint32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(vm::bitsToFloat(a) + vm::bitsToFloat(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t subtractFloat(uint32_t b, uint32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(vm::bitsToFloat(a) - vm::bitsToFloat(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t multiplyFloat(uint32_t b, uint32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(vm::bitsToFloat(a) * vm::bitsToFloat(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t divideFloat(uint32_t b, uint32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(vm::bitsToFloat(a) / vm::bitsToFloat(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t moduloFloat(uint32_t b, uint32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(fmod(vm::bitsToFloat(a), vm::bitsToFloat(b)));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t negateFloat(uint32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(-vm::bitsToFloat(a));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t absoluteFloat(uint32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(fabsf(vm::bitsToFloat(a)));
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t absoluteLong(int64_t a)
|
||||||
|
{
|
||||||
|
return a > 0 ? a : -a;
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t absoluteInt(int32_t a)
|
||||||
|
{
|
||||||
|
return a > 0 ? a : -a;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t floatToDouble(int32_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(static_cast<double>(vm::bitsToFloat(a)));
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t floatToInt(int32_t a)
|
||||||
|
{
|
||||||
|
float f = vm::bitsToFloat(a);
|
||||||
|
switch (fpclassify(f)) {
|
||||||
|
case FP_NAN:
|
||||||
|
return 0;
|
||||||
|
case FP_INFINITE:
|
||||||
|
return signbit(f) ? INT32_MIN : INT32_MAX;
|
||||||
|
default:
|
||||||
|
return f >= INT32_MAX
|
||||||
|
? INT32_MAX
|
||||||
|
: (f <= INT32_MIN ? INT32_MIN : static_cast<int32_t>(f));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t floatToLong(int32_t a)
|
||||||
|
{
|
||||||
|
float f = vm::bitsToFloat(a);
|
||||||
|
switch (fpclassify(f)) {
|
||||||
|
case FP_NAN:
|
||||||
|
return 0;
|
||||||
|
case FP_INFINITE:
|
||||||
|
return signbit(f) ? INT64_MIN : INT64_MAX;
|
||||||
|
default:
|
||||||
|
return static_cast<int64_t>(f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t intToDouble(int32_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(static_cast<double>(a));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t intToFloat(int32_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(static_cast<float>(a));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t longToDouble(int64_t a)
|
||||||
|
{
|
||||||
|
return vm::doubleToBits(static_cast<double>(a));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t longToFloat(int64_t a)
|
||||||
|
{
|
||||||
|
return vm::floatToBits(static_cast<float>(a));
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace runtime
|
||||||
|
} // namespace codegen
|
||||||
|
} // namespace avian
|
884
src/compile.cpp
884
src/compile.cpp
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user