xref: /aosp_15_r20/external/libopus/silk/x86/VAD_sse4_1.c (revision a58d3d2adb790c104798cd88c8a3aff4fa8b82cc)
1*a58d3d2aSXin Li /* Copyright (c) 2014-2020, Cisco Systems, INC
2*a58d3d2aSXin Li    Written by XiangMingZhu WeiZhou MinPeng YanWang FrancisQuiers
3*a58d3d2aSXin Li 
4*a58d3d2aSXin Li    Redistribution and use in source and binary forms, with or without
5*a58d3d2aSXin Li    modification, are permitted provided that the following conditions
6*a58d3d2aSXin Li    are met:
7*a58d3d2aSXin Li 
8*a58d3d2aSXin Li    - Redistributions of source code must retain the above copyright
9*a58d3d2aSXin Li    notice, this list of conditions and the following disclaimer.
10*a58d3d2aSXin Li 
11*a58d3d2aSXin Li    - Redistributions in binary form must reproduce the above copyright
12*a58d3d2aSXin Li    notice, this list of conditions and the following disclaimer in the
13*a58d3d2aSXin Li    documentation and/or other materials provided with the distribution.
14*a58d3d2aSXin Li 
15*a58d3d2aSXin Li    THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16*a58d3d2aSXin Li    ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17*a58d3d2aSXin Li    LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
18*a58d3d2aSXin Li    A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
19*a58d3d2aSXin Li    OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20*a58d3d2aSXin Li    EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21*a58d3d2aSXin Li    PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
22*a58d3d2aSXin Li    PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
23*a58d3d2aSXin Li    LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
24*a58d3d2aSXin Li    NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25*a58d3d2aSXin Li    SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26*a58d3d2aSXin Li */
27*a58d3d2aSXin Li 
28*a58d3d2aSXin Li #ifdef HAVE_CONFIG_H
29*a58d3d2aSXin Li #include "config.h"
30*a58d3d2aSXin Li #endif
31*a58d3d2aSXin Li 
32*a58d3d2aSXin Li #include <xmmintrin.h>
33*a58d3d2aSXin Li #include <emmintrin.h>
34*a58d3d2aSXin Li #include <smmintrin.h>
35*a58d3d2aSXin Li 
36*a58d3d2aSXin Li #include "main.h"
37*a58d3d2aSXin Li #include "stack_alloc.h"
38*a58d3d2aSXin Li 
39*a58d3d2aSXin Li /* Weighting factors for tilt measure */
40*a58d3d2aSXin Li static const opus_int32 tiltWeights[ VAD_N_BANDS ] = { 30000, 6000, -12000, -12000 };
41*a58d3d2aSXin Li 
42*a58d3d2aSXin Li /***************************************/
43*a58d3d2aSXin Li /* Get the speech activity level in Q8 */
44*a58d3d2aSXin Li /***************************************/
silk_VAD_GetSA_Q8_sse4_1(silk_encoder_state * psEncC,const opus_int16 pIn[])45*a58d3d2aSXin Li opus_int silk_VAD_GetSA_Q8_sse4_1(                  /* O    Return value, 0 if success                  */
46*a58d3d2aSXin Li     silk_encoder_state          *psEncC,            /* I/O  Encoder state                               */
47*a58d3d2aSXin Li     const opus_int16            pIn[]               /* I    PCM input                                   */
48*a58d3d2aSXin Li )
49*a58d3d2aSXin Li {
50*a58d3d2aSXin Li     opus_int   SA_Q15, pSNR_dB_Q7, input_tilt;
51*a58d3d2aSXin Li     opus_int   decimated_framelength1, decimated_framelength2;
52*a58d3d2aSXin Li     opus_int   decimated_framelength;
53*a58d3d2aSXin Li     opus_int   dec_subframe_length, dec_subframe_offset, SNR_Q7, i, b, s;
54*a58d3d2aSXin Li     opus_int32 sumSquared, smooth_coef_Q16;
55*a58d3d2aSXin Li     opus_int16 HPstateTmp;
56*a58d3d2aSXin Li     VARDECL( opus_int16, X );
57*a58d3d2aSXin Li     opus_int32 Xnrg[ VAD_N_BANDS ];
58*a58d3d2aSXin Li     opus_int32 NrgToNoiseRatio_Q8[ VAD_N_BANDS ];
59*a58d3d2aSXin Li     opus_int32 speech_nrg, x_tmp;
60*a58d3d2aSXin Li     opus_int   X_offset[ VAD_N_BANDS ];
61*a58d3d2aSXin Li     opus_int   ret = 0;
62*a58d3d2aSXin Li     silk_VAD_state *psSilk_VAD = &psEncC->sVAD;
63*a58d3d2aSXin Li 
64*a58d3d2aSXin Li     SAVE_STACK;
65*a58d3d2aSXin Li 
66*a58d3d2aSXin Li #ifdef OPUS_CHECK_ASM
67*a58d3d2aSXin Li     silk_encoder_state psEncC_c;
68*a58d3d2aSXin Li     opus_int ret_c;
69*a58d3d2aSXin Li 
70*a58d3d2aSXin Li     silk_memcpy( &psEncC_c, psEncC, sizeof( psEncC_c ) );
71*a58d3d2aSXin Li     ret_c = silk_VAD_GetSA_Q8_c( &psEncC_c, pIn );
72*a58d3d2aSXin Li #endif
73*a58d3d2aSXin Li 
74*a58d3d2aSXin Li     /* Safety checks */
75*a58d3d2aSXin Li     silk_assert( VAD_N_BANDS == 4 );
76*a58d3d2aSXin Li     celt_assert( MAX_FRAME_LENGTH >= psEncC->frame_length );
77*a58d3d2aSXin Li     celt_assert( psEncC->frame_length <= 512 );
78*a58d3d2aSXin Li     celt_assert( psEncC->frame_length == 8 * silk_RSHIFT( psEncC->frame_length, 3 ) );
79*a58d3d2aSXin Li 
80*a58d3d2aSXin Li     /***********************/
81*a58d3d2aSXin Li     /* Filter and Decimate */
82*a58d3d2aSXin Li     /***********************/
83*a58d3d2aSXin Li     decimated_framelength1 = silk_RSHIFT( psEncC->frame_length, 1 );
84*a58d3d2aSXin Li     decimated_framelength2 = silk_RSHIFT( psEncC->frame_length, 2 );
85*a58d3d2aSXin Li     decimated_framelength = silk_RSHIFT( psEncC->frame_length, 3 );
86*a58d3d2aSXin Li     /* Decimate into 4 bands:
87*a58d3d2aSXin Li        0       L      3L       L              3L                             5L
88*a58d3d2aSXin Li                -      --       -              --                             --
89*a58d3d2aSXin Li                8       8       2               4                              4
90*a58d3d2aSXin Li 
91*a58d3d2aSXin Li        [0-1 kHz| temp. |1-2 kHz|    2-4 kHz    |            4-8 kHz           |
92*a58d3d2aSXin Li 
93*a58d3d2aSXin Li        They're arranged to allow the minimal ( frame_length / 4 ) extra
94*a58d3d2aSXin Li        scratch space during the downsampling process */
95*a58d3d2aSXin Li     X_offset[ 0 ] = 0;
96*a58d3d2aSXin Li     X_offset[ 1 ] = decimated_framelength + decimated_framelength2;
97*a58d3d2aSXin Li     X_offset[ 2 ] = X_offset[ 1 ] + decimated_framelength;
98*a58d3d2aSXin Li     X_offset[ 3 ] = X_offset[ 2 ] + decimated_framelength2;
99*a58d3d2aSXin Li     ALLOC( X, X_offset[ 3 ] + decimated_framelength1, opus_int16 );
100*a58d3d2aSXin Li 
101*a58d3d2aSXin Li     /* 0-8 kHz to 0-4 kHz and 4-8 kHz */
102*a58d3d2aSXin Li     silk_ana_filt_bank_1( pIn, &psSilk_VAD->AnaState[  0 ],
103*a58d3d2aSXin Li         X, &X[ X_offset[ 3 ] ], psEncC->frame_length );
104*a58d3d2aSXin Li 
105*a58d3d2aSXin Li     /* 0-4 kHz to 0-2 kHz and 2-4 kHz */
106*a58d3d2aSXin Li     silk_ana_filt_bank_1( X, &psSilk_VAD->AnaState1[ 0 ],
107*a58d3d2aSXin Li         X, &X[ X_offset[ 2 ] ], decimated_framelength1 );
108*a58d3d2aSXin Li 
109*a58d3d2aSXin Li     /* 0-2 kHz to 0-1 kHz and 1-2 kHz */
110*a58d3d2aSXin Li     silk_ana_filt_bank_1( X, &psSilk_VAD->AnaState2[ 0 ],
111*a58d3d2aSXin Li         X, &X[ X_offset[ 1 ] ], decimated_framelength2 );
112*a58d3d2aSXin Li 
113*a58d3d2aSXin Li     /*********************************************/
114*a58d3d2aSXin Li     /* HP filter on lowest band (differentiator) */
115*a58d3d2aSXin Li     /*********************************************/
116*a58d3d2aSXin Li     X[ decimated_framelength - 1 ] = silk_RSHIFT( X[ decimated_framelength - 1 ], 1 );
117*a58d3d2aSXin Li     HPstateTmp = X[ decimated_framelength - 1 ];
118*a58d3d2aSXin Li     for( i = decimated_framelength - 1; i > 0; i-- ) {
119*a58d3d2aSXin Li         X[ i - 1 ]  = silk_RSHIFT( X[ i - 1 ], 1 );
120*a58d3d2aSXin Li         X[ i ]     -= X[ i - 1 ];
121*a58d3d2aSXin Li     }
122*a58d3d2aSXin Li     X[ 0 ] -= psSilk_VAD->HPstate;
123*a58d3d2aSXin Li     psSilk_VAD->HPstate = HPstateTmp;
124*a58d3d2aSXin Li 
125*a58d3d2aSXin Li     /*************************************/
126*a58d3d2aSXin Li     /* Calculate the energy in each band */
127*a58d3d2aSXin Li     /*************************************/
128*a58d3d2aSXin Li     for( b = 0; b < VAD_N_BANDS; b++ ) {
129*a58d3d2aSXin Li         /* Find the decimated framelength in the non-uniformly divided bands */
130*a58d3d2aSXin Li         decimated_framelength = silk_RSHIFT( psEncC->frame_length, silk_min_int( VAD_N_BANDS - b, VAD_N_BANDS - 1 ) );
131*a58d3d2aSXin Li 
132*a58d3d2aSXin Li         /* Split length into subframe lengths */
133*a58d3d2aSXin Li         dec_subframe_length = silk_RSHIFT( decimated_framelength, VAD_INTERNAL_SUBFRAMES_LOG2 );
134*a58d3d2aSXin Li         dec_subframe_offset = 0;
135*a58d3d2aSXin Li 
136*a58d3d2aSXin Li         /* Compute energy per sub-frame */
137*a58d3d2aSXin Li         /* initialize with summed energy of last subframe */
138*a58d3d2aSXin Li         Xnrg[ b ] = psSilk_VAD->XnrgSubfr[ b ];
139*a58d3d2aSXin Li         for( s = 0; s < VAD_INTERNAL_SUBFRAMES; s++ ) {
140*a58d3d2aSXin Li             __m128i xmm_X, xmm_acc;
141*a58d3d2aSXin Li             sumSquared = 0;
142*a58d3d2aSXin Li 
143*a58d3d2aSXin Li             xmm_acc = _mm_setzero_si128();
144*a58d3d2aSXin Li 
145*a58d3d2aSXin Li             for( i = 0; i < dec_subframe_length - 7; i += 8 )
146*a58d3d2aSXin Li             {
147*a58d3d2aSXin Li                 xmm_X   = _mm_loadu_si128( (__m128i *)(void*)&(X[ X_offset[ b ] + i + dec_subframe_offset ] ) );
148*a58d3d2aSXin Li                 xmm_X   = _mm_srai_epi16( xmm_X, 3 );
149*a58d3d2aSXin Li                 xmm_X   = _mm_madd_epi16( xmm_X, xmm_X );
150*a58d3d2aSXin Li                 xmm_acc = _mm_add_epi32( xmm_acc, xmm_X );
151*a58d3d2aSXin Li             }
152*a58d3d2aSXin Li 
153*a58d3d2aSXin Li             xmm_acc = _mm_add_epi32( xmm_acc, _mm_unpackhi_epi64( xmm_acc, xmm_acc ) );
154*a58d3d2aSXin Li             xmm_acc = _mm_add_epi32( xmm_acc, _mm_shufflelo_epi16( xmm_acc, 0x0E ) );
155*a58d3d2aSXin Li 
156*a58d3d2aSXin Li             sumSquared += _mm_cvtsi128_si32( xmm_acc );
157*a58d3d2aSXin Li 
158*a58d3d2aSXin Li             for( ; i < dec_subframe_length; i++ ) {
159*a58d3d2aSXin Li                 /* The energy will be less than dec_subframe_length * ( silk_int16_MIN / 8 ) ^ 2.            */
160*a58d3d2aSXin Li                 /* Therefore we can accumulate with no risk of overflow (unless dec_subframe_length > 128)  */
161*a58d3d2aSXin Li                 x_tmp = silk_RSHIFT(
162*a58d3d2aSXin Li                     X[ X_offset[ b ] + i + dec_subframe_offset ], 3 );
163*a58d3d2aSXin Li                 sumSquared = silk_SMLABB( sumSquared, x_tmp, x_tmp );
164*a58d3d2aSXin Li 
165*a58d3d2aSXin Li                 /* Safety check */
166*a58d3d2aSXin Li                 silk_assert( sumSquared >= 0 );
167*a58d3d2aSXin Li             }
168*a58d3d2aSXin Li 
169*a58d3d2aSXin Li             /* Add/saturate summed energy of current subframe */
170*a58d3d2aSXin Li             if( s < VAD_INTERNAL_SUBFRAMES - 1 ) {
171*a58d3d2aSXin Li                 Xnrg[ b ] = silk_ADD_POS_SAT32( Xnrg[ b ], sumSquared );
172*a58d3d2aSXin Li             } else {
173*a58d3d2aSXin Li                 /* Look-ahead subframe */
174*a58d3d2aSXin Li                 Xnrg[ b ] = silk_ADD_POS_SAT32( Xnrg[ b ], silk_RSHIFT( sumSquared, 1 ) );
175*a58d3d2aSXin Li             }
176*a58d3d2aSXin Li 
177*a58d3d2aSXin Li             dec_subframe_offset += dec_subframe_length;
178*a58d3d2aSXin Li         }
179*a58d3d2aSXin Li         psSilk_VAD->XnrgSubfr[ b ] = sumSquared;
180*a58d3d2aSXin Li     }
181*a58d3d2aSXin Li 
182*a58d3d2aSXin Li     /********************/
183*a58d3d2aSXin Li     /* Noise estimation */
184*a58d3d2aSXin Li     /********************/
185*a58d3d2aSXin Li     silk_VAD_GetNoiseLevels( &Xnrg[ 0 ], psSilk_VAD );
186*a58d3d2aSXin Li 
187*a58d3d2aSXin Li     /***********************************************/
188*a58d3d2aSXin Li     /* Signal-plus-noise to noise ratio estimation */
189*a58d3d2aSXin Li     /***********************************************/
190*a58d3d2aSXin Li     sumSquared = 0;
191*a58d3d2aSXin Li     input_tilt = 0;
192*a58d3d2aSXin Li     for( b = 0; b < VAD_N_BANDS; b++ ) {
193*a58d3d2aSXin Li         speech_nrg = Xnrg[ b ] - psSilk_VAD->NL[ b ];
194*a58d3d2aSXin Li         if( speech_nrg > 0 ) {
195*a58d3d2aSXin Li             /* Divide, with sufficient resolution */
196*a58d3d2aSXin Li             if( ( Xnrg[ b ] & 0xFF800000 ) == 0 ) {
197*a58d3d2aSXin Li                 NrgToNoiseRatio_Q8[ b ] = silk_DIV32( silk_LSHIFT( Xnrg[ b ], 8 ), psSilk_VAD->NL[ b ] + 1 );
198*a58d3d2aSXin Li             } else {
199*a58d3d2aSXin Li                 NrgToNoiseRatio_Q8[ b ] = silk_DIV32( Xnrg[ b ], silk_RSHIFT( psSilk_VAD->NL[ b ], 8 ) + 1 );
200*a58d3d2aSXin Li             }
201*a58d3d2aSXin Li 
202*a58d3d2aSXin Li             /* Convert to log domain */
203*a58d3d2aSXin Li             SNR_Q7 = silk_lin2log( NrgToNoiseRatio_Q8[ b ] ) - 8 * 128;
204*a58d3d2aSXin Li 
205*a58d3d2aSXin Li             /* Sum-of-squares */
206*a58d3d2aSXin Li             sumSquared = silk_SMLABB( sumSquared, SNR_Q7, SNR_Q7 );          /* Q14 */
207*a58d3d2aSXin Li 
208*a58d3d2aSXin Li             /* Tilt measure */
209*a58d3d2aSXin Li             if( speech_nrg < ( (opus_int32)1 << 20 ) ) {
210*a58d3d2aSXin Li                 /* Scale down SNR value for small subband speech energies */
211*a58d3d2aSXin Li                 SNR_Q7 = silk_SMULWB( silk_LSHIFT( silk_SQRT_APPROX( speech_nrg ), 6 ), SNR_Q7 );
212*a58d3d2aSXin Li             }
213*a58d3d2aSXin Li             input_tilt = silk_SMLAWB( input_tilt, tiltWeights[ b ], SNR_Q7 );
214*a58d3d2aSXin Li         } else {
215*a58d3d2aSXin Li             NrgToNoiseRatio_Q8[ b ] = 256;
216*a58d3d2aSXin Li         }
217*a58d3d2aSXin Li     }
218*a58d3d2aSXin Li 
219*a58d3d2aSXin Li     /* Mean-of-squares */
220*a58d3d2aSXin Li     sumSquared = silk_DIV32_16( sumSquared, VAD_N_BANDS ); /* Q14 */
221*a58d3d2aSXin Li 
222*a58d3d2aSXin Li     /* Root-mean-square approximation, scale to dBs, and write to output pointer */
223*a58d3d2aSXin Li     pSNR_dB_Q7 = (opus_int16)( 3 * silk_SQRT_APPROX( sumSquared ) ); /* Q7 */
224*a58d3d2aSXin Li 
225*a58d3d2aSXin Li     /*********************************/
226*a58d3d2aSXin Li     /* Speech Probability Estimation */
227*a58d3d2aSXin Li     /*********************************/
228*a58d3d2aSXin Li     SA_Q15 = silk_sigm_Q15( silk_SMULWB( VAD_SNR_FACTOR_Q16, pSNR_dB_Q7 ) - VAD_NEGATIVE_OFFSET_Q5 );
229*a58d3d2aSXin Li 
230*a58d3d2aSXin Li     /**************************/
231*a58d3d2aSXin Li     /* Frequency Tilt Measure */
232*a58d3d2aSXin Li     /**************************/
233*a58d3d2aSXin Li     psEncC->input_tilt_Q15 = silk_LSHIFT( silk_sigm_Q15( input_tilt ) - 16384, 1 );
234*a58d3d2aSXin Li 
235*a58d3d2aSXin Li     /**************************************************/
236*a58d3d2aSXin Li     /* Scale the sigmoid output based on power levels */
237*a58d3d2aSXin Li     /**************************************************/
238*a58d3d2aSXin Li     speech_nrg = 0;
239*a58d3d2aSXin Li     for( b = 0; b < VAD_N_BANDS; b++ ) {
240*a58d3d2aSXin Li         /* Accumulate signal-without-noise energies, higher frequency bands have more weight */
241*a58d3d2aSXin Li         speech_nrg += ( b + 1 ) * silk_RSHIFT( Xnrg[ b ] - psSilk_VAD->NL[ b ], 4 );
242*a58d3d2aSXin Li     }
243*a58d3d2aSXin Li 
244*a58d3d2aSXin Li     if( psEncC->frame_length == 20 * psEncC->fs_kHz ) {
245*a58d3d2aSXin Li         speech_nrg = silk_RSHIFT32( speech_nrg, 1 );
246*a58d3d2aSXin Li     }
247*a58d3d2aSXin Li     /* Power scaling */
248*a58d3d2aSXin Li     if( speech_nrg <= 0 ) {
249*a58d3d2aSXin Li         SA_Q15 = silk_RSHIFT( SA_Q15, 1 );
250*a58d3d2aSXin Li     } else if( speech_nrg < 16384 ) {
251*a58d3d2aSXin Li         speech_nrg = silk_LSHIFT32( speech_nrg, 16 );
252*a58d3d2aSXin Li 
253*a58d3d2aSXin Li         /* square-root */
254*a58d3d2aSXin Li         speech_nrg = silk_SQRT_APPROX( speech_nrg );
255*a58d3d2aSXin Li         SA_Q15 = silk_SMULWB( 32768 + speech_nrg, SA_Q15 );
256*a58d3d2aSXin Li     }
257*a58d3d2aSXin Li 
258*a58d3d2aSXin Li     /* Copy the resulting speech activity in Q8 */
259*a58d3d2aSXin Li     psEncC->speech_activity_Q8 = silk_min_int( silk_RSHIFT( SA_Q15, 7 ), silk_uint8_MAX );
260*a58d3d2aSXin Li 
261*a58d3d2aSXin Li     /***********************************/
262*a58d3d2aSXin Li     /* Energy Level and SNR estimation */
263*a58d3d2aSXin Li     /***********************************/
264*a58d3d2aSXin Li     /* Smoothing coefficient */
265*a58d3d2aSXin Li     smooth_coef_Q16 = silk_SMULWB( VAD_SNR_SMOOTH_COEF_Q18, silk_SMULWB( (opus_int32)SA_Q15, SA_Q15 ) );
266*a58d3d2aSXin Li 
267*a58d3d2aSXin Li     if( psEncC->frame_length == 10 * psEncC->fs_kHz ) {
268*a58d3d2aSXin Li         smooth_coef_Q16 >>= 1;
269*a58d3d2aSXin Li     }
270*a58d3d2aSXin Li 
271*a58d3d2aSXin Li     for( b = 0; b < VAD_N_BANDS; b++ ) {
272*a58d3d2aSXin Li         /* compute smoothed energy-to-noise ratio per band */
273*a58d3d2aSXin Li         psSilk_VAD->NrgRatioSmth_Q8[ b ] = silk_SMLAWB( psSilk_VAD->NrgRatioSmth_Q8[ b ],
274*a58d3d2aSXin Li             NrgToNoiseRatio_Q8[ b ] - psSilk_VAD->NrgRatioSmth_Q8[ b ], smooth_coef_Q16 );
275*a58d3d2aSXin Li 
276*a58d3d2aSXin Li         /* signal to noise ratio in dB per band */
277*a58d3d2aSXin Li         SNR_Q7 = 3 * ( silk_lin2log( psSilk_VAD->NrgRatioSmth_Q8[b] ) - 8 * 128 );
278*a58d3d2aSXin Li         /* quality = sigmoid( 0.25 * ( SNR_dB - 16 ) ); */
279*a58d3d2aSXin Li         psEncC->input_quality_bands_Q15[ b ] = silk_sigm_Q15( silk_RSHIFT( SNR_Q7 - 16 * 128, 4 ) );
280*a58d3d2aSXin Li     }
281*a58d3d2aSXin Li 
282*a58d3d2aSXin Li #ifdef OPUS_CHECK_ASM
283*a58d3d2aSXin Li     silk_assert( ret == ret_c );
284*a58d3d2aSXin Li     silk_assert( !memcmp( &psEncC_c, psEncC, sizeof( psEncC_c ) ) );
285*a58d3d2aSXin Li #endif
286*a58d3d2aSXin Li 
287*a58d3d2aSXin Li     RESTORE_STACK;
288*a58d3d2aSXin Li     return( ret );
289*a58d3d2aSXin Li }
290