C++程序  |  210行  |  6.08 KB

/******************************************************************************
 *
 * 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.
 *
 *****************************************************************************
 * Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
*/
/*!
******************************************************************************
* \file var_q_operator.c
*
* \brief
*    This files to be used for basic fixed Q-point functions
*
* \date
*
* \author
*    ittiam
*
******************************************************************************
*/
/*****************************************************************************/
/* File Includes                                                             */
/*****************************************************************************/
/* User include files */
#include "ia_type_def.h"
#include "defs.h"
#include "ia_basic_ops32.h"
#include "ia_basic_ops40.h"
/* #include "num_struct.h" */
#include "var_q_operator.h"
#include "sqrt_interp.h"
#include "common_rom.h"

#define NUM_BITS_MAG 32

/************************************************************************************/
/* The files to be used for basic fixed Q-point functions :                         */
/*                                                                                  */
/* audio/ia_standards/c64x/include/ia_basic_ops32.h        cvs_version : FULL_V1_16 */
/* audio/ia_standards/c64x/include/ia_basic_ops40.h     cvs_version : FULL_V1_16    */
/*                                                                                  */
/************************************************************************************/

/* Multiply */

void mult32_var_q(number_t a, number_t b, number_t *c)
{
    WORD32 Q_a;
    WORD32 Q_b;
    /* WORD32 final_Q; */
    WORD32 norm_a;
    WORD32 norm_b;

    norm_a = norm32(a.sm); /* norm32 defined in ia_basic_ops32.h */
    norm_b = norm32(b.sm);

    Q_a = norm_a + a.e;
    Q_b = norm_b + b.e;

    a.sm = shl32_sat(a.sm, norm_a);
    b.sm = shl32_sat(b.sm, norm_b);

    c->sm = mult32(a.sm, b.sm); /* mult32 defined in ia_basic_ops40.h */
    c->e = a.e + b.e + norm_a + norm_b - 32; /* mult32 decreases the Q-format by 32 */
}

/* Division */

void div32_var_q(number_t a, number_t b, number_t *c)
{
    WORD32 qoutient_q_format;

    c->sm = div32(a.sm, b.sm, &qoutient_q_format); /* div32 defined in ia_basic_ops32.h */
    c->e = (a.e - b.e) + qoutient_q_format;
}

/* Addition */

void add32_var_q(number_t a, number_t b, number_t *c)
{
    WORD32 Q_a;
    WORD32 Q_b;
    WORD32 final_Q;
    WORD32 norm_a;
    WORD32 norm_b;

    norm_a = norm32(a.sm) - 1; /* norm32 defined in ia_basic_ops32.h */
    norm_b = norm32(b.sm) - 1; /* we normalise a & b only to 30t bit
                                        instead of to 31st bit
                                        */

    Q_a = norm_a + a.e;
    Q_b = norm_b + b.e;

    if(Q_b < Q_a)
    {
        b.sm = shl32_dir_sat(b.sm, norm_b);
        a.sm = shr32_dir_sat(a.sm, ((a.e - b.e) - norm_b));
        final_Q = Q_b;
    }
    else if(Q_a < Q_b)
    {
        a.sm = shl32_dir_sat(a.sm, norm_a);
        b.sm = shr32_dir_sat(b.sm, ((b.e - a.e) - norm_a));
        final_Q = Q_a;
    }
    else
    {
        a.sm = shl32_dir_sat(a.sm, norm_a);
        b.sm = shl32_dir_sat(b.sm, norm_b);
        final_Q = Q_a;
    }

    c->sm = add32(a.sm, b.sm); /* add32_shr defined in ia_basic_ops32.h */
    c->e = final_Q; /* because add32_shr does right shift
                                            by 1 before adding */
}

/* Subtraction */

void sub32_var_q(number_t a, number_t b, number_t *c)
{
    WORD32 Q_a;
    WORD32 Q_b;
    WORD32 final_Q;
    WORD32 norm_a;
    WORD32 norm_b;

    norm_a = norm32(a.sm) - 1; /* norm32 defined in ia_basic_ops32.h */
    norm_b = norm32(b.sm) - 1; /* we normalise a & b only to 30t bit
                                        instead of to 31st bit
                                        */

    Q_a = norm_a + a.e;
    Q_b = norm_b + b.e;

    if(Q_b < Q_a)
    {
        b.sm = shl32_dir_sat(b.sm, norm_b);
        a.sm = shr32_dir_sat(a.sm, ((a.e - b.e) - norm_b));
        final_Q = Q_b;
    }
    else if(Q_a < Q_b)
    {
        a.sm = shl32_dir_sat(a.sm, norm_a);
        b.sm = shr32_dir_sat(b.sm, ((b.e - a.e) - norm_a));
        final_Q = Q_a;
    }
    else
    {
        a.sm = shl32_dir_sat(a.sm, norm_a);
        b.sm = shl32_dir_sat(b.sm, norm_b);
        final_Q = Q_a;
    }

    c->sm = sub32(a.sm, b.sm); /* add32_shr defined in ia_basic_ops32.h */
    c->e = final_Q; /* because add32_shr does right shift
                                            by 1 before adding */
}

/* square root */

void sqrt32_var_q(number_t a, number_t *c)
{
    WORD32 q_temp;
    q_temp = a.e;
    c->sm = sqrtFix_interpolate(a.sm, &q_temp, gi4_sqrt_tab);
    /* c->sm = sqrtFix(a.sm, &q_temp, gi4_sqrt_tab); */
    c->e = q_temp;
}

void number_t_to_word32(number_t num_a, WORD32 *a)
{
    *a = shr32_dir_sat(num_a.sm, num_a.e);
}

/*
convert_float_to_fix(float a_f,
                     number_t *a)
{
            double log_a_f;
            log_a_f = log(ABS(a_f))/log(2);

            a->e = 30 - (WORD32)ceil(log_a_f);
            a->sm = (WORD32) (a_f * pow(2, a->e));
}
*/

#ifdef ITT_C6678
#pragma CODE_SECTION(number_t_to_word32, "itt_varq_l1pram");
#pragma CODE_SECTION(sqrt32_var_q, "itt_varq_l1pram");
#pragma CODE_SECTION(sub32_var_q, "itt_varq_l1pram");
#pragma CODE_SECTION(add32_var_q, "itt_varq_l1pram");
#pragma CODE_SECTION(div32_var_q, "itt_varq_l1pram");
#pragma CODE_SECTION(mult32_var_q, "itt_varq_l1pram");
#endif