-
Notifications
You must be signed in to change notification settings - Fork 74
/
floatPacking.js
69 lines (61 loc) · 2.12 KB
/
floatPacking.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
/**
* This file encodes/decodes float values into 32bit rgba array
*
* It is based on https://computergraphics.stackexchange.com/questions/4151/webgl-packing-unpacking-functions-that-can-roundtrip-all-typical-32-bit-floats
* and it is not perfect. If you know how to improve it - please let me know.
*/
/**
* Encodes float value into output array
* @param {float} val - value to be encode
* @param {Uint8Array} out - array where encoded value needs to be written.
* @param {Number} writeOffset - offset in the original array where values should be written.
*/
export function encodeFloatRGBA(val, out, writeOffset) {
if (val == 0.0) {
out[writeOffset + 0] = 0; out[writeOffset + 1] = 0; out[writeOffset + 2] = 0; out[writeOffset + 3] = 0;
return;
}
var mag = Math.abs(val);
var exponent = Math.floor(Math.log2(mag));
// Correct log2 approximation errors.
exponent += (exp2(exponent) <= mag / 2.0) ? 1 : 0;
exponent -= (exp2(exponent) > mag) ? 1 : 0;
var mantissa;
if (exponent > 100.0) {
mantissa = mag / 1024.0 / exp2(exponent - 10.0) - 1.0;
} else {
mantissa = mag / (exp2(exponent)) - 1.0;
}
var a = exponent + 127.0;
mantissa *= 256.0;
var b = Math.floor(mantissa);
mantissa -= b;
mantissa *= 256.0;
var c = Math.floor(mantissa);
mantissa -= c;
mantissa *= 128.0;
var d = Math.floor(mantissa) * 2.0 + ((val < 0.0) ? 1: 0);
out[writeOffset + 0] = a; out[writeOffset + 1] = b; out[writeOffset + 2] = c; out[writeOffset + 3] = d;
}
/**
* Given byte values in range [0..255] returns decoded float value.
*
* @param {Byte} r
* @param {Byte} g
* @param {Byte} b
* @param {Byte} a
*/
export function decodeFloatRGBA(r, g, b, a) {
var A = Math.floor(r + 0.5);
var B = Math.floor(g + 0.5);
var C = Math.floor(b + 0.5);
var D = Math.floor(a + 0.5);
var exponent = A - 127.0;
var sign = 1.0 - (D % 2.0) * 2.0;
var mantissa = ((A > 0.0) ? 1 : 0)
+ B / 256.0
+ C / 65536.0
+ Math.floor(D / 2.0) / 8388608.0;
return sign * mantissa * exp2(exponent);
}
function exp2(exponent) { return Math.exp(exponent * Math.LN2); }