1 // Copyright 2016 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include <math.h>
6 #include <stdint.h>
7 #include <stdlib.h>
8 #include <limits>
9 
10 #include "include/v8config.h"
11 
12 #include "src/base/bits.h"
13 #include "src/utils.h"
14 #include "src/wasm/wasm-external-refs.h"
15 
16 namespace v8 {
17 namespace internal {
18 namespace wasm {
19 
f32_trunc_wrapper(float * param)20 void f32_trunc_wrapper(float* param) { *param = truncf(*param); }
21 
f32_floor_wrapper(float * param)22 void f32_floor_wrapper(float* param) { *param = floorf(*param); }
23 
f32_ceil_wrapper(float * param)24 void f32_ceil_wrapper(float* param) { *param = ceilf(*param); }
25 
f32_nearest_int_wrapper(float * param)26 void f32_nearest_int_wrapper(float* param) { *param = nearbyintf(*param); }
27 
f64_trunc_wrapper(double * param)28 void f64_trunc_wrapper(double* param) {
29   WriteDoubleValue(param, trunc(ReadDoubleValue(param)));
30 }
31 
f64_floor_wrapper(double * param)32 void f64_floor_wrapper(double* param) {
33   WriteDoubleValue(param, floor(ReadDoubleValue(param)));
34 }
35 
f64_ceil_wrapper(double * param)36 void f64_ceil_wrapper(double* param) {
37   WriteDoubleValue(param, ceil(ReadDoubleValue(param)));
38 }
39 
f64_nearest_int_wrapper(double * param)40 void f64_nearest_int_wrapper(double* param) {
41   WriteDoubleValue(param, nearbyint(ReadDoubleValue(param)));
42 }
43 
int64_to_float32_wrapper(int64_t * input,float * output)44 void int64_to_float32_wrapper(int64_t* input, float* output) {
45   *output = static_cast<float>(*input);
46 }
47 
uint64_to_float32_wrapper(uint64_t * input,float * output)48 void uint64_to_float32_wrapper(uint64_t* input, float* output) {
49 #if V8_CC_MSVC
50   // With MSVC we use static_cast<float>(uint32_t) instead of
51   // static_cast<float>(uint64_t) to achieve round-to-nearest-ties-even
52   // semantics. The idea is to calculate
53   // static_cast<float>(high_word) * 2^32 + static_cast<float>(low_word). To
54   // achieve proper rounding in all cases we have to adjust the high_word
55   // with a "rounding bit" sometimes. The rounding bit is stored in the LSB of
56   // the high_word if the low_word may affect the rounding of the high_word.
57   uint32_t low_word = static_cast<uint32_t>(*input & 0xffffffff);
58   uint32_t high_word = static_cast<uint32_t>(*input >> 32);
59 
60   float shift = static_cast<float>(1ull << 32);
61   // If the MSB of the high_word is set, then we make space for a rounding bit.
62   if (high_word < 0x80000000) {
63     high_word <<= 1;
64     shift = static_cast<float>(1ull << 31);
65   }
66 
67   if ((high_word & 0xfe000000) && low_word) {
68     // Set the rounding bit.
69     high_word |= 1;
70   }
71 
72   float result = static_cast<float>(high_word);
73   result *= shift;
74   result += static_cast<float>(low_word);
75   *output = result;
76 
77 #else
78   *output = static_cast<float>(*input);
79 #endif
80 }
81 
int64_to_float64_wrapper(int64_t * input,double * output)82 void int64_to_float64_wrapper(int64_t* input, double* output) {
83   *output = static_cast<double>(*input);
84 }
85 
uint64_to_float64_wrapper(uint64_t * input,double * output)86 void uint64_to_float64_wrapper(uint64_t* input, double* output) {
87 #if V8_CC_MSVC
88   // With MSVC we use static_cast<double>(uint32_t) instead of
89   // static_cast<double>(uint64_t) to achieve round-to-nearest-ties-even
90   // semantics. The idea is to calculate
91   // static_cast<double>(high_word) * 2^32 + static_cast<double>(low_word).
92   uint32_t low_word = static_cast<uint32_t>(*input & 0xffffffff);
93   uint32_t high_word = static_cast<uint32_t>(*input >> 32);
94 
95   double shift = static_cast<double>(1ull << 32);
96 
97   double result = static_cast<double>(high_word);
98   result *= shift;
99   result += static_cast<double>(low_word);
100   *output = result;
101 
102 #else
103   *output = static_cast<double>(*input);
104 #endif
105 }
106 
float32_to_int64_wrapper(float * input,int64_t * output)107 int32_t float32_to_int64_wrapper(float* input, int64_t* output) {
108   // We use "<" here to check the upper bound because of rounding problems: With
109   // "<=" some inputs would be considered within int64 range which are actually
110   // not within int64 range.
111   if (*input >= static_cast<float>(std::numeric_limits<int64_t>::min()) &&
112       *input < static_cast<float>(std::numeric_limits<int64_t>::max())) {
113     *output = static_cast<int64_t>(*input);
114     return 1;
115   }
116   return 0;
117 }
118 
float32_to_uint64_wrapper(float * input,uint64_t * output)119 int32_t float32_to_uint64_wrapper(float* input, uint64_t* output) {
120   // We use "<" here to check the upper bound because of rounding problems: With
121   // "<=" some inputs would be considered within uint64 range which are actually
122   // not within uint64 range.
123   if (*input > -1.0 &&
124       *input < static_cast<float>(std::numeric_limits<uint64_t>::max())) {
125     *output = static_cast<uint64_t>(*input);
126     return 1;
127   }
128   return 0;
129 }
130 
float64_to_int64_wrapper(double * input,int64_t * output)131 int32_t float64_to_int64_wrapper(double* input, int64_t* output) {
132   // We use "<" here to check the upper bound because of rounding problems: With
133   // "<=" some inputs would be considered within int64 range which are actually
134   // not within int64 range.
135   if (*input >= static_cast<double>(std::numeric_limits<int64_t>::min()) &&
136       *input < static_cast<double>(std::numeric_limits<int64_t>::max())) {
137     *output = static_cast<int64_t>(*input);
138     return 1;
139   }
140   return 0;
141 }
142 
float64_to_uint64_wrapper(double * input,uint64_t * output)143 int32_t float64_to_uint64_wrapper(double* input, uint64_t* output) {
144   // We use "<" here to check the upper bound because of rounding problems: With
145   // "<=" some inputs would be considered within uint64 range which are actually
146   // not within uint64 range.
147   if (*input > -1.0 &&
148       *input < static_cast<double>(std::numeric_limits<uint64_t>::max())) {
149     *output = static_cast<uint64_t>(*input);
150     return 1;
151   }
152   return 0;
153 }
154 
int64_div_wrapper(int64_t * dst,int64_t * src)155 int32_t int64_div_wrapper(int64_t* dst, int64_t* src) {
156   if (*src == 0) {
157     return 0;
158   }
159   if (*src == -1 && *dst == std::numeric_limits<int64_t>::min()) {
160     return -1;
161   }
162   *dst /= *src;
163   return 1;
164 }
165 
int64_mod_wrapper(int64_t * dst,int64_t * src)166 int32_t int64_mod_wrapper(int64_t* dst, int64_t* src) {
167   if (*src == 0) {
168     return 0;
169   }
170   *dst %= *src;
171   return 1;
172 }
173 
uint64_div_wrapper(uint64_t * dst,uint64_t * src)174 int32_t uint64_div_wrapper(uint64_t* dst, uint64_t* src) {
175   if (*src == 0) {
176     return 0;
177   }
178   *dst /= *src;
179   return 1;
180 }
181 
uint64_mod_wrapper(uint64_t * dst,uint64_t * src)182 int32_t uint64_mod_wrapper(uint64_t* dst, uint64_t* src) {
183   if (*src == 0) {
184     return 0;
185   }
186   *dst %= *src;
187   return 1;
188 }
189 
word32_ctz_wrapper(uint32_t * input)190 uint32_t word32_ctz_wrapper(uint32_t* input) {
191   return static_cast<uint32_t>(base::bits::CountTrailingZeros32(*input));
192 }
193 
word64_ctz_wrapper(uint64_t * input)194 uint32_t word64_ctz_wrapper(uint64_t* input) {
195   return static_cast<uint32_t>(base::bits::CountTrailingZeros64(*input));
196 }
197 
word32_popcnt_wrapper(uint32_t * input)198 uint32_t word32_popcnt_wrapper(uint32_t* input) {
199   return static_cast<uint32_t>(base::bits::CountPopulation(*input));
200 }
201 
word64_popcnt_wrapper(uint64_t * input)202 uint32_t word64_popcnt_wrapper(uint64_t* input) {
203   return static_cast<uint32_t>(base::bits::CountPopulation(*input));
204 }
205 
float64_pow_wrapper(double * param0,double * param1)206 void float64_pow_wrapper(double* param0, double* param1) {
207   double x = ReadDoubleValue(param0);
208   double y = ReadDoubleValue(param1);
209   WriteDoubleValue(param0, Pow(x, y));
210 }
211 }  // namespace wasm
212 }  // namespace internal
213 }  // namespace v8
214