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/minifloat.h> 19*b9df5ad1SAndroid Build Coastguard Worker 20*b9df5ad1SAndroid Build Coastguard Worker #define EXPONENT_BITS 3 21*b9df5ad1SAndroid Build Coastguard Worker #define EXPONENT_MAX ((1 << EXPONENT_BITS) - 1) 22*b9df5ad1SAndroid Build Coastguard Worker #define EXCESS ((1 << EXPONENT_BITS) - 2) 23*b9df5ad1SAndroid Build Coastguard Worker 24*b9df5ad1SAndroid Build Coastguard Worker #define MANTISSA_BITS 13 25*b9df5ad1SAndroid Build Coastguard Worker #define MANTISSA_MAX ((1 << MANTISSA_BITS) - 1) 26*b9df5ad1SAndroid Build Coastguard Worker #define HIDDEN_BIT (1 << MANTISSA_BITS) 27*b9df5ad1SAndroid Build Coastguard Worker #define ONE_FLOAT ((float) (1 << (MANTISSA_BITS + 1))) 28*b9df5ad1SAndroid Build Coastguard Worker 29*b9df5ad1SAndroid Build Coastguard Worker #define MINIFLOAT_MAX ((EXPONENT_MAX << MANTISSA_BITS) | MANTISSA_MAX) 30*b9df5ad1SAndroid Build Coastguard Worker 31*b9df5ad1SAndroid Build Coastguard Worker #if EXPONENT_BITS + MANTISSA_BITS != 16 32*b9df5ad1SAndroid Build Coastguard Worker #error EXPONENT_BITS and MANTISSA_BITS must sum to 16 33*b9df5ad1SAndroid Build Coastguard Worker #endif 34*b9df5ad1SAndroid Build Coastguard Worker gain_from_float(float v)35*b9df5ad1SAndroid Build Coastguard Workergain_minifloat_t gain_from_float(float v) 36*b9df5ad1SAndroid Build Coastguard Worker { 37*b9df5ad1SAndroid Build Coastguard Worker if (isnan(v) || v <= 0.0f) { 38*b9df5ad1SAndroid Build Coastguard Worker return 0; 39*b9df5ad1SAndroid Build Coastguard Worker } 40*b9df5ad1SAndroid Build Coastguard Worker if (v >= 2.0f) { 41*b9df5ad1SAndroid Build Coastguard Worker return MINIFLOAT_MAX; 42*b9df5ad1SAndroid Build Coastguard Worker } 43*b9df5ad1SAndroid Build Coastguard Worker int exp; 44*b9df5ad1SAndroid Build Coastguard Worker float r = frexpf(v, &exp); 45*b9df5ad1SAndroid Build Coastguard Worker if ((exp += EXCESS) > EXPONENT_MAX) { 46*b9df5ad1SAndroid Build Coastguard Worker return MINIFLOAT_MAX; 47*b9df5ad1SAndroid Build Coastguard Worker } 48*b9df5ad1SAndroid Build Coastguard Worker if (-exp >= MANTISSA_BITS) { 49*b9df5ad1SAndroid Build Coastguard Worker return 0; 50*b9df5ad1SAndroid Build Coastguard Worker } 51*b9df5ad1SAndroid Build Coastguard Worker int mantissa = (int) (r * ONE_FLOAT); 52*b9df5ad1SAndroid Build Coastguard Worker return exp > 0 ? (exp << MANTISSA_BITS) | (mantissa & ~HIDDEN_BIT) : 53*b9df5ad1SAndroid Build Coastguard Worker (mantissa >> (1 - exp)) & MANTISSA_MAX; 54*b9df5ad1SAndroid Build Coastguard Worker } 55*b9df5ad1SAndroid Build Coastguard Worker float_from_gain(gain_minifloat_t a)56*b9df5ad1SAndroid Build Coastguard Workerfloat float_from_gain(gain_minifloat_t a) 57*b9df5ad1SAndroid Build Coastguard Worker { 58*b9df5ad1SAndroid Build Coastguard Worker int mantissa = a & MANTISSA_MAX; 59*b9df5ad1SAndroid Build Coastguard Worker int exponent = (a >> MANTISSA_BITS) & EXPONENT_MAX; 60*b9df5ad1SAndroid Build Coastguard Worker return ldexpf((exponent > 0 ? HIDDEN_BIT | mantissa : mantissa << 1) / ONE_FLOAT, 61*b9df5ad1SAndroid Build Coastguard Worker exponent - EXCESS); 62*b9df5ad1SAndroid Build Coastguard Worker } 63