/* * Copyright (C) 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef RDSP_H #define RDSP_H #include <complex> #include <log/log.h> #include <vector> #include <map> using FloatVec = std::vector<float>; using IntVec = std::vector<int>; using ComplexVec = std::vector<std::complex<float>>; // ======= // Helper Functions // ======= template <class T> static T dBtoLinear(T valueDb) { return pow (10, valueDb / 20.0); } template <class T> static T linearToDb(T value) { return 20 * log10(value); } // ======= // DSP window creation // ======= #define TWOPI (M_PI * 2) enum rdsp_window_type { RDSP_WINDOW_RECTANGULAR, RDSP_WINDOW_TRIANGULAR, RDSP_WINDOW_TRIANGULAR_FLAT_TOP, RDSP_WINDOW_HAMMING, RDSP_WINDOW_HAMMING_FLAT_TOP, RDSP_WINDOW_HANNING, RDSP_WINDOW_HANNING_FLAT_TOP, }; template <typename T> static void fillRectangular(T &v) { const size_t size = v.size(); for (size_t i = 0; i < size; i++) { v[i] = 1.0; } } //rectangular template <typename T> static void fillTriangular(T &v, size_t overlap) { const size_t size = v.size(); //ramp up size_t i = 0; if (overlap > 0) { for (; i < overlap; i++) { v[i] = (2.0 * i + 1) / (2 * overlap); } } //flat top for (; i < size - overlap; i++) { v[i] = 1.0; } //ramp down if (overlap > 0) { for (; i < size; i++) { v[i] = (2.0 * (size - i) - 1) / (2 * overlap); } } } //triangular template <typename T> static void fillHamming(T &v, size_t overlap) { const size_t size = v.size(); const size_t twoOverlap = 2 * overlap; size_t i = 0; if (overlap > 0) { for (; i < overlap; i++) { v[i] = 0.54 - 0.46 * cos(TWOPI * i /(twoOverlap - 1)); } } //flat top for (; i < size - overlap; i++) { v[i] = 1.0; } //ramp down if (overlap > 0) { for (; i < size; i++) { int k = i - ((int)size - 2 * overlap); v[i] = 0.54 - 0.46 * cos(TWOPI * k / (twoOverlap - 1)); } } } //hamming template <typename T> static void fillHanning(T &v, size_t overlap) { const size_t size = v.size(); const size_t twoOverlap = 2 * overlap; //ramp up size_t i = 0; if (overlap > 0) { for (; i < overlap; i++) { v[i] = 0.5 * (1.0 - cos(TWOPI * i / (twoOverlap - 1))); } } //flat top for (; i < size - overlap; i++) { v[i] = 1.0; } //ramp down if (overlap > 0) { for (; i < size; i++) { int k = i - ((int)size - 2 * overlap); v[i] = 0.5 * (1.0 - cos(TWOPI * k / (twoOverlap - 1))); } } } template <typename T> static void fill_window(T &v, int type, size_t size, size_t overlap) { if (overlap > size / 2) { overlap = size / 2; } v.resize(size); switch (type) { case RDSP_WINDOW_RECTANGULAR: fillRectangular(v); break; case RDSP_WINDOW_TRIANGULAR: fillTriangular(v, size / 2); break; case RDSP_WINDOW_TRIANGULAR_FLAT_TOP: fillTriangular(v, overlap); break; case RDSP_WINDOW_HAMMING: fillHamming(v, size / 2); break; case RDSP_WINDOW_HAMMING_FLAT_TOP: fillHamming(v, overlap); break; case RDSP_WINDOW_HANNING: fillHanning(v, size / 2); break; case RDSP_WINDOW_HANNING_FLAT_TOP: fillHanning(v, overlap); break; default: ALOGE("Error: unknown window type %d", type); } } //}; #endif //RDSP_H