ms_stereo.c revision b676a05348e4c516fa8b57e33b10548e6142c3f8
1/*
2 ** Copyright 2003-2010, VisualOn, Inc.
3 **
4 ** Licensed under the Apache License, Version 2.0 (the "License");
5 ** you may not use this file except in compliance with the License.
6 ** You may obtain a copy of the License at
7 **
8 **     http://www.apache.org/licenses/LICENSE-2.0
9 **
10 ** Unless required by applicable law or agreed to in writing, software
11 ** distributed under the License is distributed on an "AS IS" BASIS,
12 ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 ** See the License for the specific language governing permissions and
14 ** limitations under the License.
15 */
16/*******************************************************************************
17	File:		ms_stereo.c
18
19	Content:	MS stereo processing function
20
21*******************************************************************************/
22
23#include "basic_op.h"
24#include "oper_32b.h"
25#include "psy_const.h"
26#include "ms_stereo.h"
27
28
29/********************************************************************************
30*
31* function name: MsStereoProcessing
32* description:  detect use ms stereo or not
33*				if ((min(thrLn, thrRn)*min(thrLn, thrRn))/(enMn*enSn))
34*				>= ((thrLn *thrRn)/(enLn*enRn)) then ms stereo
35*
36**********************************************************************************/
37void MsStereoProcessing(Word32       *sfbEnergyLeft,
38                        Word32       *sfbEnergyRight,
39                        const Word32 *sfbEnergyMid,
40                        const Word32 *sfbEnergySide,
41                        Word32       *mdctSpectrumLeft,
42                        Word32       *mdctSpectrumRight,
43                        Word32       *sfbThresholdLeft,
44                        Word32       *sfbThresholdRight,
45                        Word32       *sfbSpreadedEnLeft,
46                        Word32       *sfbSpreadedEnRight,
47                        Word16       *msDigest,
48                        Word16       *msMask,
49                        const Word16  sfbCnt,
50                        const Word16  sfbPerGroup,
51                        const Word16  maxSfbPerGroup,
52                        const Word16 *sfbOffset) {
53  Word32 temp;
54  Word32 sfb,sfboffs, j;
55  Word32 msMaskTrueSomewhere = 0;
56  Word32 msMaskFalseSomewhere = 0;
57
58  for (sfb=0; sfb<sfbCnt; sfb+=sfbPerGroup) {
59    for (sfboffs=0;sfboffs<maxSfbPerGroup;sfboffs++) {
60
61      Word32 temp;
62      Word32 pnlr,pnms;
63      Word32 minThreshold;
64      Word32 thrL, thrR, nrgL, nrgR;
65      Word32 idx, shift;
66
67      idx = sfb + sfboffs;
68
69      thrL = sfbThresholdLeft[idx];
70      thrR = sfbThresholdRight[idx];
71      nrgL = sfbEnergyLeft[idx];
72      nrgR = sfbEnergyRight[idx];
73
74      minThreshold = min(thrL, thrR);
75
76      nrgL = max(nrgL,thrL) + 1;
77      shift = norm_l(nrgL);
78	  nrgL = Div_32(thrL << shift, nrgL << shift);
79      nrgR = max(nrgR,thrR) + 1;
80      shift = norm_l(nrgR);
81	  nrgR = Div_32(thrR << shift, nrgR << shift);
82
83	  pnlr = fixmul(nrgL, nrgR);
84
85      nrgL = sfbEnergyMid[idx];
86      nrgR = sfbEnergySide[idx];
87
88      nrgL = max(nrgL,minThreshold) + 1;
89      shift = norm_l(nrgL);
90	  nrgL = Div_32(minThreshold << shift, nrgL << shift);
91
92      nrgR = max(nrgR,minThreshold) + 1;
93      shift = norm_l(nrgR);
94	  nrgR = Div_32(minThreshold << shift, nrgR << shift);
95
96      pnms = fixmul(nrgL, nrgR);
97
98      temp = (pnlr + 1) / ((pnms >> 8) + 1);
99
100      temp = pnms - pnlr;
101      if( temp > 0 ){
102
103        msMask[idx] = 1;
104        msMaskTrueSomewhere = 1;
105
106        for (j=sfbOffset[idx]; j<sfbOffset[idx+1]; j++) {
107          Word32 left, right;
108          left  = (mdctSpectrumLeft[j] >>  1);
109          right = (mdctSpectrumRight[j] >> 1);
110          mdctSpectrumLeft[j] =  left + right;
111          mdctSpectrumRight[j] =  left - right;
112        }
113
114        sfbThresholdLeft[idx] = minThreshold;
115        sfbThresholdRight[idx] = minThreshold;
116        sfbEnergyLeft[idx] = sfbEnergyMid[idx];
117        sfbEnergyRight[idx] = sfbEnergySide[idx];
118
119        sfbSpreadedEnRight[idx] = min(sfbSpreadedEnLeft[idx],sfbSpreadedEnRight[idx]) >> 1;
120        sfbSpreadedEnLeft[idx] = sfbSpreadedEnRight[idx];
121
122      }
123      else {
124        msMask[idx]  = 0;
125        msMaskFalseSomewhere = 1;
126      }
127    }
128    if ( msMaskTrueSomewhere ) {
129      if(msMaskFalseSomewhere ) {
130        *msDigest = SI_MS_MASK_SOME;
131      } else {
132        *msDigest = SI_MS_MASK_ALL;
133      }
134    } else {
135      *msDigest = SI_MS_MASK_NONE;
136    }
137  }
138
139}
140