1*b9df5ad1SAndroid Build Coastguard Worker /* 2*b9df5ad1SAndroid Build Coastguard Worker * Copyright (C) 2014 The Android Open Source Project 3*b9df5ad1SAndroid Build Coastguard Worker * 4*b9df5ad1SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*b9df5ad1SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*b9df5ad1SAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*b9df5ad1SAndroid Build Coastguard Worker * 8*b9df5ad1SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*b9df5ad1SAndroid Build Coastguard Worker * 10*b9df5ad1SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*b9df5ad1SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*b9df5ad1SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*b9df5ad1SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*b9df5ad1SAndroid Build Coastguard Worker * limitations under the License. 15*b9df5ad1SAndroid Build Coastguard Worker */ 16*b9df5ad1SAndroid Build Coastguard Worker 17*b9df5ad1SAndroid Build Coastguard Worker #include <math.h> 18*b9df5ad1SAndroid Build Coastguard Worker #include <audio_utils/sample.h> 19*b9df5ad1SAndroid Build Coastguard Worker 20*b9df5ad1SAndroid Build Coastguard Worker #define SAMPLE_NEG_MAX 0xFFFF 21*b9df5ad1SAndroid Build Coastguard Worker #define SAMPLE_POS_MAX 0x7FFF 22*b9df5ad1SAndroid Build Coastguard Worker 23*b9df5ad1SAndroid Build Coastguard Worker #define SIGN_MASK 0x8000 24*b9df5ad1SAndroid Build Coastguard Worker #define SIGN_BITS 1 25*b9df5ad1SAndroid Build Coastguard Worker 26*b9df5ad1SAndroid Build Coastguard Worker #define EXPONENT_BITS 3 27*b9df5ad1SAndroid Build Coastguard Worker #define EXPONENT_MAX ((1 << EXPONENT_BITS) - 1) 28*b9df5ad1SAndroid Build Coastguard Worker #define EXCESS ((1 << EXPONENT_BITS) - 2) 29*b9df5ad1SAndroid Build Coastguard Worker 30*b9df5ad1SAndroid Build Coastguard Worker #define MANTISSA_BITS 12 31*b9df5ad1SAndroid Build Coastguard Worker #define MANTISSA_MAX ((1 << MANTISSA_BITS) - 1) 32*b9df5ad1SAndroid Build Coastguard Worker #define HIDDEN_BIT (1 << MANTISSA_BITS) 33*b9df5ad1SAndroid Build Coastguard Worker #define ONE_FLOAT ((float) (1 << (MANTISSA_BITS + 1))) 34*b9df5ad1SAndroid Build Coastguard Worker 35*b9df5ad1SAndroid Build Coastguard Worker 36*b9df5ad1SAndroid Build Coastguard Worker #if SIGN_BITS + EXPONENT_BITS + MANTISSA_BITS != 16 37*b9df5ad1SAndroid Build Coastguard Worker #error SIGN_BITS, EXPONENT_BITS and MANTISSA_BITS must sum to 16 38*b9df5ad1SAndroid Build Coastguard Worker #endif 39*b9df5ad1SAndroid Build Coastguard Worker sample_from_float(float v)40*b9df5ad1SAndroid Build Coastguard Workersample_minifloat_t sample_from_float(float v) 41*b9df5ad1SAndroid Build Coastguard Worker { 42*b9df5ad1SAndroid Build Coastguard Worker if (isnan(v)) { 43*b9df5ad1SAndroid Build Coastguard Worker return 0; 44*b9df5ad1SAndroid Build Coastguard Worker } 45*b9df5ad1SAndroid Build Coastguard Worker sample_minifloat_t sign = 0; 46*b9df5ad1SAndroid Build Coastguard Worker if (v < 0.0f) { 47*b9df5ad1SAndroid Build Coastguard Worker sign = SIGN_MASK; 48*b9df5ad1SAndroid Build Coastguard Worker v = -v; 49*b9df5ad1SAndroid Build Coastguard Worker } 50*b9df5ad1SAndroid Build Coastguard Worker // This check could conceivably be tighter: v < constexpr float_from_sample(1). 51*b9df5ad1SAndroid Build Coastguard Worker // Probably only useful to be made more accurate if this is changed to 52*b9df5ad1SAndroid Build Coastguard Worker // manipulate the raw IEEE single precision float bit fields. 53*b9df5ad1SAndroid Build Coastguard Worker if (v <= 0.0f) { 54*b9df5ad1SAndroid Build Coastguard Worker // originally returned sign, but now seems better not to return negative 0 55*b9df5ad1SAndroid Build Coastguard Worker return 0; 56*b9df5ad1SAndroid Build Coastguard Worker } 57*b9df5ad1SAndroid Build Coastguard Worker if (v >= 2.0f) { 58*b9df5ad1SAndroid Build Coastguard Worker return SAMPLE_POS_MAX | sign; 59*b9df5ad1SAndroid Build Coastguard Worker } 60*b9df5ad1SAndroid Build Coastguard Worker int exp; 61*b9df5ad1SAndroid Build Coastguard Worker float r = frexpf(v, &exp); 62*b9df5ad1SAndroid Build Coastguard Worker if ((exp += EXCESS) > EXPONENT_MAX) { 63*b9df5ad1SAndroid Build Coastguard Worker return SAMPLE_POS_MAX | sign; 64*b9df5ad1SAndroid Build Coastguard Worker } 65*b9df5ad1SAndroid Build Coastguard Worker if (-exp >= MANTISSA_BITS) { 66*b9df5ad1SAndroid Build Coastguard Worker // originally returned sign, but now seems better not to return negative 0 67*b9df5ad1SAndroid Build Coastguard Worker return 0; 68*b9df5ad1SAndroid Build Coastguard Worker } 69*b9df5ad1SAndroid Build Coastguard Worker int mantissa = (int) (r * ONE_FLOAT); 70*b9df5ad1SAndroid Build Coastguard Worker sample_minifloat_t ret = exp > 0 ? (exp << MANTISSA_BITS) | (mantissa & ~HIDDEN_BIT) : 71*b9df5ad1SAndroid Build Coastguard Worker (mantissa >> (1 - exp)) & MANTISSA_MAX; 72*b9df5ad1SAndroid Build Coastguard Worker // ret != 0 73*b9df5ad1SAndroid Build Coastguard Worker return ret | sign; 74*b9df5ad1SAndroid Build Coastguard Worker } 75*b9df5ad1SAndroid Build Coastguard Worker float_from_sample(sample_minifloat_t a)76*b9df5ad1SAndroid Build Coastguard Workerfloat float_from_sample(sample_minifloat_t a) 77*b9df5ad1SAndroid Build Coastguard Worker { 78*b9df5ad1SAndroid Build Coastguard Worker int mantissa = a & MANTISSA_MAX; 79*b9df5ad1SAndroid Build Coastguard Worker int exponent = (a >> MANTISSA_BITS) & EXPONENT_MAX; 80*b9df5ad1SAndroid Build Coastguard Worker float f = ldexpf((exponent > 0 ? HIDDEN_BIT | mantissa : mantissa << 1) / ONE_FLOAT, 81*b9df5ad1SAndroid Build Coastguard Worker exponent - EXCESS); 82*b9df5ad1SAndroid Build Coastguard Worker return a & SIGN_MASK ? -f : f; 83*b9df5ad1SAndroid Build Coastguard Worker } 84