xref: /aosp_15_r20/external/libmpeg2/common/ideint.c (revision a97c2a1f0a796dc32bed80d3353c69c5fc07c750)
1*a97c2a1fSXin Li /******************************************************************************
2*a97c2a1fSXin Li  *
3*a97c2a1fSXin Li  * Copyright (C) 2015 The Android Open Source Project
4*a97c2a1fSXin Li  *
5*a97c2a1fSXin Li  * Licensed under the Apache License, Version 2.0 (the "License");
6*a97c2a1fSXin Li  * you may not use this file except in compliance with the License.
7*a97c2a1fSXin Li  * You may obtain a copy of the License at:
8*a97c2a1fSXin Li  *
9*a97c2a1fSXin Li  * http://www.apache.org/licenses/LICENSE-2.0
10*a97c2a1fSXin Li  *
11*a97c2a1fSXin Li  * Unless required by applicable law or agreed to in writing, software
12*a97c2a1fSXin Li  * distributed under the License is distributed on an "AS IS" BASIS,
13*a97c2a1fSXin Li  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*a97c2a1fSXin Li  * See the License for the specific language governing permissions and
15*a97c2a1fSXin Li  * limitations under the License.
16*a97c2a1fSXin Li  *
17*a97c2a1fSXin Li  *****************************************************************************
18*a97c2a1fSXin Li  * Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
19*a97c2a1fSXin Li */
20*a97c2a1fSXin Li /**
21*a97c2a1fSXin Li *******************************************************************************
22*a97c2a1fSXin Li * @file
23*a97c2a1fSXin Li *  ideint_api.c
24*a97c2a1fSXin Li *
25*a97c2a1fSXin Li * @brief
26*a97c2a1fSXin Li *  This file contains the definitions of the core  processing of the de-
27*a97c2a1fSXin Li * interlacer.
28*a97c2a1fSXin Li *
29*a97c2a1fSXin Li * @author
30*a97c2a1fSXin Li *  Ittiam
31*a97c2a1fSXin Li *
32*a97c2a1fSXin Li * @par List of Functions:
33*a97c2a1fSXin Li *
34*a97c2a1fSXin Li * @remarks
35*a97c2a1fSXin Li *  None
36*a97c2a1fSXin Li *
37*a97c2a1fSXin Li *******************************************************************************
38*a97c2a1fSXin Li */
39*a97c2a1fSXin Li /*****************************************************************************/
40*a97c2a1fSXin Li /* File Includes                                                             */
41*a97c2a1fSXin Li /*****************************************************************************/
42*a97c2a1fSXin Li /* System include files */
43*a97c2a1fSXin Li #include <stdio.h>
44*a97c2a1fSXin Li #include <stdint.h>
45*a97c2a1fSXin Li #include <string.h>
46*a97c2a1fSXin Li #include <stdlib.h>
47*a97c2a1fSXin Li #include <assert.h>
48*a97c2a1fSXin Li 
49*a97c2a1fSXin Li /* User include files */
50*a97c2a1fSXin Li #include "icv_datatypes.h"
51*a97c2a1fSXin Li #include "icv_macros.h"
52*a97c2a1fSXin Li #include "icv_platform_macros.h"
53*a97c2a1fSXin Li #include "icv.h"
54*a97c2a1fSXin Li #include "icv_variance.h"
55*a97c2a1fSXin Li #include "icv_sad.h"
56*a97c2a1fSXin Li #include "ideint.h"
57*a97c2a1fSXin Li 
58*a97c2a1fSXin Li #include "ideint_defs.h"
59*a97c2a1fSXin Li #include "ideint_structs.h"
60*a97c2a1fSXin Li 
61*a97c2a1fSXin Li #include "ideint_utils.h"
62*a97c2a1fSXin Li #include "ideint_cac.h"
63*a97c2a1fSXin Li #include "ideint_debug.h"
64*a97c2a1fSXin Li #include "ideint_function_selector.h"
65*a97c2a1fSXin Li 
66*a97c2a1fSXin Li /**
67*a97c2a1fSXin Li *******************************************************************************
68*a97c2a1fSXin Li *
69*a97c2a1fSXin Li * @brief
70*a97c2a1fSXin Li *  Return deinterlacer context size
71*a97c2a1fSXin Li *
72*a97c2a1fSXin Li * @par   Description
73*a97c2a1fSXin Li *  Return deinterlacer context size, application will allocate this memory
74*a97c2a1fSXin Li *  and send it as context to process call
75*a97c2a1fSXin Li *
76*a97c2a1fSXin Li * @param[in] None
77*a97c2a1fSXin Li *
78*a97c2a1fSXin Li * @returns
79*a97c2a1fSXin Li * Size of deinterlacer context
80*a97c2a1fSXin Li *
81*a97c2a1fSXin Li * @remarks
82*a97c2a1fSXin Li * None
83*a97c2a1fSXin Li *
84*a97c2a1fSXin Li *******************************************************************************
85*a97c2a1fSXin Li */
ideint_ctxt_size(void)86*a97c2a1fSXin Li WORD32 ideint_ctxt_size(void)
87*a97c2a1fSXin Li {
88*a97c2a1fSXin Li     return sizeof(ctxt_t);
89*a97c2a1fSXin Li }
90*a97c2a1fSXin Li 
91*a97c2a1fSXin Li /**
92*a97c2a1fSXin Li *******************************************************************************
93*a97c2a1fSXin Li *
94*a97c2a1fSXin Li * @brief
95*a97c2a1fSXin Li * Deinterlace given fields and produce a frame
96*a97c2a1fSXin Li *
97*a97c2a1fSXin Li * @par   Description
98*a97c2a1fSXin Li *  Deinterlacer function that deinterlaces given fields and produces a frame
99*a97c2a1fSXin Li *
100*a97c2a1fSXin Li * @param[in] pv_ctxt
101*a97c2a1fSXin Li *  Deinterlacer context returned by ideint_create()
102*a97c2a1fSXin Li *
103*a97c2a1fSXin Li * @param[in] ps_prv_fld
104*a97c2a1fSXin Li *  Previous field (can be null, in which case spatial filtering is done
105*a97c2a1fSXin Li *  unconditionally)
106*a97c2a1fSXin Li *
107*a97c2a1fSXin Li * @param[in] ps_cur_fld
108*a97c2a1fSXin Li *  Current field
109*a97c2a1fSXin Li *
110*a97c2a1fSXin Li * @param[in] ps_nxt_fld
111*a97c2a1fSXin Li *  Next field
112*a97c2a1fSXin Li *
113*a97c2a1fSXin Li * @param[in] ps_out_frm
114*a97c2a1fSXin Li *  Output frame
115*a97c2a1fSXin Li *
116*a97c2a1fSXin Li * @param[in] ps_params
117*a97c2a1fSXin Li *  Parameters
118*a97c2a1fSXin Li *
119*a97c2a1fSXin Li * @param[in] start_row
120*a97c2a1fSXin Li *  Start row
121*a97c2a1fSXin Li *
122*a97c2a1fSXin Li * @param[in] num_rows
123*a97c2a1fSXin Li *  Number of rows to be processed
124*a97c2a1fSXin Li *
125*a97c2a1fSXin Li * @returns
126*a97c2a1fSXin Li *  IDEINT_ERROR_T
127*a97c2a1fSXin Li *
128*a97c2a1fSXin Li * @remarks
129*a97c2a1fSXin Li *
130*a97c2a1fSXin Li *******************************************************************************
131*a97c2a1fSXin Li */
ideint_process(void * pv_ctxt,icv_pic_t * ps_prv_fld,icv_pic_t * ps_cur_fld,icv_pic_t * ps_nxt_fld,icv_pic_t * ps_out_frm,ideint_params_t * ps_params,WORD32 start_row,WORD32 num_rows)132*a97c2a1fSXin Li IDEINT_ERROR_T ideint_process(void *pv_ctxt,
133*a97c2a1fSXin Li                               icv_pic_t *ps_prv_fld,
134*a97c2a1fSXin Li                               icv_pic_t *ps_cur_fld,
135*a97c2a1fSXin Li                               icv_pic_t *ps_nxt_fld,
136*a97c2a1fSXin Li                               icv_pic_t *ps_out_frm,
137*a97c2a1fSXin Li                               ideint_params_t *ps_params,
138*a97c2a1fSXin Li                               WORD32 start_row,
139*a97c2a1fSXin Li                               WORD32 num_rows)
140*a97c2a1fSXin Li {
141*a97c2a1fSXin Li     ctxt_t *ps_ctxt;
142*a97c2a1fSXin Li     WORD32 num_blks_x, num_blks_y;
143*a97c2a1fSXin Li     WORD32 num_comp;
144*a97c2a1fSXin Li     WORD32 i, row, col;
145*a97c2a1fSXin Li     WORD32 rows_remaining;
146*a97c2a1fSXin Li 
147*a97c2a1fSXin Li     if(NULL == pv_ctxt)
148*a97c2a1fSXin Li         return IDEINT_INVALID_CTXT;
149*a97c2a1fSXin Li 
150*a97c2a1fSXin Li     ps_ctxt = (ctxt_t *)pv_ctxt;
151*a97c2a1fSXin Li 
152*a97c2a1fSXin Li     /* Copy the parameters */
153*a97c2a1fSXin Li     if(ps_params)
154*a97c2a1fSXin Li     {
155*a97c2a1fSXin Li         ps_ctxt->s_params = *ps_params;
156*a97c2a1fSXin Li     }
157*a97c2a1fSXin Li     else
158*a97c2a1fSXin Li     {
159*a97c2a1fSXin Li         /* Use default params if ps_params is NULL */
160*a97c2a1fSXin Li         ps_ctxt->s_params.i4_cur_fld_top = 1;
161*a97c2a1fSXin Li         ps_ctxt->s_params.e_mode = IDEINT_MODE_SPATIAL;
162*a97c2a1fSXin Li         ps_ctxt->s_params.e_arch = ideint_default_arch();
163*a97c2a1fSXin Li         ps_ctxt->s_params.e_soc = ICV_SOC_GENERIC;
164*a97c2a1fSXin Li         ps_ctxt->s_params.i4_disable_weave = 0;
165*a97c2a1fSXin Li         ps_ctxt->s_params.pf_aligned_alloc = NULL;
166*a97c2a1fSXin Li         ps_ctxt->s_params.pf_aligned_free = NULL;
167*a97c2a1fSXin Li     }
168*a97c2a1fSXin Li 
169*a97c2a1fSXin Li     /* Start row has to be multiple of 8 */
170*a97c2a1fSXin Li     if(start_row & 0x7)
171*a97c2a1fSXin Li     {
172*a97c2a1fSXin Li         return IDEINT_START_ROW_UNALIGNED;
173*a97c2a1fSXin Li     }
174*a97c2a1fSXin Li 
175*a97c2a1fSXin Li     /* Initialize variances */
176*a97c2a1fSXin Li     ps_ctxt->ai4_vrnc_avg_fb[0] = VAR_AVG_LUMA;
177*a97c2a1fSXin Li     ps_ctxt->ai4_vrnc_avg_fb[1] = VAR_AVG_CHROMA;
178*a97c2a1fSXin Li     ps_ctxt->ai4_vrnc_avg_fb[2] = VAR_AVG_CHROMA;
179*a97c2a1fSXin Li 
180*a97c2a1fSXin Li     ideint_init_function_ptr(ps_ctxt);
181*a97c2a1fSXin Li 
182*a97c2a1fSXin Li     rows_remaining = ps_out_frm->ai4_ht[0] - start_row;
183*a97c2a1fSXin Li     num_rows = MIN(num_rows,
184*a97c2a1fSXin Li                                         rows_remaining);
185*a97c2a1fSXin Li 
186*a97c2a1fSXin Li     IDEINT_CORRUPT_PIC(ps_out_frm, 0xCD);
187*a97c2a1fSXin Li 
188*a97c2a1fSXin Li     //Weave two fields to get a frame
189*a97c2a1fSXin Li     if(IDEINT_MODE_WEAVE == ps_ctxt->s_params.e_mode)
190*a97c2a1fSXin Li     {
191*a97c2a1fSXin Li         if(0 == ps_ctxt->s_params.i4_disable_weave)
192*a97c2a1fSXin Li         {
193*a97c2a1fSXin Li             if(ps_ctxt->s_params.i4_cur_fld_top)
194*a97c2a1fSXin Li                 ideint_weave_pic(ps_cur_fld, ps_nxt_fld, ps_out_frm,
195*a97c2a1fSXin Li                                  start_row,
196*a97c2a1fSXin Li                                  num_rows);
197*a97c2a1fSXin Li             else
198*a97c2a1fSXin Li                 ideint_weave_pic(ps_nxt_fld, ps_cur_fld, ps_out_frm,
199*a97c2a1fSXin Li                                  start_row,
200*a97c2a1fSXin Li                                  num_rows);
201*a97c2a1fSXin Li         }
202*a97c2a1fSXin Li         return IDEINT_ERROR_NONE;
203*a97c2a1fSXin Li     }
204*a97c2a1fSXin Li 
205*a97c2a1fSXin Li     num_comp = 3;
206*a97c2a1fSXin Li 
207*a97c2a1fSXin Li     for(i = 0; i < num_comp; i++)
208*a97c2a1fSXin Li     {
209*a97c2a1fSXin Li         UWORD8 *pu1_prv = NULL, *pu1_out;
210*a97c2a1fSXin Li         UWORD8 *pu1_top, *pu1_bot, *pu1_dst;
211*a97c2a1fSXin Li         WORD32 cur_strd, out_strd, dst_strd;
212*a97c2a1fSXin Li 
213*a97c2a1fSXin Li         WORD32 st_thresh;
214*a97c2a1fSXin Li         WORD32 vrnc_avg_st;
215*a97c2a1fSXin Li         WORD32 disable_cac_sad;
216*a97c2a1fSXin Li         WORD32 comp_row_start, comp_row_end;
217*a97c2a1fSXin Li         num_blks_x = ALIGN8(ps_out_frm->ai4_wd[i]) >> 3;
218*a97c2a1fSXin Li         num_blks_y = ALIGN8(ps_out_frm->ai4_ht[i]) >> 3;
219*a97c2a1fSXin Li         comp_row_start = start_row;
220*a97c2a1fSXin Li         comp_row_end = comp_row_start + num_rows;
221*a97c2a1fSXin Li 
222*a97c2a1fSXin Li         if(i)
223*a97c2a1fSXin Li         {
224*a97c2a1fSXin Li             comp_row_start >>= 1;
225*a97c2a1fSXin Li             comp_row_end >>= 1;
226*a97c2a1fSXin Li         }
227*a97c2a1fSXin Li 
228*a97c2a1fSXin Li         comp_row_end = MIN(comp_row_end, ps_out_frm->ai4_ht[i]);
229*a97c2a1fSXin Li 
230*a97c2a1fSXin Li         comp_row_start =  ALIGN8(comp_row_start) >> 3;
231*a97c2a1fSXin Li         comp_row_end  = ALIGN8(comp_row_end) >> 3;
232*a97c2a1fSXin Li         st_thresh        = ST_THRESH;
233*a97c2a1fSXin Li         vrnc_avg_st      = VAR_AVG_LUMA;
234*a97c2a1fSXin Li 
235*a97c2a1fSXin Li         if(i)
236*a97c2a1fSXin Li         {
237*a97c2a1fSXin Li             st_thresh = ST_THRESH >> 1;
238*a97c2a1fSXin Li             vrnc_avg_st = VAR_AVG_CHROMA;
239*a97c2a1fSXin Li         }
240*a97c2a1fSXin Li 
241*a97c2a1fSXin Li         out_strd = ps_out_frm->ai4_strd[i];
242*a97c2a1fSXin Li         if(ps_ctxt->s_params.i4_cur_fld_top)
243*a97c2a1fSXin Li         {
244*a97c2a1fSXin Li             cur_strd = ps_cur_fld->ai4_strd[i];
245*a97c2a1fSXin Li         }
246*a97c2a1fSXin Li         else
247*a97c2a1fSXin Li         {
248*a97c2a1fSXin Li             cur_strd = ps_nxt_fld->ai4_strd[i];
249*a97c2a1fSXin Li         }
250*a97c2a1fSXin Li 
251*a97c2a1fSXin Li 
252*a97c2a1fSXin Li         disable_cac_sad = 0;
253*a97c2a1fSXin Li         /* If previous field is not provided, then change to SPATIAL mode */
254*a97c2a1fSXin Li         if(ps_prv_fld->apu1_buf[i] == NULL)
255*a97c2a1fSXin Li         {
256*a97c2a1fSXin Li             disable_cac_sad = 1;
257*a97c2a1fSXin Li         }
258*a97c2a1fSXin Li         for(row = comp_row_start; row < comp_row_end; row++)
259*a97c2a1fSXin Li         {
260*a97c2a1fSXin Li             pu1_out = ps_out_frm->apu1_buf[i];
261*a97c2a1fSXin Li             pu1_out += (ps_out_frm->ai4_strd[i] * row << 3);
262*a97c2a1fSXin Li 
263*a97c2a1fSXin Li             if(0 == disable_cac_sad)
264*a97c2a1fSXin Li             {
265*a97c2a1fSXin Li                 pu1_prv = ps_prv_fld->apu1_buf[i];
266*a97c2a1fSXin Li                 pu1_prv += (ps_prv_fld->ai4_strd[i] * row << 2);
267*a97c2a1fSXin Li             }
268*a97c2a1fSXin Li 
269*a97c2a1fSXin Li             if(ps_ctxt->s_params.i4_cur_fld_top)
270*a97c2a1fSXin Li             {
271*a97c2a1fSXin Li                 pu1_top = ps_cur_fld->apu1_buf[i];
272*a97c2a1fSXin Li                 pu1_bot = ps_nxt_fld->apu1_buf[i];
273*a97c2a1fSXin Li             }
274*a97c2a1fSXin Li             else
275*a97c2a1fSXin Li             {
276*a97c2a1fSXin Li                 pu1_top = ps_nxt_fld->apu1_buf[i];
277*a97c2a1fSXin Li                 pu1_bot = ps_cur_fld->apu1_buf[i];
278*a97c2a1fSXin Li             }
279*a97c2a1fSXin Li             pu1_top += (cur_strd * row << 2);
280*a97c2a1fSXin Li             pu1_bot += (cur_strd * row << 2);
281*a97c2a1fSXin Li 
282*a97c2a1fSXin Li             for(col = 0; col < num_blks_x; col++)
283*a97c2a1fSXin Li             {
284*a97c2a1fSXin Li                 WORD32 cac, sad, vrnc;
285*a97c2a1fSXin Li                 WORD32 th_num, th_den;
286*a97c2a1fSXin Li                 UWORD8 au1_dst[BLK_WD * BLK_HT];
287*a97c2a1fSXin Li                 WORD32 blk_wd, blk_ht;
288*a97c2a1fSXin Li                 WORD32 input_boundary;
289*a97c2a1fSXin Li                 cac = 0;
290*a97c2a1fSXin Li                 sad = 0;
291*a97c2a1fSXin Li                 th_den = 0;
292*a97c2a1fSXin Li                 th_num = st_thresh;
293*a97c2a1fSXin Li                 vrnc = 0;
294*a97c2a1fSXin Li 
295*a97c2a1fSXin Li                 disable_cac_sad = 0;
296*a97c2a1fSXin Li                 /* If previous field is not provided, then change to SPATIAL mode */
297*a97c2a1fSXin Li                 if(ps_prv_fld->apu1_buf[i] == NULL)
298*a97c2a1fSXin Li                 {
299*a97c2a1fSXin Li                     disable_cac_sad = 1;
300*a97c2a1fSXin Li                 }
301*a97c2a1fSXin Li                 /* For boundary blocks when input dimensions are not multiple of 8,
302*a97c2a1fSXin Li                  * then change to spatial mode */
303*a97c2a1fSXin Li                 input_boundary = 0;
304*a97c2a1fSXin Li 
305*a97c2a1fSXin Li                 blk_wd = BLK_WD;
306*a97c2a1fSXin Li                 blk_ht = BLK_HT;
307*a97c2a1fSXin Li 
308*a97c2a1fSXin Li                 if((((num_blks_x - 1) == col) && (ps_out_frm->ai4_wd[i] & 0x7)) ||
309*a97c2a1fSXin Li                     (((num_blks_y - 1) == row) && (ps_out_frm->ai4_ht[i] & 0x7)))
310*a97c2a1fSXin Li                 {
311*a97c2a1fSXin Li                     disable_cac_sad = 1;
312*a97c2a1fSXin Li                     input_boundary = 1;
313*a97c2a1fSXin Li 
314*a97c2a1fSXin Li                     if(((num_blks_x - 1) == col) && (ps_out_frm->ai4_wd[i] & 0x7))
315*a97c2a1fSXin Li                         blk_wd = (ps_out_frm->ai4_wd[i] & 0x7);
316*a97c2a1fSXin Li 
317*a97c2a1fSXin Li                     if(((num_blks_y - 1) == row) && (ps_out_frm->ai4_ht[i] & 0x7))
318*a97c2a1fSXin Li                         blk_ht = (ps_out_frm->ai4_ht[i] & 0x7);
319*a97c2a1fSXin Li 
320*a97c2a1fSXin Li                 }
321*a97c2a1fSXin Li 
322*a97c2a1fSXin Li                 if(0 == disable_cac_sad)
323*a97c2a1fSXin Li                 {
324*a97c2a1fSXin Li                     /* Compute SAD */
325*a97c2a1fSXin Li                     PROFILE_DISABLE_SAD
326*a97c2a1fSXin Li                     sad = ps_ctxt->pf_sad_8x4(pu1_prv, pu1_bot, cur_strd,
327*a97c2a1fSXin Li                                               cur_strd,
328*a97c2a1fSXin Li                                               BLK_WD,
329*a97c2a1fSXin Li                                               BLK_HT >> 1);
330*a97c2a1fSXin Li                     /* Compute Variance */
331*a97c2a1fSXin Li                     PROFILE_DISABLE_VARIANCE
332*a97c2a1fSXin Li                     vrnc = ps_ctxt->pf_variance_8x4(pu1_top, cur_strd, BLK_WD,
333*a97c2a1fSXin Li                                                     BLK_HT >> 1);
334*a97c2a1fSXin Li 
335*a97c2a1fSXin Li                     th_num = st_thresh;
336*a97c2a1fSXin Li 
337*a97c2a1fSXin Li                     th_num *= vrnc_avg_st +
338*a97c2a1fSXin Li                               ((MOD_IDX_ST_NUM * vrnc) >> MOD_IDX_ST_SHIFT);
339*a97c2a1fSXin Li 
340*a97c2a1fSXin Li                     th_den = vrnc +
341*a97c2a1fSXin Li                              ((MOD_IDX_ST_NUM * vrnc_avg_st) >> MOD_IDX_ST_SHIFT);
342*a97c2a1fSXin Li 
343*a97c2a1fSXin Li                     if((sad * th_den) <= th_num)
344*a97c2a1fSXin Li                     {
345*a97c2a1fSXin Li                         /* Calculate Combing Artifact if SAD test fails */
346*a97c2a1fSXin Li                         PROFILE_DISABLE_CAC
347*a97c2a1fSXin Li                         cac = ps_ctxt->pf_cac_8x8(pu1_top, pu1_bot, cur_strd, cur_strd);
348*a97c2a1fSXin Li                     }
349*a97c2a1fSXin Li                 }
350*a97c2a1fSXin Li 
351*a97c2a1fSXin Li                 pu1_dst = pu1_out;
352*a97c2a1fSXin Li                 dst_strd = out_strd;
353*a97c2a1fSXin Li 
354*a97c2a1fSXin Li                 /* In case boundary blocks are not complete (dimensions non-multiple of 8)
355*a97c2a1fSXin Li                  * Use intermediate buffer as destination and copy required pixels to output
356*a97c2a1fSXin Li                  * buffer later
357*a97c2a1fSXin Li                  */
358*a97c2a1fSXin Li                 if(input_boundary)
359*a97c2a1fSXin Li                 {
360*a97c2a1fSXin Li                     pu1_dst = au1_dst;
361*a97c2a1fSXin Li                     dst_strd = BLK_WD;
362*a97c2a1fSXin Li                     ideint_weave_blk(pu1_top, pu1_bot, pu1_dst, dst_strd,
363*a97c2a1fSXin Li                                      cur_strd, blk_wd, blk_ht);
364*a97c2a1fSXin Li                 }
365*a97c2a1fSXin Li 
366*a97c2a1fSXin Li                 /* Weave the two fields unconditionally */
367*a97c2a1fSXin Li                 if(0 == ps_ctxt->s_params.i4_disable_weave)
368*a97c2a1fSXin Li                 {
369*a97c2a1fSXin Li                     ideint_weave_blk(pu1_top, pu1_bot, pu1_dst, dst_strd,
370*a97c2a1fSXin Li                                      cur_strd, blk_wd, blk_ht);
371*a97c2a1fSXin Li                 }
372*a97c2a1fSXin Li 
373*a97c2a1fSXin Li                 if(disable_cac_sad || cac || (sad * th_den > th_num))
374*a97c2a1fSXin Li                 {
375*a97c2a1fSXin Li                     /* Pad the input fields in an intermediate buffer if required */
376*a97c2a1fSXin Li                     if((0 == row) || (0 == col) ||
377*a97c2a1fSXin Li                        ((num_blks_x - 1) == col) || ((num_blks_y - 1) == row))
378*a97c2a1fSXin Li                     {
379*a97c2a1fSXin Li                         UWORD8 *pu1_dst_top;
380*a97c2a1fSXin Li                         UWORD8 au1_pad[(BLK_HT + 4) * (BLK_WD + 4)];
381*a97c2a1fSXin Li 
382*a97c2a1fSXin Li                         ideint_pad_blk(pu1_top, pu1_bot, au1_pad, cur_strd, row,
383*a97c2a1fSXin Li                                        col, num_blks_y, num_blks_x, blk_wd, blk_ht);
384*a97c2a1fSXin Li 
385*a97c2a1fSXin Li                         pu1_dst_top = au1_pad + 2 * (BLK_WD + 4) + 2;
386*a97c2a1fSXin Li 
387*a97c2a1fSXin Li                         PROFILE_DISABLE_SPATIAL
388*a97c2a1fSXin Li                         ps_ctxt->pf_spatial_filter(pu1_dst_top, pu1_dst + dst_strd,
389*a97c2a1fSXin Li                                                    (BLK_WD + 4) * 2,
390*a97c2a1fSXin Li                                                    dst_strd * 2);
391*a97c2a1fSXin Li                     }
392*a97c2a1fSXin Li                     else
393*a97c2a1fSXin Li                     {
394*a97c2a1fSXin Li                         PROFILE_DISABLE_SPATIAL
395*a97c2a1fSXin Li                         ps_ctxt->pf_spatial_filter(pu1_top, pu1_dst + dst_strd,
396*a97c2a1fSXin Li                                                    cur_strd, dst_strd * 2);
397*a97c2a1fSXin Li 
398*a97c2a1fSXin Li                     }
399*a97c2a1fSXin Li                 }
400*a97c2a1fSXin Li 
401*a97c2a1fSXin Li                 /* copy required pixels to output buffer for boundary blocks
402*a97c2a1fSXin Li                  * when dimensions are not multiple of 8
403*a97c2a1fSXin Li                  */
404*a97c2a1fSXin Li                 if(input_boundary)
405*a97c2a1fSXin Li                 {
406*a97c2a1fSXin Li                     WORD32 j;
407*a97c2a1fSXin Li 
408*a97c2a1fSXin Li                     for(j = 0; j < blk_ht; j++)
409*a97c2a1fSXin Li                     {
410*a97c2a1fSXin Li                         memcpy(pu1_out + j * out_strd, au1_dst + j * BLK_WD, blk_wd);
411*a97c2a1fSXin Li                     }
412*a97c2a1fSXin Li                 }
413*a97c2a1fSXin Li                 if(NULL != pu1_prv)
414*a97c2a1fSXin Li                 {
415*a97c2a1fSXin Li                     pu1_prv += 8;
416*a97c2a1fSXin Li                 }
417*a97c2a1fSXin Li                 pu1_top += 8;
418*a97c2a1fSXin Li                 pu1_bot += 8;
419*a97c2a1fSXin Li                 pu1_out += 8;
420*a97c2a1fSXin Li             }
421*a97c2a1fSXin Li         }
422*a97c2a1fSXin Li     }
423*a97c2a1fSXin Li     return IDEINT_ERROR_NONE;
424*a97c2a1fSXin Li }
425