xref: /OK3568_Linux_fs/kernel/drivers/media/dvb-frontends/drx39xyj/drxj.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun   Copyright (c), 2004-2005,2007-2010 Trident Microsystems, Inc.
3*4882a593Smuzhiyun   All rights reserved.
4*4882a593Smuzhiyun 
5*4882a593Smuzhiyun   Redistribution and use in source and binary forms, with or without
6*4882a593Smuzhiyun   modification, are permitted provided that the following conditions are met:
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun   * Redistributions of source code must retain the above copyright notice,
9*4882a593Smuzhiyun     this list of conditions and the following disclaimer.
10*4882a593Smuzhiyun   * Redistributions in binary form must reproduce the above copyright notice,
11*4882a593Smuzhiyun     this list of conditions and the following disclaimer in the documentation
12*4882a593Smuzhiyun 	and/or other materials provided with the distribution.
13*4882a593Smuzhiyun   * Neither the name of Trident Microsystems nor Hauppauge Computer Works
14*4882a593Smuzhiyun     nor the names of its contributors may be used to endorse or promote
15*4882a593Smuzhiyun 	products derived from this software without specific prior written
16*4882a593Smuzhiyun 	permission.
17*4882a593Smuzhiyun 
18*4882a593Smuzhiyun   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19*4882a593Smuzhiyun   AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20*4882a593Smuzhiyun   IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21*4882a593Smuzhiyun   ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
22*4882a593Smuzhiyun   LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23*4882a593Smuzhiyun   CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24*4882a593Smuzhiyun   SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25*4882a593Smuzhiyun   INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26*4882a593Smuzhiyun   CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27*4882a593Smuzhiyun   ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28*4882a593Smuzhiyun   POSSIBILITY OF SUCH DAMAGE.
29*4882a593Smuzhiyun 
30*4882a593Smuzhiyun   DRXJ specific implementation of DRX driver
31*4882a593Smuzhiyun   authors: Dragan Savic, Milos Nikolic, Mihajlo Katona, Tao Ding, Paul Janssen
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun   The Linux DVB Driver for Micronas DRX39xx family (drx3933j) was
34*4882a593Smuzhiyun   written by Devin Heitmueller <devin.heitmueller@kernellabs.com>
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun   This program is free software; you can redistribute it and/or modify
37*4882a593Smuzhiyun   it under the terms of the GNU General Public License as published by
38*4882a593Smuzhiyun   the Free Software Foundation; either version 2 of the License, or
39*4882a593Smuzhiyun   (at your option) any later version.
40*4882a593Smuzhiyun 
41*4882a593Smuzhiyun   This program is distributed in the hope that it will be useful,
42*4882a593Smuzhiyun   but WITHOUT ANY WARRANTY; without even the implied warranty of
43*4882a593Smuzhiyun   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun   GNU General Public License for more details.
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun   You should have received a copy of the GNU General Public License
48*4882a593Smuzhiyun   along with this program; if not, write to the Free Software
49*4882a593Smuzhiyun   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
50*4882a593Smuzhiyun */
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun /*-----------------------------------------------------------------------------
53*4882a593Smuzhiyun INCLUDE FILES
54*4882a593Smuzhiyun ----------------------------------------------------------------------------*/
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun #define pr_fmt(fmt) KBUILD_MODNAME ":%s: " fmt, __func__
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun #include <linux/module.h>
59*4882a593Smuzhiyun #include <linux/init.h>
60*4882a593Smuzhiyun #include <linux/string.h>
61*4882a593Smuzhiyun #include <linux/slab.h>
62*4882a593Smuzhiyun #include <asm/div64.h>
63*4882a593Smuzhiyun 
64*4882a593Smuzhiyun #include <media/dvb_frontend.h>
65*4882a593Smuzhiyun #include "drx39xxj.h"
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun #include "drxj.h"
68*4882a593Smuzhiyun #include "drxj_map.h"
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun /*============================================================================*/
71*4882a593Smuzhiyun /*=== DEFINES ================================================================*/
72*4882a593Smuzhiyun /*============================================================================*/
73*4882a593Smuzhiyun 
74*4882a593Smuzhiyun #define DRX39XX_MAIN_FIRMWARE "dvb-fe-drxj-mc-1.0.8.fw"
75*4882a593Smuzhiyun 
76*4882a593Smuzhiyun /*
77*4882a593Smuzhiyun * \brief Maximum u32 value.
78*4882a593Smuzhiyun */
79*4882a593Smuzhiyun #ifndef MAX_U32
80*4882a593Smuzhiyun #define MAX_U32  ((u32) (0xFFFFFFFFL))
81*4882a593Smuzhiyun #endif
82*4882a593Smuzhiyun 
83*4882a593Smuzhiyun /* Customer configurable hardware settings, etc */
84*4882a593Smuzhiyun #ifndef MPEG_SERIAL_OUTPUT_PIN_DRIVE_STRENGTH
85*4882a593Smuzhiyun #define MPEG_SERIAL_OUTPUT_PIN_DRIVE_STRENGTH 0x02
86*4882a593Smuzhiyun #endif
87*4882a593Smuzhiyun 
88*4882a593Smuzhiyun #ifndef MPEG_PARALLEL_OUTPUT_PIN_DRIVE_STRENGTH
89*4882a593Smuzhiyun #define MPEG_PARALLEL_OUTPUT_PIN_DRIVE_STRENGTH 0x02
90*4882a593Smuzhiyun #endif
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun #ifndef MPEG_OUTPUT_CLK_DRIVE_STRENGTH
93*4882a593Smuzhiyun #define MPEG_OUTPUT_CLK_DRIVE_STRENGTH 0x06
94*4882a593Smuzhiyun #endif
95*4882a593Smuzhiyun 
96*4882a593Smuzhiyun #ifndef OOB_CRX_DRIVE_STRENGTH
97*4882a593Smuzhiyun #define OOB_CRX_DRIVE_STRENGTH 0x02
98*4882a593Smuzhiyun #endif
99*4882a593Smuzhiyun 
100*4882a593Smuzhiyun #ifndef OOB_DRX_DRIVE_STRENGTH
101*4882a593Smuzhiyun #define OOB_DRX_DRIVE_STRENGTH 0x02
102*4882a593Smuzhiyun #endif
103*4882a593Smuzhiyun /*** START DJCOMBO patches to DRXJ registermap constants *********************/
104*4882a593Smuzhiyun /*** registermap 200706071303 from drxj **************************************/
105*4882a593Smuzhiyun #define   ATV_TOP_CR_AMP_TH_FM                                              0x0
106*4882a593Smuzhiyun #define   ATV_TOP_CR_AMP_TH_L                                               0xA
107*4882a593Smuzhiyun #define   ATV_TOP_CR_AMP_TH_LP                                              0xA
108*4882a593Smuzhiyun #define   ATV_TOP_CR_AMP_TH_BG                                              0x8
109*4882a593Smuzhiyun #define   ATV_TOP_CR_AMP_TH_DK                                              0x8
110*4882a593Smuzhiyun #define   ATV_TOP_CR_AMP_TH_I                                               0x8
111*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_D_MN                                         0x18
112*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_D_FM                                         0x0
113*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_D_L                                          0x20
114*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_D_LP                                         0x20
115*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_D_BG                                         0x18
116*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_D_DK                                         0x18
117*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_D_I                                          0x18
118*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_I_MN                                         0x80
119*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_I_FM                                         0x0
120*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_I_L                                          0x80
121*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_I_LP                                         0x80
122*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_I_BG                                         0x80
123*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_I_DK                                         0x80
124*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_I_I                                          0x80
125*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_P_MN                                         0x4
126*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_P_FM                                         0x0
127*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_P_L                                          0x4
128*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_P_LP                                         0x4
129*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_P_BG                                         0x4
130*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_P_DK                                         0x4
131*4882a593Smuzhiyun #define     ATV_TOP_CR_CONT_CR_P_I                                          0x4
132*4882a593Smuzhiyun #define   ATV_TOP_CR_OVM_TH_MN                                              0xA0
133*4882a593Smuzhiyun #define   ATV_TOP_CR_OVM_TH_FM                                              0x0
134*4882a593Smuzhiyun #define   ATV_TOP_CR_OVM_TH_L                                               0xA0
135*4882a593Smuzhiyun #define   ATV_TOP_CR_OVM_TH_LP                                              0xA0
136*4882a593Smuzhiyun #define   ATV_TOP_CR_OVM_TH_BG                                              0xA0
137*4882a593Smuzhiyun #define   ATV_TOP_CR_OVM_TH_DK                                              0xA0
138*4882a593Smuzhiyun #define   ATV_TOP_CR_OVM_TH_I                                               0xA0
139*4882a593Smuzhiyun #define     ATV_TOP_EQU0_EQU_C0_FM                                          0x0
140*4882a593Smuzhiyun #define     ATV_TOP_EQU0_EQU_C0_L                                           0x3
141*4882a593Smuzhiyun #define     ATV_TOP_EQU0_EQU_C0_LP                                          0x3
142*4882a593Smuzhiyun #define     ATV_TOP_EQU0_EQU_C0_BG                                          0x7
143*4882a593Smuzhiyun #define     ATV_TOP_EQU0_EQU_C0_DK                                          0x0
144*4882a593Smuzhiyun #define     ATV_TOP_EQU0_EQU_C0_I                                           0x3
145*4882a593Smuzhiyun #define     ATV_TOP_EQU1_EQU_C1_FM                                          0x0
146*4882a593Smuzhiyun #define     ATV_TOP_EQU1_EQU_C1_L                                           0x1F6
147*4882a593Smuzhiyun #define     ATV_TOP_EQU1_EQU_C1_LP                                          0x1F6
148*4882a593Smuzhiyun #define     ATV_TOP_EQU1_EQU_C1_BG                                          0x197
149*4882a593Smuzhiyun #define     ATV_TOP_EQU1_EQU_C1_DK                                          0x198
150*4882a593Smuzhiyun #define     ATV_TOP_EQU1_EQU_C1_I                                           0x1F6
151*4882a593Smuzhiyun #define     ATV_TOP_EQU2_EQU_C2_FM                                          0x0
152*4882a593Smuzhiyun #define     ATV_TOP_EQU2_EQU_C2_L                                           0x28
153*4882a593Smuzhiyun #define     ATV_TOP_EQU2_EQU_C2_LP                                          0x28
154*4882a593Smuzhiyun #define     ATV_TOP_EQU2_EQU_C2_BG                                          0xC5
155*4882a593Smuzhiyun #define     ATV_TOP_EQU2_EQU_C2_DK                                          0xB0
156*4882a593Smuzhiyun #define     ATV_TOP_EQU2_EQU_C2_I                                           0x28
157*4882a593Smuzhiyun #define     ATV_TOP_EQU3_EQU_C3_FM                                          0x0
158*4882a593Smuzhiyun #define     ATV_TOP_EQU3_EQU_C3_L                                           0x192
159*4882a593Smuzhiyun #define     ATV_TOP_EQU3_EQU_C3_LP                                          0x192
160*4882a593Smuzhiyun #define     ATV_TOP_EQU3_EQU_C3_BG                                          0x12E
161*4882a593Smuzhiyun #define     ATV_TOP_EQU3_EQU_C3_DK                                          0x18E
162*4882a593Smuzhiyun #define     ATV_TOP_EQU3_EQU_C3_I                                           0x192
163*4882a593Smuzhiyun #define     ATV_TOP_STD_MODE_MN                                             0x0
164*4882a593Smuzhiyun #define     ATV_TOP_STD_MODE_FM                                             0x1
165*4882a593Smuzhiyun #define     ATV_TOP_STD_MODE_L                                              0x0
166*4882a593Smuzhiyun #define     ATV_TOP_STD_MODE_LP                                             0x0
167*4882a593Smuzhiyun #define     ATV_TOP_STD_MODE_BG                                             0x0
168*4882a593Smuzhiyun #define     ATV_TOP_STD_MODE_DK                                             0x0
169*4882a593Smuzhiyun #define     ATV_TOP_STD_MODE_I                                              0x0
170*4882a593Smuzhiyun #define     ATV_TOP_STD_VID_POL_MN                                          0x0
171*4882a593Smuzhiyun #define     ATV_TOP_STD_VID_POL_FM                                          0x0
172*4882a593Smuzhiyun #define     ATV_TOP_STD_VID_POL_L                                           0x2
173*4882a593Smuzhiyun #define     ATV_TOP_STD_VID_POL_LP                                          0x2
174*4882a593Smuzhiyun #define     ATV_TOP_STD_VID_POL_BG                                          0x0
175*4882a593Smuzhiyun #define     ATV_TOP_STD_VID_POL_DK                                          0x0
176*4882a593Smuzhiyun #define     ATV_TOP_STD_VID_POL_I                                           0x0
177*4882a593Smuzhiyun #define   ATV_TOP_VID_AMP_MN                                                0x380
178*4882a593Smuzhiyun #define   ATV_TOP_VID_AMP_FM                                                0x0
179*4882a593Smuzhiyun #define   ATV_TOP_VID_AMP_L                                                 0xF50
180*4882a593Smuzhiyun #define   ATV_TOP_VID_AMP_LP                                                0xF50
181*4882a593Smuzhiyun #define   ATV_TOP_VID_AMP_BG                                                0x380
182*4882a593Smuzhiyun #define   ATV_TOP_VID_AMP_DK                                                0x394
183*4882a593Smuzhiyun #define   ATV_TOP_VID_AMP_I                                                 0x3D8
184*4882a593Smuzhiyun #define   IQM_CF_OUT_ENA_OFDM__M                                            0x4
185*4882a593Smuzhiyun #define     IQM_FS_ADJ_SEL_B_QAM                                            0x1
186*4882a593Smuzhiyun #define     IQM_FS_ADJ_SEL_B_OFF                                            0x0
187*4882a593Smuzhiyun #define     IQM_FS_ADJ_SEL_B_VSB                                            0x2
188*4882a593Smuzhiyun #define     IQM_RC_ADJ_SEL_B_OFF                                            0x0
189*4882a593Smuzhiyun #define     IQM_RC_ADJ_SEL_B_QAM                                            0x1
190*4882a593Smuzhiyun #define     IQM_RC_ADJ_SEL_B_VSB                                            0x2
191*4882a593Smuzhiyun /*** END DJCOMBO patches to DRXJ registermap *********************************/
192*4882a593Smuzhiyun 
193*4882a593Smuzhiyun #include "drx_driver_version.h"
194*4882a593Smuzhiyun 
195*4882a593Smuzhiyun /* #define DRX_DEBUG */
196*4882a593Smuzhiyun #ifdef DRX_DEBUG
197*4882a593Smuzhiyun #include <stdio.h>
198*4882a593Smuzhiyun #endif
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun /*-----------------------------------------------------------------------------
201*4882a593Smuzhiyun ENUMS
202*4882a593Smuzhiyun ----------------------------------------------------------------------------*/
203*4882a593Smuzhiyun 
204*4882a593Smuzhiyun /*-----------------------------------------------------------------------------
205*4882a593Smuzhiyun DEFINES
206*4882a593Smuzhiyun ----------------------------------------------------------------------------*/
207*4882a593Smuzhiyun #ifndef DRXJ_WAKE_UP_KEY
208*4882a593Smuzhiyun #define DRXJ_WAKE_UP_KEY (demod->my_i2c_dev_addr->i2c_addr)
209*4882a593Smuzhiyun #endif
210*4882a593Smuzhiyun 
211*4882a593Smuzhiyun /*
212*4882a593Smuzhiyun * \def DRXJ_DEF_I2C_ADDR
213*4882a593Smuzhiyun * \brief Default I2C address of a demodulator instance.
214*4882a593Smuzhiyun */
215*4882a593Smuzhiyun #define DRXJ_DEF_I2C_ADDR (0x52)
216*4882a593Smuzhiyun 
217*4882a593Smuzhiyun /*
218*4882a593Smuzhiyun * \def DRXJ_DEF_DEMOD_DEV_ID
219*4882a593Smuzhiyun * \brief Default device identifier of a demodultor instance.
220*4882a593Smuzhiyun */
221*4882a593Smuzhiyun #define DRXJ_DEF_DEMOD_DEV_ID      (1)
222*4882a593Smuzhiyun 
223*4882a593Smuzhiyun /*
224*4882a593Smuzhiyun * \def DRXJ_SCAN_TIMEOUT
225*4882a593Smuzhiyun * \brief Timeout value for waiting on demod lock during channel scan (millisec).
226*4882a593Smuzhiyun */
227*4882a593Smuzhiyun #define DRXJ_SCAN_TIMEOUT    1000
228*4882a593Smuzhiyun 
229*4882a593Smuzhiyun /*
230*4882a593Smuzhiyun * \def HI_I2C_DELAY
231*4882a593Smuzhiyun * \brief HI timing delay for I2C timing (in nano seconds)
232*4882a593Smuzhiyun *
233*4882a593Smuzhiyun *  Used to compute HI_CFG_DIV
234*4882a593Smuzhiyun */
235*4882a593Smuzhiyun #define HI_I2C_DELAY    42
236*4882a593Smuzhiyun 
237*4882a593Smuzhiyun /*
238*4882a593Smuzhiyun * \def HI_I2C_BRIDGE_DELAY
239*4882a593Smuzhiyun * \brief HI timing delay for I2C timing (in nano seconds)
240*4882a593Smuzhiyun *
241*4882a593Smuzhiyun *  Used to compute HI_CFG_BDL
242*4882a593Smuzhiyun */
243*4882a593Smuzhiyun #define HI_I2C_BRIDGE_DELAY   750
244*4882a593Smuzhiyun 
245*4882a593Smuzhiyun /*
246*4882a593Smuzhiyun * \brief Time Window for MER and SER Measurement in Units of Segment duration.
247*4882a593Smuzhiyun */
248*4882a593Smuzhiyun #define VSB_TOP_MEASUREMENT_PERIOD  64
249*4882a593Smuzhiyun #define SYMBOLS_PER_SEGMENT         832
250*4882a593Smuzhiyun 
251*4882a593Smuzhiyun /*
252*4882a593Smuzhiyun * \brief bit rate and segment rate constants used for SER and BER.
253*4882a593Smuzhiyun */
254*4882a593Smuzhiyun /* values taken from the QAM microcode */
255*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QAM_UNKNOWN 0
256*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QPSK        32768
257*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QAM8        24576
258*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QAM16       40960
259*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QAM32       20480
260*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QAM64       43008
261*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QAM128      20992
262*4882a593Smuzhiyun #define DRXJ_QAM_SL_SIG_POWER_QAM256      43520
263*4882a593Smuzhiyun /*
264*4882a593Smuzhiyun * \brief Min supported symbolrates.
265*4882a593Smuzhiyun */
266*4882a593Smuzhiyun #ifndef DRXJ_QAM_SYMBOLRATE_MIN
267*4882a593Smuzhiyun #define DRXJ_QAM_SYMBOLRATE_MIN          (520000)
268*4882a593Smuzhiyun #endif
269*4882a593Smuzhiyun 
270*4882a593Smuzhiyun /*
271*4882a593Smuzhiyun * \brief Max supported symbolrates.
272*4882a593Smuzhiyun */
273*4882a593Smuzhiyun #ifndef DRXJ_QAM_SYMBOLRATE_MAX
274*4882a593Smuzhiyun #define DRXJ_QAM_SYMBOLRATE_MAX         (7233000)
275*4882a593Smuzhiyun #endif
276*4882a593Smuzhiyun 
277*4882a593Smuzhiyun /*
278*4882a593Smuzhiyun * \def DRXJ_QAM_MAX_WAITTIME
279*4882a593Smuzhiyun * \brief Maximal wait time for QAM auto constellation in ms
280*4882a593Smuzhiyun */
281*4882a593Smuzhiyun #ifndef DRXJ_QAM_MAX_WAITTIME
282*4882a593Smuzhiyun #define DRXJ_QAM_MAX_WAITTIME 900
283*4882a593Smuzhiyun #endif
284*4882a593Smuzhiyun 
285*4882a593Smuzhiyun #ifndef DRXJ_QAM_FEC_LOCK_WAITTIME
286*4882a593Smuzhiyun #define DRXJ_QAM_FEC_LOCK_WAITTIME 150
287*4882a593Smuzhiyun #endif
288*4882a593Smuzhiyun 
289*4882a593Smuzhiyun #ifndef DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME
290*4882a593Smuzhiyun #define DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME 200
291*4882a593Smuzhiyun #endif
292*4882a593Smuzhiyun 
293*4882a593Smuzhiyun /*
294*4882a593Smuzhiyun * \def SCU status and results
295*4882a593Smuzhiyun * \brief SCU
296*4882a593Smuzhiyun */
297*4882a593Smuzhiyun #define DRX_SCU_READY               0
298*4882a593Smuzhiyun #define DRXJ_MAX_WAITTIME           100	/* ms */
299*4882a593Smuzhiyun #define FEC_RS_MEASUREMENT_PERIOD   12894	/* 1 sec */
300*4882a593Smuzhiyun #define FEC_RS_MEASUREMENT_PRESCALE 1	/* n sec */
301*4882a593Smuzhiyun 
302*4882a593Smuzhiyun /*
303*4882a593Smuzhiyun * \def DRX_AUD_MAX_DEVIATION
304*4882a593Smuzhiyun * \brief Needed for calculation of prescale feature in AUD
305*4882a593Smuzhiyun */
306*4882a593Smuzhiyun #ifndef DRXJ_AUD_MAX_FM_DEVIATION
307*4882a593Smuzhiyun #define DRXJ_AUD_MAX_FM_DEVIATION  100	/* kHz */
308*4882a593Smuzhiyun #endif
309*4882a593Smuzhiyun 
310*4882a593Smuzhiyun /*
311*4882a593Smuzhiyun * \brief Needed for calculation of NICAM prescale feature in AUD
312*4882a593Smuzhiyun */
313*4882a593Smuzhiyun #ifndef DRXJ_AUD_MAX_NICAM_PRESCALE
314*4882a593Smuzhiyun #define DRXJ_AUD_MAX_NICAM_PRESCALE  (9)	/* dB */
315*4882a593Smuzhiyun #endif
316*4882a593Smuzhiyun 
317*4882a593Smuzhiyun /*
318*4882a593Smuzhiyun * \brief Needed for calculation of NICAM prescale feature in AUD
319*4882a593Smuzhiyun */
320*4882a593Smuzhiyun #ifndef DRXJ_AUD_MAX_WAITTIME
321*4882a593Smuzhiyun #define DRXJ_AUD_MAX_WAITTIME  250	/* ms */
322*4882a593Smuzhiyun #endif
323*4882a593Smuzhiyun 
324*4882a593Smuzhiyun /* ATV config changed flags */
325*4882a593Smuzhiyun #define DRXJ_ATV_CHANGED_COEF          (0x00000001UL)
326*4882a593Smuzhiyun #define DRXJ_ATV_CHANGED_PEAK_FLT      (0x00000008UL)
327*4882a593Smuzhiyun #define DRXJ_ATV_CHANGED_NOISE_FLT     (0x00000010UL)
328*4882a593Smuzhiyun #define DRXJ_ATV_CHANGED_OUTPUT        (0x00000020UL)
329*4882a593Smuzhiyun #define DRXJ_ATV_CHANGED_SIF_ATT       (0x00000040UL)
330*4882a593Smuzhiyun 
331*4882a593Smuzhiyun /* UIO define */
332*4882a593Smuzhiyun #define DRX_UIO_MODE_FIRMWARE_SMA DRX_UIO_MODE_FIRMWARE0
333*4882a593Smuzhiyun #define DRX_UIO_MODE_FIRMWARE_SAW DRX_UIO_MODE_FIRMWARE1
334*4882a593Smuzhiyun 
335*4882a593Smuzhiyun /*
336*4882a593Smuzhiyun  * MICROCODE RELATED DEFINES
337*4882a593Smuzhiyun  */
338*4882a593Smuzhiyun 
339*4882a593Smuzhiyun /* Magic word for checking correct Endianness of microcode data */
340*4882a593Smuzhiyun #define DRX_UCODE_MAGIC_WORD         ((((u16)'H')<<8)+((u16)'L'))
341*4882a593Smuzhiyun 
342*4882a593Smuzhiyun /* CRC flag in ucode header, flags field. */
343*4882a593Smuzhiyun #define DRX_UCODE_CRC_FLAG           (0x0001)
344*4882a593Smuzhiyun 
345*4882a593Smuzhiyun /*
346*4882a593Smuzhiyun  * Maximum size of buffer used to verify the microcode.
347*4882a593Smuzhiyun  * Must be an even number
348*4882a593Smuzhiyun  */
349*4882a593Smuzhiyun #define DRX_UCODE_MAX_BUF_SIZE       (DRXDAP_MAX_RCHUNKSIZE)
350*4882a593Smuzhiyun 
351*4882a593Smuzhiyun #if DRX_UCODE_MAX_BUF_SIZE & 1
352*4882a593Smuzhiyun #error DRX_UCODE_MAX_BUF_SIZE must be an even number
353*4882a593Smuzhiyun #endif
354*4882a593Smuzhiyun 
355*4882a593Smuzhiyun /*
356*4882a593Smuzhiyun  * Power mode macros
357*4882a593Smuzhiyun  */
358*4882a593Smuzhiyun 
359*4882a593Smuzhiyun #define DRX_ISPOWERDOWNMODE(mode) ((mode == DRX_POWER_MODE_9) || \
360*4882a593Smuzhiyun 				       (mode == DRX_POWER_MODE_10) || \
361*4882a593Smuzhiyun 				       (mode == DRX_POWER_MODE_11) || \
362*4882a593Smuzhiyun 				       (mode == DRX_POWER_MODE_12) || \
363*4882a593Smuzhiyun 				       (mode == DRX_POWER_MODE_13) || \
364*4882a593Smuzhiyun 				       (mode == DRX_POWER_MODE_14) || \
365*4882a593Smuzhiyun 				       (mode == DRX_POWER_MODE_15) || \
366*4882a593Smuzhiyun 				       (mode == DRX_POWER_MODE_16) || \
367*4882a593Smuzhiyun 				       (mode == DRX_POWER_DOWN))
368*4882a593Smuzhiyun 
369*4882a593Smuzhiyun /* Pin safe mode macro */
370*4882a593Smuzhiyun #define DRXJ_PIN_SAFE_MODE 0x0000
371*4882a593Smuzhiyun /*============================================================================*/
372*4882a593Smuzhiyun /*=== GLOBAL VARIABLEs =======================================================*/
373*4882a593Smuzhiyun /*============================================================================*/
374*4882a593Smuzhiyun /*
375*4882a593Smuzhiyun */
376*4882a593Smuzhiyun 
377*4882a593Smuzhiyun /*
378*4882a593Smuzhiyun * \brief Temporary register definitions.
379*4882a593Smuzhiyun *        (register definitions that are not yet available in register master)
380*4882a593Smuzhiyun */
381*4882a593Smuzhiyun 
382*4882a593Smuzhiyun /*****************************************************************************/
383*4882a593Smuzhiyun /* Audio block 0x103 is write only. To avoid shadowing in driver accessing   */
384*4882a593Smuzhiyun /* RAM addresses directly. This must be READ ONLY to avoid problems.         */
385*4882a593Smuzhiyun /* Writing to the interface addresses are more than only writing the RAM     */
386*4882a593Smuzhiyun /* locations                                                                 */
387*4882a593Smuzhiyun /*****************************************************************************/
388*4882a593Smuzhiyun /*
389*4882a593Smuzhiyun * \brief RAM location of MODUS registers
390*4882a593Smuzhiyun */
391*4882a593Smuzhiyun #define AUD_DEM_RAM_MODUS_HI__A              0x10204A3
392*4882a593Smuzhiyun #define AUD_DEM_RAM_MODUS_HI__M              0xF000
393*4882a593Smuzhiyun 
394*4882a593Smuzhiyun #define AUD_DEM_RAM_MODUS_LO__A              0x10204A4
395*4882a593Smuzhiyun #define AUD_DEM_RAM_MODUS_LO__M              0x0FFF
396*4882a593Smuzhiyun 
397*4882a593Smuzhiyun /*
398*4882a593Smuzhiyun * \brief RAM location of I2S config registers
399*4882a593Smuzhiyun */
400*4882a593Smuzhiyun #define AUD_DEM_RAM_I2S_CONFIG1__A           0x10204B1
401*4882a593Smuzhiyun #define AUD_DEM_RAM_I2S_CONFIG2__A           0x10204B2
402*4882a593Smuzhiyun 
403*4882a593Smuzhiyun /*
404*4882a593Smuzhiyun * \brief RAM location of DCO config registers
405*4882a593Smuzhiyun */
406*4882a593Smuzhiyun #define AUD_DEM_RAM_DCO_B_HI__A              0x1020461
407*4882a593Smuzhiyun #define AUD_DEM_RAM_DCO_B_LO__A              0x1020462
408*4882a593Smuzhiyun #define AUD_DEM_RAM_DCO_A_HI__A              0x1020463
409*4882a593Smuzhiyun #define AUD_DEM_RAM_DCO_A_LO__A              0x1020464
410*4882a593Smuzhiyun 
411*4882a593Smuzhiyun /*
412*4882a593Smuzhiyun * \brief RAM location of Threshold registers
413*4882a593Smuzhiyun */
414*4882a593Smuzhiyun #define AUD_DEM_RAM_NICAM_THRSHLD__A         0x102045A
415*4882a593Smuzhiyun #define AUD_DEM_RAM_A2_THRSHLD__A            0x10204BB
416*4882a593Smuzhiyun #define AUD_DEM_RAM_BTSC_THRSHLD__A          0x10204A6
417*4882a593Smuzhiyun 
418*4882a593Smuzhiyun /*
419*4882a593Smuzhiyun * \brief RAM location of Carrier Threshold registers
420*4882a593Smuzhiyun */
421*4882a593Smuzhiyun #define AUD_DEM_RAM_CM_A_THRSHLD__A          0x10204AF
422*4882a593Smuzhiyun #define AUD_DEM_RAM_CM_B_THRSHLD__A          0x10204B0
423*4882a593Smuzhiyun 
424*4882a593Smuzhiyun /*
425*4882a593Smuzhiyun * \brief FM Matrix register fix
426*4882a593Smuzhiyun */
427*4882a593Smuzhiyun #ifdef AUD_DEM_WR_FM_MATRIX__A
428*4882a593Smuzhiyun #undef  AUD_DEM_WR_FM_MATRIX__A
429*4882a593Smuzhiyun #endif
430*4882a593Smuzhiyun #define AUD_DEM_WR_FM_MATRIX__A              0x105006F
431*4882a593Smuzhiyun 
432*4882a593Smuzhiyun /*============================================================================*/
433*4882a593Smuzhiyun /*
434*4882a593Smuzhiyun * \brief Defines required for audio
435*4882a593Smuzhiyun */
436*4882a593Smuzhiyun #define AUD_VOLUME_ZERO_DB                      115
437*4882a593Smuzhiyun #define AUD_VOLUME_DB_MIN                       -60
438*4882a593Smuzhiyun #define AUD_VOLUME_DB_MAX                       12
439*4882a593Smuzhiyun #define AUD_CARRIER_STRENGTH_QP_0DB             0x4000
440*4882a593Smuzhiyun #define AUD_CARRIER_STRENGTH_QP_0DB_LOG10T100   421
441*4882a593Smuzhiyun #define AUD_MAX_AVC_REF_LEVEL                   15
442*4882a593Smuzhiyun #define AUD_I2S_FREQUENCY_MAX                   48000UL
443*4882a593Smuzhiyun #define AUD_I2S_FREQUENCY_MIN                   12000UL
444*4882a593Smuzhiyun #define AUD_RDS_ARRAY_SIZE                      18
445*4882a593Smuzhiyun 
446*4882a593Smuzhiyun /*
447*4882a593Smuzhiyun * \brief Needed for calculation of prescale feature in AUD
448*4882a593Smuzhiyun */
449*4882a593Smuzhiyun #ifndef DRX_AUD_MAX_FM_DEVIATION
450*4882a593Smuzhiyun #define DRX_AUD_MAX_FM_DEVIATION  (100)	/* kHz */
451*4882a593Smuzhiyun #endif
452*4882a593Smuzhiyun 
453*4882a593Smuzhiyun /*
454*4882a593Smuzhiyun * \brief Needed for calculation of NICAM prescale feature in AUD
455*4882a593Smuzhiyun */
456*4882a593Smuzhiyun #ifndef DRX_AUD_MAX_NICAM_PRESCALE
457*4882a593Smuzhiyun #define DRX_AUD_MAX_NICAM_PRESCALE  (9)	/* dB */
458*4882a593Smuzhiyun #endif
459*4882a593Smuzhiyun 
460*4882a593Smuzhiyun /*============================================================================*/
461*4882a593Smuzhiyun /* Values for I2S Master/Slave pin configurations */
462*4882a593Smuzhiyun #define SIO_PDR_I2S_CL_CFG_MODE__MASTER      0x0004
463*4882a593Smuzhiyun #define SIO_PDR_I2S_CL_CFG_DRIVE__MASTER     0x0008
464*4882a593Smuzhiyun #define SIO_PDR_I2S_CL_CFG_MODE__SLAVE       0x0004
465*4882a593Smuzhiyun #define SIO_PDR_I2S_CL_CFG_DRIVE__SLAVE      0x0000
466*4882a593Smuzhiyun 
467*4882a593Smuzhiyun #define SIO_PDR_I2S_DA_CFG_MODE__MASTER      0x0003
468*4882a593Smuzhiyun #define SIO_PDR_I2S_DA_CFG_DRIVE__MASTER     0x0008
469*4882a593Smuzhiyun #define SIO_PDR_I2S_DA_CFG_MODE__SLAVE       0x0003
470*4882a593Smuzhiyun #define SIO_PDR_I2S_DA_CFG_DRIVE__SLAVE      0x0008
471*4882a593Smuzhiyun 
472*4882a593Smuzhiyun #define SIO_PDR_I2S_WS_CFG_MODE__MASTER      0x0004
473*4882a593Smuzhiyun #define SIO_PDR_I2S_WS_CFG_DRIVE__MASTER     0x0008
474*4882a593Smuzhiyun #define SIO_PDR_I2S_WS_CFG_MODE__SLAVE       0x0004
475*4882a593Smuzhiyun #define SIO_PDR_I2S_WS_CFG_DRIVE__SLAVE      0x0000
476*4882a593Smuzhiyun 
477*4882a593Smuzhiyun /*============================================================================*/
478*4882a593Smuzhiyun /*=== REGISTER ACCESS MACROS =================================================*/
479*4882a593Smuzhiyun /*============================================================================*/
480*4882a593Smuzhiyun 
481*4882a593Smuzhiyun /*
482*4882a593Smuzhiyun * This macro is used to create byte arrays for block writes.
483*4882a593Smuzhiyun * Block writes speed up I2C traffic between host and demod.
484*4882a593Smuzhiyun * The macro takes care of the required byte order in a 16 bits word.
485*4882a593Smuzhiyun * x -> lowbyte(x), highbyte(x)
486*4882a593Smuzhiyun */
487*4882a593Smuzhiyun #define DRXJ_16TO8(x) ((u8) (((u16)x) & 0xFF)), \
488*4882a593Smuzhiyun 		       ((u8)((((u16)x)>>8)&0xFF))
489*4882a593Smuzhiyun /*
490*4882a593Smuzhiyun * This macro is used to convert byte array to 16 bit register value for block read.
491*4882a593Smuzhiyun * Block read speed up I2C traffic between host and demod.
492*4882a593Smuzhiyun * The macro takes care of the required byte order in a 16 bits word.
493*4882a593Smuzhiyun */
494*4882a593Smuzhiyun #define DRXJ_8TO16(x) ((u16) (x[0] | (x[1] << 8)))
495*4882a593Smuzhiyun 
496*4882a593Smuzhiyun /*============================================================================*/
497*4882a593Smuzhiyun /*=== MISC DEFINES ===========================================================*/
498*4882a593Smuzhiyun /*============================================================================*/
499*4882a593Smuzhiyun 
500*4882a593Smuzhiyun /*============================================================================*/
501*4882a593Smuzhiyun /*=== HI COMMAND RELATED DEFINES =============================================*/
502*4882a593Smuzhiyun /*============================================================================*/
503*4882a593Smuzhiyun 
504*4882a593Smuzhiyun /*
505*4882a593Smuzhiyun * \brief General maximum number of retries for ucode command interfaces
506*4882a593Smuzhiyun */
507*4882a593Smuzhiyun #define DRXJ_MAX_RETRIES (100)
508*4882a593Smuzhiyun 
509*4882a593Smuzhiyun /*============================================================================*/
510*4882a593Smuzhiyun /*=== STANDARD RELATED MACROS ================================================*/
511*4882a593Smuzhiyun /*============================================================================*/
512*4882a593Smuzhiyun 
513*4882a593Smuzhiyun #define DRXJ_ISATVSTD(std) ((std == DRX_STANDARD_PAL_SECAM_BG) || \
514*4882a593Smuzhiyun 			       (std == DRX_STANDARD_PAL_SECAM_DK) || \
515*4882a593Smuzhiyun 			       (std == DRX_STANDARD_PAL_SECAM_I) || \
516*4882a593Smuzhiyun 			       (std == DRX_STANDARD_PAL_SECAM_L) || \
517*4882a593Smuzhiyun 			       (std == DRX_STANDARD_PAL_SECAM_LP) || \
518*4882a593Smuzhiyun 			       (std == DRX_STANDARD_NTSC) || \
519*4882a593Smuzhiyun 			       (std == DRX_STANDARD_FM))
520*4882a593Smuzhiyun 
521*4882a593Smuzhiyun #define DRXJ_ISQAMSTD(std) ((std == DRX_STANDARD_ITU_A) || \
522*4882a593Smuzhiyun 			       (std == DRX_STANDARD_ITU_B) || \
523*4882a593Smuzhiyun 			       (std == DRX_STANDARD_ITU_C) || \
524*4882a593Smuzhiyun 			       (std == DRX_STANDARD_ITU_D))
525*4882a593Smuzhiyun 
526*4882a593Smuzhiyun /*-----------------------------------------------------------------------------
527*4882a593Smuzhiyun GLOBAL VARIABLES
528*4882a593Smuzhiyun ----------------------------------------------------------------------------*/
529*4882a593Smuzhiyun /*
530*4882a593Smuzhiyun  * DRXJ DAP structures
531*4882a593Smuzhiyun  */
532*4882a593Smuzhiyun 
533*4882a593Smuzhiyun static int drxdap_fasi_read_block(struct i2c_device_addr *dev_addr,
534*4882a593Smuzhiyun 				      u32 addr,
535*4882a593Smuzhiyun 				      u16 datasize,
536*4882a593Smuzhiyun 				      u8 *data, u32 flags);
537*4882a593Smuzhiyun 
538*4882a593Smuzhiyun 
539*4882a593Smuzhiyun static int drxj_dap_read_modify_write_reg16(struct i2c_device_addr *dev_addr,
540*4882a593Smuzhiyun 						 u32 waddr,
541*4882a593Smuzhiyun 						 u32 raddr,
542*4882a593Smuzhiyun 						 u16 wdata, u16 *rdata);
543*4882a593Smuzhiyun 
544*4882a593Smuzhiyun static int drxj_dap_read_reg16(struct i2c_device_addr *dev_addr,
545*4882a593Smuzhiyun 				      u32 addr,
546*4882a593Smuzhiyun 				      u16 *data, u32 flags);
547*4882a593Smuzhiyun 
548*4882a593Smuzhiyun static int drxdap_fasi_read_reg32(struct i2c_device_addr *dev_addr,
549*4882a593Smuzhiyun 				      u32 addr,
550*4882a593Smuzhiyun 				      u32 *data, u32 flags);
551*4882a593Smuzhiyun 
552*4882a593Smuzhiyun static int drxdap_fasi_write_block(struct i2c_device_addr *dev_addr,
553*4882a593Smuzhiyun 				       u32 addr,
554*4882a593Smuzhiyun 				       u16 datasize,
555*4882a593Smuzhiyun 				       u8 *data, u32 flags);
556*4882a593Smuzhiyun 
557*4882a593Smuzhiyun static int drxj_dap_write_reg16(struct i2c_device_addr *dev_addr,
558*4882a593Smuzhiyun 				       u32 addr,
559*4882a593Smuzhiyun 				       u16 data, u32 flags);
560*4882a593Smuzhiyun 
561*4882a593Smuzhiyun static int drxdap_fasi_write_reg32(struct i2c_device_addr *dev_addr,
562*4882a593Smuzhiyun 				       u32 addr,
563*4882a593Smuzhiyun 				       u32 data, u32 flags);
564*4882a593Smuzhiyun 
565*4882a593Smuzhiyun static struct drxj_data drxj_data_g = {
566*4882a593Smuzhiyun 	false,			/* has_lna : true if LNA (aka PGA) present      */
567*4882a593Smuzhiyun 	false,			/* has_oob : true if OOB supported              */
568*4882a593Smuzhiyun 	false,			/* has_ntsc: true if NTSC supported             */
569*4882a593Smuzhiyun 	false,			/* has_btsc: true if BTSC supported             */
570*4882a593Smuzhiyun 	false,			/* has_smatx: true if SMA_TX pin is available   */
571*4882a593Smuzhiyun 	false,			/* has_smarx: true if SMA_RX pin is available   */
572*4882a593Smuzhiyun 	false,			/* has_gpio : true if GPIO pin is available     */
573*4882a593Smuzhiyun 	false,			/* has_irqn : true if IRQN pin is available     */
574*4882a593Smuzhiyun 	0,			/* mfx A1/A2/A... */
575*4882a593Smuzhiyun 
576*4882a593Smuzhiyun 	/* tuner settings */
577*4882a593Smuzhiyun 	false,			/* tuner mirrors RF signal    */
578*4882a593Smuzhiyun 	/* standard/channel settings */
579*4882a593Smuzhiyun 	DRX_STANDARD_UNKNOWN,	/* current standard           */
580*4882a593Smuzhiyun 	DRX_CONSTELLATION_AUTO,	/* constellation              */
581*4882a593Smuzhiyun 	0,			/* frequency in KHz           */
582*4882a593Smuzhiyun 	DRX_BANDWIDTH_UNKNOWN,	/* curr_bandwidth              */
583*4882a593Smuzhiyun 	DRX_MIRROR_NO,		/* mirror                     */
584*4882a593Smuzhiyun 
585*4882a593Smuzhiyun 	/* signal quality information: */
586*4882a593Smuzhiyun 	/* default values taken from the QAM Programming guide */
587*4882a593Smuzhiyun 	/*   fec_bits_desired should not be less than 4000000    */
588*4882a593Smuzhiyun 	4000000,		/* fec_bits_desired    */
589*4882a593Smuzhiyun 	5,			/* fec_vd_plen         */
590*4882a593Smuzhiyun 	4,			/* qam_vd_prescale     */
591*4882a593Smuzhiyun 	0xFFFF,			/* qamVDPeriod       */
592*4882a593Smuzhiyun 	204 * 8,		/* fec_rs_plen annex A */
593*4882a593Smuzhiyun 	1,			/* fec_rs_prescale     */
594*4882a593Smuzhiyun 	FEC_RS_MEASUREMENT_PERIOD,	/* fec_rs_period     */
595*4882a593Smuzhiyun 	true,			/* reset_pkt_err_acc    */
596*4882a593Smuzhiyun 	0,			/* pkt_err_acc_start    */
597*4882a593Smuzhiyun 
598*4882a593Smuzhiyun 	/* HI configuration */
599*4882a593Smuzhiyun 	0,			/* hi_cfg_timing_div    */
600*4882a593Smuzhiyun 	0,			/* hi_cfg_bridge_delay  */
601*4882a593Smuzhiyun 	0,			/* hi_cfg_wake_up_key    */
602*4882a593Smuzhiyun 	0,			/* hi_cfg_ctrl         */
603*4882a593Smuzhiyun 	0,			/* HICfgTimeout      */
604*4882a593Smuzhiyun 	/* UIO configuration */
605*4882a593Smuzhiyun 	DRX_UIO_MODE_DISABLE,	/* uio_sma_rx_mode      */
606*4882a593Smuzhiyun 	DRX_UIO_MODE_DISABLE,	/* uio_sma_tx_mode      */
607*4882a593Smuzhiyun 	DRX_UIO_MODE_DISABLE,	/* uioASELMode       */
608*4882a593Smuzhiyun 	DRX_UIO_MODE_DISABLE,	/* uio_irqn_mode       */
609*4882a593Smuzhiyun 	/* FS setting */
610*4882a593Smuzhiyun 	0UL,			/* iqm_fs_rate_ofs      */
611*4882a593Smuzhiyun 	false,			/* pos_image          */
612*4882a593Smuzhiyun 	/* RC setting */
613*4882a593Smuzhiyun 	0UL,			/* iqm_rc_rate_ofs      */
614*4882a593Smuzhiyun 	/* AUD information */
615*4882a593Smuzhiyun /*   false,                  * flagSetAUDdone    */
616*4882a593Smuzhiyun /*   false,                  * detectedRDS       */
617*4882a593Smuzhiyun /*   true,                   * flagASDRequest    */
618*4882a593Smuzhiyun /*   false,                  * flagHDevClear     */
619*4882a593Smuzhiyun /*   false,                  * flagHDevSet       */
620*4882a593Smuzhiyun /*   (u16) 0xFFF,          * rdsLastCount      */
621*4882a593Smuzhiyun 
622*4882a593Smuzhiyun 	/* ATV configuration */
623*4882a593Smuzhiyun 	0UL,			/* flags cfg changes */
624*4882a593Smuzhiyun 	/* shadow of ATV_TOP_EQU0__A */
625*4882a593Smuzhiyun 	{-5,
626*4882a593Smuzhiyun 	 ATV_TOP_EQU0_EQU_C0_FM,
627*4882a593Smuzhiyun 	 ATV_TOP_EQU0_EQU_C0_L,
628*4882a593Smuzhiyun 	 ATV_TOP_EQU0_EQU_C0_LP,
629*4882a593Smuzhiyun 	 ATV_TOP_EQU0_EQU_C0_BG,
630*4882a593Smuzhiyun 	 ATV_TOP_EQU0_EQU_C0_DK,
631*4882a593Smuzhiyun 	 ATV_TOP_EQU0_EQU_C0_I},
632*4882a593Smuzhiyun 	/* shadow of ATV_TOP_EQU1__A */
633*4882a593Smuzhiyun 	{-50,
634*4882a593Smuzhiyun 	 ATV_TOP_EQU1_EQU_C1_FM,
635*4882a593Smuzhiyun 	 ATV_TOP_EQU1_EQU_C1_L,
636*4882a593Smuzhiyun 	 ATV_TOP_EQU1_EQU_C1_LP,
637*4882a593Smuzhiyun 	 ATV_TOP_EQU1_EQU_C1_BG,
638*4882a593Smuzhiyun 	 ATV_TOP_EQU1_EQU_C1_DK,
639*4882a593Smuzhiyun 	 ATV_TOP_EQU1_EQU_C1_I},
640*4882a593Smuzhiyun 	/* shadow of ATV_TOP_EQU2__A */
641*4882a593Smuzhiyun 	{210,
642*4882a593Smuzhiyun 	 ATV_TOP_EQU2_EQU_C2_FM,
643*4882a593Smuzhiyun 	 ATV_TOP_EQU2_EQU_C2_L,
644*4882a593Smuzhiyun 	 ATV_TOP_EQU2_EQU_C2_LP,
645*4882a593Smuzhiyun 	 ATV_TOP_EQU2_EQU_C2_BG,
646*4882a593Smuzhiyun 	 ATV_TOP_EQU2_EQU_C2_DK,
647*4882a593Smuzhiyun 	 ATV_TOP_EQU2_EQU_C2_I},
648*4882a593Smuzhiyun 	/* shadow of ATV_TOP_EQU3__A */
649*4882a593Smuzhiyun 	{-160,
650*4882a593Smuzhiyun 	 ATV_TOP_EQU3_EQU_C3_FM,
651*4882a593Smuzhiyun 	 ATV_TOP_EQU3_EQU_C3_L,
652*4882a593Smuzhiyun 	 ATV_TOP_EQU3_EQU_C3_LP,
653*4882a593Smuzhiyun 	 ATV_TOP_EQU3_EQU_C3_BG,
654*4882a593Smuzhiyun 	 ATV_TOP_EQU3_EQU_C3_DK,
655*4882a593Smuzhiyun 	 ATV_TOP_EQU3_EQU_C3_I},
656*4882a593Smuzhiyun 	false,			/* flag: true=bypass             */
657*4882a593Smuzhiyun 	ATV_TOP_VID_PEAK__PRE,	/* shadow of ATV_TOP_VID_PEAK__A */
658*4882a593Smuzhiyun 	ATV_TOP_NOISE_TH__PRE,	/* shadow of ATV_TOP_NOISE_TH__A */
659*4882a593Smuzhiyun 	true,			/* flag CVBS output enable       */
660*4882a593Smuzhiyun 	false,			/* flag SIF output enable        */
661*4882a593Smuzhiyun 	DRXJ_SIF_ATTENUATION_0DB,	/* current SIF att setting       */
662*4882a593Smuzhiyun 	{			/* qam_rf_agc_cfg */
663*4882a593Smuzhiyun 	 DRX_STANDARD_ITU_B,	/* standard            */
664*4882a593Smuzhiyun 	 DRX_AGC_CTRL_AUTO,	/* ctrl_mode            */
665*4882a593Smuzhiyun 	 0,			/* output_level         */
666*4882a593Smuzhiyun 	 0,			/* min_output_level      */
667*4882a593Smuzhiyun 	 0xFFFF,		/* max_output_level      */
668*4882a593Smuzhiyun 	 0x0000,		/* speed               */
669*4882a593Smuzhiyun 	 0x0000,		/* top                 */
670*4882a593Smuzhiyun 	 0x0000			/* c.o.c.              */
671*4882a593Smuzhiyun 	 },
672*4882a593Smuzhiyun 	{			/* qam_if_agc_cfg */
673*4882a593Smuzhiyun 	 DRX_STANDARD_ITU_B,	/* standard            */
674*4882a593Smuzhiyun 	 DRX_AGC_CTRL_AUTO,	/* ctrl_mode            */
675*4882a593Smuzhiyun 	 0,			/* output_level         */
676*4882a593Smuzhiyun 	 0,			/* min_output_level      */
677*4882a593Smuzhiyun 	 0xFFFF,		/* max_output_level      */
678*4882a593Smuzhiyun 	 0x0000,		/* speed               */
679*4882a593Smuzhiyun 	 0x0000,		/* top    (don't care) */
680*4882a593Smuzhiyun 	 0x0000			/* c.o.c. (don't care) */
681*4882a593Smuzhiyun 	 },
682*4882a593Smuzhiyun 	{			/* vsb_rf_agc_cfg */
683*4882a593Smuzhiyun 	 DRX_STANDARD_8VSB,	/* standard       */
684*4882a593Smuzhiyun 	 DRX_AGC_CTRL_AUTO,	/* ctrl_mode       */
685*4882a593Smuzhiyun 	 0,			/* output_level    */
686*4882a593Smuzhiyun 	 0,			/* min_output_level */
687*4882a593Smuzhiyun 	 0xFFFF,		/* max_output_level */
688*4882a593Smuzhiyun 	 0x0000,		/* speed          */
689*4882a593Smuzhiyun 	 0x0000,		/* top    (don't care) */
690*4882a593Smuzhiyun 	 0x0000			/* c.o.c. (don't care) */
691*4882a593Smuzhiyun 	 },
692*4882a593Smuzhiyun 	{			/* vsb_if_agc_cfg */
693*4882a593Smuzhiyun 	 DRX_STANDARD_8VSB,	/* standard       */
694*4882a593Smuzhiyun 	 DRX_AGC_CTRL_AUTO,	/* ctrl_mode       */
695*4882a593Smuzhiyun 	 0,			/* output_level    */
696*4882a593Smuzhiyun 	 0,			/* min_output_level */
697*4882a593Smuzhiyun 	 0xFFFF,		/* max_output_level */
698*4882a593Smuzhiyun 	 0x0000,		/* speed          */
699*4882a593Smuzhiyun 	 0x0000,		/* top    (don't care) */
700*4882a593Smuzhiyun 	 0x0000			/* c.o.c. (don't care) */
701*4882a593Smuzhiyun 	 },
702*4882a593Smuzhiyun 	0,			/* qam_pga_cfg */
703*4882a593Smuzhiyun 	0,			/* vsb_pga_cfg */
704*4882a593Smuzhiyun 	{			/* qam_pre_saw_cfg */
705*4882a593Smuzhiyun 	 DRX_STANDARD_ITU_B,	/* standard  */
706*4882a593Smuzhiyun 	 0,			/* reference */
707*4882a593Smuzhiyun 	 false			/* use_pre_saw */
708*4882a593Smuzhiyun 	 },
709*4882a593Smuzhiyun 	{			/* vsb_pre_saw_cfg */
710*4882a593Smuzhiyun 	 DRX_STANDARD_8VSB,	/* standard  */
711*4882a593Smuzhiyun 	 0,			/* reference */
712*4882a593Smuzhiyun 	 false			/* use_pre_saw */
713*4882a593Smuzhiyun 	 },
714*4882a593Smuzhiyun 
715*4882a593Smuzhiyun 	/* Version information */
716*4882a593Smuzhiyun #ifndef _CH_
717*4882a593Smuzhiyun 	{
718*4882a593Smuzhiyun 	 "01234567890",		/* human readable version microcode             */
719*4882a593Smuzhiyun 	 "01234567890"		/* human readable version device specific code  */
720*4882a593Smuzhiyun 	 },
721*4882a593Smuzhiyun 	{
722*4882a593Smuzhiyun 	 {			/* struct drx_version for microcode                   */
723*4882a593Smuzhiyun 	  DRX_MODULE_UNKNOWN,
724*4882a593Smuzhiyun 	  (char *)(NULL),
725*4882a593Smuzhiyun 	  0,
726*4882a593Smuzhiyun 	  0,
727*4882a593Smuzhiyun 	  0,
728*4882a593Smuzhiyun 	  (char *)(NULL)
729*4882a593Smuzhiyun 	  },
730*4882a593Smuzhiyun 	 {			/* struct drx_version for device specific code */
731*4882a593Smuzhiyun 	  DRX_MODULE_UNKNOWN,
732*4882a593Smuzhiyun 	  (char *)(NULL),
733*4882a593Smuzhiyun 	  0,
734*4882a593Smuzhiyun 	  0,
735*4882a593Smuzhiyun 	  0,
736*4882a593Smuzhiyun 	  (char *)(NULL)
737*4882a593Smuzhiyun 	  }
738*4882a593Smuzhiyun 	 },
739*4882a593Smuzhiyun 	{
740*4882a593Smuzhiyun 	 {			/* struct drx_version_list for microcode */
741*4882a593Smuzhiyun 	  (struct drx_version *) (NULL),
742*4882a593Smuzhiyun 	  (struct drx_version_list *) (NULL)
743*4882a593Smuzhiyun 	  },
744*4882a593Smuzhiyun 	 {			/* struct drx_version_list for device specific code */
745*4882a593Smuzhiyun 	  (struct drx_version *) (NULL),
746*4882a593Smuzhiyun 	  (struct drx_version_list *) (NULL)
747*4882a593Smuzhiyun 	  }
748*4882a593Smuzhiyun 	 },
749*4882a593Smuzhiyun #endif
750*4882a593Smuzhiyun 	false,			/* smart_ant_inverted */
751*4882a593Smuzhiyun 	/* Tracking filter setting for OOB  */
752*4882a593Smuzhiyun 	{
753*4882a593Smuzhiyun 	 12000,
754*4882a593Smuzhiyun 	 9300,
755*4882a593Smuzhiyun 	 6600,
756*4882a593Smuzhiyun 	 5280,
757*4882a593Smuzhiyun 	 3700,
758*4882a593Smuzhiyun 	 3000,
759*4882a593Smuzhiyun 	 2000,
760*4882a593Smuzhiyun 	 0},
761*4882a593Smuzhiyun 	false,			/* oob_power_on           */
762*4882a593Smuzhiyun 	0,			/* mpeg_ts_static_bitrate  */
763*4882a593Smuzhiyun 	false,			/* disable_te_ihandling   */
764*4882a593Smuzhiyun 	false,			/* bit_reverse_mpeg_outout */
765*4882a593Smuzhiyun 	DRXJ_MPEGOUTPUT_CLOCK_RATE_AUTO,	/* mpeg_output_clock_rate */
766*4882a593Smuzhiyun 	DRXJ_MPEG_START_WIDTH_1CLKCYC,	/* mpeg_start_width */
767*4882a593Smuzhiyun 
768*4882a593Smuzhiyun 	/* Pre SAW & Agc configuration for ATV */
769*4882a593Smuzhiyun 	{
770*4882a593Smuzhiyun 	 DRX_STANDARD_NTSC,	/* standard     */
771*4882a593Smuzhiyun 	 7,			/* reference    */
772*4882a593Smuzhiyun 	 true			/* use_pre_saw    */
773*4882a593Smuzhiyun 	 },
774*4882a593Smuzhiyun 	{			/* ATV RF-AGC */
775*4882a593Smuzhiyun 	 DRX_STANDARD_NTSC,	/* standard              */
776*4882a593Smuzhiyun 	 DRX_AGC_CTRL_AUTO,	/* ctrl_mode              */
777*4882a593Smuzhiyun 	 0,			/* output_level           */
778*4882a593Smuzhiyun 	 0,			/* min_output_level (d.c.) */
779*4882a593Smuzhiyun 	 0,			/* max_output_level (d.c.) */
780*4882a593Smuzhiyun 	 3,			/* speed                 */
781*4882a593Smuzhiyun 	 9500,			/* top                   */
782*4882a593Smuzhiyun 	 4000			/* cut-off current       */
783*4882a593Smuzhiyun 	 },
784*4882a593Smuzhiyun 	{			/* ATV IF-AGC */
785*4882a593Smuzhiyun 	 DRX_STANDARD_NTSC,	/* standard              */
786*4882a593Smuzhiyun 	 DRX_AGC_CTRL_AUTO,	/* ctrl_mode              */
787*4882a593Smuzhiyun 	 0,			/* output_level           */
788*4882a593Smuzhiyun 	 0,			/* min_output_level (d.c.) */
789*4882a593Smuzhiyun 	 0,			/* max_output_level (d.c.) */
790*4882a593Smuzhiyun 	 3,			/* speed                 */
791*4882a593Smuzhiyun 	 2400,			/* top                   */
792*4882a593Smuzhiyun 	 0			/* c.o.c.         (d.c.) */
793*4882a593Smuzhiyun 	 },
794*4882a593Smuzhiyun 	140,			/* ATV PGA config */
795*4882a593Smuzhiyun 	0,			/* curr_symbol_rate */
796*4882a593Smuzhiyun 
797*4882a593Smuzhiyun 	false,			/* pdr_safe_mode     */
798*4882a593Smuzhiyun 	SIO_PDR_GPIO_CFG__PRE,	/* pdr_safe_restore_val_gpio  */
799*4882a593Smuzhiyun 	SIO_PDR_VSYNC_CFG__PRE,	/* pdr_safe_restore_val_v_sync */
800*4882a593Smuzhiyun 	SIO_PDR_SMA_RX_CFG__PRE,	/* pdr_safe_restore_val_sma_rx */
801*4882a593Smuzhiyun 	SIO_PDR_SMA_TX_CFG__PRE,	/* pdr_safe_restore_val_sma_tx */
802*4882a593Smuzhiyun 
803*4882a593Smuzhiyun 	4,			/* oob_pre_saw            */
804*4882a593Smuzhiyun 	DRXJ_OOB_LO_POW_MINUS10DB,	/* oob_lo_pow             */
805*4882a593Smuzhiyun 	{
806*4882a593Smuzhiyun 	 false			/* aud_data, only first member */
807*4882a593Smuzhiyun 	 },
808*4882a593Smuzhiyun };
809*4882a593Smuzhiyun 
810*4882a593Smuzhiyun /*
811*4882a593Smuzhiyun * \var drxj_default_addr_g
812*4882a593Smuzhiyun * \brief Default I2C address and device identifier.
813*4882a593Smuzhiyun */
814*4882a593Smuzhiyun static struct i2c_device_addr drxj_default_addr_g = {
815*4882a593Smuzhiyun 	DRXJ_DEF_I2C_ADDR,	/* i2c address */
816*4882a593Smuzhiyun 	DRXJ_DEF_DEMOD_DEV_ID	/* device id */
817*4882a593Smuzhiyun };
818*4882a593Smuzhiyun 
819*4882a593Smuzhiyun /*
820*4882a593Smuzhiyun * \var drxj_default_comm_attr_g
821*4882a593Smuzhiyun * \brief Default common attributes of a drxj demodulator instance.
822*4882a593Smuzhiyun */
823*4882a593Smuzhiyun static struct drx_common_attr drxj_default_comm_attr_g = {
824*4882a593Smuzhiyun 	NULL,			/* ucode file           */
825*4882a593Smuzhiyun 	true,			/* ucode verify switch  */
826*4882a593Smuzhiyun 	{0},			/* version record       */
827*4882a593Smuzhiyun 
828*4882a593Smuzhiyun 	44000,			/* IF in kHz in case no tuner instance is used  */
829*4882a593Smuzhiyun 	(151875 - 0),		/* system clock frequency in kHz                */
830*4882a593Smuzhiyun 	0,			/* oscillator frequency kHz                     */
831*4882a593Smuzhiyun 	0,			/* oscillator deviation in ppm, signed          */
832*4882a593Smuzhiyun 	false,			/* If true mirror frequency spectrum            */
833*4882a593Smuzhiyun 	{
834*4882a593Smuzhiyun 	 /* MPEG output configuration */
835*4882a593Smuzhiyun 	 true,			/* If true, enable MPEG output   */
836*4882a593Smuzhiyun 	 false,			/* If true, insert RS byte       */
837*4882a593Smuzhiyun 	 false,			/* If true, parallel out otherwise serial */
838*4882a593Smuzhiyun 	 false,			/* If true, invert DATA signals  */
839*4882a593Smuzhiyun 	 false,			/* If true, invert ERR signal    */
840*4882a593Smuzhiyun 	 false,			/* If true, invert STR signals   */
841*4882a593Smuzhiyun 	 false,			/* If true, invert VAL signals   */
842*4882a593Smuzhiyun 	 false,			/* If true, invert CLK signals   */
843*4882a593Smuzhiyun 	 true,			/* If true, static MPEG clockrate will
844*4882a593Smuzhiyun 				   be used, otherwise clockrate will
845*4882a593Smuzhiyun 				   adapt to the bitrate of the TS */
846*4882a593Smuzhiyun 	 19392658UL,		/* Maximum bitrate in b/s in case
847*4882a593Smuzhiyun 				   static clockrate is selected */
848*4882a593Smuzhiyun 	 DRX_MPEG_STR_WIDTH_1	/* MPEG Start width in clock cycles */
849*4882a593Smuzhiyun 	 },
850*4882a593Smuzhiyun 	/* Initilisations below can be omitted, they require no user input and
851*4882a593Smuzhiyun 	   are initially 0, NULL or false. The compiler will initialize them to these
852*4882a593Smuzhiyun 	   values when omitted.  */
853*4882a593Smuzhiyun 	false,			/* is_opened */
854*4882a593Smuzhiyun 
855*4882a593Smuzhiyun 	/* SCAN */
856*4882a593Smuzhiyun 	NULL,			/* no scan params yet               */
857*4882a593Smuzhiyun 	0,			/* current scan index               */
858*4882a593Smuzhiyun 	0,			/* next scan frequency              */
859*4882a593Smuzhiyun 	false,			/* scan ready flag                  */
860*4882a593Smuzhiyun 	0,			/* max channels to scan             */
861*4882a593Smuzhiyun 	0,			/* nr of channels scanned           */
862*4882a593Smuzhiyun 	NULL,			/* default scan function            */
863*4882a593Smuzhiyun 	NULL,			/* default context pointer          */
864*4882a593Smuzhiyun 	0,			/* millisec to wait for demod lock  */
865*4882a593Smuzhiyun 	DRXJ_DEMOD_LOCK,	/* desired lock               */
866*4882a593Smuzhiyun 	false,
867*4882a593Smuzhiyun 
868*4882a593Smuzhiyun 	/* Power management */
869*4882a593Smuzhiyun 	DRX_POWER_UP,
870*4882a593Smuzhiyun 
871*4882a593Smuzhiyun 	/* Tuner */
872*4882a593Smuzhiyun 	1,			/* nr of I2C port to which tuner is    */
873*4882a593Smuzhiyun 	0L,			/* minimum RF input frequency, in kHz  */
874*4882a593Smuzhiyun 	0L,			/* maximum RF input frequency, in kHz  */
875*4882a593Smuzhiyun 	false,			/* Rf Agc Polarity                     */
876*4882a593Smuzhiyun 	false,			/* If Agc Polarity                     */
877*4882a593Smuzhiyun 	false,			/* tuner slow mode                     */
878*4882a593Smuzhiyun 
879*4882a593Smuzhiyun 	{			/* current channel (all 0)             */
880*4882a593Smuzhiyun 	 0UL			/* channel.frequency */
881*4882a593Smuzhiyun 	 },
882*4882a593Smuzhiyun 	DRX_STANDARD_UNKNOWN,	/* current standard */
883*4882a593Smuzhiyun 	DRX_STANDARD_UNKNOWN,	/* previous standard */
884*4882a593Smuzhiyun 	DRX_STANDARD_UNKNOWN,	/* di_cache_standard   */
885*4882a593Smuzhiyun 	false,			/* use_bootloader */
886*4882a593Smuzhiyun 	0UL,			/* capabilities */
887*4882a593Smuzhiyun 	0			/* mfx */
888*4882a593Smuzhiyun };
889*4882a593Smuzhiyun 
890*4882a593Smuzhiyun /*
891*4882a593Smuzhiyun * \var drxj_default_demod_g
892*4882a593Smuzhiyun * \brief Default drxj demodulator instance.
893*4882a593Smuzhiyun */
894*4882a593Smuzhiyun static struct drx_demod_instance drxj_default_demod_g = {
895*4882a593Smuzhiyun 	&drxj_default_addr_g,	/* i2c address & device id */
896*4882a593Smuzhiyun 	&drxj_default_comm_attr_g,	/* demod common attributes */
897*4882a593Smuzhiyun 	&drxj_data_g		/* demod device specific attributes */
898*4882a593Smuzhiyun };
899*4882a593Smuzhiyun 
900*4882a593Smuzhiyun /*
901*4882a593Smuzhiyun * \brief Default audio data structure for DRK demodulator instance.
902*4882a593Smuzhiyun *
903*4882a593Smuzhiyun * This structure is DRXK specific.
904*4882a593Smuzhiyun *
905*4882a593Smuzhiyun */
906*4882a593Smuzhiyun static struct drx_aud_data drxj_default_aud_data_g = {
907*4882a593Smuzhiyun 	false,			/* audio_is_active */
908*4882a593Smuzhiyun 	DRX_AUD_STANDARD_AUTO,	/* audio_standard  */
909*4882a593Smuzhiyun 
910*4882a593Smuzhiyun 	/* i2sdata */
911*4882a593Smuzhiyun 	{
912*4882a593Smuzhiyun 	 false,			/* output_enable   */
913*4882a593Smuzhiyun 	 48000,			/* frequency      */
914*4882a593Smuzhiyun 	 DRX_I2S_MODE_MASTER,	/* mode           */
915*4882a593Smuzhiyun 	 DRX_I2S_WORDLENGTH_32,	/* word_length     */
916*4882a593Smuzhiyun 	 DRX_I2S_POLARITY_RIGHT,	/* polarity       */
917*4882a593Smuzhiyun 	 DRX_I2S_FORMAT_WS_WITH_DATA	/* format         */
918*4882a593Smuzhiyun 	 },
919*4882a593Smuzhiyun 	/* volume            */
920*4882a593Smuzhiyun 	{
921*4882a593Smuzhiyun 	 true,			/* mute;          */
922*4882a593Smuzhiyun 	 0,			/* volume         */
923*4882a593Smuzhiyun 	 DRX_AUD_AVC_OFF,	/* avc_mode        */
924*4882a593Smuzhiyun 	 0,			/* avc_ref_level    */
925*4882a593Smuzhiyun 	 DRX_AUD_AVC_MAX_GAIN_12DB,	/* avc_max_gain     */
926*4882a593Smuzhiyun 	 DRX_AUD_AVC_MAX_ATTEN_24DB,	/* avc_max_atten    */
927*4882a593Smuzhiyun 	 0,			/* strength_left   */
928*4882a593Smuzhiyun 	 0			/* strength_right  */
929*4882a593Smuzhiyun 	 },
930*4882a593Smuzhiyun 	DRX_AUD_AUTO_SOUND_SELECT_ON_CHANGE_ON,	/* auto_sound */
931*4882a593Smuzhiyun 	/*  ass_thresholds */
932*4882a593Smuzhiyun 	{
933*4882a593Smuzhiyun 	 440,			/* A2    */
934*4882a593Smuzhiyun 	 12,			/* BTSC  */
935*4882a593Smuzhiyun 	 700,			/* NICAM */
936*4882a593Smuzhiyun 	 },
937*4882a593Smuzhiyun 	/* carrier */
938*4882a593Smuzhiyun 	{
939*4882a593Smuzhiyun 	 /* a */
940*4882a593Smuzhiyun 	 {
941*4882a593Smuzhiyun 	  42,			/* thres */
942*4882a593Smuzhiyun 	  DRX_NO_CARRIER_NOISE,	/* opt   */
943*4882a593Smuzhiyun 	  0,			/* shift */
944*4882a593Smuzhiyun 	  0			/* dco   */
945*4882a593Smuzhiyun 	  },
946*4882a593Smuzhiyun 	 /* b */
947*4882a593Smuzhiyun 	 {
948*4882a593Smuzhiyun 	  42,			/* thres */
949*4882a593Smuzhiyun 	  DRX_NO_CARRIER_MUTE,	/* opt   */
950*4882a593Smuzhiyun 	  0,			/* shift */
951*4882a593Smuzhiyun 	  0			/* dco   */
952*4882a593Smuzhiyun 	  },
953*4882a593Smuzhiyun 
954*4882a593Smuzhiyun 	 },
955*4882a593Smuzhiyun 	/* mixer */
956*4882a593Smuzhiyun 	{
957*4882a593Smuzhiyun 	 DRX_AUD_SRC_STEREO_OR_A,	/* source_i2s */
958*4882a593Smuzhiyun 	 DRX_AUD_I2S_MATRIX_STEREO,	/* matrix_i2s */
959*4882a593Smuzhiyun 	 DRX_AUD_FM_MATRIX_SOUND_A	/* matrix_fm  */
960*4882a593Smuzhiyun 	 },
961*4882a593Smuzhiyun 	DRX_AUD_DEVIATION_NORMAL,	/* deviation */
962*4882a593Smuzhiyun 	DRX_AUD_AVSYNC_OFF,	/* av_sync */
963*4882a593Smuzhiyun 
964*4882a593Smuzhiyun 	/* prescale */
965*4882a593Smuzhiyun 	{
966*4882a593Smuzhiyun 	 DRX_AUD_MAX_FM_DEVIATION,	/* fm_deviation */
967*4882a593Smuzhiyun 	 DRX_AUD_MAX_NICAM_PRESCALE	/* nicam_gain */
968*4882a593Smuzhiyun 	 },
969*4882a593Smuzhiyun 	DRX_AUD_FM_DEEMPH_75US,	/* deemph */
970*4882a593Smuzhiyun 	DRX_BTSC_STEREO,	/* btsc_detect */
971*4882a593Smuzhiyun 	0,			/* rds_data_counter */
972*4882a593Smuzhiyun 	false			/* rds_data_present */
973*4882a593Smuzhiyun };
974*4882a593Smuzhiyun 
975*4882a593Smuzhiyun /*-----------------------------------------------------------------------------
976*4882a593Smuzhiyun STRUCTURES
977*4882a593Smuzhiyun ----------------------------------------------------------------------------*/
978*4882a593Smuzhiyun struct drxjeq_stat {
979*4882a593Smuzhiyun 	u16 eq_mse;
980*4882a593Smuzhiyun 	u8 eq_mode;
981*4882a593Smuzhiyun 	u8 eq_ctrl;
982*4882a593Smuzhiyun 	u8 eq_stat;
983*4882a593Smuzhiyun };
984*4882a593Smuzhiyun 
985*4882a593Smuzhiyun /* HI command */
986*4882a593Smuzhiyun struct drxj_hi_cmd {
987*4882a593Smuzhiyun 	u16 cmd;
988*4882a593Smuzhiyun 	u16 param1;
989*4882a593Smuzhiyun 	u16 param2;
990*4882a593Smuzhiyun 	u16 param3;
991*4882a593Smuzhiyun 	u16 param4;
992*4882a593Smuzhiyun 	u16 param5;
993*4882a593Smuzhiyun 	u16 param6;
994*4882a593Smuzhiyun };
995*4882a593Smuzhiyun 
996*4882a593Smuzhiyun /*============================================================================*/
997*4882a593Smuzhiyun /*=== MICROCODE RELATED STRUCTURES ===========================================*/
998*4882a593Smuzhiyun /*============================================================================*/
999*4882a593Smuzhiyun 
1000*4882a593Smuzhiyun /*
1001*4882a593Smuzhiyun  * struct drxu_code_block_hdr - Structure of the microcode block headers
1002*4882a593Smuzhiyun  *
1003*4882a593Smuzhiyun  * @addr:	Destination address of the data in this block
1004*4882a593Smuzhiyun  * @size:	Size of the block data following this header counted in
1005*4882a593Smuzhiyun  *		16 bits words
1006*4882a593Smuzhiyun  * @CRC:	CRC value of the data block, only valid if CRC flag is
1007*4882a593Smuzhiyun  *		set.
1008*4882a593Smuzhiyun  */
1009*4882a593Smuzhiyun struct drxu_code_block_hdr {
1010*4882a593Smuzhiyun 	u32 addr;
1011*4882a593Smuzhiyun 	u16 size;
1012*4882a593Smuzhiyun 	u16 flags;
1013*4882a593Smuzhiyun 	u16 CRC;
1014*4882a593Smuzhiyun };
1015*4882a593Smuzhiyun 
1016*4882a593Smuzhiyun /*-----------------------------------------------------------------------------
1017*4882a593Smuzhiyun FUNCTIONS
1018*4882a593Smuzhiyun ----------------------------------------------------------------------------*/
1019*4882a593Smuzhiyun /* Some prototypes */
1020*4882a593Smuzhiyun static int
1021*4882a593Smuzhiyun hi_command(struct i2c_device_addr *dev_addr,
1022*4882a593Smuzhiyun 	   const struct drxj_hi_cmd *cmd, u16 *result);
1023*4882a593Smuzhiyun 
1024*4882a593Smuzhiyun static int
1025*4882a593Smuzhiyun ctrl_lock_status(struct drx_demod_instance *demod, enum drx_lock_status *lock_stat);
1026*4882a593Smuzhiyun 
1027*4882a593Smuzhiyun static int
1028*4882a593Smuzhiyun ctrl_power_mode(struct drx_demod_instance *demod, enum drx_power_mode *mode);
1029*4882a593Smuzhiyun 
1030*4882a593Smuzhiyun static int power_down_aud(struct drx_demod_instance *demod);
1031*4882a593Smuzhiyun 
1032*4882a593Smuzhiyun static int
1033*4882a593Smuzhiyun ctrl_set_cfg_pre_saw(struct drx_demod_instance *demod, struct drxj_cfg_pre_saw *pre_saw);
1034*4882a593Smuzhiyun 
1035*4882a593Smuzhiyun static int
1036*4882a593Smuzhiyun ctrl_set_cfg_afe_gain(struct drx_demod_instance *demod, struct drxj_cfg_afe_gain *afe_gain);
1037*4882a593Smuzhiyun 
1038*4882a593Smuzhiyun /*============================================================================*/
1039*4882a593Smuzhiyun /*============================================================================*/
1040*4882a593Smuzhiyun /*==                          HELPER FUNCTIONS                              ==*/
1041*4882a593Smuzhiyun /*============================================================================*/
1042*4882a593Smuzhiyun /*============================================================================*/
1043*4882a593Smuzhiyun 
1044*4882a593Smuzhiyun 
1045*4882a593Smuzhiyun /*============================================================================*/
1046*4882a593Smuzhiyun 
1047*4882a593Smuzhiyun /*
1048*4882a593Smuzhiyun * \fn u32 frac28(u32 N, u32 D)
1049*4882a593Smuzhiyun * \brief Compute: (1<<28)*N/D
1050*4882a593Smuzhiyun * \param N 32 bits
1051*4882a593Smuzhiyun * \param D 32 bits
1052*4882a593Smuzhiyun * \return (1<<28)*N/D
1053*4882a593Smuzhiyun * This function is used to avoid floating-point calculations as they may
1054*4882a593Smuzhiyun * not be present on the target platform.
1055*4882a593Smuzhiyun 
1056*4882a593Smuzhiyun * frac28 performs an unsigned 28/28 bits division to 32-bit fixed point
1057*4882a593Smuzhiyun * fraction used for setting the Frequency Shifter registers.
1058*4882a593Smuzhiyun * N and D can hold numbers up to width: 28-bits.
1059*4882a593Smuzhiyun * The 4 bits integer part and the 28 bits fractional part are calculated.
1060*4882a593Smuzhiyun 
1061*4882a593Smuzhiyun * Usage condition: ((1<<28)*n)/d < ((1<<32)-1) => (n/d) < 15.999
1062*4882a593Smuzhiyun 
1063*4882a593Smuzhiyun * N: 0...(1<<28)-1 = 268435454
1064*4882a593Smuzhiyun * D: 0...(1<<28)-1
1065*4882a593Smuzhiyun * Q: 0...(1<<32)-1
1066*4882a593Smuzhiyun */
frac28(u32 N,u32 D)1067*4882a593Smuzhiyun static u32 frac28(u32 N, u32 D)
1068*4882a593Smuzhiyun {
1069*4882a593Smuzhiyun 	int i = 0;
1070*4882a593Smuzhiyun 	u32 Q1 = 0;
1071*4882a593Smuzhiyun 	u32 R0 = 0;
1072*4882a593Smuzhiyun 
1073*4882a593Smuzhiyun 	R0 = (N % D) << 4;	/* 32-28 == 4 shifts possible at max */
1074*4882a593Smuzhiyun 	Q1 = N / D;		/* integer part, only the 4 least significant bits
1075*4882a593Smuzhiyun 				   will be visible in the result */
1076*4882a593Smuzhiyun 
1077*4882a593Smuzhiyun 	/* division using radix 16, 7 nibbles in the result */
1078*4882a593Smuzhiyun 	for (i = 0; i < 7; i++) {
1079*4882a593Smuzhiyun 		Q1 = (Q1 << 4) | R0 / D;
1080*4882a593Smuzhiyun 		R0 = (R0 % D) << 4;
1081*4882a593Smuzhiyun 	}
1082*4882a593Smuzhiyun 	/* rounding */
1083*4882a593Smuzhiyun 	if ((R0 >> 3) >= D)
1084*4882a593Smuzhiyun 		Q1++;
1085*4882a593Smuzhiyun 
1086*4882a593Smuzhiyun 	return Q1;
1087*4882a593Smuzhiyun }
1088*4882a593Smuzhiyun 
1089*4882a593Smuzhiyun /*
1090*4882a593Smuzhiyun * \fn u32 log1_times100( u32 x)
1091*4882a593Smuzhiyun * \brief Compute: 100*log10(x)
1092*4882a593Smuzhiyun * \param x 32 bits
1093*4882a593Smuzhiyun * \return 100*log10(x)
1094*4882a593Smuzhiyun *
1095*4882a593Smuzhiyun * 100*log10(x)
1096*4882a593Smuzhiyun * = 100*(log2(x)/log2(10)))
1097*4882a593Smuzhiyun * = (100*(2^15)*log2(x))/((2^15)*log2(10))
1098*4882a593Smuzhiyun * = ((200*(2^15)*log2(x))/((2^15)*log2(10)))/2
1099*4882a593Smuzhiyun * = ((200*(2^15)*(log2(x/y)+log2(y)))/((2^15)*log2(10)))/2
1100*4882a593Smuzhiyun * = ((200*(2^15)*log2(x/y))+(200*(2^15)*log2(y)))/((2^15)*log2(10)))/2
1101*4882a593Smuzhiyun *
1102*4882a593Smuzhiyun * where y = 2^k and 1<= (x/y) < 2
1103*4882a593Smuzhiyun */
1104*4882a593Smuzhiyun 
log1_times100(u32 x)1105*4882a593Smuzhiyun static u32 log1_times100(u32 x)
1106*4882a593Smuzhiyun {
1107*4882a593Smuzhiyun 	static const u8 scale = 15;
1108*4882a593Smuzhiyun 	static const u8 index_width = 5;
1109*4882a593Smuzhiyun 	/*
1110*4882a593Smuzhiyun 	   log2lut[n] = (1<<scale) * 200 * log2( 1.0 + ( (1.0/(1<<INDEXWIDTH)) * n ))
1111*4882a593Smuzhiyun 	   0 <= n < ((1<<INDEXWIDTH)+1)
1112*4882a593Smuzhiyun 	 */
1113*4882a593Smuzhiyun 
1114*4882a593Smuzhiyun 	static const u32 log2lut[] = {
1115*4882a593Smuzhiyun 		0,		/* 0.000000 */
1116*4882a593Smuzhiyun 		290941,		/* 290941.300628 */
1117*4882a593Smuzhiyun 		573196,		/* 573196.476418 */
1118*4882a593Smuzhiyun 		847269,		/* 847269.179851 */
1119*4882a593Smuzhiyun 		1113620,	/* 1113620.489452 */
1120*4882a593Smuzhiyun 		1372674,	/* 1372673.576986 */
1121*4882a593Smuzhiyun 		1624818,	/* 1624817.752104 */
1122*4882a593Smuzhiyun 		1870412,	/* 1870411.981536 */
1123*4882a593Smuzhiyun 		2109788,	/* 2109787.962654 */
1124*4882a593Smuzhiyun 		2343253,	/* 2343252.817465 */
1125*4882a593Smuzhiyun 		2571091,	/* 2571091.461923 */
1126*4882a593Smuzhiyun 		2793569,	/* 2793568.696416 */
1127*4882a593Smuzhiyun 		3010931,	/* 3010931.055901 */
1128*4882a593Smuzhiyun 		3223408,	/* 3223408.452106 */
1129*4882a593Smuzhiyun 		3431216,	/* 3431215.635215 */
1130*4882a593Smuzhiyun 		3634553,	/* 3634553.498355 */
1131*4882a593Smuzhiyun 		3833610,	/* 3833610.244726 */
1132*4882a593Smuzhiyun 		4028562,	/* 4028562.434393 */
1133*4882a593Smuzhiyun 		4219576,	/* 4219575.925308 */
1134*4882a593Smuzhiyun 		4406807,	/* 4406806.721144 */
1135*4882a593Smuzhiyun 		4590402,	/* 4590401.736809 */
1136*4882a593Smuzhiyun 		4770499,	/* 4770499.491025 */
1137*4882a593Smuzhiyun 		4947231,	/* 4947230.734179 */
1138*4882a593Smuzhiyun 		5120719,	/* 5120719.018555 */
1139*4882a593Smuzhiyun 		5291081,	/* 5291081.217197 */
1140*4882a593Smuzhiyun 		5458428,	/* 5458427.996830 */
1141*4882a593Smuzhiyun 		5622864,	/* 5622864.249668 */
1142*4882a593Smuzhiyun 		5784489,	/* 5784489.488298 */
1143*4882a593Smuzhiyun 		5943398,	/* 5943398.207380 */
1144*4882a593Smuzhiyun 		6099680,	/* 6099680.215452 */
1145*4882a593Smuzhiyun 		6253421,	/* 6253420.939751 */
1146*4882a593Smuzhiyun 		6404702,	/* 6404701.706649 */
1147*4882a593Smuzhiyun 		6553600,	/* 6553600.000000 */
1148*4882a593Smuzhiyun 	};
1149*4882a593Smuzhiyun 
1150*4882a593Smuzhiyun 	u8 i = 0;
1151*4882a593Smuzhiyun 	u32 y = 0;
1152*4882a593Smuzhiyun 	u32 d = 0;
1153*4882a593Smuzhiyun 	u32 k = 0;
1154*4882a593Smuzhiyun 	u32 r = 0;
1155*4882a593Smuzhiyun 
1156*4882a593Smuzhiyun 	if (x == 0)
1157*4882a593Smuzhiyun 		return 0;
1158*4882a593Smuzhiyun 
1159*4882a593Smuzhiyun 	/* Scale x (normalize) */
1160*4882a593Smuzhiyun 	/* computing y in log(x/y) = log(x) - log(y) */
1161*4882a593Smuzhiyun 	if ((x & (((u32) (-1)) << (scale + 1))) == 0) {
1162*4882a593Smuzhiyun 		for (k = scale; k > 0; k--) {
1163*4882a593Smuzhiyun 			if (x & (((u32) 1) << scale))
1164*4882a593Smuzhiyun 				break;
1165*4882a593Smuzhiyun 			x <<= 1;
1166*4882a593Smuzhiyun 		}
1167*4882a593Smuzhiyun 	} else {
1168*4882a593Smuzhiyun 		for (k = scale; k < 31; k++) {
1169*4882a593Smuzhiyun 			if ((x & (((u32) (-1)) << (scale + 1))) == 0)
1170*4882a593Smuzhiyun 				break;
1171*4882a593Smuzhiyun 			x >>= 1;
1172*4882a593Smuzhiyun 		}
1173*4882a593Smuzhiyun 	}
1174*4882a593Smuzhiyun 	/*
1175*4882a593Smuzhiyun 	   Now x has binary point between bit[scale] and bit[scale-1]
1176*4882a593Smuzhiyun 	   and 1.0 <= x < 2.0 */
1177*4882a593Smuzhiyun 
1178*4882a593Smuzhiyun 	/* correction for division: log(x) = log(x/y)+log(y) */
1179*4882a593Smuzhiyun 	y = k * ((((u32) 1) << scale) * 200);
1180*4882a593Smuzhiyun 
1181*4882a593Smuzhiyun 	/* remove integer part */
1182*4882a593Smuzhiyun 	x &= ((((u32) 1) << scale) - 1);
1183*4882a593Smuzhiyun 	/* get index */
1184*4882a593Smuzhiyun 	i = (u8) (x >> (scale - index_width));
1185*4882a593Smuzhiyun 	/* compute delta (x-a) */
1186*4882a593Smuzhiyun 	d = x & ((((u32) 1) << (scale - index_width)) - 1);
1187*4882a593Smuzhiyun 	/* compute log, multiplication ( d* (.. )) must be within range ! */
1188*4882a593Smuzhiyun 	y += log2lut[i] +
1189*4882a593Smuzhiyun 	    ((d * (log2lut[i + 1] - log2lut[i])) >> (scale - index_width));
1190*4882a593Smuzhiyun 	/* Conver to log10() */
1191*4882a593Smuzhiyun 	y /= 108853;		/* (log2(10) << scale) */
1192*4882a593Smuzhiyun 	r = (y >> 1);
1193*4882a593Smuzhiyun 	/* rounding */
1194*4882a593Smuzhiyun 	if (y & ((u32)1))
1195*4882a593Smuzhiyun 		r++;
1196*4882a593Smuzhiyun 
1197*4882a593Smuzhiyun 	return r;
1198*4882a593Smuzhiyun 
1199*4882a593Smuzhiyun }
1200*4882a593Smuzhiyun 
1201*4882a593Smuzhiyun /*
1202*4882a593Smuzhiyun * \fn u32 frac_times1e6( u16 N, u32 D)
1203*4882a593Smuzhiyun * \brief Compute: (N/D) * 1000000.
1204*4882a593Smuzhiyun * \param N nominator 16-bits.
1205*4882a593Smuzhiyun * \param D denominator 32-bits.
1206*4882a593Smuzhiyun * \return u32
1207*4882a593Smuzhiyun * \retval ((N/D) * 1000000), 32 bits
1208*4882a593Smuzhiyun *
1209*4882a593Smuzhiyun * No check on D=0!
1210*4882a593Smuzhiyun */
frac_times1e6(u32 N,u32 D)1211*4882a593Smuzhiyun static u32 frac_times1e6(u32 N, u32 D)
1212*4882a593Smuzhiyun {
1213*4882a593Smuzhiyun 	u32 remainder = 0;
1214*4882a593Smuzhiyun 	u32 frac = 0;
1215*4882a593Smuzhiyun 
1216*4882a593Smuzhiyun 	/*
1217*4882a593Smuzhiyun 	   frac = (N * 1000000) / D
1218*4882a593Smuzhiyun 	   To let it fit in a 32 bits computation:
1219*4882a593Smuzhiyun 	   frac = (N * (1000000 >> 4)) / (D >> 4)
1220*4882a593Smuzhiyun 	   This would result in a problem in case D < 16 (div by 0).
1221*4882a593Smuzhiyun 	   So we do it more elaborate as shown below.
1222*4882a593Smuzhiyun 	 */
1223*4882a593Smuzhiyun 	frac = (((u32) N) * (1000000 >> 4)) / D;
1224*4882a593Smuzhiyun 	frac <<= 4;
1225*4882a593Smuzhiyun 	remainder = (((u32) N) * (1000000 >> 4)) % D;
1226*4882a593Smuzhiyun 	remainder <<= 4;
1227*4882a593Smuzhiyun 	frac += remainder / D;
1228*4882a593Smuzhiyun 	remainder = remainder % D;
1229*4882a593Smuzhiyun 	if ((remainder * 2) > D)
1230*4882a593Smuzhiyun 		frac++;
1231*4882a593Smuzhiyun 
1232*4882a593Smuzhiyun 	return frac;
1233*4882a593Smuzhiyun }
1234*4882a593Smuzhiyun 
1235*4882a593Smuzhiyun /*============================================================================*/
1236*4882a593Smuzhiyun 
1237*4882a593Smuzhiyun 
1238*4882a593Smuzhiyun /*
1239*4882a593Smuzhiyun * \brief Values for NICAM prescaler gain. Computed from dB to integer
1240*4882a593Smuzhiyun *        and rounded. For calc used formula: 16*10^(prescaleGain[dB]/20).
1241*4882a593Smuzhiyun *
1242*4882a593Smuzhiyun */
1243*4882a593Smuzhiyun #if 0
1244*4882a593Smuzhiyun /* Currently, unused as we lack support for analog TV */
1245*4882a593Smuzhiyun static const u16 nicam_presc_table_val[43] = {
1246*4882a593Smuzhiyun 	1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 4, 4,
1247*4882a593Smuzhiyun 	5, 5, 6, 6, 7, 8, 9, 10, 11, 13, 14, 16,
1248*4882a593Smuzhiyun 	18, 20, 23, 25, 28, 32, 36, 40, 45,
1249*4882a593Smuzhiyun 	51, 57, 64, 71, 80, 90, 101, 113, 127
1250*4882a593Smuzhiyun };
1251*4882a593Smuzhiyun #endif
1252*4882a593Smuzhiyun 
1253*4882a593Smuzhiyun /*============================================================================*/
1254*4882a593Smuzhiyun /*==                        END HELPER FUNCTIONS                            ==*/
1255*4882a593Smuzhiyun /*============================================================================*/
1256*4882a593Smuzhiyun 
1257*4882a593Smuzhiyun /*============================================================================*/
1258*4882a593Smuzhiyun /*============================================================================*/
1259*4882a593Smuzhiyun /*==                      DRXJ DAP FUNCTIONS                                ==*/
1260*4882a593Smuzhiyun /*============================================================================*/
1261*4882a593Smuzhiyun /*============================================================================*/
1262*4882a593Smuzhiyun 
1263*4882a593Smuzhiyun /*
1264*4882a593Smuzhiyun    This layer takes care of some device specific register access protocols:
1265*4882a593Smuzhiyun    -conversion to short address format
1266*4882a593Smuzhiyun    -access to audio block
1267*4882a593Smuzhiyun    This layer is placed between the drx_dap_fasi and the rest of the drxj
1268*4882a593Smuzhiyun    specific implementation. This layer can use address map knowledge whereas
1269*4882a593Smuzhiyun    dap_fasi may not use memory map knowledge.
1270*4882a593Smuzhiyun 
1271*4882a593Smuzhiyun    * For audio currently only 16 bits read and write register access is
1272*4882a593Smuzhiyun      supported. More is not needed. RMW and 32 or 8 bit access on audio
1273*4882a593Smuzhiyun      registers will have undefined behaviour. Flags (RMW, CRC reset, broadcast
1274*4882a593Smuzhiyun      single/multi master) will be ignored.
1275*4882a593Smuzhiyun 
1276*4882a593Smuzhiyun    TODO: check ignoring single/multimaster is ok for AUD access ?
1277*4882a593Smuzhiyun */
1278*4882a593Smuzhiyun 
1279*4882a593Smuzhiyun #define DRXJ_ISAUDWRITE(addr) (((((addr)>>16)&1) == 1) ? true : false)
1280*4882a593Smuzhiyun #define DRXJ_DAP_AUDTRIF_TIMEOUT 80	/* millisec */
1281*4882a593Smuzhiyun /*============================================================================*/
1282*4882a593Smuzhiyun 
1283*4882a593Smuzhiyun /*
1284*4882a593Smuzhiyun * \fn bool is_handled_by_aud_tr_if( u32 addr )
1285*4882a593Smuzhiyun * \brief Check if this address is handled by the audio token ring interface.
1286*4882a593Smuzhiyun * \param addr
1287*4882a593Smuzhiyun * \return bool
1288*4882a593Smuzhiyun * \retval true  Yes, handled by audio token ring interface
1289*4882a593Smuzhiyun * \retval false No, not handled by audio token ring interface
1290*4882a593Smuzhiyun *
1291*4882a593Smuzhiyun */
1292*4882a593Smuzhiyun static
is_handled_by_aud_tr_if(u32 addr)1293*4882a593Smuzhiyun bool is_handled_by_aud_tr_if(u32 addr)
1294*4882a593Smuzhiyun {
1295*4882a593Smuzhiyun 	bool retval = false;
1296*4882a593Smuzhiyun 
1297*4882a593Smuzhiyun 	if ((DRXDAP_FASI_ADDR2BLOCK(addr) == 4) &&
1298*4882a593Smuzhiyun 	    (DRXDAP_FASI_ADDR2BANK(addr) > 1) &&
1299*4882a593Smuzhiyun 	    (DRXDAP_FASI_ADDR2BANK(addr) < 6)) {
1300*4882a593Smuzhiyun 		retval = true;
1301*4882a593Smuzhiyun 	}
1302*4882a593Smuzhiyun 
1303*4882a593Smuzhiyun 	return retval;
1304*4882a593Smuzhiyun }
1305*4882a593Smuzhiyun 
1306*4882a593Smuzhiyun /*============================================================================*/
1307*4882a593Smuzhiyun 
drxbsp_i2c_write_read(struct i2c_device_addr * w_dev_addr,u16 w_count,u8 * wData,struct i2c_device_addr * r_dev_addr,u16 r_count,u8 * r_data)1308*4882a593Smuzhiyun int drxbsp_i2c_write_read(struct i2c_device_addr *w_dev_addr,
1309*4882a593Smuzhiyun 				 u16 w_count,
1310*4882a593Smuzhiyun 				 u8 *wData,
1311*4882a593Smuzhiyun 				 struct i2c_device_addr *r_dev_addr,
1312*4882a593Smuzhiyun 				 u16 r_count, u8 *r_data)
1313*4882a593Smuzhiyun {
1314*4882a593Smuzhiyun 	struct drx39xxj_state *state;
1315*4882a593Smuzhiyun 	struct i2c_msg msg[2];
1316*4882a593Smuzhiyun 	unsigned int num_msgs;
1317*4882a593Smuzhiyun 
1318*4882a593Smuzhiyun 	if (w_dev_addr == NULL) {
1319*4882a593Smuzhiyun 		/* Read only */
1320*4882a593Smuzhiyun 		state = r_dev_addr->user_data;
1321*4882a593Smuzhiyun 		msg[0].addr = r_dev_addr->i2c_addr >> 1;
1322*4882a593Smuzhiyun 		msg[0].flags = I2C_M_RD;
1323*4882a593Smuzhiyun 		msg[0].buf = r_data;
1324*4882a593Smuzhiyun 		msg[0].len = r_count;
1325*4882a593Smuzhiyun 		num_msgs = 1;
1326*4882a593Smuzhiyun 	} else if (r_dev_addr == NULL) {
1327*4882a593Smuzhiyun 		/* Write only */
1328*4882a593Smuzhiyun 		state = w_dev_addr->user_data;
1329*4882a593Smuzhiyun 		msg[0].addr = w_dev_addr->i2c_addr >> 1;
1330*4882a593Smuzhiyun 		msg[0].flags = 0;
1331*4882a593Smuzhiyun 		msg[0].buf = wData;
1332*4882a593Smuzhiyun 		msg[0].len = w_count;
1333*4882a593Smuzhiyun 		num_msgs = 1;
1334*4882a593Smuzhiyun 	} else {
1335*4882a593Smuzhiyun 		/* Both write and read */
1336*4882a593Smuzhiyun 		state = w_dev_addr->user_data;
1337*4882a593Smuzhiyun 		msg[0].addr = w_dev_addr->i2c_addr >> 1;
1338*4882a593Smuzhiyun 		msg[0].flags = 0;
1339*4882a593Smuzhiyun 		msg[0].buf = wData;
1340*4882a593Smuzhiyun 		msg[0].len = w_count;
1341*4882a593Smuzhiyun 		msg[1].addr = r_dev_addr->i2c_addr >> 1;
1342*4882a593Smuzhiyun 		msg[1].flags = I2C_M_RD;
1343*4882a593Smuzhiyun 		msg[1].buf = r_data;
1344*4882a593Smuzhiyun 		msg[1].len = r_count;
1345*4882a593Smuzhiyun 		num_msgs = 2;
1346*4882a593Smuzhiyun 	}
1347*4882a593Smuzhiyun 
1348*4882a593Smuzhiyun 	if (state->i2c == NULL) {
1349*4882a593Smuzhiyun 		pr_err("i2c was zero, aborting\n");
1350*4882a593Smuzhiyun 		return 0;
1351*4882a593Smuzhiyun 	}
1352*4882a593Smuzhiyun 	if (i2c_transfer(state->i2c, msg, num_msgs) != num_msgs) {
1353*4882a593Smuzhiyun 		pr_warn("drx3933: I2C write/read failed\n");
1354*4882a593Smuzhiyun 		return -EREMOTEIO;
1355*4882a593Smuzhiyun 	}
1356*4882a593Smuzhiyun 
1357*4882a593Smuzhiyun #ifdef DJH_DEBUG
1358*4882a593Smuzhiyun 	if (w_dev_addr == NULL || r_dev_addr == NULL)
1359*4882a593Smuzhiyun 		return 0;
1360*4882a593Smuzhiyun 
1361*4882a593Smuzhiyun 	state = w_dev_addr->user_data;
1362*4882a593Smuzhiyun 
1363*4882a593Smuzhiyun 	if (state->i2c == NULL)
1364*4882a593Smuzhiyun 		return 0;
1365*4882a593Smuzhiyun 
1366*4882a593Smuzhiyun 	msg[0].addr = w_dev_addr->i2c_addr;
1367*4882a593Smuzhiyun 	msg[0].flags = 0;
1368*4882a593Smuzhiyun 	msg[0].buf = wData;
1369*4882a593Smuzhiyun 	msg[0].len = w_count;
1370*4882a593Smuzhiyun 	msg[1].addr = r_dev_addr->i2c_addr;
1371*4882a593Smuzhiyun 	msg[1].flags = I2C_M_RD;
1372*4882a593Smuzhiyun 	msg[1].buf = r_data;
1373*4882a593Smuzhiyun 	msg[1].len = r_count;
1374*4882a593Smuzhiyun 	num_msgs = 2;
1375*4882a593Smuzhiyun 
1376*4882a593Smuzhiyun 	pr_debug("drx3933 i2c operation addr=%x i2c=%p, wc=%x rc=%x\n",
1377*4882a593Smuzhiyun 	       w_dev_addr->i2c_addr, state->i2c, w_count, r_count);
1378*4882a593Smuzhiyun 
1379*4882a593Smuzhiyun 	if (i2c_transfer(state->i2c, msg, 2) != 2) {
1380*4882a593Smuzhiyun 		pr_warn("drx3933: I2C write/read failed\n");
1381*4882a593Smuzhiyun 		return -EREMOTEIO;
1382*4882a593Smuzhiyun 	}
1383*4882a593Smuzhiyun #endif
1384*4882a593Smuzhiyun 	return 0;
1385*4882a593Smuzhiyun }
1386*4882a593Smuzhiyun 
1387*4882a593Smuzhiyun /*============================================================================*/
1388*4882a593Smuzhiyun 
1389*4882a593Smuzhiyun /*****************************
1390*4882a593Smuzhiyun *
1391*4882a593Smuzhiyun * int drxdap_fasi_read_block (
1392*4882a593Smuzhiyun *      struct i2c_device_addr *dev_addr,      -- address of I2C device
1393*4882a593Smuzhiyun *      u32 addr,         -- address of chip register/memory
1394*4882a593Smuzhiyun *      u16            datasize,     -- number of bytes to read
1395*4882a593Smuzhiyun *      u8 *data,         -- data to receive
1396*4882a593Smuzhiyun *      u32 flags)        -- special device flags
1397*4882a593Smuzhiyun *
1398*4882a593Smuzhiyun * Read block data from chip address. Because the chip is word oriented,
1399*4882a593Smuzhiyun * the number of bytes to read must be even.
1400*4882a593Smuzhiyun *
1401*4882a593Smuzhiyun * Make sure that the buffer to receive the data is large enough.
1402*4882a593Smuzhiyun *
1403*4882a593Smuzhiyun * Although this function expects an even number of bytes, it is still byte
1404*4882a593Smuzhiyun * oriented, and the data read back is NOT translated to the endianness of
1405*4882a593Smuzhiyun * the target platform.
1406*4882a593Smuzhiyun *
1407*4882a593Smuzhiyun * Output:
1408*4882a593Smuzhiyun * - 0     if reading was successful
1409*4882a593Smuzhiyun *                  in that case: data read is in *data.
1410*4882a593Smuzhiyun * - -EIO  if anything went wrong
1411*4882a593Smuzhiyun *
1412*4882a593Smuzhiyun ******************************/
1413*4882a593Smuzhiyun 
drxdap_fasi_read_block(struct i2c_device_addr * dev_addr,u32 addr,u16 datasize,u8 * data,u32 flags)1414*4882a593Smuzhiyun static int drxdap_fasi_read_block(struct i2c_device_addr *dev_addr,
1415*4882a593Smuzhiyun 					 u32 addr,
1416*4882a593Smuzhiyun 					 u16 datasize,
1417*4882a593Smuzhiyun 					 u8 *data, u32 flags)
1418*4882a593Smuzhiyun {
1419*4882a593Smuzhiyun 	u8 buf[4];
1420*4882a593Smuzhiyun 	u16 bufx;
1421*4882a593Smuzhiyun 	int rc;
1422*4882a593Smuzhiyun 	u16 overhead_size = 0;
1423*4882a593Smuzhiyun 
1424*4882a593Smuzhiyun 	/* Check parameters ******************************************************* */
1425*4882a593Smuzhiyun 	if (dev_addr == NULL)
1426*4882a593Smuzhiyun 		return -EINVAL;
1427*4882a593Smuzhiyun 
1428*4882a593Smuzhiyun 	overhead_size = (IS_I2C_10BIT(dev_addr->i2c_addr) ? 2 : 1) +
1429*4882a593Smuzhiyun 	    (DRXDAP_FASI_LONG_FORMAT(addr) ? 4 : 2);
1430*4882a593Smuzhiyun 
1431*4882a593Smuzhiyun 	if ((DRXDAP_FASI_OFFSET_TOO_LARGE(addr)) ||
1432*4882a593Smuzhiyun 	    ((!(DRXDAPFASI_LONG_ADDR_ALLOWED)) &&
1433*4882a593Smuzhiyun 	     DRXDAP_FASI_LONG_FORMAT(addr)) ||
1434*4882a593Smuzhiyun 	    (overhead_size > (DRXDAP_MAX_WCHUNKSIZE)) ||
1435*4882a593Smuzhiyun 	    ((datasize != 0) && (data == NULL)) || ((datasize & 1) == 1)) {
1436*4882a593Smuzhiyun 		return -EINVAL;
1437*4882a593Smuzhiyun 	}
1438*4882a593Smuzhiyun 
1439*4882a593Smuzhiyun 	/* ReadModifyWrite & mode flag bits are not allowed */
1440*4882a593Smuzhiyun 	flags &= (~DRXDAP_FASI_RMW & ~DRXDAP_FASI_MODEFLAGS);
1441*4882a593Smuzhiyun #if DRXDAP_SINGLE_MASTER
1442*4882a593Smuzhiyun 	flags |= DRXDAP_FASI_SINGLE_MASTER;
1443*4882a593Smuzhiyun #endif
1444*4882a593Smuzhiyun 
1445*4882a593Smuzhiyun 	/* Read block from I2C **************************************************** */
1446*4882a593Smuzhiyun 	do {
1447*4882a593Smuzhiyun 		u16 todo = (datasize < DRXDAP_MAX_RCHUNKSIZE ?
1448*4882a593Smuzhiyun 			      datasize : DRXDAP_MAX_RCHUNKSIZE);
1449*4882a593Smuzhiyun 
1450*4882a593Smuzhiyun 		bufx = 0;
1451*4882a593Smuzhiyun 
1452*4882a593Smuzhiyun 		addr &= ~DRXDAP_FASI_FLAGS;
1453*4882a593Smuzhiyun 		addr |= flags;
1454*4882a593Smuzhiyun 
1455*4882a593Smuzhiyun #if ((DRXDAPFASI_LONG_ADDR_ALLOWED == 1) && (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1))
1456*4882a593Smuzhiyun 		/* short format address preferred but long format otherwise */
1457*4882a593Smuzhiyun 		if (DRXDAP_FASI_LONG_FORMAT(addr)) {
1458*4882a593Smuzhiyun #endif
1459*4882a593Smuzhiyun #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 1)
1460*4882a593Smuzhiyun 			buf[bufx++] = (u8) (((addr << 1) & 0xFF) | 0x01);
1461*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr >> 16) & 0xFF);
1462*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr >> 24) & 0xFF);
1463*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr >> 7) & 0xFF);
1464*4882a593Smuzhiyun #endif
1465*4882a593Smuzhiyun #if ((DRXDAPFASI_LONG_ADDR_ALLOWED == 1) && (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1))
1466*4882a593Smuzhiyun 		} else {
1467*4882a593Smuzhiyun #endif
1468*4882a593Smuzhiyun #if (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1)
1469*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr << 1) & 0xFF);
1470*4882a593Smuzhiyun 			buf[bufx++] =
1471*4882a593Smuzhiyun 			    (u8) (((addr >> 16) & 0x0F) |
1472*4882a593Smuzhiyun 				    ((addr >> 18) & 0xF0));
1473*4882a593Smuzhiyun #endif
1474*4882a593Smuzhiyun #if ((DRXDAPFASI_LONG_ADDR_ALLOWED == 1) && (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1))
1475*4882a593Smuzhiyun 		}
1476*4882a593Smuzhiyun #endif
1477*4882a593Smuzhiyun 
1478*4882a593Smuzhiyun #if DRXDAP_SINGLE_MASTER
1479*4882a593Smuzhiyun 		/*
1480*4882a593Smuzhiyun 		 * In single master mode, split the read and write actions.
1481*4882a593Smuzhiyun 		 * No special action is needed for write chunks here.
1482*4882a593Smuzhiyun 		 */
1483*4882a593Smuzhiyun 		rc = drxbsp_i2c_write_read(dev_addr, bufx, buf,
1484*4882a593Smuzhiyun 					   NULL, 0, NULL);
1485*4882a593Smuzhiyun 		if (rc == 0)
1486*4882a593Smuzhiyun 			rc = drxbsp_i2c_write_read(NULL, 0, NULL, dev_addr, todo, data);
1487*4882a593Smuzhiyun #else
1488*4882a593Smuzhiyun 		/* In multi master mode, do everything in one RW action */
1489*4882a593Smuzhiyun 		rc = drxbsp_i2c_write_read(dev_addr, bufx, buf, dev_addr, todo,
1490*4882a593Smuzhiyun 					  data);
1491*4882a593Smuzhiyun #endif
1492*4882a593Smuzhiyun 		data += todo;
1493*4882a593Smuzhiyun 		addr += (todo >> 1);
1494*4882a593Smuzhiyun 		datasize -= todo;
1495*4882a593Smuzhiyun 	} while (datasize && rc == 0);
1496*4882a593Smuzhiyun 
1497*4882a593Smuzhiyun 	return rc;
1498*4882a593Smuzhiyun }
1499*4882a593Smuzhiyun 
1500*4882a593Smuzhiyun 
1501*4882a593Smuzhiyun /*****************************
1502*4882a593Smuzhiyun *
1503*4882a593Smuzhiyun * int drxdap_fasi_read_reg16 (
1504*4882a593Smuzhiyun *     struct i2c_device_addr *dev_addr, -- address of I2C device
1505*4882a593Smuzhiyun *     u32 addr,    -- address of chip register/memory
1506*4882a593Smuzhiyun *     u16 *data,    -- data to receive
1507*4882a593Smuzhiyun *     u32 flags)   -- special device flags
1508*4882a593Smuzhiyun *
1509*4882a593Smuzhiyun * Read one 16-bit register or memory location. The data received back is
1510*4882a593Smuzhiyun * converted back to the target platform's endianness.
1511*4882a593Smuzhiyun *
1512*4882a593Smuzhiyun * Output:
1513*4882a593Smuzhiyun * - 0     if reading was successful
1514*4882a593Smuzhiyun *                  in that case: read data is at *data
1515*4882a593Smuzhiyun * - -EIO  if anything went wrong
1516*4882a593Smuzhiyun *
1517*4882a593Smuzhiyun ******************************/
1518*4882a593Smuzhiyun 
drxdap_fasi_read_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 * data,u32 flags)1519*4882a593Smuzhiyun static int drxdap_fasi_read_reg16(struct i2c_device_addr *dev_addr,
1520*4882a593Smuzhiyun 					 u32 addr,
1521*4882a593Smuzhiyun 					 u16 *data, u32 flags)
1522*4882a593Smuzhiyun {
1523*4882a593Smuzhiyun 	u8 buf[sizeof(*data)];
1524*4882a593Smuzhiyun 	int rc;
1525*4882a593Smuzhiyun 
1526*4882a593Smuzhiyun 	if (!data)
1527*4882a593Smuzhiyun 		return -EINVAL;
1528*4882a593Smuzhiyun 
1529*4882a593Smuzhiyun 	rc = drxdap_fasi_read_block(dev_addr, addr, sizeof(*data), buf, flags);
1530*4882a593Smuzhiyun 	*data = buf[0] + (((u16) buf[1]) << 8);
1531*4882a593Smuzhiyun 	return rc;
1532*4882a593Smuzhiyun }
1533*4882a593Smuzhiyun 
1534*4882a593Smuzhiyun /*****************************
1535*4882a593Smuzhiyun *
1536*4882a593Smuzhiyun * int drxdap_fasi_read_reg32 (
1537*4882a593Smuzhiyun *     struct i2c_device_addr *dev_addr, -- address of I2C device
1538*4882a593Smuzhiyun *     u32 addr,    -- address of chip register/memory
1539*4882a593Smuzhiyun *     u32 *data,    -- data to receive
1540*4882a593Smuzhiyun *     u32 flags)   -- special device flags
1541*4882a593Smuzhiyun *
1542*4882a593Smuzhiyun * Read one 32-bit register or memory location. The data received back is
1543*4882a593Smuzhiyun * converted back to the target platform's endianness.
1544*4882a593Smuzhiyun *
1545*4882a593Smuzhiyun * Output:
1546*4882a593Smuzhiyun * - 0     if reading was successful
1547*4882a593Smuzhiyun *                  in that case: read data is at *data
1548*4882a593Smuzhiyun * - -EIO  if anything went wrong
1549*4882a593Smuzhiyun *
1550*4882a593Smuzhiyun ******************************/
1551*4882a593Smuzhiyun 
drxdap_fasi_read_reg32(struct i2c_device_addr * dev_addr,u32 addr,u32 * data,u32 flags)1552*4882a593Smuzhiyun static int drxdap_fasi_read_reg32(struct i2c_device_addr *dev_addr,
1553*4882a593Smuzhiyun 					 u32 addr,
1554*4882a593Smuzhiyun 					 u32 *data, u32 flags)
1555*4882a593Smuzhiyun {
1556*4882a593Smuzhiyun 	u8 buf[sizeof(*data)];
1557*4882a593Smuzhiyun 	int rc;
1558*4882a593Smuzhiyun 
1559*4882a593Smuzhiyun 	if (!data)
1560*4882a593Smuzhiyun 		return -EINVAL;
1561*4882a593Smuzhiyun 
1562*4882a593Smuzhiyun 	rc = drxdap_fasi_read_block(dev_addr, addr, sizeof(*data), buf, flags);
1563*4882a593Smuzhiyun 	*data = (((u32) buf[0]) << 0) +
1564*4882a593Smuzhiyun 	    (((u32) buf[1]) << 8) +
1565*4882a593Smuzhiyun 	    (((u32) buf[2]) << 16) + (((u32) buf[3]) << 24);
1566*4882a593Smuzhiyun 	return rc;
1567*4882a593Smuzhiyun }
1568*4882a593Smuzhiyun 
1569*4882a593Smuzhiyun /*****************************
1570*4882a593Smuzhiyun *
1571*4882a593Smuzhiyun * int drxdap_fasi_write_block (
1572*4882a593Smuzhiyun *      struct i2c_device_addr *dev_addr,    -- address of I2C device
1573*4882a593Smuzhiyun *      u32 addr,       -- address of chip register/memory
1574*4882a593Smuzhiyun *      u16            datasize,   -- number of bytes to read
1575*4882a593Smuzhiyun *      u8 *data,       -- data to receive
1576*4882a593Smuzhiyun *      u32 flags)      -- special device flags
1577*4882a593Smuzhiyun *
1578*4882a593Smuzhiyun * Write block data to chip address. Because the chip is word oriented,
1579*4882a593Smuzhiyun * the number of bytes to write must be even.
1580*4882a593Smuzhiyun *
1581*4882a593Smuzhiyun * Although this function expects an even number of bytes, it is still byte
1582*4882a593Smuzhiyun * oriented, and the data being written is NOT translated from the endianness of
1583*4882a593Smuzhiyun * the target platform.
1584*4882a593Smuzhiyun *
1585*4882a593Smuzhiyun * Output:
1586*4882a593Smuzhiyun * - 0     if writing was successful
1587*4882a593Smuzhiyun * - -EIO  if anything went wrong
1588*4882a593Smuzhiyun *
1589*4882a593Smuzhiyun ******************************/
1590*4882a593Smuzhiyun 
drxdap_fasi_write_block(struct i2c_device_addr * dev_addr,u32 addr,u16 datasize,u8 * data,u32 flags)1591*4882a593Smuzhiyun static int drxdap_fasi_write_block(struct i2c_device_addr *dev_addr,
1592*4882a593Smuzhiyun 					  u32 addr,
1593*4882a593Smuzhiyun 					  u16 datasize,
1594*4882a593Smuzhiyun 					  u8 *data, u32 flags)
1595*4882a593Smuzhiyun {
1596*4882a593Smuzhiyun 	u8 buf[DRXDAP_MAX_WCHUNKSIZE];
1597*4882a593Smuzhiyun 	int st = -EIO;
1598*4882a593Smuzhiyun 	int first_err = 0;
1599*4882a593Smuzhiyun 	u16 overhead_size = 0;
1600*4882a593Smuzhiyun 	u16 block_size = 0;
1601*4882a593Smuzhiyun 
1602*4882a593Smuzhiyun 	/* Check parameters ******************************************************* */
1603*4882a593Smuzhiyun 	if (dev_addr == NULL)
1604*4882a593Smuzhiyun 		return -EINVAL;
1605*4882a593Smuzhiyun 
1606*4882a593Smuzhiyun 	overhead_size = (IS_I2C_10BIT(dev_addr->i2c_addr) ? 2 : 1) +
1607*4882a593Smuzhiyun 	    (DRXDAP_FASI_LONG_FORMAT(addr) ? 4 : 2);
1608*4882a593Smuzhiyun 
1609*4882a593Smuzhiyun 	if ((DRXDAP_FASI_OFFSET_TOO_LARGE(addr)) ||
1610*4882a593Smuzhiyun 	    ((!(DRXDAPFASI_LONG_ADDR_ALLOWED)) &&
1611*4882a593Smuzhiyun 	     DRXDAP_FASI_LONG_FORMAT(addr)) ||
1612*4882a593Smuzhiyun 	    (overhead_size > (DRXDAP_MAX_WCHUNKSIZE)) ||
1613*4882a593Smuzhiyun 	    ((datasize != 0) && (data == NULL)) || ((datasize & 1) == 1))
1614*4882a593Smuzhiyun 		return -EINVAL;
1615*4882a593Smuzhiyun 
1616*4882a593Smuzhiyun 	flags &= DRXDAP_FASI_FLAGS;
1617*4882a593Smuzhiyun 	flags &= ~DRXDAP_FASI_MODEFLAGS;
1618*4882a593Smuzhiyun #if DRXDAP_SINGLE_MASTER
1619*4882a593Smuzhiyun 	flags |= DRXDAP_FASI_SINGLE_MASTER;
1620*4882a593Smuzhiyun #endif
1621*4882a593Smuzhiyun 
1622*4882a593Smuzhiyun 	/* Write block to I2C ***************************************************** */
1623*4882a593Smuzhiyun 	block_size = ((DRXDAP_MAX_WCHUNKSIZE) - overhead_size) & ~1;
1624*4882a593Smuzhiyun 	do {
1625*4882a593Smuzhiyun 		u16 todo = 0;
1626*4882a593Smuzhiyun 		u16 bufx = 0;
1627*4882a593Smuzhiyun 
1628*4882a593Smuzhiyun 		/* Buffer device address */
1629*4882a593Smuzhiyun 		addr &= ~DRXDAP_FASI_FLAGS;
1630*4882a593Smuzhiyun 		addr |= flags;
1631*4882a593Smuzhiyun #if (((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1) && ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1))
1632*4882a593Smuzhiyun 		/* short format address preferred but long format otherwise */
1633*4882a593Smuzhiyun 		if (DRXDAP_FASI_LONG_FORMAT(addr)) {
1634*4882a593Smuzhiyun #endif
1635*4882a593Smuzhiyun #if ((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1)
1636*4882a593Smuzhiyun 			buf[bufx++] = (u8) (((addr << 1) & 0xFF) | 0x01);
1637*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr >> 16) & 0xFF);
1638*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr >> 24) & 0xFF);
1639*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr >> 7) & 0xFF);
1640*4882a593Smuzhiyun #endif
1641*4882a593Smuzhiyun #if (((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1) && ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1))
1642*4882a593Smuzhiyun 		} else {
1643*4882a593Smuzhiyun #endif
1644*4882a593Smuzhiyun #if ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1)
1645*4882a593Smuzhiyun 			buf[bufx++] = (u8) ((addr << 1) & 0xFF);
1646*4882a593Smuzhiyun 			buf[bufx++] =
1647*4882a593Smuzhiyun 			    (u8) (((addr >> 16) & 0x0F) |
1648*4882a593Smuzhiyun 				    ((addr >> 18) & 0xF0));
1649*4882a593Smuzhiyun #endif
1650*4882a593Smuzhiyun #if (((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1) && ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1))
1651*4882a593Smuzhiyun 		}
1652*4882a593Smuzhiyun #endif
1653*4882a593Smuzhiyun 
1654*4882a593Smuzhiyun 		/*
1655*4882a593Smuzhiyun 		   In single master mode block_size can be 0. In such a case this I2C
1656*4882a593Smuzhiyun 		   sequense will be visible: (1) write address {i2c addr,
1657*4882a593Smuzhiyun 		   4 bytes chip address} (2) write data {i2c addr, 4 bytes data }
1658*4882a593Smuzhiyun 		   (3) write address (4) write data etc...
1659*4882a593Smuzhiyun 		   Address must be rewritten because HI is reset after data transport and
1660*4882a593Smuzhiyun 		   expects an address.
1661*4882a593Smuzhiyun 		 */
1662*4882a593Smuzhiyun 		todo = (block_size < datasize ? block_size : datasize);
1663*4882a593Smuzhiyun 		if (todo == 0) {
1664*4882a593Smuzhiyun 			u16 overhead_size_i2c_addr = 0;
1665*4882a593Smuzhiyun 			u16 data_block_size = 0;
1666*4882a593Smuzhiyun 
1667*4882a593Smuzhiyun 			overhead_size_i2c_addr =
1668*4882a593Smuzhiyun 			    (IS_I2C_10BIT(dev_addr->i2c_addr) ? 2 : 1);
1669*4882a593Smuzhiyun 			data_block_size =
1670*4882a593Smuzhiyun 			    (DRXDAP_MAX_WCHUNKSIZE - overhead_size_i2c_addr) & ~1;
1671*4882a593Smuzhiyun 
1672*4882a593Smuzhiyun 			/* write device address */
1673*4882a593Smuzhiyun 			st = drxbsp_i2c_write_read(dev_addr,
1674*4882a593Smuzhiyun 						  (u16) (bufx),
1675*4882a593Smuzhiyun 						  buf,
1676*4882a593Smuzhiyun 						  (struct i2c_device_addr *)(NULL),
1677*4882a593Smuzhiyun 						  0, (u8 *)(NULL));
1678*4882a593Smuzhiyun 
1679*4882a593Smuzhiyun 			if ((st != 0) && (first_err == 0)) {
1680*4882a593Smuzhiyun 				/* at the end, return the first error encountered */
1681*4882a593Smuzhiyun 				first_err = st;
1682*4882a593Smuzhiyun 			}
1683*4882a593Smuzhiyun 			bufx = 0;
1684*4882a593Smuzhiyun 			todo =
1685*4882a593Smuzhiyun 			    (data_block_size <
1686*4882a593Smuzhiyun 			     datasize ? data_block_size : datasize);
1687*4882a593Smuzhiyun 		}
1688*4882a593Smuzhiyun 		memcpy(&buf[bufx], data, todo);
1689*4882a593Smuzhiyun 		/* write (address if can do and) data */
1690*4882a593Smuzhiyun 		st = drxbsp_i2c_write_read(dev_addr,
1691*4882a593Smuzhiyun 					  (u16) (bufx + todo),
1692*4882a593Smuzhiyun 					  buf,
1693*4882a593Smuzhiyun 					  (struct i2c_device_addr *)(NULL),
1694*4882a593Smuzhiyun 					  0, (u8 *)(NULL));
1695*4882a593Smuzhiyun 
1696*4882a593Smuzhiyun 		if ((st != 0) && (first_err == 0)) {
1697*4882a593Smuzhiyun 			/* at the end, return the first error encountered */
1698*4882a593Smuzhiyun 			first_err = st;
1699*4882a593Smuzhiyun 		}
1700*4882a593Smuzhiyun 		datasize -= todo;
1701*4882a593Smuzhiyun 		data += todo;
1702*4882a593Smuzhiyun 		addr += (todo >> 1);
1703*4882a593Smuzhiyun 	} while (datasize);
1704*4882a593Smuzhiyun 
1705*4882a593Smuzhiyun 	return first_err;
1706*4882a593Smuzhiyun }
1707*4882a593Smuzhiyun 
1708*4882a593Smuzhiyun /*****************************
1709*4882a593Smuzhiyun *
1710*4882a593Smuzhiyun * int drxdap_fasi_write_reg16 (
1711*4882a593Smuzhiyun *     struct i2c_device_addr *dev_addr, -- address of I2C device
1712*4882a593Smuzhiyun *     u32 addr,    -- address of chip register/memory
1713*4882a593Smuzhiyun *     u16            data,    -- data to send
1714*4882a593Smuzhiyun *     u32 flags)   -- special device flags
1715*4882a593Smuzhiyun *
1716*4882a593Smuzhiyun * Write one 16-bit register or memory location. The data being written is
1717*4882a593Smuzhiyun * converted from the target platform's endianness to little endian.
1718*4882a593Smuzhiyun *
1719*4882a593Smuzhiyun * Output:
1720*4882a593Smuzhiyun * - 0     if writing was successful
1721*4882a593Smuzhiyun * - -EIO  if anything went wrong
1722*4882a593Smuzhiyun *
1723*4882a593Smuzhiyun ******************************/
1724*4882a593Smuzhiyun 
drxdap_fasi_write_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 data,u32 flags)1725*4882a593Smuzhiyun static int drxdap_fasi_write_reg16(struct i2c_device_addr *dev_addr,
1726*4882a593Smuzhiyun 					  u32 addr,
1727*4882a593Smuzhiyun 					  u16 data, u32 flags)
1728*4882a593Smuzhiyun {
1729*4882a593Smuzhiyun 	u8 buf[sizeof(data)];
1730*4882a593Smuzhiyun 
1731*4882a593Smuzhiyun 	buf[0] = (u8) ((data >> 0) & 0xFF);
1732*4882a593Smuzhiyun 	buf[1] = (u8) ((data >> 8) & 0xFF);
1733*4882a593Smuzhiyun 
1734*4882a593Smuzhiyun 	return drxdap_fasi_write_block(dev_addr, addr, sizeof(data), buf, flags);
1735*4882a593Smuzhiyun }
1736*4882a593Smuzhiyun 
1737*4882a593Smuzhiyun /*****************************
1738*4882a593Smuzhiyun *
1739*4882a593Smuzhiyun * int drxdap_fasi_read_modify_write_reg16 (
1740*4882a593Smuzhiyun *      struct i2c_device_addr *dev_addr,   -- address of I2C device
1741*4882a593Smuzhiyun *      u32 waddr,     -- address of chip register/memory
1742*4882a593Smuzhiyun *      u32 raddr,     -- chip address to read back from
1743*4882a593Smuzhiyun *      u16            wdata,     -- data to send
1744*4882a593Smuzhiyun *      u16 *rdata)     -- data to receive back
1745*4882a593Smuzhiyun *
1746*4882a593Smuzhiyun * Write 16-bit data, then read back the original contents of that location.
1747*4882a593Smuzhiyun * Requires long addressing format to be allowed.
1748*4882a593Smuzhiyun *
1749*4882a593Smuzhiyun * Before sending data, the data is converted to little endian. The
1750*4882a593Smuzhiyun * data received back is converted back to the target platform's endianness.
1751*4882a593Smuzhiyun *
1752*4882a593Smuzhiyun * WARNING: This function is only guaranteed to work if there is one
1753*4882a593Smuzhiyun * master on the I2C bus.
1754*4882a593Smuzhiyun *
1755*4882a593Smuzhiyun * Output:
1756*4882a593Smuzhiyun * - 0     if reading was successful
1757*4882a593Smuzhiyun *                  in that case: read back data is at *rdata
1758*4882a593Smuzhiyun * - -EIO  if anything went wrong
1759*4882a593Smuzhiyun *
1760*4882a593Smuzhiyun ******************************/
1761*4882a593Smuzhiyun 
drxdap_fasi_read_modify_write_reg16(struct i2c_device_addr * dev_addr,u32 waddr,u32 raddr,u16 wdata,u16 * rdata)1762*4882a593Smuzhiyun static int drxdap_fasi_read_modify_write_reg16(struct i2c_device_addr *dev_addr,
1763*4882a593Smuzhiyun 						    u32 waddr,
1764*4882a593Smuzhiyun 						    u32 raddr,
1765*4882a593Smuzhiyun 						    u16 wdata, u16 *rdata)
1766*4882a593Smuzhiyun {
1767*4882a593Smuzhiyun 	int rc = -EIO;
1768*4882a593Smuzhiyun 
1769*4882a593Smuzhiyun #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 1)
1770*4882a593Smuzhiyun 	if (rdata == NULL)
1771*4882a593Smuzhiyun 		return -EINVAL;
1772*4882a593Smuzhiyun 
1773*4882a593Smuzhiyun 	rc = drxdap_fasi_write_reg16(dev_addr, waddr, wdata, DRXDAP_FASI_RMW);
1774*4882a593Smuzhiyun 	if (rc == 0)
1775*4882a593Smuzhiyun 		rc = drxdap_fasi_read_reg16(dev_addr, raddr, rdata, 0);
1776*4882a593Smuzhiyun #endif
1777*4882a593Smuzhiyun 
1778*4882a593Smuzhiyun 	return rc;
1779*4882a593Smuzhiyun }
1780*4882a593Smuzhiyun 
1781*4882a593Smuzhiyun /*****************************
1782*4882a593Smuzhiyun *
1783*4882a593Smuzhiyun * int drxdap_fasi_write_reg32 (
1784*4882a593Smuzhiyun *     struct i2c_device_addr *dev_addr, -- address of I2C device
1785*4882a593Smuzhiyun *     u32 addr,    -- address of chip register/memory
1786*4882a593Smuzhiyun *     u32            data,    -- data to send
1787*4882a593Smuzhiyun *     u32 flags)   -- special device flags
1788*4882a593Smuzhiyun *
1789*4882a593Smuzhiyun * Write one 32-bit register or memory location. The data being written is
1790*4882a593Smuzhiyun * converted from the target platform's endianness to little endian.
1791*4882a593Smuzhiyun *
1792*4882a593Smuzhiyun * Output:
1793*4882a593Smuzhiyun * - 0     if writing was successful
1794*4882a593Smuzhiyun * - -EIO  if anything went wrong
1795*4882a593Smuzhiyun *
1796*4882a593Smuzhiyun ******************************/
1797*4882a593Smuzhiyun 
drxdap_fasi_write_reg32(struct i2c_device_addr * dev_addr,u32 addr,u32 data,u32 flags)1798*4882a593Smuzhiyun static int drxdap_fasi_write_reg32(struct i2c_device_addr *dev_addr,
1799*4882a593Smuzhiyun 					  u32 addr,
1800*4882a593Smuzhiyun 					  u32 data, u32 flags)
1801*4882a593Smuzhiyun {
1802*4882a593Smuzhiyun 	u8 buf[sizeof(data)];
1803*4882a593Smuzhiyun 
1804*4882a593Smuzhiyun 	buf[0] = (u8) ((data >> 0) & 0xFF);
1805*4882a593Smuzhiyun 	buf[1] = (u8) ((data >> 8) & 0xFF);
1806*4882a593Smuzhiyun 	buf[2] = (u8) ((data >> 16) & 0xFF);
1807*4882a593Smuzhiyun 	buf[3] = (u8) ((data >> 24) & 0xFF);
1808*4882a593Smuzhiyun 
1809*4882a593Smuzhiyun 	return drxdap_fasi_write_block(dev_addr, addr, sizeof(data), buf, flags);
1810*4882a593Smuzhiyun }
1811*4882a593Smuzhiyun 
1812*4882a593Smuzhiyun /*============================================================================*/
1813*4882a593Smuzhiyun 
1814*4882a593Smuzhiyun /*
1815*4882a593Smuzhiyun * \fn int drxj_dap_rm_write_reg16short
1816*4882a593Smuzhiyun * \brief Read modify write 16 bits audio register using short format only.
1817*4882a593Smuzhiyun * \param dev_addr
1818*4882a593Smuzhiyun * \param waddr    Address to write to
1819*4882a593Smuzhiyun * \param raddr    Address to read from (usually SIO_HI_RA_RAM_S0_RMWBUF__A)
1820*4882a593Smuzhiyun * \param wdata    Data to write
1821*4882a593Smuzhiyun * \param rdata    Buffer for data to read
1822*4882a593Smuzhiyun * \return int
1823*4882a593Smuzhiyun * \retval 0 Success
1824*4882a593Smuzhiyun * \retval -EIO Timeout, I2C error, illegal bank
1825*4882a593Smuzhiyun *
1826*4882a593Smuzhiyun * 16 bits register read modify write access using short addressing format only.
1827*4882a593Smuzhiyun * Requires knowledge of the registermap, thus device dependent.
1828*4882a593Smuzhiyun * Using DAP FASI directly to avoid endless recursion of RMWs to audio registers.
1829*4882a593Smuzhiyun *
1830*4882a593Smuzhiyun */
1831*4882a593Smuzhiyun 
1832*4882a593Smuzhiyun /* TODO correct define should be #if ( DRXDAPFASI_SHORT_ADDR_ALLOWED==1 )
1833*4882a593Smuzhiyun    See comments drxj_dap_read_modify_write_reg16 */
1834*4882a593Smuzhiyun #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 0)
drxj_dap_rm_write_reg16short(struct i2c_device_addr * dev_addr,u32 waddr,u32 raddr,u16 wdata,u16 * rdata)1835*4882a593Smuzhiyun static int drxj_dap_rm_write_reg16short(struct i2c_device_addr *dev_addr,
1836*4882a593Smuzhiyun 					      u32 waddr,
1837*4882a593Smuzhiyun 					      u32 raddr,
1838*4882a593Smuzhiyun 					      u16 wdata, u16 *rdata)
1839*4882a593Smuzhiyun {
1840*4882a593Smuzhiyun 	int rc;
1841*4882a593Smuzhiyun 
1842*4882a593Smuzhiyun 	if (rdata == NULL)
1843*4882a593Smuzhiyun 		return -EINVAL;
1844*4882a593Smuzhiyun 
1845*4882a593Smuzhiyun 	/* Set RMW flag */
1846*4882a593Smuzhiyun 	rc = drxdap_fasi_write_reg16(dev_addr,
1847*4882a593Smuzhiyun 					      SIO_HI_RA_RAM_S0_FLG_ACC__A,
1848*4882a593Smuzhiyun 					      SIO_HI_RA_RAM_S0_FLG_ACC_S0_RWM__M,
1849*4882a593Smuzhiyun 					      0x0000);
1850*4882a593Smuzhiyun 	if (rc == 0) {
1851*4882a593Smuzhiyun 		/* Write new data: triggers RMW */
1852*4882a593Smuzhiyun 		rc = drxdap_fasi_write_reg16(dev_addr, waddr, wdata,
1853*4882a593Smuzhiyun 						      0x0000);
1854*4882a593Smuzhiyun 	}
1855*4882a593Smuzhiyun 	if (rc == 0) {
1856*4882a593Smuzhiyun 		/* Read old data */
1857*4882a593Smuzhiyun 		rc = drxdap_fasi_read_reg16(dev_addr, raddr, rdata,
1858*4882a593Smuzhiyun 						     0x0000);
1859*4882a593Smuzhiyun 	}
1860*4882a593Smuzhiyun 	if (rc == 0) {
1861*4882a593Smuzhiyun 		/* Reset RMW flag */
1862*4882a593Smuzhiyun 		rc = drxdap_fasi_write_reg16(dev_addr,
1863*4882a593Smuzhiyun 						      SIO_HI_RA_RAM_S0_FLG_ACC__A,
1864*4882a593Smuzhiyun 						      0, 0x0000);
1865*4882a593Smuzhiyun 	}
1866*4882a593Smuzhiyun 
1867*4882a593Smuzhiyun 	return rc;
1868*4882a593Smuzhiyun }
1869*4882a593Smuzhiyun #endif
1870*4882a593Smuzhiyun 
1871*4882a593Smuzhiyun /*============================================================================*/
1872*4882a593Smuzhiyun 
drxj_dap_read_modify_write_reg16(struct i2c_device_addr * dev_addr,u32 waddr,u32 raddr,u16 wdata,u16 * rdata)1873*4882a593Smuzhiyun static int drxj_dap_read_modify_write_reg16(struct i2c_device_addr *dev_addr,
1874*4882a593Smuzhiyun 						 u32 waddr,
1875*4882a593Smuzhiyun 						 u32 raddr,
1876*4882a593Smuzhiyun 						 u16 wdata, u16 *rdata)
1877*4882a593Smuzhiyun {
1878*4882a593Smuzhiyun 	/* TODO: correct short/long addressing format decision,
1879*4882a593Smuzhiyun 	   now long format has higher prio then short because short also
1880*4882a593Smuzhiyun 	   needs virt bnks (not impl yet) for certain audio registers */
1881*4882a593Smuzhiyun #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 1)
1882*4882a593Smuzhiyun 	return drxdap_fasi_read_modify_write_reg16(dev_addr,
1883*4882a593Smuzhiyun 							  waddr,
1884*4882a593Smuzhiyun 							  raddr, wdata, rdata);
1885*4882a593Smuzhiyun #else
1886*4882a593Smuzhiyun 	return drxj_dap_rm_write_reg16short(dev_addr, waddr, raddr, wdata, rdata);
1887*4882a593Smuzhiyun #endif
1888*4882a593Smuzhiyun }
1889*4882a593Smuzhiyun 
1890*4882a593Smuzhiyun 
1891*4882a593Smuzhiyun /*============================================================================*/
1892*4882a593Smuzhiyun 
1893*4882a593Smuzhiyun /*
1894*4882a593Smuzhiyun * \fn int drxj_dap_read_aud_reg16
1895*4882a593Smuzhiyun * \brief Read 16 bits audio register
1896*4882a593Smuzhiyun * \param dev_addr
1897*4882a593Smuzhiyun * \param addr
1898*4882a593Smuzhiyun * \param data
1899*4882a593Smuzhiyun * \return int
1900*4882a593Smuzhiyun * \retval 0 Success
1901*4882a593Smuzhiyun * \retval -EIO Timeout, I2C error, illegal bank
1902*4882a593Smuzhiyun *
1903*4882a593Smuzhiyun * 16 bits register read access via audio token ring interface.
1904*4882a593Smuzhiyun *
1905*4882a593Smuzhiyun */
drxj_dap_read_aud_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 * data)1906*4882a593Smuzhiyun static int drxj_dap_read_aud_reg16(struct i2c_device_addr *dev_addr,
1907*4882a593Smuzhiyun 					 u32 addr, u16 *data)
1908*4882a593Smuzhiyun {
1909*4882a593Smuzhiyun 	u32 start_timer = 0;
1910*4882a593Smuzhiyun 	u32 current_timer = 0;
1911*4882a593Smuzhiyun 	u32 delta_timer = 0;
1912*4882a593Smuzhiyun 	u16 tr_status = 0;
1913*4882a593Smuzhiyun 	int stat = -EIO;
1914*4882a593Smuzhiyun 
1915*4882a593Smuzhiyun 	/* No read possible for bank 3, return with error */
1916*4882a593Smuzhiyun 	if (DRXDAP_FASI_ADDR2BANK(addr) == 3) {
1917*4882a593Smuzhiyun 		stat = -EINVAL;
1918*4882a593Smuzhiyun 	} else {
1919*4882a593Smuzhiyun 		const u32 write_bit = ((dr_xaddr_t) 1) << 16;
1920*4882a593Smuzhiyun 
1921*4882a593Smuzhiyun 		/* Force reset write bit */
1922*4882a593Smuzhiyun 		addr &= (~write_bit);
1923*4882a593Smuzhiyun 
1924*4882a593Smuzhiyun 		/* Set up read */
1925*4882a593Smuzhiyun 		start_timer = jiffies_to_msecs(jiffies);
1926*4882a593Smuzhiyun 		do {
1927*4882a593Smuzhiyun 			/* RMW to aud TR IF until request is granted or timeout */
1928*4882a593Smuzhiyun 			stat = drxj_dap_read_modify_write_reg16(dev_addr,
1929*4882a593Smuzhiyun 							     addr,
1930*4882a593Smuzhiyun 							     SIO_HI_RA_RAM_S0_RMWBUF__A,
1931*4882a593Smuzhiyun 							     0x0000, &tr_status);
1932*4882a593Smuzhiyun 
1933*4882a593Smuzhiyun 			if (stat != 0)
1934*4882a593Smuzhiyun 				break;
1935*4882a593Smuzhiyun 
1936*4882a593Smuzhiyun 			current_timer = jiffies_to_msecs(jiffies);
1937*4882a593Smuzhiyun 			delta_timer = current_timer - start_timer;
1938*4882a593Smuzhiyun 			if (delta_timer > DRXJ_DAP_AUDTRIF_TIMEOUT) {
1939*4882a593Smuzhiyun 				stat = -EIO;
1940*4882a593Smuzhiyun 				break;
1941*4882a593Smuzhiyun 			}
1942*4882a593Smuzhiyun 
1943*4882a593Smuzhiyun 		} while (((tr_status & AUD_TOP_TR_CTR_FIFO_LOCK__M) ==
1944*4882a593Smuzhiyun 			  AUD_TOP_TR_CTR_FIFO_LOCK_LOCKED) ||
1945*4882a593Smuzhiyun 			 ((tr_status & AUD_TOP_TR_CTR_FIFO_FULL__M) ==
1946*4882a593Smuzhiyun 			  AUD_TOP_TR_CTR_FIFO_FULL_FULL));
1947*4882a593Smuzhiyun 	}			/* if ( DRXDAP_FASI_ADDR2BANK(addr)!=3 ) */
1948*4882a593Smuzhiyun 
1949*4882a593Smuzhiyun 	/* Wait for read ready status or timeout */
1950*4882a593Smuzhiyun 	if (stat == 0) {
1951*4882a593Smuzhiyun 		start_timer = jiffies_to_msecs(jiffies);
1952*4882a593Smuzhiyun 
1953*4882a593Smuzhiyun 		while ((tr_status & AUD_TOP_TR_CTR_FIFO_RD_RDY__M) !=
1954*4882a593Smuzhiyun 		       AUD_TOP_TR_CTR_FIFO_RD_RDY_READY) {
1955*4882a593Smuzhiyun 			stat = drxj_dap_read_reg16(dev_addr,
1956*4882a593Smuzhiyun 						  AUD_TOP_TR_CTR__A,
1957*4882a593Smuzhiyun 						  &tr_status, 0x0000);
1958*4882a593Smuzhiyun 			if (stat != 0)
1959*4882a593Smuzhiyun 				break;
1960*4882a593Smuzhiyun 
1961*4882a593Smuzhiyun 			current_timer = jiffies_to_msecs(jiffies);
1962*4882a593Smuzhiyun 			delta_timer = current_timer - start_timer;
1963*4882a593Smuzhiyun 			if (delta_timer > DRXJ_DAP_AUDTRIF_TIMEOUT) {
1964*4882a593Smuzhiyun 				stat = -EIO;
1965*4882a593Smuzhiyun 				break;
1966*4882a593Smuzhiyun 			}
1967*4882a593Smuzhiyun 		}		/* while ( ... ) */
1968*4882a593Smuzhiyun 	}
1969*4882a593Smuzhiyun 
1970*4882a593Smuzhiyun 	/* Read value */
1971*4882a593Smuzhiyun 	if (stat == 0)
1972*4882a593Smuzhiyun 		stat = drxj_dap_read_modify_write_reg16(dev_addr,
1973*4882a593Smuzhiyun 						     AUD_TOP_TR_RD_REG__A,
1974*4882a593Smuzhiyun 						     SIO_HI_RA_RAM_S0_RMWBUF__A,
1975*4882a593Smuzhiyun 						     0x0000, data);
1976*4882a593Smuzhiyun 	return stat;
1977*4882a593Smuzhiyun }
1978*4882a593Smuzhiyun 
1979*4882a593Smuzhiyun /*============================================================================*/
1980*4882a593Smuzhiyun 
drxj_dap_read_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 * data,u32 flags)1981*4882a593Smuzhiyun static int drxj_dap_read_reg16(struct i2c_device_addr *dev_addr,
1982*4882a593Smuzhiyun 				      u32 addr,
1983*4882a593Smuzhiyun 				      u16 *data, u32 flags)
1984*4882a593Smuzhiyun {
1985*4882a593Smuzhiyun 	int stat = -EIO;
1986*4882a593Smuzhiyun 
1987*4882a593Smuzhiyun 	/* Check param */
1988*4882a593Smuzhiyun 	if ((dev_addr == NULL) || (data == NULL))
1989*4882a593Smuzhiyun 		return -EINVAL;
1990*4882a593Smuzhiyun 
1991*4882a593Smuzhiyun 	if (is_handled_by_aud_tr_if(addr))
1992*4882a593Smuzhiyun 		stat = drxj_dap_read_aud_reg16(dev_addr, addr, data);
1993*4882a593Smuzhiyun 	else
1994*4882a593Smuzhiyun 		stat = drxdap_fasi_read_reg16(dev_addr, addr, data, flags);
1995*4882a593Smuzhiyun 
1996*4882a593Smuzhiyun 	return stat;
1997*4882a593Smuzhiyun }
1998*4882a593Smuzhiyun /*============================================================================*/
1999*4882a593Smuzhiyun 
2000*4882a593Smuzhiyun /*
2001*4882a593Smuzhiyun * \fn int drxj_dap_write_aud_reg16
2002*4882a593Smuzhiyun * \brief Write 16 bits audio register
2003*4882a593Smuzhiyun * \param dev_addr
2004*4882a593Smuzhiyun * \param addr
2005*4882a593Smuzhiyun * \param data
2006*4882a593Smuzhiyun * \return int
2007*4882a593Smuzhiyun * \retval 0 Success
2008*4882a593Smuzhiyun * \retval -EIO Timeout, I2C error, illegal bank
2009*4882a593Smuzhiyun *
2010*4882a593Smuzhiyun * 16 bits register write access via audio token ring interface.
2011*4882a593Smuzhiyun *
2012*4882a593Smuzhiyun */
drxj_dap_write_aud_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 data)2013*4882a593Smuzhiyun static int drxj_dap_write_aud_reg16(struct i2c_device_addr *dev_addr,
2014*4882a593Smuzhiyun 					  u32 addr, u16 data)
2015*4882a593Smuzhiyun {
2016*4882a593Smuzhiyun 	int stat = -EIO;
2017*4882a593Smuzhiyun 
2018*4882a593Smuzhiyun 	/* No write possible for bank 2, return with error */
2019*4882a593Smuzhiyun 	if (DRXDAP_FASI_ADDR2BANK(addr) == 2) {
2020*4882a593Smuzhiyun 		stat = -EINVAL;
2021*4882a593Smuzhiyun 	} else {
2022*4882a593Smuzhiyun 		u32 start_timer = 0;
2023*4882a593Smuzhiyun 		u32 current_timer = 0;
2024*4882a593Smuzhiyun 		u32 delta_timer = 0;
2025*4882a593Smuzhiyun 		u16 tr_status = 0;
2026*4882a593Smuzhiyun 		const u32 write_bit = ((dr_xaddr_t) 1) << 16;
2027*4882a593Smuzhiyun 
2028*4882a593Smuzhiyun 		/* Force write bit */
2029*4882a593Smuzhiyun 		addr |= write_bit;
2030*4882a593Smuzhiyun 		start_timer = jiffies_to_msecs(jiffies);
2031*4882a593Smuzhiyun 		do {
2032*4882a593Smuzhiyun 			/* RMW to aud TR IF until request is granted or timeout */
2033*4882a593Smuzhiyun 			stat = drxj_dap_read_modify_write_reg16(dev_addr,
2034*4882a593Smuzhiyun 							     addr,
2035*4882a593Smuzhiyun 							     SIO_HI_RA_RAM_S0_RMWBUF__A,
2036*4882a593Smuzhiyun 							     data, &tr_status);
2037*4882a593Smuzhiyun 			if (stat != 0)
2038*4882a593Smuzhiyun 				break;
2039*4882a593Smuzhiyun 
2040*4882a593Smuzhiyun 			current_timer = jiffies_to_msecs(jiffies);
2041*4882a593Smuzhiyun 			delta_timer = current_timer - start_timer;
2042*4882a593Smuzhiyun 			if (delta_timer > DRXJ_DAP_AUDTRIF_TIMEOUT) {
2043*4882a593Smuzhiyun 				stat = -EIO;
2044*4882a593Smuzhiyun 				break;
2045*4882a593Smuzhiyun 			}
2046*4882a593Smuzhiyun 
2047*4882a593Smuzhiyun 		} while (((tr_status & AUD_TOP_TR_CTR_FIFO_LOCK__M) ==
2048*4882a593Smuzhiyun 			  AUD_TOP_TR_CTR_FIFO_LOCK_LOCKED) ||
2049*4882a593Smuzhiyun 			 ((tr_status & AUD_TOP_TR_CTR_FIFO_FULL__M) ==
2050*4882a593Smuzhiyun 			  AUD_TOP_TR_CTR_FIFO_FULL_FULL));
2051*4882a593Smuzhiyun 
2052*4882a593Smuzhiyun 	}			/* if ( DRXDAP_FASI_ADDR2BANK(addr)!=2 ) */
2053*4882a593Smuzhiyun 
2054*4882a593Smuzhiyun 	return stat;
2055*4882a593Smuzhiyun }
2056*4882a593Smuzhiyun 
2057*4882a593Smuzhiyun /*============================================================================*/
2058*4882a593Smuzhiyun 
drxj_dap_write_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 data,u32 flags)2059*4882a593Smuzhiyun static int drxj_dap_write_reg16(struct i2c_device_addr *dev_addr,
2060*4882a593Smuzhiyun 				       u32 addr,
2061*4882a593Smuzhiyun 				       u16 data, u32 flags)
2062*4882a593Smuzhiyun {
2063*4882a593Smuzhiyun 	int stat = -EIO;
2064*4882a593Smuzhiyun 
2065*4882a593Smuzhiyun 	/* Check param */
2066*4882a593Smuzhiyun 	if (dev_addr == NULL)
2067*4882a593Smuzhiyun 		return -EINVAL;
2068*4882a593Smuzhiyun 
2069*4882a593Smuzhiyun 	if (is_handled_by_aud_tr_if(addr))
2070*4882a593Smuzhiyun 		stat = drxj_dap_write_aud_reg16(dev_addr, addr, data);
2071*4882a593Smuzhiyun 	else
2072*4882a593Smuzhiyun 		stat = drxdap_fasi_write_reg16(dev_addr,
2073*4882a593Smuzhiyun 							    addr, data, flags);
2074*4882a593Smuzhiyun 
2075*4882a593Smuzhiyun 	return stat;
2076*4882a593Smuzhiyun }
2077*4882a593Smuzhiyun 
2078*4882a593Smuzhiyun /*============================================================================*/
2079*4882a593Smuzhiyun 
2080*4882a593Smuzhiyun /* Free data ram in SIO HI */
2081*4882a593Smuzhiyun #define SIO_HI_RA_RAM_USR_BEGIN__A 0x420040
2082*4882a593Smuzhiyun #define SIO_HI_RA_RAM_USR_END__A   0x420060
2083*4882a593Smuzhiyun 
2084*4882a593Smuzhiyun #define DRXJ_HI_ATOMIC_BUF_START (SIO_HI_RA_RAM_USR_BEGIN__A)
2085*4882a593Smuzhiyun #define DRXJ_HI_ATOMIC_BUF_END   (SIO_HI_RA_RAM_USR_BEGIN__A + 7)
2086*4882a593Smuzhiyun #define DRXJ_HI_ATOMIC_READ      SIO_HI_RA_RAM_PAR_3_ACP_RW_READ
2087*4882a593Smuzhiyun #define DRXJ_HI_ATOMIC_WRITE     SIO_HI_RA_RAM_PAR_3_ACP_RW_WRITE
2088*4882a593Smuzhiyun 
2089*4882a593Smuzhiyun /*
2090*4882a593Smuzhiyun * \fn int drxj_dap_atomic_read_write_block()
2091*4882a593Smuzhiyun * \brief Basic access routine for atomic read or write access
2092*4882a593Smuzhiyun * \param dev_addr  pointer to i2c dev address
2093*4882a593Smuzhiyun * \param addr     destination/source address
2094*4882a593Smuzhiyun * \param datasize size of data buffer in bytes
2095*4882a593Smuzhiyun * \param data     pointer to data buffer
2096*4882a593Smuzhiyun * \return int
2097*4882a593Smuzhiyun * \retval 0 Success
2098*4882a593Smuzhiyun * \retval -EIO Timeout, I2C error, illegal bank
2099*4882a593Smuzhiyun *
2100*4882a593Smuzhiyun */
2101*4882a593Smuzhiyun static
drxj_dap_atomic_read_write_block(struct i2c_device_addr * dev_addr,u32 addr,u16 datasize,u8 * data,bool read_flag)2102*4882a593Smuzhiyun int drxj_dap_atomic_read_write_block(struct i2c_device_addr *dev_addr,
2103*4882a593Smuzhiyun 					  u32 addr,
2104*4882a593Smuzhiyun 					  u16 datasize,
2105*4882a593Smuzhiyun 					  u8 *data, bool read_flag)
2106*4882a593Smuzhiyun {
2107*4882a593Smuzhiyun 	struct drxj_hi_cmd hi_cmd;
2108*4882a593Smuzhiyun 	int rc;
2109*4882a593Smuzhiyun 	u16 word;
2110*4882a593Smuzhiyun 	u16 dummy = 0;
2111*4882a593Smuzhiyun 	u16 i = 0;
2112*4882a593Smuzhiyun 
2113*4882a593Smuzhiyun 	/* Parameter check */
2114*4882a593Smuzhiyun 	if (!data || !dev_addr || ((datasize % 2)) || ((datasize / 2) > 8))
2115*4882a593Smuzhiyun 		return -EINVAL;
2116*4882a593Smuzhiyun 
2117*4882a593Smuzhiyun 	/* Set up HI parameters to read or write n bytes */
2118*4882a593Smuzhiyun 	hi_cmd.cmd = SIO_HI_RA_RAM_CMD_ATOMIC_COPY;
2119*4882a593Smuzhiyun 	hi_cmd.param1 =
2120*4882a593Smuzhiyun 	    (u16) ((DRXDAP_FASI_ADDR2BLOCK(DRXJ_HI_ATOMIC_BUF_START) << 6) +
2121*4882a593Smuzhiyun 		     DRXDAP_FASI_ADDR2BANK(DRXJ_HI_ATOMIC_BUF_START));
2122*4882a593Smuzhiyun 	hi_cmd.param2 =
2123*4882a593Smuzhiyun 	    (u16) DRXDAP_FASI_ADDR2OFFSET(DRXJ_HI_ATOMIC_BUF_START);
2124*4882a593Smuzhiyun 	hi_cmd.param3 = (u16) ((datasize / 2) - 1);
2125*4882a593Smuzhiyun 	if (!read_flag)
2126*4882a593Smuzhiyun 		hi_cmd.param3 |= DRXJ_HI_ATOMIC_WRITE;
2127*4882a593Smuzhiyun 	else
2128*4882a593Smuzhiyun 		hi_cmd.param3 |= DRXJ_HI_ATOMIC_READ;
2129*4882a593Smuzhiyun 	hi_cmd.param4 = (u16) ((DRXDAP_FASI_ADDR2BLOCK(addr) << 6) +
2130*4882a593Smuzhiyun 				DRXDAP_FASI_ADDR2BANK(addr));
2131*4882a593Smuzhiyun 	hi_cmd.param5 = (u16) DRXDAP_FASI_ADDR2OFFSET(addr);
2132*4882a593Smuzhiyun 
2133*4882a593Smuzhiyun 	if (!read_flag) {
2134*4882a593Smuzhiyun 		/* write data to buffer */
2135*4882a593Smuzhiyun 		for (i = 0; i < (datasize / 2); i++) {
2136*4882a593Smuzhiyun 
2137*4882a593Smuzhiyun 			word = ((u16) data[2 * i]);
2138*4882a593Smuzhiyun 			word += (((u16) data[(2 * i) + 1]) << 8);
2139*4882a593Smuzhiyun 			drxj_dap_write_reg16(dev_addr,
2140*4882a593Smuzhiyun 					     (DRXJ_HI_ATOMIC_BUF_START + i),
2141*4882a593Smuzhiyun 					    word, 0);
2142*4882a593Smuzhiyun 		}
2143*4882a593Smuzhiyun 	}
2144*4882a593Smuzhiyun 
2145*4882a593Smuzhiyun 	rc = hi_command(dev_addr, &hi_cmd, &dummy);
2146*4882a593Smuzhiyun 	if (rc != 0) {
2147*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2148*4882a593Smuzhiyun 		goto rw_error;
2149*4882a593Smuzhiyun 	}
2150*4882a593Smuzhiyun 
2151*4882a593Smuzhiyun 	if (read_flag) {
2152*4882a593Smuzhiyun 		/* read data from buffer */
2153*4882a593Smuzhiyun 		for (i = 0; i < (datasize / 2); i++) {
2154*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr,
2155*4882a593Smuzhiyun 						 (DRXJ_HI_ATOMIC_BUF_START + i),
2156*4882a593Smuzhiyun 						 &word, 0);
2157*4882a593Smuzhiyun 			if (rc) {
2158*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2159*4882a593Smuzhiyun 				goto rw_error;
2160*4882a593Smuzhiyun 			}
2161*4882a593Smuzhiyun 			data[2 * i] = (u8) (word & 0xFF);
2162*4882a593Smuzhiyun 			data[(2 * i) + 1] = (u8) (word >> 8);
2163*4882a593Smuzhiyun 		}
2164*4882a593Smuzhiyun 	}
2165*4882a593Smuzhiyun 
2166*4882a593Smuzhiyun 	return 0;
2167*4882a593Smuzhiyun 
2168*4882a593Smuzhiyun rw_error:
2169*4882a593Smuzhiyun 	return rc;
2170*4882a593Smuzhiyun 
2171*4882a593Smuzhiyun }
2172*4882a593Smuzhiyun 
2173*4882a593Smuzhiyun /*============================================================================*/
2174*4882a593Smuzhiyun 
2175*4882a593Smuzhiyun /*
2176*4882a593Smuzhiyun * \fn int drxj_dap_atomic_read_reg32()
2177*4882a593Smuzhiyun * \brief Atomic read of 32 bits words
2178*4882a593Smuzhiyun */
2179*4882a593Smuzhiyun static
drxj_dap_atomic_read_reg32(struct i2c_device_addr * dev_addr,u32 addr,u32 * data,u32 flags)2180*4882a593Smuzhiyun int drxj_dap_atomic_read_reg32(struct i2c_device_addr *dev_addr,
2181*4882a593Smuzhiyun 				     u32 addr,
2182*4882a593Smuzhiyun 				     u32 *data, u32 flags)
2183*4882a593Smuzhiyun {
2184*4882a593Smuzhiyun 	u8 buf[sizeof(*data)] = { 0 };
2185*4882a593Smuzhiyun 	int rc;
2186*4882a593Smuzhiyun 	u32 word = 0;
2187*4882a593Smuzhiyun 
2188*4882a593Smuzhiyun 	if (!data)
2189*4882a593Smuzhiyun 		return -EINVAL;
2190*4882a593Smuzhiyun 
2191*4882a593Smuzhiyun 	rc = drxj_dap_atomic_read_write_block(dev_addr, addr,
2192*4882a593Smuzhiyun 					      sizeof(*data), buf, true);
2193*4882a593Smuzhiyun 
2194*4882a593Smuzhiyun 	if (rc < 0)
2195*4882a593Smuzhiyun 		return 0;
2196*4882a593Smuzhiyun 
2197*4882a593Smuzhiyun 	word = (u32) buf[3];
2198*4882a593Smuzhiyun 	word <<= 8;
2199*4882a593Smuzhiyun 	word |= (u32) buf[2];
2200*4882a593Smuzhiyun 	word <<= 8;
2201*4882a593Smuzhiyun 	word |= (u32) buf[1];
2202*4882a593Smuzhiyun 	word <<= 8;
2203*4882a593Smuzhiyun 	word |= (u32) buf[0];
2204*4882a593Smuzhiyun 
2205*4882a593Smuzhiyun 	*data = word;
2206*4882a593Smuzhiyun 
2207*4882a593Smuzhiyun 	return rc;
2208*4882a593Smuzhiyun }
2209*4882a593Smuzhiyun 
2210*4882a593Smuzhiyun /*============================================================================*/
2211*4882a593Smuzhiyun 
2212*4882a593Smuzhiyun /*============================================================================*/
2213*4882a593Smuzhiyun /*==                        END DRXJ DAP FUNCTIONS                          ==*/
2214*4882a593Smuzhiyun /*============================================================================*/
2215*4882a593Smuzhiyun 
2216*4882a593Smuzhiyun /*============================================================================*/
2217*4882a593Smuzhiyun /*============================================================================*/
2218*4882a593Smuzhiyun /*==                      HOST INTERFACE FUNCTIONS                          ==*/
2219*4882a593Smuzhiyun /*============================================================================*/
2220*4882a593Smuzhiyun /*============================================================================*/
2221*4882a593Smuzhiyun 
2222*4882a593Smuzhiyun /*
2223*4882a593Smuzhiyun * \fn int hi_cfg_command()
2224*4882a593Smuzhiyun * \brief Configure HI with settings stored in the demod structure.
2225*4882a593Smuzhiyun * \param demod Demodulator.
2226*4882a593Smuzhiyun * \return int.
2227*4882a593Smuzhiyun *
2228*4882a593Smuzhiyun * This routine was created because to much orthogonal settings have
2229*4882a593Smuzhiyun * been put into one HI API function (configure). Especially the I2C bridge
2230*4882a593Smuzhiyun * enable/disable should not need re-configuration of the HI.
2231*4882a593Smuzhiyun *
2232*4882a593Smuzhiyun */
hi_cfg_command(const struct drx_demod_instance * demod)2233*4882a593Smuzhiyun static int hi_cfg_command(const struct drx_demod_instance *demod)
2234*4882a593Smuzhiyun {
2235*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
2236*4882a593Smuzhiyun 	struct drxj_hi_cmd hi_cmd;
2237*4882a593Smuzhiyun 	u16 result = 0;
2238*4882a593Smuzhiyun 	int rc;
2239*4882a593Smuzhiyun 
2240*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
2241*4882a593Smuzhiyun 
2242*4882a593Smuzhiyun 	hi_cmd.cmd = SIO_HI_RA_RAM_CMD_CONFIG;
2243*4882a593Smuzhiyun 	hi_cmd.param1 = SIO_HI_RA_RAM_PAR_1_PAR1_SEC_KEY;
2244*4882a593Smuzhiyun 	hi_cmd.param2 = ext_attr->hi_cfg_timing_div;
2245*4882a593Smuzhiyun 	hi_cmd.param3 = ext_attr->hi_cfg_bridge_delay;
2246*4882a593Smuzhiyun 	hi_cmd.param4 = ext_attr->hi_cfg_wake_up_key;
2247*4882a593Smuzhiyun 	hi_cmd.param5 = ext_attr->hi_cfg_ctrl;
2248*4882a593Smuzhiyun 	hi_cmd.param6 = ext_attr->hi_cfg_transmit;
2249*4882a593Smuzhiyun 
2250*4882a593Smuzhiyun 	rc = hi_command(demod->my_i2c_dev_addr, &hi_cmd, &result);
2251*4882a593Smuzhiyun 	if (rc != 0) {
2252*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2253*4882a593Smuzhiyun 		goto rw_error;
2254*4882a593Smuzhiyun 	}
2255*4882a593Smuzhiyun 
2256*4882a593Smuzhiyun 	/* Reset power down flag (set one call only) */
2257*4882a593Smuzhiyun 	ext_attr->hi_cfg_ctrl &= (~(SIO_HI_RA_RAM_PAR_5_CFG_SLEEP_ZZZ));
2258*4882a593Smuzhiyun 
2259*4882a593Smuzhiyun 	return 0;
2260*4882a593Smuzhiyun 
2261*4882a593Smuzhiyun rw_error:
2262*4882a593Smuzhiyun 	return rc;
2263*4882a593Smuzhiyun }
2264*4882a593Smuzhiyun 
2265*4882a593Smuzhiyun /*
2266*4882a593Smuzhiyun * \fn int hi_command()
2267*4882a593Smuzhiyun * \brief Configure HI with settings stored in the demod structure.
2268*4882a593Smuzhiyun * \param dev_addr I2C address.
2269*4882a593Smuzhiyun * \param cmd HI command.
2270*4882a593Smuzhiyun * \param result HI command result.
2271*4882a593Smuzhiyun * \return int.
2272*4882a593Smuzhiyun *
2273*4882a593Smuzhiyun * Sends command to HI
2274*4882a593Smuzhiyun *
2275*4882a593Smuzhiyun */
2276*4882a593Smuzhiyun static int
hi_command(struct i2c_device_addr * dev_addr,const struct drxj_hi_cmd * cmd,u16 * result)2277*4882a593Smuzhiyun hi_command(struct i2c_device_addr *dev_addr, const struct drxj_hi_cmd *cmd, u16 *result)
2278*4882a593Smuzhiyun {
2279*4882a593Smuzhiyun 	u16 wait_cmd = 0;
2280*4882a593Smuzhiyun 	u16 nr_retries = 0;
2281*4882a593Smuzhiyun 	bool powerdown_cmd = false;
2282*4882a593Smuzhiyun 	int rc;
2283*4882a593Smuzhiyun 
2284*4882a593Smuzhiyun 	/* Write parameters */
2285*4882a593Smuzhiyun 	switch (cmd->cmd) {
2286*4882a593Smuzhiyun 
2287*4882a593Smuzhiyun 	case SIO_HI_RA_RAM_CMD_CONFIG:
2288*4882a593Smuzhiyun 	case SIO_HI_RA_RAM_CMD_ATOMIC_COPY:
2289*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_6__A, cmd->param6, 0);
2290*4882a593Smuzhiyun 		if (rc != 0) {
2291*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2292*4882a593Smuzhiyun 			goto rw_error;
2293*4882a593Smuzhiyun 		}
2294*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_5__A, cmd->param5, 0);
2295*4882a593Smuzhiyun 		if (rc != 0) {
2296*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2297*4882a593Smuzhiyun 			goto rw_error;
2298*4882a593Smuzhiyun 		}
2299*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_4__A, cmd->param4, 0);
2300*4882a593Smuzhiyun 		if (rc != 0) {
2301*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2302*4882a593Smuzhiyun 			goto rw_error;
2303*4882a593Smuzhiyun 		}
2304*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_3__A, cmd->param3, 0);
2305*4882a593Smuzhiyun 		if (rc != 0) {
2306*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2307*4882a593Smuzhiyun 			goto rw_error;
2308*4882a593Smuzhiyun 		}
2309*4882a593Smuzhiyun 		fallthrough;
2310*4882a593Smuzhiyun 	case SIO_HI_RA_RAM_CMD_BRDCTRL:
2311*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_2__A, cmd->param2, 0);
2312*4882a593Smuzhiyun 		if (rc != 0) {
2313*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2314*4882a593Smuzhiyun 			goto rw_error;
2315*4882a593Smuzhiyun 		}
2316*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_1__A, cmd->param1, 0);
2317*4882a593Smuzhiyun 		if (rc != 0) {
2318*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2319*4882a593Smuzhiyun 			goto rw_error;
2320*4882a593Smuzhiyun 		}
2321*4882a593Smuzhiyun 		fallthrough;
2322*4882a593Smuzhiyun 	case SIO_HI_RA_RAM_CMD_NULL:
2323*4882a593Smuzhiyun 		/* No parameters */
2324*4882a593Smuzhiyun 		break;
2325*4882a593Smuzhiyun 
2326*4882a593Smuzhiyun 	default:
2327*4882a593Smuzhiyun 		return -EINVAL;
2328*4882a593Smuzhiyun 		break;
2329*4882a593Smuzhiyun 	}
2330*4882a593Smuzhiyun 
2331*4882a593Smuzhiyun 	/* Write command */
2332*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_CMD__A, cmd->cmd, 0);
2333*4882a593Smuzhiyun 	if (rc != 0) {
2334*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2335*4882a593Smuzhiyun 		goto rw_error;
2336*4882a593Smuzhiyun 	}
2337*4882a593Smuzhiyun 
2338*4882a593Smuzhiyun 	if ((cmd->cmd) == SIO_HI_RA_RAM_CMD_RESET)
2339*4882a593Smuzhiyun 		msleep(1);
2340*4882a593Smuzhiyun 
2341*4882a593Smuzhiyun 	/* Detect power down to omit reading result */
2342*4882a593Smuzhiyun 	powerdown_cmd = (bool) ((cmd->cmd == SIO_HI_RA_RAM_CMD_CONFIG) &&
2343*4882a593Smuzhiyun 				  (((cmd->
2344*4882a593Smuzhiyun 				     param5) & SIO_HI_RA_RAM_PAR_5_CFG_SLEEP__M)
2345*4882a593Smuzhiyun 				   == SIO_HI_RA_RAM_PAR_5_CFG_SLEEP_ZZZ));
2346*4882a593Smuzhiyun 	if (!powerdown_cmd) {
2347*4882a593Smuzhiyun 		/* Wait until command rdy */
2348*4882a593Smuzhiyun 		do {
2349*4882a593Smuzhiyun 			nr_retries++;
2350*4882a593Smuzhiyun 			if (nr_retries > DRXJ_MAX_RETRIES) {
2351*4882a593Smuzhiyun 				pr_err("timeout\n");
2352*4882a593Smuzhiyun 				goto rw_error;
2353*4882a593Smuzhiyun 			}
2354*4882a593Smuzhiyun 
2355*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, SIO_HI_RA_RAM_CMD__A, &wait_cmd, 0);
2356*4882a593Smuzhiyun 			if (rc != 0) {
2357*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2358*4882a593Smuzhiyun 				goto rw_error;
2359*4882a593Smuzhiyun 			}
2360*4882a593Smuzhiyun 		} while (wait_cmd != 0);
2361*4882a593Smuzhiyun 
2362*4882a593Smuzhiyun 		/* Read result */
2363*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, SIO_HI_RA_RAM_RES__A, result, 0);
2364*4882a593Smuzhiyun 		if (rc != 0) {
2365*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2366*4882a593Smuzhiyun 			goto rw_error;
2367*4882a593Smuzhiyun 		}
2368*4882a593Smuzhiyun 
2369*4882a593Smuzhiyun 	}
2370*4882a593Smuzhiyun 	/* if ( powerdown_cmd == true ) */
2371*4882a593Smuzhiyun 	return 0;
2372*4882a593Smuzhiyun rw_error:
2373*4882a593Smuzhiyun 	return rc;
2374*4882a593Smuzhiyun }
2375*4882a593Smuzhiyun 
2376*4882a593Smuzhiyun /*
2377*4882a593Smuzhiyun * \fn int init_hi( const struct drx_demod_instance *demod )
2378*4882a593Smuzhiyun * \brief Initialise and configurate HI.
2379*4882a593Smuzhiyun * \param demod pointer to demod data.
2380*4882a593Smuzhiyun * \return int Return status.
2381*4882a593Smuzhiyun * \retval 0 Success.
2382*4882a593Smuzhiyun * \retval -EIO Failure.
2383*4882a593Smuzhiyun *
2384*4882a593Smuzhiyun * Needs to know Psys (System Clock period) and Posc (Osc Clock period)
2385*4882a593Smuzhiyun * Need to store configuration in driver because of the way I2C
2386*4882a593Smuzhiyun * bridging is controlled.
2387*4882a593Smuzhiyun *
2388*4882a593Smuzhiyun */
init_hi(const struct drx_demod_instance * demod)2389*4882a593Smuzhiyun static int init_hi(const struct drx_demod_instance *demod)
2390*4882a593Smuzhiyun {
2391*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
2392*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = (struct drx_common_attr *) (NULL);
2393*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
2394*4882a593Smuzhiyun 	int rc;
2395*4882a593Smuzhiyun 
2396*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
2397*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
2398*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
2399*4882a593Smuzhiyun 
2400*4882a593Smuzhiyun 	/* PATCH for bug 5003, HI ucode v3.1.0 */
2401*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, 0x4301D7, 0x801, 0);
2402*4882a593Smuzhiyun 	if (rc != 0) {
2403*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2404*4882a593Smuzhiyun 		goto rw_error;
2405*4882a593Smuzhiyun 	}
2406*4882a593Smuzhiyun 
2407*4882a593Smuzhiyun 	/* Timing div, 250ns/Psys */
2408*4882a593Smuzhiyun 	/* Timing div, = ( delay (nano seconds) * sysclk (kHz) )/ 1000 */
2409*4882a593Smuzhiyun 	ext_attr->hi_cfg_timing_div =
2410*4882a593Smuzhiyun 	    (u16) ((common_attr->sys_clock_freq / 1000) * HI_I2C_DELAY) / 1000;
2411*4882a593Smuzhiyun 	/* Clipping */
2412*4882a593Smuzhiyun 	if ((ext_attr->hi_cfg_timing_div) > SIO_HI_RA_RAM_PAR_2_CFG_DIV__M)
2413*4882a593Smuzhiyun 		ext_attr->hi_cfg_timing_div = SIO_HI_RA_RAM_PAR_2_CFG_DIV__M;
2414*4882a593Smuzhiyun 	/* Bridge delay, uses oscilator clock */
2415*4882a593Smuzhiyun 	/* Delay = ( delay (nano seconds) * oscclk (kHz) )/ 1000 */
2416*4882a593Smuzhiyun 	/* SDA brdige delay */
2417*4882a593Smuzhiyun 	ext_attr->hi_cfg_bridge_delay =
2418*4882a593Smuzhiyun 	    (u16) ((common_attr->osc_clock_freq / 1000) * HI_I2C_BRIDGE_DELAY) /
2419*4882a593Smuzhiyun 	    1000;
2420*4882a593Smuzhiyun 	/* Clipping */
2421*4882a593Smuzhiyun 	if ((ext_attr->hi_cfg_bridge_delay) > SIO_HI_RA_RAM_PAR_3_CFG_DBL_SDA__M)
2422*4882a593Smuzhiyun 		ext_attr->hi_cfg_bridge_delay = SIO_HI_RA_RAM_PAR_3_CFG_DBL_SDA__M;
2423*4882a593Smuzhiyun 	/* SCL bridge delay, same as SDA for now */
2424*4882a593Smuzhiyun 	ext_attr->hi_cfg_bridge_delay += ((ext_attr->hi_cfg_bridge_delay) <<
2425*4882a593Smuzhiyun 				      SIO_HI_RA_RAM_PAR_3_CFG_DBL_SCL__B);
2426*4882a593Smuzhiyun 	/* Wakeup key, setting the read flag (as suggest in the documentation) does
2427*4882a593Smuzhiyun 	   not always result into a working solution (barebones worked VI2C failed).
2428*4882a593Smuzhiyun 	   Not setting the bit works in all cases . */
2429*4882a593Smuzhiyun 	ext_attr->hi_cfg_wake_up_key = DRXJ_WAKE_UP_KEY;
2430*4882a593Smuzhiyun 	/* port/bridge/power down ctrl */
2431*4882a593Smuzhiyun 	ext_attr->hi_cfg_ctrl = (SIO_HI_RA_RAM_PAR_5_CFG_SLV0_SLAVE);
2432*4882a593Smuzhiyun 	/* transit mode time out delay and watch dog divider */
2433*4882a593Smuzhiyun 	ext_attr->hi_cfg_transmit = SIO_HI_RA_RAM_PAR_6__PRE;
2434*4882a593Smuzhiyun 
2435*4882a593Smuzhiyun 	rc = hi_cfg_command(demod);
2436*4882a593Smuzhiyun 	if (rc != 0) {
2437*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2438*4882a593Smuzhiyun 		goto rw_error;
2439*4882a593Smuzhiyun 	}
2440*4882a593Smuzhiyun 
2441*4882a593Smuzhiyun 	return 0;
2442*4882a593Smuzhiyun 
2443*4882a593Smuzhiyun rw_error:
2444*4882a593Smuzhiyun 	return rc;
2445*4882a593Smuzhiyun }
2446*4882a593Smuzhiyun 
2447*4882a593Smuzhiyun /*============================================================================*/
2448*4882a593Smuzhiyun /*==                   END HOST INTERFACE FUNCTIONS                         ==*/
2449*4882a593Smuzhiyun /*============================================================================*/
2450*4882a593Smuzhiyun 
2451*4882a593Smuzhiyun /*============================================================================*/
2452*4882a593Smuzhiyun /*============================================================================*/
2453*4882a593Smuzhiyun /*==                        AUXILIARY FUNCTIONS                             ==*/
2454*4882a593Smuzhiyun /*============================================================================*/
2455*4882a593Smuzhiyun /*============================================================================*/
2456*4882a593Smuzhiyun 
2457*4882a593Smuzhiyun /*
2458*4882a593Smuzhiyun * \fn int get_device_capabilities()
2459*4882a593Smuzhiyun * \brief Get and store device capabilities.
2460*4882a593Smuzhiyun * \param demod  Pointer to demodulator instance.
2461*4882a593Smuzhiyun * \return int.
2462*4882a593Smuzhiyun * \return 0    Success
2463*4882a593Smuzhiyun * \retval -EIO Failure
2464*4882a593Smuzhiyun *
2465*4882a593Smuzhiyun * Depending on pulldowns on MDx pins the following internals are set:
2466*4882a593Smuzhiyun *  * common_attr->osc_clock_freq
2467*4882a593Smuzhiyun *  * ext_attr->has_lna
2468*4882a593Smuzhiyun *  * ext_attr->has_ntsc
2469*4882a593Smuzhiyun *  * ext_attr->has_btsc
2470*4882a593Smuzhiyun *  * ext_attr->has_oob
2471*4882a593Smuzhiyun *
2472*4882a593Smuzhiyun */
get_device_capabilities(struct drx_demod_instance * demod)2473*4882a593Smuzhiyun static int get_device_capabilities(struct drx_demod_instance *demod)
2474*4882a593Smuzhiyun {
2475*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = (struct drx_common_attr *) (NULL);
2476*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) NULL;
2477*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
2478*4882a593Smuzhiyun 	u16 sio_pdr_ohw_cfg = 0;
2479*4882a593Smuzhiyun 	u32 sio_top_jtagid_lo = 0;
2480*4882a593Smuzhiyun 	u16 bid = 0;
2481*4882a593Smuzhiyun 	int rc;
2482*4882a593Smuzhiyun 
2483*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
2484*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
2485*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
2486*4882a593Smuzhiyun 
2487*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
2488*4882a593Smuzhiyun 	if (rc != 0) {
2489*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2490*4882a593Smuzhiyun 		goto rw_error;
2491*4882a593Smuzhiyun 	}
2492*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, SIO_PDR_OHW_CFG__A, &sio_pdr_ohw_cfg, 0);
2493*4882a593Smuzhiyun 	if (rc != 0) {
2494*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2495*4882a593Smuzhiyun 		goto rw_error;
2496*4882a593Smuzhiyun 	}
2497*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY__PRE, 0);
2498*4882a593Smuzhiyun 	if (rc != 0) {
2499*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2500*4882a593Smuzhiyun 		goto rw_error;
2501*4882a593Smuzhiyun 	}
2502*4882a593Smuzhiyun 
2503*4882a593Smuzhiyun 	switch ((sio_pdr_ohw_cfg & SIO_PDR_OHW_CFG_FREF_SEL__M)) {
2504*4882a593Smuzhiyun 	case 0:
2505*4882a593Smuzhiyun 		/* ignore (bypass ?) */
2506*4882a593Smuzhiyun 		break;
2507*4882a593Smuzhiyun 	case 1:
2508*4882a593Smuzhiyun 		/* 27 MHz */
2509*4882a593Smuzhiyun 		common_attr->osc_clock_freq = 27000;
2510*4882a593Smuzhiyun 		break;
2511*4882a593Smuzhiyun 	case 2:
2512*4882a593Smuzhiyun 		/* 20.25 MHz */
2513*4882a593Smuzhiyun 		common_attr->osc_clock_freq = 20250;
2514*4882a593Smuzhiyun 		break;
2515*4882a593Smuzhiyun 	case 3:
2516*4882a593Smuzhiyun 		/* 4 MHz */
2517*4882a593Smuzhiyun 		common_attr->osc_clock_freq = 4000;
2518*4882a593Smuzhiyun 		break;
2519*4882a593Smuzhiyun 	default:
2520*4882a593Smuzhiyun 		return -EIO;
2521*4882a593Smuzhiyun 	}
2522*4882a593Smuzhiyun 
2523*4882a593Smuzhiyun 	/*
2524*4882a593Smuzhiyun 	   Determine device capabilities
2525*4882a593Smuzhiyun 	   Based on pinning v47
2526*4882a593Smuzhiyun 	 */
2527*4882a593Smuzhiyun 	rc = drxdap_fasi_read_reg32(dev_addr, SIO_TOP_JTAGID_LO__A, &sio_top_jtagid_lo, 0);
2528*4882a593Smuzhiyun 	if (rc != 0) {
2529*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
2530*4882a593Smuzhiyun 		goto rw_error;
2531*4882a593Smuzhiyun 	}
2532*4882a593Smuzhiyun 	ext_attr->mfx = (u8) ((sio_top_jtagid_lo >> 29) & 0xF);
2533*4882a593Smuzhiyun 
2534*4882a593Smuzhiyun 	switch ((sio_top_jtagid_lo >> 12) & 0xFF) {
2535*4882a593Smuzhiyun 	case 0x31:
2536*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
2537*4882a593Smuzhiyun 		if (rc != 0) {
2538*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2539*4882a593Smuzhiyun 			goto rw_error;
2540*4882a593Smuzhiyun 		}
2541*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, SIO_PDR_UIO_IN_HI__A, &bid, 0);
2542*4882a593Smuzhiyun 		if (rc != 0) {
2543*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2544*4882a593Smuzhiyun 			goto rw_error;
2545*4882a593Smuzhiyun 		}
2546*4882a593Smuzhiyun 		bid = (bid >> 10) & 0xf;
2547*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY__PRE, 0);
2548*4882a593Smuzhiyun 		if (rc != 0) {
2549*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2550*4882a593Smuzhiyun 			goto rw_error;
2551*4882a593Smuzhiyun 		}
2552*4882a593Smuzhiyun 
2553*4882a593Smuzhiyun 		ext_attr->has_lna = true;
2554*4882a593Smuzhiyun 		ext_attr->has_ntsc = false;
2555*4882a593Smuzhiyun 		ext_attr->has_btsc = false;
2556*4882a593Smuzhiyun 		ext_attr->has_oob = false;
2557*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2558*4882a593Smuzhiyun 		ext_attr->has_smarx = false;
2559*4882a593Smuzhiyun 		ext_attr->has_gpio = false;
2560*4882a593Smuzhiyun 		ext_attr->has_irqn = false;
2561*4882a593Smuzhiyun 		break;
2562*4882a593Smuzhiyun 	case 0x33:
2563*4882a593Smuzhiyun 		ext_attr->has_lna = false;
2564*4882a593Smuzhiyun 		ext_attr->has_ntsc = false;
2565*4882a593Smuzhiyun 		ext_attr->has_btsc = false;
2566*4882a593Smuzhiyun 		ext_attr->has_oob = false;
2567*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2568*4882a593Smuzhiyun 		ext_attr->has_smarx = false;
2569*4882a593Smuzhiyun 		ext_attr->has_gpio = false;
2570*4882a593Smuzhiyun 		ext_attr->has_irqn = false;
2571*4882a593Smuzhiyun 		break;
2572*4882a593Smuzhiyun 	case 0x45:
2573*4882a593Smuzhiyun 		ext_attr->has_lna = true;
2574*4882a593Smuzhiyun 		ext_attr->has_ntsc = true;
2575*4882a593Smuzhiyun 		ext_attr->has_btsc = false;
2576*4882a593Smuzhiyun 		ext_attr->has_oob = false;
2577*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2578*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2579*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2580*4882a593Smuzhiyun 		ext_attr->has_irqn = false;
2581*4882a593Smuzhiyun 		break;
2582*4882a593Smuzhiyun 	case 0x46:
2583*4882a593Smuzhiyun 		ext_attr->has_lna = false;
2584*4882a593Smuzhiyun 		ext_attr->has_ntsc = true;
2585*4882a593Smuzhiyun 		ext_attr->has_btsc = false;
2586*4882a593Smuzhiyun 		ext_attr->has_oob = false;
2587*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2588*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2589*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2590*4882a593Smuzhiyun 		ext_attr->has_irqn = false;
2591*4882a593Smuzhiyun 		break;
2592*4882a593Smuzhiyun 	case 0x41:
2593*4882a593Smuzhiyun 		ext_attr->has_lna = true;
2594*4882a593Smuzhiyun 		ext_attr->has_ntsc = true;
2595*4882a593Smuzhiyun 		ext_attr->has_btsc = true;
2596*4882a593Smuzhiyun 		ext_attr->has_oob = false;
2597*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2598*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2599*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2600*4882a593Smuzhiyun 		ext_attr->has_irqn = false;
2601*4882a593Smuzhiyun 		break;
2602*4882a593Smuzhiyun 	case 0x43:
2603*4882a593Smuzhiyun 		ext_attr->has_lna = false;
2604*4882a593Smuzhiyun 		ext_attr->has_ntsc = true;
2605*4882a593Smuzhiyun 		ext_attr->has_btsc = true;
2606*4882a593Smuzhiyun 		ext_attr->has_oob = false;
2607*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2608*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2609*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2610*4882a593Smuzhiyun 		ext_attr->has_irqn = false;
2611*4882a593Smuzhiyun 		break;
2612*4882a593Smuzhiyun 	case 0x32:
2613*4882a593Smuzhiyun 		ext_attr->has_lna = true;
2614*4882a593Smuzhiyun 		ext_attr->has_ntsc = false;
2615*4882a593Smuzhiyun 		ext_attr->has_btsc = false;
2616*4882a593Smuzhiyun 		ext_attr->has_oob = true;
2617*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2618*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2619*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2620*4882a593Smuzhiyun 		ext_attr->has_irqn = true;
2621*4882a593Smuzhiyun 		break;
2622*4882a593Smuzhiyun 	case 0x34:
2623*4882a593Smuzhiyun 		ext_attr->has_lna = false;
2624*4882a593Smuzhiyun 		ext_attr->has_ntsc = true;
2625*4882a593Smuzhiyun 		ext_attr->has_btsc = true;
2626*4882a593Smuzhiyun 		ext_attr->has_oob = true;
2627*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2628*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2629*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2630*4882a593Smuzhiyun 		ext_attr->has_irqn = true;
2631*4882a593Smuzhiyun 		break;
2632*4882a593Smuzhiyun 	case 0x42:
2633*4882a593Smuzhiyun 		ext_attr->has_lna = true;
2634*4882a593Smuzhiyun 		ext_attr->has_ntsc = true;
2635*4882a593Smuzhiyun 		ext_attr->has_btsc = true;
2636*4882a593Smuzhiyun 		ext_attr->has_oob = true;
2637*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2638*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2639*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2640*4882a593Smuzhiyun 		ext_attr->has_irqn = true;
2641*4882a593Smuzhiyun 		break;
2642*4882a593Smuzhiyun 	case 0x44:
2643*4882a593Smuzhiyun 		ext_attr->has_lna = false;
2644*4882a593Smuzhiyun 		ext_attr->has_ntsc = true;
2645*4882a593Smuzhiyun 		ext_attr->has_btsc = true;
2646*4882a593Smuzhiyun 		ext_attr->has_oob = true;
2647*4882a593Smuzhiyun 		ext_attr->has_smatx = true;
2648*4882a593Smuzhiyun 		ext_attr->has_smarx = true;
2649*4882a593Smuzhiyun 		ext_attr->has_gpio = true;
2650*4882a593Smuzhiyun 		ext_attr->has_irqn = true;
2651*4882a593Smuzhiyun 		break;
2652*4882a593Smuzhiyun 	default:
2653*4882a593Smuzhiyun 		/* Unknown device variant */
2654*4882a593Smuzhiyun 		return -EIO;
2655*4882a593Smuzhiyun 		break;
2656*4882a593Smuzhiyun 	}
2657*4882a593Smuzhiyun 
2658*4882a593Smuzhiyun 	return 0;
2659*4882a593Smuzhiyun rw_error:
2660*4882a593Smuzhiyun 	return rc;
2661*4882a593Smuzhiyun }
2662*4882a593Smuzhiyun 
2663*4882a593Smuzhiyun /*
2664*4882a593Smuzhiyun * \fn int power_up_device()
2665*4882a593Smuzhiyun * \brief Power up device.
2666*4882a593Smuzhiyun * \param demod  Pointer to demodulator instance.
2667*4882a593Smuzhiyun * \return int.
2668*4882a593Smuzhiyun * \return 0    Success
2669*4882a593Smuzhiyun * \retval -EIO Failure, I2C or max retries reached
2670*4882a593Smuzhiyun *
2671*4882a593Smuzhiyun */
2672*4882a593Smuzhiyun 
2673*4882a593Smuzhiyun #ifndef DRXJ_MAX_RETRIES_POWERUP
2674*4882a593Smuzhiyun #define DRXJ_MAX_RETRIES_POWERUP 10
2675*4882a593Smuzhiyun #endif
2676*4882a593Smuzhiyun 
power_up_device(struct drx_demod_instance * demod)2677*4882a593Smuzhiyun static int power_up_device(struct drx_demod_instance *demod)
2678*4882a593Smuzhiyun {
2679*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
2680*4882a593Smuzhiyun 	u8 data = 0;
2681*4882a593Smuzhiyun 	u16 retry_count = 0;
2682*4882a593Smuzhiyun 	struct i2c_device_addr wake_up_addr;
2683*4882a593Smuzhiyun 
2684*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
2685*4882a593Smuzhiyun 	wake_up_addr.i2c_addr = DRXJ_WAKE_UP_KEY;
2686*4882a593Smuzhiyun 	wake_up_addr.i2c_dev_id = dev_addr->i2c_dev_id;
2687*4882a593Smuzhiyun 	wake_up_addr.user_data = dev_addr->user_data;
2688*4882a593Smuzhiyun 	/*
2689*4882a593Smuzhiyun 	 * I2C access may fail in this case: no ack
2690*4882a593Smuzhiyun 	 * dummy write must be used to wake uop device, dummy read must be used to
2691*4882a593Smuzhiyun 	 * reset HI state machine (avoiding actual writes)
2692*4882a593Smuzhiyun 	 */
2693*4882a593Smuzhiyun 	do {
2694*4882a593Smuzhiyun 		data = 0;
2695*4882a593Smuzhiyun 		drxbsp_i2c_write_read(&wake_up_addr, 1, &data,
2696*4882a593Smuzhiyun 				      (struct i2c_device_addr *)(NULL), 0,
2697*4882a593Smuzhiyun 				     (u8 *)(NULL));
2698*4882a593Smuzhiyun 		msleep(10);
2699*4882a593Smuzhiyun 		retry_count++;
2700*4882a593Smuzhiyun 	} while ((drxbsp_i2c_write_read
2701*4882a593Smuzhiyun 		  ((struct i2c_device_addr *) (NULL), 0, (u8 *)(NULL), dev_addr, 1,
2702*4882a593Smuzhiyun 		   &data)
2703*4882a593Smuzhiyun 		  != 0) && (retry_count < DRXJ_MAX_RETRIES_POWERUP));
2704*4882a593Smuzhiyun 
2705*4882a593Smuzhiyun 	/* Need some recovery time .... */
2706*4882a593Smuzhiyun 	msleep(10);
2707*4882a593Smuzhiyun 
2708*4882a593Smuzhiyun 	if (retry_count == DRXJ_MAX_RETRIES_POWERUP)
2709*4882a593Smuzhiyun 		return -EIO;
2710*4882a593Smuzhiyun 
2711*4882a593Smuzhiyun 	return 0;
2712*4882a593Smuzhiyun }
2713*4882a593Smuzhiyun 
2714*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
2715*4882a593Smuzhiyun /* MPEG Output Configuration Functions - begin                                */
2716*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
2717*4882a593Smuzhiyun /*
2718*4882a593Smuzhiyun * \fn int ctrl_set_cfg_mpeg_output()
2719*4882a593Smuzhiyun * \brief Set MPEG output configuration of the device.
2720*4882a593Smuzhiyun * \param devmod  Pointer to demodulator instance.
2721*4882a593Smuzhiyun * \param cfg_data Pointer to mpeg output configuaration.
2722*4882a593Smuzhiyun * \return int.
2723*4882a593Smuzhiyun *
2724*4882a593Smuzhiyun *  Configure MPEG output parameters.
2725*4882a593Smuzhiyun *
2726*4882a593Smuzhiyun */
2727*4882a593Smuzhiyun static int
ctrl_set_cfg_mpeg_output(struct drx_demod_instance * demod,struct drx_cfg_mpeg_output * cfg_data)2728*4882a593Smuzhiyun ctrl_set_cfg_mpeg_output(struct drx_demod_instance *demod, struct drx_cfg_mpeg_output *cfg_data)
2729*4882a593Smuzhiyun {
2730*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
2731*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
2732*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = (struct drx_common_attr *) (NULL);
2733*4882a593Smuzhiyun 	int rc;
2734*4882a593Smuzhiyun 	u16 fec_oc_reg_mode = 0;
2735*4882a593Smuzhiyun 	u16 fec_oc_reg_ipr_mode = 0;
2736*4882a593Smuzhiyun 	u16 fec_oc_reg_ipr_invert = 0;
2737*4882a593Smuzhiyun 	u32 max_bit_rate = 0;
2738*4882a593Smuzhiyun 	u32 rcn_rate = 0;
2739*4882a593Smuzhiyun 	u32 nr_bits = 0;
2740*4882a593Smuzhiyun 	u16 sio_pdr_md_cfg = 0;
2741*4882a593Smuzhiyun 	/* data mask for the output data byte */
2742*4882a593Smuzhiyun 	u16 invert_data_mask =
2743*4882a593Smuzhiyun 	    FEC_OC_IPR_INVERT_MD7__M | FEC_OC_IPR_INVERT_MD6__M |
2744*4882a593Smuzhiyun 	    FEC_OC_IPR_INVERT_MD5__M | FEC_OC_IPR_INVERT_MD4__M |
2745*4882a593Smuzhiyun 	    FEC_OC_IPR_INVERT_MD3__M | FEC_OC_IPR_INVERT_MD2__M |
2746*4882a593Smuzhiyun 	    FEC_OC_IPR_INVERT_MD1__M | FEC_OC_IPR_INVERT_MD0__M;
2747*4882a593Smuzhiyun 
2748*4882a593Smuzhiyun 	/* check arguments */
2749*4882a593Smuzhiyun 	if ((demod == NULL) || (cfg_data == NULL))
2750*4882a593Smuzhiyun 		return -EINVAL;
2751*4882a593Smuzhiyun 
2752*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
2753*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
2754*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
2755*4882a593Smuzhiyun 
2756*4882a593Smuzhiyun 	if (cfg_data->enable_mpeg_output == true) {
2757*4882a593Smuzhiyun 		/* quick and dirty patch to set MPEG in case current std is not
2758*4882a593Smuzhiyun 		   producing MPEG */
2759*4882a593Smuzhiyun 		switch (ext_attr->standard) {
2760*4882a593Smuzhiyun 		case DRX_STANDARD_8VSB:
2761*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_A:
2762*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_B:
2763*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_C:
2764*4882a593Smuzhiyun 			break;
2765*4882a593Smuzhiyun 		default:
2766*4882a593Smuzhiyun 			return 0;
2767*4882a593Smuzhiyun 		}
2768*4882a593Smuzhiyun 
2769*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_OC_OCR_INVERT__A, 0, 0);
2770*4882a593Smuzhiyun 		if (rc != 0) {
2771*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2772*4882a593Smuzhiyun 			goto rw_error;
2773*4882a593Smuzhiyun 		}
2774*4882a593Smuzhiyun 		switch (ext_attr->standard) {
2775*4882a593Smuzhiyun 		case DRX_STANDARD_8VSB:
2776*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_USAGE__A, 7, 0);
2777*4882a593Smuzhiyun 			if (rc != 0) {
2778*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2779*4882a593Smuzhiyun 				goto rw_error;
2780*4882a593Smuzhiyun 			}	/* 2048 bytes fifo ram */
2781*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_CTL_UPD_RATE__A, 10, 0);
2782*4882a593Smuzhiyun 			if (rc != 0) {
2783*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2784*4882a593Smuzhiyun 				goto rw_error;
2785*4882a593Smuzhiyun 			}
2786*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_INT_UPD_RATE__A, 10, 0);
2787*4882a593Smuzhiyun 			if (rc != 0) {
2788*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2789*4882a593Smuzhiyun 				goto rw_error;
2790*4882a593Smuzhiyun 			}
2791*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_A__A, 5, 0);
2792*4882a593Smuzhiyun 			if (rc != 0) {
2793*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2794*4882a593Smuzhiyun 				goto rw_error;
2795*4882a593Smuzhiyun 			}
2796*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_B__A, 7, 0);
2797*4882a593Smuzhiyun 			if (rc != 0) {
2798*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2799*4882a593Smuzhiyun 				goto rw_error;
2800*4882a593Smuzhiyun 			}
2801*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_RCN_GAIN__A, 10, 0);
2802*4882a593Smuzhiyun 			if (rc != 0) {
2803*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2804*4882a593Smuzhiyun 				goto rw_error;
2805*4882a593Smuzhiyun 			}
2806*4882a593Smuzhiyun 			/* Low Water Mark for synchronization  */
2807*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_LWM__A, 3, 0);
2808*4882a593Smuzhiyun 			if (rc != 0) {
2809*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2810*4882a593Smuzhiyun 				goto rw_error;
2811*4882a593Smuzhiyun 			}
2812*4882a593Smuzhiyun 			/* High Water Mark for synchronization */
2813*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_HWM__A, 5, 0);
2814*4882a593Smuzhiyun 			if (rc != 0) {
2815*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2816*4882a593Smuzhiyun 				goto rw_error;
2817*4882a593Smuzhiyun 			}
2818*4882a593Smuzhiyun 			break;
2819*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_A:
2820*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_C:
2821*4882a593Smuzhiyun 			switch (ext_attr->constellation) {
2822*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM256:
2823*4882a593Smuzhiyun 				nr_bits = 8;
2824*4882a593Smuzhiyun 				break;
2825*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM128:
2826*4882a593Smuzhiyun 				nr_bits = 7;
2827*4882a593Smuzhiyun 				break;
2828*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM64:
2829*4882a593Smuzhiyun 				nr_bits = 6;
2830*4882a593Smuzhiyun 				break;
2831*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM32:
2832*4882a593Smuzhiyun 				nr_bits = 5;
2833*4882a593Smuzhiyun 				break;
2834*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM16:
2835*4882a593Smuzhiyun 				nr_bits = 4;
2836*4882a593Smuzhiyun 				break;
2837*4882a593Smuzhiyun 			default:
2838*4882a593Smuzhiyun 				return -EIO;
2839*4882a593Smuzhiyun 			}	/* ext_attr->constellation */
2840*4882a593Smuzhiyun 			/* max_bit_rate = symbol_rate * nr_bits * coef */
2841*4882a593Smuzhiyun 			/* coef = 188/204                          */
2842*4882a593Smuzhiyun 			max_bit_rate =
2843*4882a593Smuzhiyun 			    (ext_attr->curr_symbol_rate / 8) * nr_bits * 188;
2844*4882a593Smuzhiyun 			fallthrough;	/* as b/c Annex A/C need following settings */
2845*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_B:
2846*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_USAGE__A, FEC_OC_FCT_USAGE__PRE, 0);
2847*4882a593Smuzhiyun 			if (rc != 0) {
2848*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2849*4882a593Smuzhiyun 				goto rw_error;
2850*4882a593Smuzhiyun 			}
2851*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_CTL_UPD_RATE__A, FEC_OC_TMD_CTL_UPD_RATE__PRE, 0);
2852*4882a593Smuzhiyun 			if (rc != 0) {
2853*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2854*4882a593Smuzhiyun 				goto rw_error;
2855*4882a593Smuzhiyun 			}
2856*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_INT_UPD_RATE__A, 5, 0);
2857*4882a593Smuzhiyun 			if (rc != 0) {
2858*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2859*4882a593Smuzhiyun 				goto rw_error;
2860*4882a593Smuzhiyun 			}
2861*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_A__A, FEC_OC_AVR_PARM_A__PRE, 0);
2862*4882a593Smuzhiyun 			if (rc != 0) {
2863*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2864*4882a593Smuzhiyun 				goto rw_error;
2865*4882a593Smuzhiyun 			}
2866*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_B__A, FEC_OC_AVR_PARM_B__PRE, 0);
2867*4882a593Smuzhiyun 			if (rc != 0) {
2868*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2869*4882a593Smuzhiyun 				goto rw_error;
2870*4882a593Smuzhiyun 			}
2871*4882a593Smuzhiyun 			if (cfg_data->static_clk == true) {
2872*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, FEC_OC_RCN_GAIN__A, 0xD, 0);
2873*4882a593Smuzhiyun 				if (rc != 0) {
2874*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
2875*4882a593Smuzhiyun 					goto rw_error;
2876*4882a593Smuzhiyun 				}
2877*4882a593Smuzhiyun 			} else {
2878*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, FEC_OC_RCN_GAIN__A, FEC_OC_RCN_GAIN__PRE, 0);
2879*4882a593Smuzhiyun 				if (rc != 0) {
2880*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
2881*4882a593Smuzhiyun 					goto rw_error;
2882*4882a593Smuzhiyun 				}
2883*4882a593Smuzhiyun 			}
2884*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_LWM__A, 2, 0);
2885*4882a593Smuzhiyun 			if (rc != 0) {
2886*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2887*4882a593Smuzhiyun 				goto rw_error;
2888*4882a593Smuzhiyun 			}
2889*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_HWM__A, 12, 0);
2890*4882a593Smuzhiyun 			if (rc != 0) {
2891*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
2892*4882a593Smuzhiyun 				goto rw_error;
2893*4882a593Smuzhiyun 			}
2894*4882a593Smuzhiyun 			break;
2895*4882a593Smuzhiyun 		default:
2896*4882a593Smuzhiyun 			break;
2897*4882a593Smuzhiyun 		}		/* switch (standard) */
2898*4882a593Smuzhiyun 
2899*4882a593Smuzhiyun 		/* Check insertion of the Reed-Solomon parity bytes */
2900*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, FEC_OC_MODE__A, &fec_oc_reg_mode, 0);
2901*4882a593Smuzhiyun 		if (rc != 0) {
2902*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2903*4882a593Smuzhiyun 			goto rw_error;
2904*4882a593Smuzhiyun 		}
2905*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, FEC_OC_IPR_MODE__A, &fec_oc_reg_ipr_mode, 0);
2906*4882a593Smuzhiyun 		if (rc != 0) {
2907*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
2908*4882a593Smuzhiyun 			goto rw_error;
2909*4882a593Smuzhiyun 		}
2910*4882a593Smuzhiyun 		if (cfg_data->insert_rs_byte == true) {
2911*4882a593Smuzhiyun 			/* enable parity symbol forward */
2912*4882a593Smuzhiyun 			fec_oc_reg_mode |= FEC_OC_MODE_PARITY__M;
2913*4882a593Smuzhiyun 			/* MVAL disable during parity bytes */
2914*4882a593Smuzhiyun 			fec_oc_reg_ipr_mode |= FEC_OC_IPR_MODE_MVAL_DIS_PAR__M;
2915*4882a593Smuzhiyun 			switch (ext_attr->standard) {
2916*4882a593Smuzhiyun 			case DRX_STANDARD_8VSB:
2917*4882a593Smuzhiyun 				rcn_rate = 0x004854D3;
2918*4882a593Smuzhiyun 				break;
2919*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_B:
2920*4882a593Smuzhiyun 				fec_oc_reg_mode |= FEC_OC_MODE_TRANSPARENT__M;
2921*4882a593Smuzhiyun 				switch (ext_attr->constellation) {
2922*4882a593Smuzhiyun 				case DRX_CONSTELLATION_QAM256:
2923*4882a593Smuzhiyun 					rcn_rate = 0x008945E7;
2924*4882a593Smuzhiyun 					break;
2925*4882a593Smuzhiyun 				case DRX_CONSTELLATION_QAM64:
2926*4882a593Smuzhiyun 					rcn_rate = 0x005F64D4;
2927*4882a593Smuzhiyun 					break;
2928*4882a593Smuzhiyun 				default:
2929*4882a593Smuzhiyun 					return -EIO;
2930*4882a593Smuzhiyun 				}
2931*4882a593Smuzhiyun 				break;
2932*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_A:
2933*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_C:
2934*4882a593Smuzhiyun 				/* insert_rs_byte = true -> coef = 188/188 -> 1, RS bits are in MPEG output */
2935*4882a593Smuzhiyun 				rcn_rate =
2936*4882a593Smuzhiyun 				    (frac28
2937*4882a593Smuzhiyun 				     (max_bit_rate,
2938*4882a593Smuzhiyun 				      (u32) (common_attr->sys_clock_freq / 8))) /
2939*4882a593Smuzhiyun 				    188;
2940*4882a593Smuzhiyun 				break;
2941*4882a593Smuzhiyun 			default:
2942*4882a593Smuzhiyun 				return -EIO;
2943*4882a593Smuzhiyun 			}	/* ext_attr->standard */
2944*4882a593Smuzhiyun 		} else {	/* insert_rs_byte == false */
2945*4882a593Smuzhiyun 
2946*4882a593Smuzhiyun 			/* disable parity symbol forward */
2947*4882a593Smuzhiyun 			fec_oc_reg_mode &= (~FEC_OC_MODE_PARITY__M);
2948*4882a593Smuzhiyun 			/* MVAL enable during parity bytes */
2949*4882a593Smuzhiyun 			fec_oc_reg_ipr_mode &= (~FEC_OC_IPR_MODE_MVAL_DIS_PAR__M);
2950*4882a593Smuzhiyun 			switch (ext_attr->standard) {
2951*4882a593Smuzhiyun 			case DRX_STANDARD_8VSB:
2952*4882a593Smuzhiyun 				rcn_rate = 0x0041605C;
2953*4882a593Smuzhiyun 				break;
2954*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_B:
2955*4882a593Smuzhiyun 				fec_oc_reg_mode &= (~FEC_OC_MODE_TRANSPARENT__M);
2956*4882a593Smuzhiyun 				switch (ext_attr->constellation) {
2957*4882a593Smuzhiyun 				case DRX_CONSTELLATION_QAM256:
2958*4882a593Smuzhiyun 					rcn_rate = 0x0082D6A0;
2959*4882a593Smuzhiyun 					break;
2960*4882a593Smuzhiyun 				case DRX_CONSTELLATION_QAM64:
2961*4882a593Smuzhiyun 					rcn_rate = 0x005AEC1A;
2962*4882a593Smuzhiyun 					break;
2963*4882a593Smuzhiyun 				default:
2964*4882a593Smuzhiyun 					return -EIO;
2965*4882a593Smuzhiyun 				}
2966*4882a593Smuzhiyun 				break;
2967*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_A:
2968*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_C:
2969*4882a593Smuzhiyun 				/* insert_rs_byte = false -> coef = 188/204, RS bits not in MPEG output */
2970*4882a593Smuzhiyun 				rcn_rate =
2971*4882a593Smuzhiyun 				    (frac28
2972*4882a593Smuzhiyun 				     (max_bit_rate,
2973*4882a593Smuzhiyun 				      (u32) (common_attr->sys_clock_freq / 8))) /
2974*4882a593Smuzhiyun 				    204;
2975*4882a593Smuzhiyun 				break;
2976*4882a593Smuzhiyun 			default:
2977*4882a593Smuzhiyun 				return -EIO;
2978*4882a593Smuzhiyun 			}	/* ext_attr->standard */
2979*4882a593Smuzhiyun 		}
2980*4882a593Smuzhiyun 
2981*4882a593Smuzhiyun 		if (cfg_data->enable_parallel == true) {	/* MPEG data output is parallel -> clear ipr_mode[0] */
2982*4882a593Smuzhiyun 			fec_oc_reg_ipr_mode &= (~(FEC_OC_IPR_MODE_SERIAL__M));
2983*4882a593Smuzhiyun 		} else {	/* MPEG data output is serial -> set ipr_mode[0] */
2984*4882a593Smuzhiyun 			fec_oc_reg_ipr_mode |= FEC_OC_IPR_MODE_SERIAL__M;
2985*4882a593Smuzhiyun 		}
2986*4882a593Smuzhiyun 
2987*4882a593Smuzhiyun 		/* Control slective inversion of output bits */
2988*4882a593Smuzhiyun 		if (cfg_data->invert_data == true)
2989*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert |= invert_data_mask;
2990*4882a593Smuzhiyun 		else
2991*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert &= (~(invert_data_mask));
2992*4882a593Smuzhiyun 
2993*4882a593Smuzhiyun 		if (cfg_data->invert_err == true)
2994*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MERR__M;
2995*4882a593Smuzhiyun 		else
2996*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MERR__M));
2997*4882a593Smuzhiyun 
2998*4882a593Smuzhiyun 		if (cfg_data->invert_str == true)
2999*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MSTRT__M;
3000*4882a593Smuzhiyun 		else
3001*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MSTRT__M));
3002*4882a593Smuzhiyun 
3003*4882a593Smuzhiyun 		if (cfg_data->invert_val == true)
3004*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MVAL__M;
3005*4882a593Smuzhiyun 		else
3006*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MVAL__M));
3007*4882a593Smuzhiyun 
3008*4882a593Smuzhiyun 		if (cfg_data->invert_clk == true)
3009*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MCLK__M;
3010*4882a593Smuzhiyun 		else
3011*4882a593Smuzhiyun 			fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MCLK__M));
3012*4882a593Smuzhiyun 
3013*4882a593Smuzhiyun 
3014*4882a593Smuzhiyun 		if (cfg_data->static_clk == true) {	/* Static mode */
3015*4882a593Smuzhiyun 			u32 dto_rate = 0;
3016*4882a593Smuzhiyun 			u32 bit_rate = 0;
3017*4882a593Smuzhiyun 			u16 fec_oc_dto_burst_len = 0;
3018*4882a593Smuzhiyun 			u16 fec_oc_dto_period = 0;
3019*4882a593Smuzhiyun 
3020*4882a593Smuzhiyun 			fec_oc_dto_burst_len = FEC_OC_DTO_BURST_LEN__PRE;
3021*4882a593Smuzhiyun 
3022*4882a593Smuzhiyun 			switch (ext_attr->standard) {
3023*4882a593Smuzhiyun 			case DRX_STANDARD_8VSB:
3024*4882a593Smuzhiyun 				fec_oc_dto_period = 4;
3025*4882a593Smuzhiyun 				if (cfg_data->insert_rs_byte == true)
3026*4882a593Smuzhiyun 					fec_oc_dto_burst_len = 208;
3027*4882a593Smuzhiyun 				break;
3028*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_A:
3029*4882a593Smuzhiyun 				{
3030*4882a593Smuzhiyun 					u32 symbol_rate_th = 6400000;
3031*4882a593Smuzhiyun 					if (cfg_data->insert_rs_byte == true) {
3032*4882a593Smuzhiyun 						fec_oc_dto_burst_len = 204;
3033*4882a593Smuzhiyun 						symbol_rate_th = 5900000;
3034*4882a593Smuzhiyun 					}
3035*4882a593Smuzhiyun 					if (ext_attr->curr_symbol_rate >=
3036*4882a593Smuzhiyun 					    symbol_rate_th) {
3037*4882a593Smuzhiyun 						fec_oc_dto_period = 0;
3038*4882a593Smuzhiyun 					} else {
3039*4882a593Smuzhiyun 						fec_oc_dto_period = 1;
3040*4882a593Smuzhiyun 					}
3041*4882a593Smuzhiyun 				}
3042*4882a593Smuzhiyun 				break;
3043*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_B:
3044*4882a593Smuzhiyun 				fec_oc_dto_period = 1;
3045*4882a593Smuzhiyun 				if (cfg_data->insert_rs_byte == true)
3046*4882a593Smuzhiyun 					fec_oc_dto_burst_len = 128;
3047*4882a593Smuzhiyun 				break;
3048*4882a593Smuzhiyun 			case DRX_STANDARD_ITU_C:
3049*4882a593Smuzhiyun 				fec_oc_dto_period = 1;
3050*4882a593Smuzhiyun 				if (cfg_data->insert_rs_byte == true)
3051*4882a593Smuzhiyun 					fec_oc_dto_burst_len = 204;
3052*4882a593Smuzhiyun 				break;
3053*4882a593Smuzhiyun 			default:
3054*4882a593Smuzhiyun 				return -EIO;
3055*4882a593Smuzhiyun 			}
3056*4882a593Smuzhiyun 			bit_rate =
3057*4882a593Smuzhiyun 			    common_attr->sys_clock_freq * 1000 / (fec_oc_dto_period +
3058*4882a593Smuzhiyun 							       2);
3059*4882a593Smuzhiyun 			dto_rate =
3060*4882a593Smuzhiyun 			    frac28(bit_rate, common_attr->sys_clock_freq * 1000);
3061*4882a593Smuzhiyun 			dto_rate >>= 3;
3062*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_RATE_HI__A, (u16)((dto_rate >> 16) & FEC_OC_DTO_RATE_HI__M), 0);
3063*4882a593Smuzhiyun 			if (rc != 0) {
3064*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3065*4882a593Smuzhiyun 				goto rw_error;
3066*4882a593Smuzhiyun 			}
3067*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_RATE_LO__A, (u16)(dto_rate & FEC_OC_DTO_RATE_LO_RATE_LO__M), 0);
3068*4882a593Smuzhiyun 			if (rc != 0) {
3069*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3070*4882a593Smuzhiyun 				goto rw_error;
3071*4882a593Smuzhiyun 			}
3072*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_MODE__A, FEC_OC_DTO_MODE_DYNAMIC__M | FEC_OC_DTO_MODE_OFFSET_ENABLE__M, 0);
3073*4882a593Smuzhiyun 			if (rc != 0) {
3074*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3075*4882a593Smuzhiyun 				goto rw_error;
3076*4882a593Smuzhiyun 			}
3077*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_MODE__A, FEC_OC_FCT_MODE_RAT_ENA__M | FEC_OC_FCT_MODE_VIRT_ENA__M, 0);
3078*4882a593Smuzhiyun 			if (rc != 0) {
3079*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3080*4882a593Smuzhiyun 				goto rw_error;
3081*4882a593Smuzhiyun 			}
3082*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_BURST_LEN__A, fec_oc_dto_burst_len, 0);
3083*4882a593Smuzhiyun 			if (rc != 0) {
3084*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3085*4882a593Smuzhiyun 				goto rw_error;
3086*4882a593Smuzhiyun 			}
3087*4882a593Smuzhiyun 			if (ext_attr->mpeg_output_clock_rate != DRXJ_MPEGOUTPUT_CLOCK_RATE_AUTO)
3088*4882a593Smuzhiyun 				fec_oc_dto_period = ext_attr->mpeg_output_clock_rate - 1;
3089*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_PERIOD__A, fec_oc_dto_period, 0);
3090*4882a593Smuzhiyun 			if (rc != 0) {
3091*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3092*4882a593Smuzhiyun 				goto rw_error;
3093*4882a593Smuzhiyun 			}
3094*4882a593Smuzhiyun 		} else {	/* Dynamic mode */
3095*4882a593Smuzhiyun 
3096*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_MODE__A, FEC_OC_DTO_MODE_DYNAMIC__M, 0);
3097*4882a593Smuzhiyun 			if (rc != 0) {
3098*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3099*4882a593Smuzhiyun 				goto rw_error;
3100*4882a593Smuzhiyun 			}
3101*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_MODE__A, 0, 0);
3102*4882a593Smuzhiyun 			if (rc != 0) {
3103*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3104*4882a593Smuzhiyun 				goto rw_error;
3105*4882a593Smuzhiyun 			}
3106*4882a593Smuzhiyun 		}
3107*4882a593Smuzhiyun 
3108*4882a593Smuzhiyun 		rc = drxdap_fasi_write_reg32(dev_addr, FEC_OC_RCN_CTL_RATE_LO__A, rcn_rate, 0);
3109*4882a593Smuzhiyun 		if (rc != 0) {
3110*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3111*4882a593Smuzhiyun 			goto rw_error;
3112*4882a593Smuzhiyun 		}
3113*4882a593Smuzhiyun 
3114*4882a593Smuzhiyun 		/* Write appropriate registers with requested configuration */
3115*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_OC_MODE__A, fec_oc_reg_mode, 0);
3116*4882a593Smuzhiyun 		if (rc != 0) {
3117*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3118*4882a593Smuzhiyun 			goto rw_error;
3119*4882a593Smuzhiyun 		}
3120*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_OC_IPR_MODE__A, fec_oc_reg_ipr_mode, 0);
3121*4882a593Smuzhiyun 		if (rc != 0) {
3122*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3123*4882a593Smuzhiyun 			goto rw_error;
3124*4882a593Smuzhiyun 		}
3125*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_OC_IPR_INVERT__A, fec_oc_reg_ipr_invert, 0);
3126*4882a593Smuzhiyun 		if (rc != 0) {
3127*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3128*4882a593Smuzhiyun 			goto rw_error;
3129*4882a593Smuzhiyun 		}
3130*4882a593Smuzhiyun 
3131*4882a593Smuzhiyun 		/* enabling for both parallel and serial now */
3132*4882a593Smuzhiyun 		/*  Write magic word to enable pdr reg write */
3133*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0xFABA, 0);
3134*4882a593Smuzhiyun 		if (rc != 0) {
3135*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3136*4882a593Smuzhiyun 			goto rw_error;
3137*4882a593Smuzhiyun 		}
3138*4882a593Smuzhiyun 		/*  Set MPEG TS pads to outputmode */
3139*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MSTRT_CFG__A, 0x0013, 0);
3140*4882a593Smuzhiyun 		if (rc != 0) {
3141*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3142*4882a593Smuzhiyun 			goto rw_error;
3143*4882a593Smuzhiyun 		}
3144*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MERR_CFG__A, 0x0013, 0);
3145*4882a593Smuzhiyun 		if (rc != 0) {
3146*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3147*4882a593Smuzhiyun 			goto rw_error;
3148*4882a593Smuzhiyun 		}
3149*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MCLK_CFG__A, MPEG_OUTPUT_CLK_DRIVE_STRENGTH << SIO_PDR_MCLK_CFG_DRIVE__B | 0x03 << SIO_PDR_MCLK_CFG_MODE__B, 0);
3150*4882a593Smuzhiyun 		if (rc != 0) {
3151*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3152*4882a593Smuzhiyun 			goto rw_error;
3153*4882a593Smuzhiyun 		}
3154*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MVAL_CFG__A, 0x0013, 0);
3155*4882a593Smuzhiyun 		if (rc != 0) {
3156*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3157*4882a593Smuzhiyun 			goto rw_error;
3158*4882a593Smuzhiyun 		}
3159*4882a593Smuzhiyun 		sio_pdr_md_cfg =
3160*4882a593Smuzhiyun 		    MPEG_SERIAL_OUTPUT_PIN_DRIVE_STRENGTH <<
3161*4882a593Smuzhiyun 		    SIO_PDR_MD0_CFG_DRIVE__B | 0x03 << SIO_PDR_MD0_CFG_MODE__B;
3162*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD0_CFG__A, sio_pdr_md_cfg, 0);
3163*4882a593Smuzhiyun 		if (rc != 0) {
3164*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3165*4882a593Smuzhiyun 			goto rw_error;
3166*4882a593Smuzhiyun 		}
3167*4882a593Smuzhiyun 		if (cfg_data->enable_parallel == true) {	/* MPEG data output is parallel -> set MD1 to MD7 to output mode */
3168*4882a593Smuzhiyun 			sio_pdr_md_cfg =
3169*4882a593Smuzhiyun 			    MPEG_PARALLEL_OUTPUT_PIN_DRIVE_STRENGTH <<
3170*4882a593Smuzhiyun 			    SIO_PDR_MD0_CFG_DRIVE__B | 0x03 <<
3171*4882a593Smuzhiyun 			    SIO_PDR_MD0_CFG_MODE__B;
3172*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD0_CFG__A, sio_pdr_md_cfg, 0);
3173*4882a593Smuzhiyun 			if (rc != 0) {
3174*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3175*4882a593Smuzhiyun 				goto rw_error;
3176*4882a593Smuzhiyun 			}
3177*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD1_CFG__A, sio_pdr_md_cfg, 0);
3178*4882a593Smuzhiyun 			if (rc != 0) {
3179*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3180*4882a593Smuzhiyun 				goto rw_error;
3181*4882a593Smuzhiyun 			}
3182*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD2_CFG__A, sio_pdr_md_cfg, 0);
3183*4882a593Smuzhiyun 			if (rc != 0) {
3184*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3185*4882a593Smuzhiyun 				goto rw_error;
3186*4882a593Smuzhiyun 			}
3187*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD3_CFG__A, sio_pdr_md_cfg, 0);
3188*4882a593Smuzhiyun 			if (rc != 0) {
3189*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3190*4882a593Smuzhiyun 				goto rw_error;
3191*4882a593Smuzhiyun 			}
3192*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD4_CFG__A, sio_pdr_md_cfg, 0);
3193*4882a593Smuzhiyun 			if (rc != 0) {
3194*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3195*4882a593Smuzhiyun 				goto rw_error;
3196*4882a593Smuzhiyun 			}
3197*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD5_CFG__A, sio_pdr_md_cfg, 0);
3198*4882a593Smuzhiyun 			if (rc != 0) {
3199*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3200*4882a593Smuzhiyun 				goto rw_error;
3201*4882a593Smuzhiyun 			}
3202*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD6_CFG__A, sio_pdr_md_cfg, 0);
3203*4882a593Smuzhiyun 			if (rc != 0) {
3204*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3205*4882a593Smuzhiyun 				goto rw_error;
3206*4882a593Smuzhiyun 			}
3207*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD7_CFG__A, sio_pdr_md_cfg, 0);
3208*4882a593Smuzhiyun 			if (rc != 0) {
3209*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3210*4882a593Smuzhiyun 				goto rw_error;
3211*4882a593Smuzhiyun 			}
3212*4882a593Smuzhiyun 		} else {	/* MPEG data output is serial -> set MD1 to MD7 to tri-state */
3213*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD1_CFG__A, 0x0000, 0);
3214*4882a593Smuzhiyun 			if (rc != 0) {
3215*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3216*4882a593Smuzhiyun 				goto rw_error;
3217*4882a593Smuzhiyun 			}
3218*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD2_CFG__A, 0x0000, 0);
3219*4882a593Smuzhiyun 			if (rc != 0) {
3220*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3221*4882a593Smuzhiyun 				goto rw_error;
3222*4882a593Smuzhiyun 			}
3223*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD3_CFG__A, 0x0000, 0);
3224*4882a593Smuzhiyun 			if (rc != 0) {
3225*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3226*4882a593Smuzhiyun 				goto rw_error;
3227*4882a593Smuzhiyun 			}
3228*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD4_CFG__A, 0x0000, 0);
3229*4882a593Smuzhiyun 			if (rc != 0) {
3230*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3231*4882a593Smuzhiyun 				goto rw_error;
3232*4882a593Smuzhiyun 			}
3233*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD5_CFG__A, 0x0000, 0);
3234*4882a593Smuzhiyun 			if (rc != 0) {
3235*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3236*4882a593Smuzhiyun 				goto rw_error;
3237*4882a593Smuzhiyun 			}
3238*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD6_CFG__A, 0x0000, 0);
3239*4882a593Smuzhiyun 			if (rc != 0) {
3240*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3241*4882a593Smuzhiyun 				goto rw_error;
3242*4882a593Smuzhiyun 			}
3243*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD7_CFG__A, 0x0000, 0);
3244*4882a593Smuzhiyun 			if (rc != 0) {
3245*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3246*4882a593Smuzhiyun 				goto rw_error;
3247*4882a593Smuzhiyun 			}
3248*4882a593Smuzhiyun 		}
3249*4882a593Smuzhiyun 		/*  Enable Monitor Bus output over MPEG pads and ctl input */
3250*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MON_CFG__A, 0x0000, 0);
3251*4882a593Smuzhiyun 		if (rc != 0) {
3252*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3253*4882a593Smuzhiyun 			goto rw_error;
3254*4882a593Smuzhiyun 		}
3255*4882a593Smuzhiyun 		/*  Write nomagic word to enable pdr reg write */
3256*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
3257*4882a593Smuzhiyun 		if (rc != 0) {
3258*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3259*4882a593Smuzhiyun 			goto rw_error;
3260*4882a593Smuzhiyun 		}
3261*4882a593Smuzhiyun 	} else {
3262*4882a593Smuzhiyun 		/*  Write magic word to enable pdr reg write */
3263*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0xFABA, 0);
3264*4882a593Smuzhiyun 		if (rc != 0) {
3265*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3266*4882a593Smuzhiyun 			goto rw_error;
3267*4882a593Smuzhiyun 		}
3268*4882a593Smuzhiyun 		/*  Set MPEG TS pads to inputmode */
3269*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MSTRT_CFG__A, 0x0000, 0);
3270*4882a593Smuzhiyun 		if (rc != 0) {
3271*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3272*4882a593Smuzhiyun 			goto rw_error;
3273*4882a593Smuzhiyun 		}
3274*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MERR_CFG__A, 0x0000, 0);
3275*4882a593Smuzhiyun 		if (rc != 0) {
3276*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3277*4882a593Smuzhiyun 			goto rw_error;
3278*4882a593Smuzhiyun 		}
3279*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MCLK_CFG__A, 0x0000, 0);
3280*4882a593Smuzhiyun 		if (rc != 0) {
3281*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3282*4882a593Smuzhiyun 			goto rw_error;
3283*4882a593Smuzhiyun 		}
3284*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MVAL_CFG__A, 0x0000, 0);
3285*4882a593Smuzhiyun 		if (rc != 0) {
3286*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3287*4882a593Smuzhiyun 			goto rw_error;
3288*4882a593Smuzhiyun 		}
3289*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD0_CFG__A, 0x0000, 0);
3290*4882a593Smuzhiyun 		if (rc != 0) {
3291*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3292*4882a593Smuzhiyun 			goto rw_error;
3293*4882a593Smuzhiyun 		}
3294*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD1_CFG__A, 0x0000, 0);
3295*4882a593Smuzhiyun 		if (rc != 0) {
3296*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3297*4882a593Smuzhiyun 			goto rw_error;
3298*4882a593Smuzhiyun 		}
3299*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD2_CFG__A, 0x0000, 0);
3300*4882a593Smuzhiyun 		if (rc != 0) {
3301*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3302*4882a593Smuzhiyun 			goto rw_error;
3303*4882a593Smuzhiyun 		}
3304*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD3_CFG__A, 0x0000, 0);
3305*4882a593Smuzhiyun 		if (rc != 0) {
3306*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3307*4882a593Smuzhiyun 			goto rw_error;
3308*4882a593Smuzhiyun 		}
3309*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD4_CFG__A, 0x0000, 0);
3310*4882a593Smuzhiyun 		if (rc != 0) {
3311*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3312*4882a593Smuzhiyun 			goto rw_error;
3313*4882a593Smuzhiyun 		}
3314*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD5_CFG__A, 0x0000, 0);
3315*4882a593Smuzhiyun 		if (rc != 0) {
3316*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3317*4882a593Smuzhiyun 			goto rw_error;
3318*4882a593Smuzhiyun 		}
3319*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD6_CFG__A, 0x0000, 0);
3320*4882a593Smuzhiyun 		if (rc != 0) {
3321*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3322*4882a593Smuzhiyun 			goto rw_error;
3323*4882a593Smuzhiyun 		}
3324*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD7_CFG__A, 0x0000, 0);
3325*4882a593Smuzhiyun 		if (rc != 0) {
3326*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3327*4882a593Smuzhiyun 			goto rw_error;
3328*4882a593Smuzhiyun 		}
3329*4882a593Smuzhiyun 		/* Enable Monitor Bus output over MPEG pads and ctl input */
3330*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MON_CFG__A, 0x0000, 0);
3331*4882a593Smuzhiyun 		if (rc != 0) {
3332*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3333*4882a593Smuzhiyun 			goto rw_error;
3334*4882a593Smuzhiyun 		}
3335*4882a593Smuzhiyun 		/* Write nomagic word to enable pdr reg write */
3336*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
3337*4882a593Smuzhiyun 		if (rc != 0) {
3338*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3339*4882a593Smuzhiyun 			goto rw_error;
3340*4882a593Smuzhiyun 		}
3341*4882a593Smuzhiyun 	}
3342*4882a593Smuzhiyun 
3343*4882a593Smuzhiyun 	/* save values for restore after re-acquire */
3344*4882a593Smuzhiyun 	common_attr->mpeg_cfg.enable_mpeg_output = cfg_data->enable_mpeg_output;
3345*4882a593Smuzhiyun 
3346*4882a593Smuzhiyun 	return 0;
3347*4882a593Smuzhiyun rw_error:
3348*4882a593Smuzhiyun 	return rc;
3349*4882a593Smuzhiyun }
3350*4882a593Smuzhiyun 
3351*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3352*4882a593Smuzhiyun 
3353*4882a593Smuzhiyun 
3354*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3355*4882a593Smuzhiyun /* MPEG Output Configuration Functions - end                                  */
3356*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3357*4882a593Smuzhiyun 
3358*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3359*4882a593Smuzhiyun /* miscellaneous configurations - begin                           */
3360*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3361*4882a593Smuzhiyun 
3362*4882a593Smuzhiyun /*
3363*4882a593Smuzhiyun * \fn int set_mpegtei_handling()
3364*4882a593Smuzhiyun * \brief Activate MPEG TEI handling settings.
3365*4882a593Smuzhiyun * \param devmod  Pointer to demodulator instance.
3366*4882a593Smuzhiyun * \return int.
3367*4882a593Smuzhiyun *
3368*4882a593Smuzhiyun * This routine should be called during a set channel of QAM/VSB
3369*4882a593Smuzhiyun *
3370*4882a593Smuzhiyun */
set_mpegtei_handling(struct drx_demod_instance * demod)3371*4882a593Smuzhiyun static int set_mpegtei_handling(struct drx_demod_instance *demod)
3372*4882a593Smuzhiyun {
3373*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
3374*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
3375*4882a593Smuzhiyun 	int rc;
3376*4882a593Smuzhiyun 	u16 fec_oc_dpr_mode = 0;
3377*4882a593Smuzhiyun 	u16 fec_oc_snc_mode = 0;
3378*4882a593Smuzhiyun 	u16 fec_oc_ems_mode = 0;
3379*4882a593Smuzhiyun 
3380*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
3381*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
3382*4882a593Smuzhiyun 
3383*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_OC_DPR_MODE__A, &fec_oc_dpr_mode, 0);
3384*4882a593Smuzhiyun 	if (rc != 0) {
3385*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3386*4882a593Smuzhiyun 		goto rw_error;
3387*4882a593Smuzhiyun 	}
3388*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_MODE__A, &fec_oc_snc_mode, 0);
3389*4882a593Smuzhiyun 	if (rc != 0) {
3390*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3391*4882a593Smuzhiyun 		goto rw_error;
3392*4882a593Smuzhiyun 	}
3393*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_OC_EMS_MODE__A, &fec_oc_ems_mode, 0);
3394*4882a593Smuzhiyun 	if (rc != 0) {
3395*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3396*4882a593Smuzhiyun 		goto rw_error;
3397*4882a593Smuzhiyun 	}
3398*4882a593Smuzhiyun 
3399*4882a593Smuzhiyun 	/* reset to default, allow TEI bit to be changed */
3400*4882a593Smuzhiyun 	fec_oc_dpr_mode &= (~FEC_OC_DPR_MODE_ERR_DISABLE__M);
3401*4882a593Smuzhiyun 	fec_oc_snc_mode &= (~(FEC_OC_SNC_MODE_ERROR_CTL__M |
3402*4882a593Smuzhiyun 			   FEC_OC_SNC_MODE_CORR_DISABLE__M));
3403*4882a593Smuzhiyun 	fec_oc_ems_mode &= (~FEC_OC_EMS_MODE_MODE__M);
3404*4882a593Smuzhiyun 
3405*4882a593Smuzhiyun 	if (ext_attr->disable_te_ihandling) {
3406*4882a593Smuzhiyun 		/* do not change TEI bit */
3407*4882a593Smuzhiyun 		fec_oc_dpr_mode |= FEC_OC_DPR_MODE_ERR_DISABLE__M;
3408*4882a593Smuzhiyun 		fec_oc_snc_mode |= FEC_OC_SNC_MODE_CORR_DISABLE__M |
3409*4882a593Smuzhiyun 		    ((0x2) << (FEC_OC_SNC_MODE_ERROR_CTL__B));
3410*4882a593Smuzhiyun 		fec_oc_ems_mode |= ((0x01) << (FEC_OC_EMS_MODE_MODE__B));
3411*4882a593Smuzhiyun 	}
3412*4882a593Smuzhiyun 
3413*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DPR_MODE__A, fec_oc_dpr_mode, 0);
3414*4882a593Smuzhiyun 	if (rc != 0) {
3415*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3416*4882a593Smuzhiyun 		goto rw_error;
3417*4882a593Smuzhiyun 	}
3418*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_MODE__A, fec_oc_snc_mode, 0);
3419*4882a593Smuzhiyun 	if (rc != 0) {
3420*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3421*4882a593Smuzhiyun 		goto rw_error;
3422*4882a593Smuzhiyun 	}
3423*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_OC_EMS_MODE__A, fec_oc_ems_mode, 0);
3424*4882a593Smuzhiyun 	if (rc != 0) {
3425*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3426*4882a593Smuzhiyun 		goto rw_error;
3427*4882a593Smuzhiyun 	}
3428*4882a593Smuzhiyun 
3429*4882a593Smuzhiyun 	return 0;
3430*4882a593Smuzhiyun rw_error:
3431*4882a593Smuzhiyun 	return rc;
3432*4882a593Smuzhiyun }
3433*4882a593Smuzhiyun 
3434*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3435*4882a593Smuzhiyun /*
3436*4882a593Smuzhiyun * \fn int bit_reverse_mpeg_output()
3437*4882a593Smuzhiyun * \brief Set MPEG output bit-endian settings.
3438*4882a593Smuzhiyun * \param devmod  Pointer to demodulator instance.
3439*4882a593Smuzhiyun * \return int.
3440*4882a593Smuzhiyun *
3441*4882a593Smuzhiyun * This routine should be called during a set channel of QAM/VSB
3442*4882a593Smuzhiyun *
3443*4882a593Smuzhiyun */
bit_reverse_mpeg_output(struct drx_demod_instance * demod)3444*4882a593Smuzhiyun static int bit_reverse_mpeg_output(struct drx_demod_instance *demod)
3445*4882a593Smuzhiyun {
3446*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
3447*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
3448*4882a593Smuzhiyun 	int rc;
3449*4882a593Smuzhiyun 	u16 fec_oc_ipr_mode = 0;
3450*4882a593Smuzhiyun 
3451*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
3452*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
3453*4882a593Smuzhiyun 
3454*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_OC_IPR_MODE__A, &fec_oc_ipr_mode, 0);
3455*4882a593Smuzhiyun 	if (rc != 0) {
3456*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3457*4882a593Smuzhiyun 		goto rw_error;
3458*4882a593Smuzhiyun 	}
3459*4882a593Smuzhiyun 
3460*4882a593Smuzhiyun 	/* reset to default (normal bit order) */
3461*4882a593Smuzhiyun 	fec_oc_ipr_mode &= (~FEC_OC_IPR_MODE_REVERSE_ORDER__M);
3462*4882a593Smuzhiyun 
3463*4882a593Smuzhiyun 	if (ext_attr->bit_reverse_mpeg_outout)
3464*4882a593Smuzhiyun 		fec_oc_ipr_mode |= FEC_OC_IPR_MODE_REVERSE_ORDER__M;
3465*4882a593Smuzhiyun 
3466*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_OC_IPR_MODE__A, fec_oc_ipr_mode, 0);
3467*4882a593Smuzhiyun 	if (rc != 0) {
3468*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3469*4882a593Smuzhiyun 		goto rw_error;
3470*4882a593Smuzhiyun 	}
3471*4882a593Smuzhiyun 
3472*4882a593Smuzhiyun 	return 0;
3473*4882a593Smuzhiyun rw_error:
3474*4882a593Smuzhiyun 	return rc;
3475*4882a593Smuzhiyun }
3476*4882a593Smuzhiyun 
3477*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3478*4882a593Smuzhiyun /*
3479*4882a593Smuzhiyun * \fn int set_mpeg_start_width()
3480*4882a593Smuzhiyun * \brief Set MPEG start width.
3481*4882a593Smuzhiyun * \param devmod  Pointer to demodulator instance.
3482*4882a593Smuzhiyun * \return int.
3483*4882a593Smuzhiyun *
3484*4882a593Smuzhiyun * This routine should be called during a set channel of QAM/VSB
3485*4882a593Smuzhiyun *
3486*4882a593Smuzhiyun */
set_mpeg_start_width(struct drx_demod_instance * demod)3487*4882a593Smuzhiyun static int set_mpeg_start_width(struct drx_demod_instance *demod)
3488*4882a593Smuzhiyun {
3489*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
3490*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
3491*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = (struct drx_common_attr *) NULL;
3492*4882a593Smuzhiyun 	int rc;
3493*4882a593Smuzhiyun 	u16 fec_oc_comm_mb = 0;
3494*4882a593Smuzhiyun 
3495*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
3496*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
3497*4882a593Smuzhiyun 	common_attr = demod->my_common_attr;
3498*4882a593Smuzhiyun 
3499*4882a593Smuzhiyun 	if ((common_attr->mpeg_cfg.static_clk == true)
3500*4882a593Smuzhiyun 	    && (common_attr->mpeg_cfg.enable_parallel == false)) {
3501*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, FEC_OC_COMM_MB__A, &fec_oc_comm_mb, 0);
3502*4882a593Smuzhiyun 		if (rc != 0) {
3503*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3504*4882a593Smuzhiyun 			goto rw_error;
3505*4882a593Smuzhiyun 		}
3506*4882a593Smuzhiyun 		fec_oc_comm_mb &= ~FEC_OC_COMM_MB_CTL_ON;
3507*4882a593Smuzhiyun 		if (ext_attr->mpeg_start_width == DRXJ_MPEG_START_WIDTH_8CLKCYC)
3508*4882a593Smuzhiyun 			fec_oc_comm_mb |= FEC_OC_COMM_MB_CTL_ON;
3509*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_OC_COMM_MB__A, fec_oc_comm_mb, 0);
3510*4882a593Smuzhiyun 		if (rc != 0) {
3511*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3512*4882a593Smuzhiyun 			goto rw_error;
3513*4882a593Smuzhiyun 		}
3514*4882a593Smuzhiyun 	}
3515*4882a593Smuzhiyun 
3516*4882a593Smuzhiyun 	return 0;
3517*4882a593Smuzhiyun rw_error:
3518*4882a593Smuzhiyun 	return rc;
3519*4882a593Smuzhiyun }
3520*4882a593Smuzhiyun 
3521*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3522*4882a593Smuzhiyun /* miscellaneous configurations - end                             */
3523*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3524*4882a593Smuzhiyun 
3525*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3526*4882a593Smuzhiyun /* UIO Configuration Functions - begin                                        */
3527*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3528*4882a593Smuzhiyun /*
3529*4882a593Smuzhiyun * \fn int ctrl_set_uio_cfg()
3530*4882a593Smuzhiyun * \brief Configure modus oprandi UIO.
3531*4882a593Smuzhiyun * \param demod Pointer to demodulator instance.
3532*4882a593Smuzhiyun * \param uio_cfg Pointer to a configuration setting for a certain UIO.
3533*4882a593Smuzhiyun * \return int.
3534*4882a593Smuzhiyun */
ctrl_set_uio_cfg(struct drx_demod_instance * demod,struct drxuio_cfg * uio_cfg)3535*4882a593Smuzhiyun static int ctrl_set_uio_cfg(struct drx_demod_instance *demod, struct drxuio_cfg *uio_cfg)
3536*4882a593Smuzhiyun {
3537*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
3538*4882a593Smuzhiyun 	int rc;
3539*4882a593Smuzhiyun 
3540*4882a593Smuzhiyun 	if ((uio_cfg == NULL) || (demod == NULL))
3541*4882a593Smuzhiyun 		return -EINVAL;
3542*4882a593Smuzhiyun 
3543*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
3544*4882a593Smuzhiyun 
3545*4882a593Smuzhiyun 	/*  Write magic word to enable pdr reg write               */
3546*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
3547*4882a593Smuzhiyun 	if (rc != 0) {
3548*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3549*4882a593Smuzhiyun 		goto rw_error;
3550*4882a593Smuzhiyun 	}
3551*4882a593Smuzhiyun 	switch (uio_cfg->uio) {
3552*4882a593Smuzhiyun       /*====================================================================*/
3553*4882a593Smuzhiyun 	case DRX_UIO1:
3554*4882a593Smuzhiyun 		/* DRX_UIO1: SMA_TX UIO-1 */
3555*4882a593Smuzhiyun 		if (!ext_attr->has_smatx)
3556*4882a593Smuzhiyun 			return -EIO;
3557*4882a593Smuzhiyun 		switch (uio_cfg->mode) {
3558*4882a593Smuzhiyun 		case DRX_UIO_MODE_FIRMWARE_SMA:
3559*4882a593Smuzhiyun 		case DRX_UIO_MODE_FIRMWARE_SAW:
3560*4882a593Smuzhiyun 		case DRX_UIO_MODE_READWRITE:
3561*4882a593Smuzhiyun 			ext_attr->uio_sma_tx_mode = uio_cfg->mode;
3562*4882a593Smuzhiyun 			break;
3563*4882a593Smuzhiyun 		case DRX_UIO_MODE_DISABLE:
3564*4882a593Smuzhiyun 			ext_attr->uio_sma_tx_mode = uio_cfg->mode;
3565*4882a593Smuzhiyun 			/* pad configuration register is set 0 - input mode */
3566*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_CFG__A, 0, 0);
3567*4882a593Smuzhiyun 			if (rc != 0) {
3568*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3569*4882a593Smuzhiyun 				goto rw_error;
3570*4882a593Smuzhiyun 			}
3571*4882a593Smuzhiyun 			break;
3572*4882a593Smuzhiyun 		default:
3573*4882a593Smuzhiyun 			return -EINVAL;
3574*4882a593Smuzhiyun 		}		/* switch ( uio_cfg->mode ) */
3575*4882a593Smuzhiyun 		break;
3576*4882a593Smuzhiyun       /*====================================================================*/
3577*4882a593Smuzhiyun 	case DRX_UIO2:
3578*4882a593Smuzhiyun 		/* DRX_UIO2: SMA_RX UIO-2 */
3579*4882a593Smuzhiyun 		if (!ext_attr->has_smarx)
3580*4882a593Smuzhiyun 			return -EIO;
3581*4882a593Smuzhiyun 		switch (uio_cfg->mode) {
3582*4882a593Smuzhiyun 		case DRX_UIO_MODE_FIRMWARE0:
3583*4882a593Smuzhiyun 		case DRX_UIO_MODE_READWRITE:
3584*4882a593Smuzhiyun 			ext_attr->uio_sma_rx_mode = uio_cfg->mode;
3585*4882a593Smuzhiyun 			break;
3586*4882a593Smuzhiyun 		case DRX_UIO_MODE_DISABLE:
3587*4882a593Smuzhiyun 			ext_attr->uio_sma_rx_mode = uio_cfg->mode;
3588*4882a593Smuzhiyun 			/* pad configuration register is set 0 - input mode */
3589*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_RX_CFG__A, 0, 0);
3590*4882a593Smuzhiyun 			if (rc != 0) {
3591*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3592*4882a593Smuzhiyun 				goto rw_error;
3593*4882a593Smuzhiyun 			}
3594*4882a593Smuzhiyun 			break;
3595*4882a593Smuzhiyun 		default:
3596*4882a593Smuzhiyun 			return -EINVAL;
3597*4882a593Smuzhiyun 			break;
3598*4882a593Smuzhiyun 		}		/* switch ( uio_cfg->mode ) */
3599*4882a593Smuzhiyun 		break;
3600*4882a593Smuzhiyun       /*====================================================================*/
3601*4882a593Smuzhiyun 	case DRX_UIO3:
3602*4882a593Smuzhiyun 		/* DRX_UIO3: GPIO UIO-3 */
3603*4882a593Smuzhiyun 		if (!ext_attr->has_gpio)
3604*4882a593Smuzhiyun 			return -EIO;
3605*4882a593Smuzhiyun 		switch (uio_cfg->mode) {
3606*4882a593Smuzhiyun 		case DRX_UIO_MODE_FIRMWARE0:
3607*4882a593Smuzhiyun 		case DRX_UIO_MODE_READWRITE:
3608*4882a593Smuzhiyun 			ext_attr->uio_gpio_mode = uio_cfg->mode;
3609*4882a593Smuzhiyun 			break;
3610*4882a593Smuzhiyun 		case DRX_UIO_MODE_DISABLE:
3611*4882a593Smuzhiyun 			ext_attr->uio_gpio_mode = uio_cfg->mode;
3612*4882a593Smuzhiyun 			/* pad configuration register is set 0 - input mode */
3613*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_GPIO_CFG__A, 0, 0);
3614*4882a593Smuzhiyun 			if (rc != 0) {
3615*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3616*4882a593Smuzhiyun 				goto rw_error;
3617*4882a593Smuzhiyun 			}
3618*4882a593Smuzhiyun 			break;
3619*4882a593Smuzhiyun 		default:
3620*4882a593Smuzhiyun 			return -EINVAL;
3621*4882a593Smuzhiyun 			break;
3622*4882a593Smuzhiyun 		}		/* switch ( uio_cfg->mode ) */
3623*4882a593Smuzhiyun 		break;
3624*4882a593Smuzhiyun       /*====================================================================*/
3625*4882a593Smuzhiyun 	case DRX_UIO4:
3626*4882a593Smuzhiyun 		/* DRX_UIO4: IRQN UIO-4 */
3627*4882a593Smuzhiyun 		if (!ext_attr->has_irqn)
3628*4882a593Smuzhiyun 			return -EIO;
3629*4882a593Smuzhiyun 		switch (uio_cfg->mode) {
3630*4882a593Smuzhiyun 		case DRX_UIO_MODE_READWRITE:
3631*4882a593Smuzhiyun 			ext_attr->uio_irqn_mode = uio_cfg->mode;
3632*4882a593Smuzhiyun 			break;
3633*4882a593Smuzhiyun 		case DRX_UIO_MODE_DISABLE:
3634*4882a593Smuzhiyun 			/* pad configuration register is set 0 - input mode */
3635*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_IRQN_CFG__A, 0, 0);
3636*4882a593Smuzhiyun 			if (rc != 0) {
3637*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
3638*4882a593Smuzhiyun 				goto rw_error;
3639*4882a593Smuzhiyun 			}
3640*4882a593Smuzhiyun 			ext_attr->uio_irqn_mode = uio_cfg->mode;
3641*4882a593Smuzhiyun 			break;
3642*4882a593Smuzhiyun 		case DRX_UIO_MODE_FIRMWARE0:
3643*4882a593Smuzhiyun 		default:
3644*4882a593Smuzhiyun 			return -EINVAL;
3645*4882a593Smuzhiyun 			break;
3646*4882a593Smuzhiyun 		}		/* switch ( uio_cfg->mode ) */
3647*4882a593Smuzhiyun 		break;
3648*4882a593Smuzhiyun       /*====================================================================*/
3649*4882a593Smuzhiyun 	default:
3650*4882a593Smuzhiyun 		return -EINVAL;
3651*4882a593Smuzhiyun 	}			/* switch ( uio_cfg->uio ) */
3652*4882a593Smuzhiyun 
3653*4882a593Smuzhiyun 	/*  Write magic word to disable pdr reg write               */
3654*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
3655*4882a593Smuzhiyun 	if (rc != 0) {
3656*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3657*4882a593Smuzhiyun 		goto rw_error;
3658*4882a593Smuzhiyun 	}
3659*4882a593Smuzhiyun 
3660*4882a593Smuzhiyun 	return 0;
3661*4882a593Smuzhiyun rw_error:
3662*4882a593Smuzhiyun 	return rc;
3663*4882a593Smuzhiyun }
3664*4882a593Smuzhiyun 
3665*4882a593Smuzhiyun /*
3666*4882a593Smuzhiyun * \fn int ctrl_uio_write()
3667*4882a593Smuzhiyun * \brief Write to a UIO.
3668*4882a593Smuzhiyun * \param demod Pointer to demodulator instance.
3669*4882a593Smuzhiyun * \param uio_data Pointer to data container for a certain UIO.
3670*4882a593Smuzhiyun * \return int.
3671*4882a593Smuzhiyun */
3672*4882a593Smuzhiyun static int
ctrl_uio_write(struct drx_demod_instance * demod,struct drxuio_data * uio_data)3673*4882a593Smuzhiyun ctrl_uio_write(struct drx_demod_instance *demod, struct drxuio_data *uio_data)
3674*4882a593Smuzhiyun {
3675*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
3676*4882a593Smuzhiyun 	int rc;
3677*4882a593Smuzhiyun 	u16 pin_cfg_value = 0;
3678*4882a593Smuzhiyun 	u16 value = 0;
3679*4882a593Smuzhiyun 
3680*4882a593Smuzhiyun 	if ((uio_data == NULL) || (demod == NULL))
3681*4882a593Smuzhiyun 		return -EINVAL;
3682*4882a593Smuzhiyun 
3683*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
3684*4882a593Smuzhiyun 
3685*4882a593Smuzhiyun 	/*  Write magic word to enable pdr reg write               */
3686*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
3687*4882a593Smuzhiyun 	if (rc != 0) {
3688*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3689*4882a593Smuzhiyun 		goto rw_error;
3690*4882a593Smuzhiyun 	}
3691*4882a593Smuzhiyun 	switch (uio_data->uio) {
3692*4882a593Smuzhiyun       /*====================================================================*/
3693*4882a593Smuzhiyun 	case DRX_UIO1:
3694*4882a593Smuzhiyun 		/* DRX_UIO1: SMA_TX UIO-1 */
3695*4882a593Smuzhiyun 		if (!ext_attr->has_smatx)
3696*4882a593Smuzhiyun 			return -EIO;
3697*4882a593Smuzhiyun 		if ((ext_attr->uio_sma_tx_mode != DRX_UIO_MODE_READWRITE)
3698*4882a593Smuzhiyun 		    && (ext_attr->uio_sma_tx_mode != DRX_UIO_MODE_FIRMWARE_SAW)) {
3699*4882a593Smuzhiyun 			return -EIO;
3700*4882a593Smuzhiyun 		}
3701*4882a593Smuzhiyun 		pin_cfg_value = 0;
3702*4882a593Smuzhiyun 		/* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
3703*4882a593Smuzhiyun 		pin_cfg_value |= 0x0113;
3704*4882a593Smuzhiyun 		/* io_pad_cfg_mode output mode is drive always */
3705*4882a593Smuzhiyun 		/* io_pad_cfg_drive is set to power 2 (23 mA) */
3706*4882a593Smuzhiyun 
3707*4882a593Smuzhiyun 		/* write to io pad configuration register - output mode */
3708*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_CFG__A, pin_cfg_value, 0);
3709*4882a593Smuzhiyun 		if (rc != 0) {
3710*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3711*4882a593Smuzhiyun 			goto rw_error;
3712*4882a593Smuzhiyun 		}
3713*4882a593Smuzhiyun 
3714*4882a593Smuzhiyun 		/* use corresponding bit in io data output registar */
3715*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, &value, 0);
3716*4882a593Smuzhiyun 		if (rc != 0) {
3717*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3718*4882a593Smuzhiyun 			goto rw_error;
3719*4882a593Smuzhiyun 		}
3720*4882a593Smuzhiyun 		if (!uio_data->value)
3721*4882a593Smuzhiyun 			value &= 0x7FFF;	/* write zero to 15th bit - 1st UIO */
3722*4882a593Smuzhiyun 		else
3723*4882a593Smuzhiyun 			value |= 0x8000;	/* write one to 15th bit - 1st UIO */
3724*4882a593Smuzhiyun 
3725*4882a593Smuzhiyun 		/* write back to io data output register */
3726*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, value, 0);
3727*4882a593Smuzhiyun 		if (rc != 0) {
3728*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3729*4882a593Smuzhiyun 			goto rw_error;
3730*4882a593Smuzhiyun 		}
3731*4882a593Smuzhiyun 		break;
3732*4882a593Smuzhiyun    /*======================================================================*/
3733*4882a593Smuzhiyun 	case DRX_UIO2:
3734*4882a593Smuzhiyun 		/* DRX_UIO2: SMA_RX UIO-2 */
3735*4882a593Smuzhiyun 		if (!ext_attr->has_smarx)
3736*4882a593Smuzhiyun 			return -EIO;
3737*4882a593Smuzhiyun 		if (ext_attr->uio_sma_rx_mode != DRX_UIO_MODE_READWRITE)
3738*4882a593Smuzhiyun 			return -EIO;
3739*4882a593Smuzhiyun 
3740*4882a593Smuzhiyun 		pin_cfg_value = 0;
3741*4882a593Smuzhiyun 		/* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
3742*4882a593Smuzhiyun 		pin_cfg_value |= 0x0113;
3743*4882a593Smuzhiyun 		/* io_pad_cfg_mode output mode is drive always */
3744*4882a593Smuzhiyun 		/* io_pad_cfg_drive is set to power 2 (23 mA) */
3745*4882a593Smuzhiyun 
3746*4882a593Smuzhiyun 		/* write to io pad configuration register - output mode */
3747*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_RX_CFG__A, pin_cfg_value, 0);
3748*4882a593Smuzhiyun 		if (rc != 0) {
3749*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3750*4882a593Smuzhiyun 			goto rw_error;
3751*4882a593Smuzhiyun 		}
3752*4882a593Smuzhiyun 
3753*4882a593Smuzhiyun 		/* use corresponding bit in io data output registar */
3754*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, &value, 0);
3755*4882a593Smuzhiyun 		if (rc != 0) {
3756*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3757*4882a593Smuzhiyun 			goto rw_error;
3758*4882a593Smuzhiyun 		}
3759*4882a593Smuzhiyun 		if (!uio_data->value)
3760*4882a593Smuzhiyun 			value &= 0xBFFF;	/* write zero to 14th bit - 2nd UIO */
3761*4882a593Smuzhiyun 		else
3762*4882a593Smuzhiyun 			value |= 0x4000;	/* write one to 14th bit - 2nd UIO */
3763*4882a593Smuzhiyun 
3764*4882a593Smuzhiyun 		/* write back to io data output register */
3765*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, value, 0);
3766*4882a593Smuzhiyun 		if (rc != 0) {
3767*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3768*4882a593Smuzhiyun 			goto rw_error;
3769*4882a593Smuzhiyun 		}
3770*4882a593Smuzhiyun 		break;
3771*4882a593Smuzhiyun    /*====================================================================*/
3772*4882a593Smuzhiyun 	case DRX_UIO3:
3773*4882a593Smuzhiyun 		/* DRX_UIO3: ASEL UIO-3 */
3774*4882a593Smuzhiyun 		if (!ext_attr->has_gpio)
3775*4882a593Smuzhiyun 			return -EIO;
3776*4882a593Smuzhiyun 		if (ext_attr->uio_gpio_mode != DRX_UIO_MODE_READWRITE)
3777*4882a593Smuzhiyun 			return -EIO;
3778*4882a593Smuzhiyun 
3779*4882a593Smuzhiyun 		pin_cfg_value = 0;
3780*4882a593Smuzhiyun 		/* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
3781*4882a593Smuzhiyun 		pin_cfg_value |= 0x0113;
3782*4882a593Smuzhiyun 		/* io_pad_cfg_mode output mode is drive always */
3783*4882a593Smuzhiyun 		/* io_pad_cfg_drive is set to power 2 (23 mA) */
3784*4882a593Smuzhiyun 
3785*4882a593Smuzhiyun 		/* write to io pad configuration register - output mode */
3786*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_GPIO_CFG__A, pin_cfg_value, 0);
3787*4882a593Smuzhiyun 		if (rc != 0) {
3788*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3789*4882a593Smuzhiyun 			goto rw_error;
3790*4882a593Smuzhiyun 		}
3791*4882a593Smuzhiyun 
3792*4882a593Smuzhiyun 		/* use corresponding bit in io data output registar */
3793*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_HI__A, &value, 0);
3794*4882a593Smuzhiyun 		if (rc != 0) {
3795*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3796*4882a593Smuzhiyun 			goto rw_error;
3797*4882a593Smuzhiyun 		}
3798*4882a593Smuzhiyun 		if (!uio_data->value)
3799*4882a593Smuzhiyun 			value &= 0xFFFB;	/* write zero to 2nd bit - 3rd UIO */
3800*4882a593Smuzhiyun 		else
3801*4882a593Smuzhiyun 			value |= 0x0004;	/* write one to 2nd bit - 3rd UIO */
3802*4882a593Smuzhiyun 
3803*4882a593Smuzhiyun 		/* write back to io data output register */
3804*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_HI__A, value, 0);
3805*4882a593Smuzhiyun 		if (rc != 0) {
3806*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3807*4882a593Smuzhiyun 			goto rw_error;
3808*4882a593Smuzhiyun 		}
3809*4882a593Smuzhiyun 		break;
3810*4882a593Smuzhiyun    /*=====================================================================*/
3811*4882a593Smuzhiyun 	case DRX_UIO4:
3812*4882a593Smuzhiyun 		/* DRX_UIO4: IRQN UIO-4 */
3813*4882a593Smuzhiyun 		if (!ext_attr->has_irqn)
3814*4882a593Smuzhiyun 			return -EIO;
3815*4882a593Smuzhiyun 
3816*4882a593Smuzhiyun 		if (ext_attr->uio_irqn_mode != DRX_UIO_MODE_READWRITE)
3817*4882a593Smuzhiyun 			return -EIO;
3818*4882a593Smuzhiyun 
3819*4882a593Smuzhiyun 		pin_cfg_value = 0;
3820*4882a593Smuzhiyun 		/* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
3821*4882a593Smuzhiyun 		pin_cfg_value |= 0x0113;
3822*4882a593Smuzhiyun 		/* io_pad_cfg_mode output mode is drive always */
3823*4882a593Smuzhiyun 		/* io_pad_cfg_drive is set to power 2 (23 mA) */
3824*4882a593Smuzhiyun 
3825*4882a593Smuzhiyun 		/* write to io pad configuration register - output mode */
3826*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_IRQN_CFG__A, pin_cfg_value, 0);
3827*4882a593Smuzhiyun 		if (rc != 0) {
3828*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3829*4882a593Smuzhiyun 			goto rw_error;
3830*4882a593Smuzhiyun 		}
3831*4882a593Smuzhiyun 
3832*4882a593Smuzhiyun 		/* use corresponding bit in io data output registar */
3833*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, &value, 0);
3834*4882a593Smuzhiyun 		if (rc != 0) {
3835*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3836*4882a593Smuzhiyun 			goto rw_error;
3837*4882a593Smuzhiyun 		}
3838*4882a593Smuzhiyun 		if (uio_data->value == false)
3839*4882a593Smuzhiyun 			value &= 0xEFFF;	/* write zero to 12th bit - 4th UIO */
3840*4882a593Smuzhiyun 		else
3841*4882a593Smuzhiyun 			value |= 0x1000;	/* write one to 12th bit - 4th UIO */
3842*4882a593Smuzhiyun 
3843*4882a593Smuzhiyun 		/* write back to io data output register */
3844*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, value, 0);
3845*4882a593Smuzhiyun 		if (rc != 0) {
3846*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3847*4882a593Smuzhiyun 			goto rw_error;
3848*4882a593Smuzhiyun 		}
3849*4882a593Smuzhiyun 		break;
3850*4882a593Smuzhiyun       /*=====================================================================*/
3851*4882a593Smuzhiyun 	default:
3852*4882a593Smuzhiyun 		return -EINVAL;
3853*4882a593Smuzhiyun 	}			/* switch ( uio_data->uio ) */
3854*4882a593Smuzhiyun 
3855*4882a593Smuzhiyun 	/*  Write magic word to disable pdr reg write               */
3856*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
3857*4882a593Smuzhiyun 	if (rc != 0) {
3858*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3859*4882a593Smuzhiyun 		goto rw_error;
3860*4882a593Smuzhiyun 	}
3861*4882a593Smuzhiyun 
3862*4882a593Smuzhiyun 	return 0;
3863*4882a593Smuzhiyun rw_error:
3864*4882a593Smuzhiyun 	return rc;
3865*4882a593Smuzhiyun }
3866*4882a593Smuzhiyun 
3867*4882a593Smuzhiyun /*---------------------------------------------------------------------------*/
3868*4882a593Smuzhiyun /* UIO Configuration Functions - end                                         */
3869*4882a593Smuzhiyun /*---------------------------------------------------------------------------*/
3870*4882a593Smuzhiyun 
3871*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3872*4882a593Smuzhiyun /* I2C Bridge Functions - begin                                               */
3873*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3874*4882a593Smuzhiyun /*
3875*4882a593Smuzhiyun * \fn int ctrl_i2c_bridge()
3876*4882a593Smuzhiyun * \brief Open or close the I2C switch to tuner.
3877*4882a593Smuzhiyun * \param demod Pointer to demodulator instance.
3878*4882a593Smuzhiyun * \param bridge_closed Pointer to bool indication if bridge is closed not.
3879*4882a593Smuzhiyun * \return int.
3880*4882a593Smuzhiyun 
3881*4882a593Smuzhiyun */
3882*4882a593Smuzhiyun static int
ctrl_i2c_bridge(struct drx_demod_instance * demod,bool * bridge_closed)3883*4882a593Smuzhiyun ctrl_i2c_bridge(struct drx_demod_instance *demod, bool *bridge_closed)
3884*4882a593Smuzhiyun {
3885*4882a593Smuzhiyun 	struct drxj_hi_cmd hi_cmd;
3886*4882a593Smuzhiyun 	u16 result = 0;
3887*4882a593Smuzhiyun 
3888*4882a593Smuzhiyun 	/* check arguments */
3889*4882a593Smuzhiyun 	if (bridge_closed == NULL)
3890*4882a593Smuzhiyun 		return -EINVAL;
3891*4882a593Smuzhiyun 
3892*4882a593Smuzhiyun 	hi_cmd.cmd = SIO_HI_RA_RAM_CMD_BRDCTRL;
3893*4882a593Smuzhiyun 	hi_cmd.param1 = SIO_HI_RA_RAM_PAR_1_PAR1_SEC_KEY;
3894*4882a593Smuzhiyun 	if (*bridge_closed)
3895*4882a593Smuzhiyun 		hi_cmd.param2 = SIO_HI_RA_RAM_PAR_2_BRD_CFG_CLOSED;
3896*4882a593Smuzhiyun 	else
3897*4882a593Smuzhiyun 		hi_cmd.param2 = SIO_HI_RA_RAM_PAR_2_BRD_CFG_OPEN;
3898*4882a593Smuzhiyun 
3899*4882a593Smuzhiyun 	return hi_command(demod->my_i2c_dev_addr, &hi_cmd, &result);
3900*4882a593Smuzhiyun }
3901*4882a593Smuzhiyun 
3902*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3903*4882a593Smuzhiyun /* I2C Bridge Functions - end                                                 */
3904*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3905*4882a593Smuzhiyun 
3906*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3907*4882a593Smuzhiyun /* Smart antenna Functions - begin                                            */
3908*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
3909*4882a593Smuzhiyun /*
3910*4882a593Smuzhiyun * \fn int smart_ant_init()
3911*4882a593Smuzhiyun * \brief Initialize Smart Antenna.
3912*4882a593Smuzhiyun * \param pointer to struct drx_demod_instance.
3913*4882a593Smuzhiyun * \return int.
3914*4882a593Smuzhiyun *
3915*4882a593Smuzhiyun */
smart_ant_init(struct drx_demod_instance * demod)3916*4882a593Smuzhiyun static int smart_ant_init(struct drx_demod_instance *demod)
3917*4882a593Smuzhiyun {
3918*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
3919*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
3920*4882a593Smuzhiyun 	struct drxuio_cfg uio_cfg = { DRX_UIO1, DRX_UIO_MODE_FIRMWARE_SMA };
3921*4882a593Smuzhiyun 	int rc;
3922*4882a593Smuzhiyun 	u16 data = 0;
3923*4882a593Smuzhiyun 
3924*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
3925*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
3926*4882a593Smuzhiyun 
3927*4882a593Smuzhiyun 	/*  Write magic word to enable pdr reg write               */
3928*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
3929*4882a593Smuzhiyun 	if (rc != 0) {
3930*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3931*4882a593Smuzhiyun 		goto rw_error;
3932*4882a593Smuzhiyun 	}
3933*4882a593Smuzhiyun 	/* init smart antenna */
3934*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, SIO_SA_TX_COMMAND__A, &data, 0);
3935*4882a593Smuzhiyun 	if (rc != 0) {
3936*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3937*4882a593Smuzhiyun 		goto rw_error;
3938*4882a593Smuzhiyun 	}
3939*4882a593Smuzhiyun 	if (ext_attr->smart_ant_inverted) {
3940*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_SA_TX_COMMAND__A, (data | SIO_SA_TX_COMMAND_TX_INVERT__M) | SIO_SA_TX_COMMAND_TX_ENABLE__M, 0);
3941*4882a593Smuzhiyun 		if (rc != 0) {
3942*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3943*4882a593Smuzhiyun 			goto rw_error;
3944*4882a593Smuzhiyun 		}
3945*4882a593Smuzhiyun 	} else {
3946*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_SA_TX_COMMAND__A, (data & (~SIO_SA_TX_COMMAND_TX_INVERT__M)) | SIO_SA_TX_COMMAND_TX_ENABLE__M, 0);
3947*4882a593Smuzhiyun 		if (rc != 0) {
3948*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
3949*4882a593Smuzhiyun 			goto rw_error;
3950*4882a593Smuzhiyun 		}
3951*4882a593Smuzhiyun 	}
3952*4882a593Smuzhiyun 
3953*4882a593Smuzhiyun 	/* config SMA_TX pin to smart antenna mode */
3954*4882a593Smuzhiyun 	rc = ctrl_set_uio_cfg(demod, &uio_cfg);
3955*4882a593Smuzhiyun 	if (rc != 0) {
3956*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3957*4882a593Smuzhiyun 		goto rw_error;
3958*4882a593Smuzhiyun 	}
3959*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_CFG__A, 0x13, 0);
3960*4882a593Smuzhiyun 	if (rc != 0) {
3961*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3962*4882a593Smuzhiyun 		goto rw_error;
3963*4882a593Smuzhiyun 	}
3964*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_GPIO_FNC__A, 0x03, 0);
3965*4882a593Smuzhiyun 	if (rc != 0) {
3966*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3967*4882a593Smuzhiyun 		goto rw_error;
3968*4882a593Smuzhiyun 	}
3969*4882a593Smuzhiyun 
3970*4882a593Smuzhiyun 	/*  Write magic word to disable pdr reg write               */
3971*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
3972*4882a593Smuzhiyun 	if (rc != 0) {
3973*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3974*4882a593Smuzhiyun 		goto rw_error;
3975*4882a593Smuzhiyun 	}
3976*4882a593Smuzhiyun 
3977*4882a593Smuzhiyun 	return 0;
3978*4882a593Smuzhiyun rw_error:
3979*4882a593Smuzhiyun 	return rc;
3980*4882a593Smuzhiyun }
3981*4882a593Smuzhiyun 
scu_command(struct i2c_device_addr * dev_addr,struct drxjscu_cmd * cmd)3982*4882a593Smuzhiyun static int scu_command(struct i2c_device_addr *dev_addr, struct drxjscu_cmd *cmd)
3983*4882a593Smuzhiyun {
3984*4882a593Smuzhiyun 	int rc;
3985*4882a593Smuzhiyun 	u16 cur_cmd = 0;
3986*4882a593Smuzhiyun 	unsigned long timeout;
3987*4882a593Smuzhiyun 
3988*4882a593Smuzhiyun 	/* Check param */
3989*4882a593Smuzhiyun 	if (cmd == NULL)
3990*4882a593Smuzhiyun 		return -EINVAL;
3991*4882a593Smuzhiyun 
3992*4882a593Smuzhiyun 	/* Wait until SCU command interface is ready to receive command */
3993*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_COMMAND__A, &cur_cmd, 0);
3994*4882a593Smuzhiyun 	if (rc != 0) {
3995*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
3996*4882a593Smuzhiyun 		goto rw_error;
3997*4882a593Smuzhiyun 	}
3998*4882a593Smuzhiyun 	if (cur_cmd != DRX_SCU_READY)
3999*4882a593Smuzhiyun 		return -EIO;
4000*4882a593Smuzhiyun 
4001*4882a593Smuzhiyun 	switch (cmd->parameter_len) {
4002*4882a593Smuzhiyun 	case 5:
4003*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_4__A, *(cmd->parameter + 4), 0);
4004*4882a593Smuzhiyun 		if (rc != 0) {
4005*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4006*4882a593Smuzhiyun 			goto rw_error;
4007*4882a593Smuzhiyun 		}
4008*4882a593Smuzhiyun 		fallthrough;
4009*4882a593Smuzhiyun 	case 4:
4010*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_3__A, *(cmd->parameter + 3), 0);
4011*4882a593Smuzhiyun 		if (rc != 0) {
4012*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4013*4882a593Smuzhiyun 			goto rw_error;
4014*4882a593Smuzhiyun 		}
4015*4882a593Smuzhiyun 		fallthrough;
4016*4882a593Smuzhiyun 	case 3:
4017*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_2__A, *(cmd->parameter + 2), 0);
4018*4882a593Smuzhiyun 		if (rc != 0) {
4019*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4020*4882a593Smuzhiyun 			goto rw_error;
4021*4882a593Smuzhiyun 		}
4022*4882a593Smuzhiyun 		fallthrough;
4023*4882a593Smuzhiyun 	case 2:
4024*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_1__A, *(cmd->parameter + 1), 0);
4025*4882a593Smuzhiyun 		if (rc != 0) {
4026*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4027*4882a593Smuzhiyun 			goto rw_error;
4028*4882a593Smuzhiyun 		}
4029*4882a593Smuzhiyun 		fallthrough;
4030*4882a593Smuzhiyun 	case 1:
4031*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_0__A, *(cmd->parameter + 0), 0);
4032*4882a593Smuzhiyun 		if (rc != 0) {
4033*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4034*4882a593Smuzhiyun 			goto rw_error;
4035*4882a593Smuzhiyun 		}
4036*4882a593Smuzhiyun 		fallthrough;
4037*4882a593Smuzhiyun 	case 0:
4038*4882a593Smuzhiyun 		/* do nothing */
4039*4882a593Smuzhiyun 		break;
4040*4882a593Smuzhiyun 	default:
4041*4882a593Smuzhiyun 		/* this number of parameters is not supported */
4042*4882a593Smuzhiyun 		return -EIO;
4043*4882a593Smuzhiyun 	}
4044*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_COMMAND__A, cmd->command, 0);
4045*4882a593Smuzhiyun 	if (rc != 0) {
4046*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4047*4882a593Smuzhiyun 		goto rw_error;
4048*4882a593Smuzhiyun 	}
4049*4882a593Smuzhiyun 
4050*4882a593Smuzhiyun 	/* Wait until SCU has processed command */
4051*4882a593Smuzhiyun 	timeout = jiffies + msecs_to_jiffies(DRXJ_MAX_WAITTIME);
4052*4882a593Smuzhiyun 	while (time_is_after_jiffies(timeout)) {
4053*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_COMMAND__A, &cur_cmd, 0);
4054*4882a593Smuzhiyun 		if (rc != 0) {
4055*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4056*4882a593Smuzhiyun 			goto rw_error;
4057*4882a593Smuzhiyun 		}
4058*4882a593Smuzhiyun 		if (cur_cmd == DRX_SCU_READY)
4059*4882a593Smuzhiyun 			break;
4060*4882a593Smuzhiyun 		usleep_range(1000, 2000);
4061*4882a593Smuzhiyun 	}
4062*4882a593Smuzhiyun 
4063*4882a593Smuzhiyun 	if (cur_cmd != DRX_SCU_READY)
4064*4882a593Smuzhiyun 		return -EIO;
4065*4882a593Smuzhiyun 
4066*4882a593Smuzhiyun 	/* read results */
4067*4882a593Smuzhiyun 	if ((cmd->result_len > 0) && (cmd->result != NULL)) {
4068*4882a593Smuzhiyun 		s16 err;
4069*4882a593Smuzhiyun 
4070*4882a593Smuzhiyun 		switch (cmd->result_len) {
4071*4882a593Smuzhiyun 		case 4:
4072*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_3__A, cmd->result + 3, 0);
4073*4882a593Smuzhiyun 			if (rc != 0) {
4074*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4075*4882a593Smuzhiyun 				goto rw_error;
4076*4882a593Smuzhiyun 			}
4077*4882a593Smuzhiyun 			fallthrough;
4078*4882a593Smuzhiyun 		case 3:
4079*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_2__A, cmd->result + 2, 0);
4080*4882a593Smuzhiyun 			if (rc != 0) {
4081*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4082*4882a593Smuzhiyun 				goto rw_error;
4083*4882a593Smuzhiyun 			}
4084*4882a593Smuzhiyun 			fallthrough;
4085*4882a593Smuzhiyun 		case 2:
4086*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_1__A, cmd->result + 1, 0);
4087*4882a593Smuzhiyun 			if (rc != 0) {
4088*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4089*4882a593Smuzhiyun 				goto rw_error;
4090*4882a593Smuzhiyun 			}
4091*4882a593Smuzhiyun 			fallthrough;
4092*4882a593Smuzhiyun 		case 1:
4093*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_0__A, cmd->result + 0, 0);
4094*4882a593Smuzhiyun 			if (rc != 0) {
4095*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4096*4882a593Smuzhiyun 				goto rw_error;
4097*4882a593Smuzhiyun 			}
4098*4882a593Smuzhiyun 			fallthrough;
4099*4882a593Smuzhiyun 		case 0:
4100*4882a593Smuzhiyun 			/* do nothing */
4101*4882a593Smuzhiyun 			break;
4102*4882a593Smuzhiyun 		default:
4103*4882a593Smuzhiyun 			/* this number of parameters is not supported */
4104*4882a593Smuzhiyun 			return -EIO;
4105*4882a593Smuzhiyun 		}
4106*4882a593Smuzhiyun 
4107*4882a593Smuzhiyun 		/* Check if an error was reported by SCU */
4108*4882a593Smuzhiyun 		err = cmd->result[0];
4109*4882a593Smuzhiyun 
4110*4882a593Smuzhiyun 		/* check a few fixed error codes */
4111*4882a593Smuzhiyun 		if ((err == (s16) SCU_RAM_PARAM_0_RESULT_UNKSTD)
4112*4882a593Smuzhiyun 		    || (err == (s16) SCU_RAM_PARAM_0_RESULT_UNKCMD)
4113*4882a593Smuzhiyun 		    || (err == (s16) SCU_RAM_PARAM_0_RESULT_INVPAR)
4114*4882a593Smuzhiyun 		    || (err == (s16) SCU_RAM_PARAM_0_RESULT_SIZE)
4115*4882a593Smuzhiyun 		    ) {
4116*4882a593Smuzhiyun 			return -EINVAL;
4117*4882a593Smuzhiyun 		}
4118*4882a593Smuzhiyun 		/* here it is assumed that negative means error, and positive no error */
4119*4882a593Smuzhiyun 		else if (err < 0)
4120*4882a593Smuzhiyun 			return -EIO;
4121*4882a593Smuzhiyun 		else
4122*4882a593Smuzhiyun 			return 0;
4123*4882a593Smuzhiyun 	}
4124*4882a593Smuzhiyun 
4125*4882a593Smuzhiyun 	return 0;
4126*4882a593Smuzhiyun 
4127*4882a593Smuzhiyun rw_error:
4128*4882a593Smuzhiyun 	return rc;
4129*4882a593Smuzhiyun }
4130*4882a593Smuzhiyun 
4131*4882a593Smuzhiyun /*
4132*4882a593Smuzhiyun * \fn int DRXJ_DAP_SCUAtomicReadWriteBlock()
4133*4882a593Smuzhiyun * \brief Basic access routine for SCU atomic read or write access
4134*4882a593Smuzhiyun * \param dev_addr  pointer to i2c dev address
4135*4882a593Smuzhiyun * \param addr     destination/source address
4136*4882a593Smuzhiyun * \param datasize size of data buffer in bytes
4137*4882a593Smuzhiyun * \param data     pointer to data buffer
4138*4882a593Smuzhiyun * \return int
4139*4882a593Smuzhiyun * \retval 0 Success
4140*4882a593Smuzhiyun * \retval -EIO Timeout, I2C error, illegal bank
4141*4882a593Smuzhiyun *
4142*4882a593Smuzhiyun */
4143*4882a593Smuzhiyun #define ADDR_AT_SCU_SPACE(x) ((x - 0x82E000) * 2)
4144*4882a593Smuzhiyun static
drxj_dap_scu_atomic_read_write_block(struct i2c_device_addr * dev_addr,u32 addr,u16 datasize,u8 * data,bool read_flag)4145*4882a593Smuzhiyun int drxj_dap_scu_atomic_read_write_block(struct i2c_device_addr *dev_addr, u32 addr, u16 datasize,	/* max 30 bytes because the limit of SCU parameter */
4146*4882a593Smuzhiyun 					      u8 *data, bool read_flag)
4147*4882a593Smuzhiyun {
4148*4882a593Smuzhiyun 	struct drxjscu_cmd scu_cmd;
4149*4882a593Smuzhiyun 	int rc;
4150*4882a593Smuzhiyun 	u16 set_param_parameters[18];
4151*4882a593Smuzhiyun 	u16 cmd_result[15];
4152*4882a593Smuzhiyun 
4153*4882a593Smuzhiyun 	/* Parameter check */
4154*4882a593Smuzhiyun 	if (!data || !dev_addr || (datasize % 2) || ((datasize / 2) > 16))
4155*4882a593Smuzhiyun 		return -EINVAL;
4156*4882a593Smuzhiyun 
4157*4882a593Smuzhiyun 	set_param_parameters[1] = (u16) ADDR_AT_SCU_SPACE(addr);
4158*4882a593Smuzhiyun 	if (read_flag) {		/* read */
4159*4882a593Smuzhiyun 		set_param_parameters[0] = ((~(0x0080)) & datasize);
4160*4882a593Smuzhiyun 		scu_cmd.parameter_len = 2;
4161*4882a593Smuzhiyun 		scu_cmd.result_len = datasize / 2 + 2;
4162*4882a593Smuzhiyun 	} else {
4163*4882a593Smuzhiyun 		int i = 0;
4164*4882a593Smuzhiyun 
4165*4882a593Smuzhiyun 		set_param_parameters[0] = 0x0080 | datasize;
4166*4882a593Smuzhiyun 		for (i = 0; i < (datasize / 2); i++) {
4167*4882a593Smuzhiyun 			set_param_parameters[i + 2] =
4168*4882a593Smuzhiyun 			    (data[2 * i] | (data[(2 * i) + 1] << 8));
4169*4882a593Smuzhiyun 		}
4170*4882a593Smuzhiyun 		scu_cmd.parameter_len = datasize / 2 + 2;
4171*4882a593Smuzhiyun 		scu_cmd.result_len = 1;
4172*4882a593Smuzhiyun 	}
4173*4882a593Smuzhiyun 
4174*4882a593Smuzhiyun 	scu_cmd.command =
4175*4882a593Smuzhiyun 	    SCU_RAM_COMMAND_STANDARD_TOP |
4176*4882a593Smuzhiyun 	    SCU_RAM_COMMAND_CMD_AUX_SCU_ATOMIC_ACCESS;
4177*4882a593Smuzhiyun 	scu_cmd.result = cmd_result;
4178*4882a593Smuzhiyun 	scu_cmd.parameter = set_param_parameters;
4179*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &scu_cmd);
4180*4882a593Smuzhiyun 	if (rc != 0) {
4181*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4182*4882a593Smuzhiyun 		goto rw_error;
4183*4882a593Smuzhiyun 	}
4184*4882a593Smuzhiyun 
4185*4882a593Smuzhiyun 	if (read_flag) {
4186*4882a593Smuzhiyun 		int i = 0;
4187*4882a593Smuzhiyun 		/* read data from buffer */
4188*4882a593Smuzhiyun 		for (i = 0; i < (datasize / 2); i++) {
4189*4882a593Smuzhiyun 			data[2 * i] = (u8) (scu_cmd.result[i + 2] & 0xFF);
4190*4882a593Smuzhiyun 			data[(2 * i) + 1] = (u8) (scu_cmd.result[i + 2] >> 8);
4191*4882a593Smuzhiyun 		}
4192*4882a593Smuzhiyun 	}
4193*4882a593Smuzhiyun 
4194*4882a593Smuzhiyun 	return 0;
4195*4882a593Smuzhiyun 
4196*4882a593Smuzhiyun rw_error:
4197*4882a593Smuzhiyun 	return rc;
4198*4882a593Smuzhiyun 
4199*4882a593Smuzhiyun }
4200*4882a593Smuzhiyun 
4201*4882a593Smuzhiyun /*============================================================================*/
4202*4882a593Smuzhiyun 
4203*4882a593Smuzhiyun /*
4204*4882a593Smuzhiyun * \fn int DRXJ_DAP_AtomicReadReg16()
4205*4882a593Smuzhiyun * \brief Atomic read of 16 bits words
4206*4882a593Smuzhiyun */
4207*4882a593Smuzhiyun static
drxj_dap_scu_atomic_read_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 * data,u32 flags)4208*4882a593Smuzhiyun int drxj_dap_scu_atomic_read_reg16(struct i2c_device_addr *dev_addr,
4209*4882a593Smuzhiyun 					 u32 addr,
4210*4882a593Smuzhiyun 					 u16 *data, u32 flags)
4211*4882a593Smuzhiyun {
4212*4882a593Smuzhiyun 	u8 buf[2] = { 0 };
4213*4882a593Smuzhiyun 	int rc;
4214*4882a593Smuzhiyun 	u16 word = 0;
4215*4882a593Smuzhiyun 
4216*4882a593Smuzhiyun 	if (!data)
4217*4882a593Smuzhiyun 		return -EINVAL;
4218*4882a593Smuzhiyun 
4219*4882a593Smuzhiyun 	rc = drxj_dap_scu_atomic_read_write_block(dev_addr, addr, 2, buf, true);
4220*4882a593Smuzhiyun 	if (rc < 0)
4221*4882a593Smuzhiyun 		return rc;
4222*4882a593Smuzhiyun 
4223*4882a593Smuzhiyun 	word = (u16) (buf[0] + (buf[1] << 8));
4224*4882a593Smuzhiyun 
4225*4882a593Smuzhiyun 	*data = word;
4226*4882a593Smuzhiyun 
4227*4882a593Smuzhiyun 	return rc;
4228*4882a593Smuzhiyun }
4229*4882a593Smuzhiyun 
4230*4882a593Smuzhiyun /*============================================================================*/
4231*4882a593Smuzhiyun /*
4232*4882a593Smuzhiyun * \fn int drxj_dap_scu_atomic_write_reg16()
4233*4882a593Smuzhiyun * \brief Atomic read of 16 bits words
4234*4882a593Smuzhiyun */
4235*4882a593Smuzhiyun static
drxj_dap_scu_atomic_write_reg16(struct i2c_device_addr * dev_addr,u32 addr,u16 data,u32 flags)4236*4882a593Smuzhiyun int drxj_dap_scu_atomic_write_reg16(struct i2c_device_addr *dev_addr,
4237*4882a593Smuzhiyun 					  u32 addr,
4238*4882a593Smuzhiyun 					  u16 data, u32 flags)
4239*4882a593Smuzhiyun {
4240*4882a593Smuzhiyun 	u8 buf[2];
4241*4882a593Smuzhiyun 	int rc;
4242*4882a593Smuzhiyun 
4243*4882a593Smuzhiyun 	buf[0] = (u8) (data & 0xff);
4244*4882a593Smuzhiyun 	buf[1] = (u8) ((data >> 8) & 0xff);
4245*4882a593Smuzhiyun 
4246*4882a593Smuzhiyun 	rc = drxj_dap_scu_atomic_read_write_block(dev_addr, addr, 2, buf, false);
4247*4882a593Smuzhiyun 
4248*4882a593Smuzhiyun 	return rc;
4249*4882a593Smuzhiyun }
4250*4882a593Smuzhiyun 
4251*4882a593Smuzhiyun /* -------------------------------------------------------------------------- */
4252*4882a593Smuzhiyun /*
4253*4882a593Smuzhiyun * \brief Measure result of ADC synchronisation
4254*4882a593Smuzhiyun * \param demod demod instance
4255*4882a593Smuzhiyun * \param count (returned) count
4256*4882a593Smuzhiyun * \return int.
4257*4882a593Smuzhiyun * \retval 0    Success
4258*4882a593Smuzhiyun * \retval -EIO Failure: I2C error
4259*4882a593Smuzhiyun *
4260*4882a593Smuzhiyun */
adc_sync_measurement(struct drx_demod_instance * demod,u16 * count)4261*4882a593Smuzhiyun static int adc_sync_measurement(struct drx_demod_instance *demod, u16 *count)
4262*4882a593Smuzhiyun {
4263*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
4264*4882a593Smuzhiyun 	int rc;
4265*4882a593Smuzhiyun 	u16 data = 0;
4266*4882a593Smuzhiyun 
4267*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
4268*4882a593Smuzhiyun 
4269*4882a593Smuzhiyun 	/* Start measurement */
4270*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_COMM_EXEC__A, IQM_AF_COMM_EXEC_ACTIVE, 0);
4271*4882a593Smuzhiyun 	if (rc != 0) {
4272*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4273*4882a593Smuzhiyun 		goto rw_error;
4274*4882a593Smuzhiyun 	}
4275*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_START_LOCK__A, 1, 0);
4276*4882a593Smuzhiyun 	if (rc != 0) {
4277*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4278*4882a593Smuzhiyun 		goto rw_error;
4279*4882a593Smuzhiyun 	}
4280*4882a593Smuzhiyun 
4281*4882a593Smuzhiyun 	/* Wait at least 3*128*(1/sysclk) <<< 1 millisec */
4282*4882a593Smuzhiyun 	msleep(1);
4283*4882a593Smuzhiyun 
4284*4882a593Smuzhiyun 	*count = 0;
4285*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, IQM_AF_PHASE0__A, &data, 0);
4286*4882a593Smuzhiyun 	if (rc != 0) {
4287*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4288*4882a593Smuzhiyun 		goto rw_error;
4289*4882a593Smuzhiyun 	}
4290*4882a593Smuzhiyun 	if (data == 127)
4291*4882a593Smuzhiyun 		*count = *count + 1;
4292*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, IQM_AF_PHASE1__A, &data, 0);
4293*4882a593Smuzhiyun 	if (rc != 0) {
4294*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4295*4882a593Smuzhiyun 		goto rw_error;
4296*4882a593Smuzhiyun 	}
4297*4882a593Smuzhiyun 	if (data == 127)
4298*4882a593Smuzhiyun 		*count = *count + 1;
4299*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, IQM_AF_PHASE2__A, &data, 0);
4300*4882a593Smuzhiyun 	if (rc != 0) {
4301*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4302*4882a593Smuzhiyun 		goto rw_error;
4303*4882a593Smuzhiyun 	}
4304*4882a593Smuzhiyun 	if (data == 127)
4305*4882a593Smuzhiyun 		*count = *count + 1;
4306*4882a593Smuzhiyun 
4307*4882a593Smuzhiyun 	return 0;
4308*4882a593Smuzhiyun rw_error:
4309*4882a593Smuzhiyun 	return rc;
4310*4882a593Smuzhiyun }
4311*4882a593Smuzhiyun 
4312*4882a593Smuzhiyun /*
4313*4882a593Smuzhiyun * \brief Synchronize analog and digital clock domains
4314*4882a593Smuzhiyun * \param demod demod instance
4315*4882a593Smuzhiyun * \return int.
4316*4882a593Smuzhiyun * \retval 0    Success
4317*4882a593Smuzhiyun * \retval -EIO Failure: I2C error or failure to synchronize
4318*4882a593Smuzhiyun *
4319*4882a593Smuzhiyun * An IQM reset will also reset the results of this synchronization.
4320*4882a593Smuzhiyun * After an IQM reset this routine needs to be called again.
4321*4882a593Smuzhiyun *
4322*4882a593Smuzhiyun */
4323*4882a593Smuzhiyun 
adc_synchronization(struct drx_demod_instance * demod)4324*4882a593Smuzhiyun static int adc_synchronization(struct drx_demod_instance *demod)
4325*4882a593Smuzhiyun {
4326*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
4327*4882a593Smuzhiyun 	int rc;
4328*4882a593Smuzhiyun 	u16 count = 0;
4329*4882a593Smuzhiyun 
4330*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
4331*4882a593Smuzhiyun 
4332*4882a593Smuzhiyun 	rc = adc_sync_measurement(demod, &count);
4333*4882a593Smuzhiyun 	if (rc != 0) {
4334*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4335*4882a593Smuzhiyun 		goto rw_error;
4336*4882a593Smuzhiyun 	}
4337*4882a593Smuzhiyun 
4338*4882a593Smuzhiyun 	if (count == 1) {
4339*4882a593Smuzhiyun 		/* Try sampling on a different edge */
4340*4882a593Smuzhiyun 		u16 clk_neg = 0;
4341*4882a593Smuzhiyun 
4342*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, IQM_AF_CLKNEG__A, &clk_neg, 0);
4343*4882a593Smuzhiyun 		if (rc != 0) {
4344*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4345*4882a593Smuzhiyun 			goto rw_error;
4346*4882a593Smuzhiyun 		}
4347*4882a593Smuzhiyun 
4348*4882a593Smuzhiyun 		clk_neg ^= IQM_AF_CLKNEG_CLKNEGDATA__M;
4349*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLKNEG__A, clk_neg, 0);
4350*4882a593Smuzhiyun 		if (rc != 0) {
4351*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4352*4882a593Smuzhiyun 			goto rw_error;
4353*4882a593Smuzhiyun 		}
4354*4882a593Smuzhiyun 
4355*4882a593Smuzhiyun 		rc = adc_sync_measurement(demod, &count);
4356*4882a593Smuzhiyun 		if (rc != 0) {
4357*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4358*4882a593Smuzhiyun 			goto rw_error;
4359*4882a593Smuzhiyun 		}
4360*4882a593Smuzhiyun 	}
4361*4882a593Smuzhiyun 
4362*4882a593Smuzhiyun 	/* TODO: implement fallback scenarios */
4363*4882a593Smuzhiyun 	if (count < 2)
4364*4882a593Smuzhiyun 		return -EIO;
4365*4882a593Smuzhiyun 
4366*4882a593Smuzhiyun 	return 0;
4367*4882a593Smuzhiyun rw_error:
4368*4882a593Smuzhiyun 	return rc;
4369*4882a593Smuzhiyun }
4370*4882a593Smuzhiyun 
4371*4882a593Smuzhiyun /*============================================================================*/
4372*4882a593Smuzhiyun /*==                      END AUXILIARY FUNCTIONS                           ==*/
4373*4882a593Smuzhiyun /*============================================================================*/
4374*4882a593Smuzhiyun 
4375*4882a593Smuzhiyun /*============================================================================*/
4376*4882a593Smuzhiyun /*============================================================================*/
4377*4882a593Smuzhiyun /*==                8VSB & QAM COMMON DATAPATH FUNCTIONS                    ==*/
4378*4882a593Smuzhiyun /*============================================================================*/
4379*4882a593Smuzhiyun /*============================================================================*/
4380*4882a593Smuzhiyun /*
4381*4882a593Smuzhiyun * \fn int init_agc ()
4382*4882a593Smuzhiyun * \brief Initialize AGC for all standards.
4383*4882a593Smuzhiyun * \param demod instance of demodulator.
4384*4882a593Smuzhiyun * \param channel pointer to channel data.
4385*4882a593Smuzhiyun * \return int.
4386*4882a593Smuzhiyun */
init_agc(struct drx_demod_instance * demod)4387*4882a593Smuzhiyun static int init_agc(struct drx_demod_instance *demod)
4388*4882a593Smuzhiyun {
4389*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
4390*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = NULL;
4391*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
4392*4882a593Smuzhiyun 	struct drxj_cfg_agc *p_agc_rf_settings = NULL;
4393*4882a593Smuzhiyun 	struct drxj_cfg_agc *p_agc_if_settings = NULL;
4394*4882a593Smuzhiyun 	int rc;
4395*4882a593Smuzhiyun 	u16 ingain_tgt_max = 0;
4396*4882a593Smuzhiyun 	u16 clp_dir_to = 0;
4397*4882a593Smuzhiyun 	u16 sns_sum_max = 0;
4398*4882a593Smuzhiyun 	u16 clp_sum_max = 0;
4399*4882a593Smuzhiyun 	u16 sns_dir_to = 0;
4400*4882a593Smuzhiyun 	u16 ki_innergain_min = 0;
4401*4882a593Smuzhiyun 	u16 agc_ki = 0;
4402*4882a593Smuzhiyun 	u16 ki_max = 0;
4403*4882a593Smuzhiyun 	u16 if_iaccu_hi_tgt_min = 0;
4404*4882a593Smuzhiyun 	u16 data = 0;
4405*4882a593Smuzhiyun 	u16 agc_ki_dgain = 0;
4406*4882a593Smuzhiyun 	u16 ki_min = 0;
4407*4882a593Smuzhiyun 	u16 clp_ctrl_mode = 0;
4408*4882a593Smuzhiyun 	u16 agc_rf = 0;
4409*4882a593Smuzhiyun 	u16 agc_if = 0;
4410*4882a593Smuzhiyun 
4411*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
4412*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
4413*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
4414*4882a593Smuzhiyun 
4415*4882a593Smuzhiyun 	switch (ext_attr->standard) {
4416*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
4417*4882a593Smuzhiyun 		clp_sum_max = 1023;
4418*4882a593Smuzhiyun 		clp_dir_to = (u16) (-9);
4419*4882a593Smuzhiyun 		sns_sum_max = 1023;
4420*4882a593Smuzhiyun 		sns_dir_to = (u16) (-9);
4421*4882a593Smuzhiyun 		ki_innergain_min = (u16) (-32768);
4422*4882a593Smuzhiyun 		ki_max = 0x032C;
4423*4882a593Smuzhiyun 		agc_ki_dgain = 0xC;
4424*4882a593Smuzhiyun 		if_iaccu_hi_tgt_min = 2047;
4425*4882a593Smuzhiyun 		ki_min = 0x0117;
4426*4882a593Smuzhiyun 		ingain_tgt_max = 16383;
4427*4882a593Smuzhiyun 		clp_ctrl_mode = 0;
4428*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MINGAIN__A, 0x7fff, 0);
4429*4882a593Smuzhiyun 		if (rc != 0) {
4430*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4431*4882a593Smuzhiyun 			goto rw_error;
4432*4882a593Smuzhiyun 		}
4433*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAXGAIN__A, 0x0, 0);
4434*4882a593Smuzhiyun 		if (rc != 0) {
4435*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4436*4882a593Smuzhiyun 			goto rw_error;
4437*4882a593Smuzhiyun 		}
4438*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM__A, 0, 0);
4439*4882a593Smuzhiyun 		if (rc != 0) {
4440*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4441*4882a593Smuzhiyun 			goto rw_error;
4442*4882a593Smuzhiyun 		}
4443*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CYCCNT__A, 0, 0);
4444*4882a593Smuzhiyun 		if (rc != 0) {
4445*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4446*4882a593Smuzhiyun 			goto rw_error;
4447*4882a593Smuzhiyun 		}
4448*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_WD__A, 0, 0);
4449*4882a593Smuzhiyun 		if (rc != 0) {
4450*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4451*4882a593Smuzhiyun 			goto rw_error;
4452*4882a593Smuzhiyun 		}
4453*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_STP__A, 1, 0);
4454*4882a593Smuzhiyun 		if (rc != 0) {
4455*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4456*4882a593Smuzhiyun 			goto rw_error;
4457*4882a593Smuzhiyun 		}
4458*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM__A, 0, 0);
4459*4882a593Smuzhiyun 		if (rc != 0) {
4460*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4461*4882a593Smuzhiyun 			goto rw_error;
4462*4882a593Smuzhiyun 		}
4463*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_CYCCNT__A, 0, 0);
4464*4882a593Smuzhiyun 		if (rc != 0) {
4465*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4466*4882a593Smuzhiyun 			goto rw_error;
4467*4882a593Smuzhiyun 		}
4468*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_WD__A, 0, 0);
4469*4882a593Smuzhiyun 		if (rc != 0) {
4470*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4471*4882a593Smuzhiyun 			goto rw_error;
4472*4882a593Smuzhiyun 		}
4473*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_STP__A, 1, 0);
4474*4882a593Smuzhiyun 		if (rc != 0) {
4475*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4476*4882a593Smuzhiyun 			goto rw_error;
4477*4882a593Smuzhiyun 		}
4478*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN__A, 1024, 0);
4479*4882a593Smuzhiyun 		if (rc != 0) {
4480*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4481*4882a593Smuzhiyun 			goto rw_error;
4482*4882a593Smuzhiyun 		}
4483*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_VSB_AGC_POW_TGT__A, 22600, 0);
4484*4882a593Smuzhiyun 		if (rc != 0) {
4485*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4486*4882a593Smuzhiyun 			goto rw_error;
4487*4882a593Smuzhiyun 		}
4488*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT__A, 13200, 0);
4489*4882a593Smuzhiyun 		if (rc != 0) {
4490*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4491*4882a593Smuzhiyun 			goto rw_error;
4492*4882a593Smuzhiyun 		}
4493*4882a593Smuzhiyun 		p_agc_if_settings = &(ext_attr->vsb_if_agc_cfg);
4494*4882a593Smuzhiyun 		p_agc_rf_settings = &(ext_attr->vsb_rf_agc_cfg);
4495*4882a593Smuzhiyun 		break;
4496*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
4497*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
4498*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
4499*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
4500*4882a593Smuzhiyun 		ingain_tgt_max = 5119;
4501*4882a593Smuzhiyun 		clp_sum_max = 1023;
4502*4882a593Smuzhiyun 		clp_dir_to = (u16) (-5);
4503*4882a593Smuzhiyun 		sns_sum_max = 127;
4504*4882a593Smuzhiyun 		sns_dir_to = (u16) (-3);
4505*4882a593Smuzhiyun 		ki_innergain_min = 0;
4506*4882a593Smuzhiyun 		ki_max = 0x0657;
4507*4882a593Smuzhiyun 		if_iaccu_hi_tgt_min = 2047;
4508*4882a593Smuzhiyun 		agc_ki_dgain = 0x7;
4509*4882a593Smuzhiyun 		ki_min = 0x0117;
4510*4882a593Smuzhiyun 		clp_ctrl_mode = 0;
4511*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MINGAIN__A, 0x7fff, 0);
4512*4882a593Smuzhiyun 		if (rc != 0) {
4513*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4514*4882a593Smuzhiyun 			goto rw_error;
4515*4882a593Smuzhiyun 		}
4516*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAXGAIN__A, 0x0, 0);
4517*4882a593Smuzhiyun 		if (rc != 0) {
4518*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4519*4882a593Smuzhiyun 			goto rw_error;
4520*4882a593Smuzhiyun 		}
4521*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM__A, 0, 0);
4522*4882a593Smuzhiyun 		if (rc != 0) {
4523*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4524*4882a593Smuzhiyun 			goto rw_error;
4525*4882a593Smuzhiyun 		}
4526*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CYCCNT__A, 0, 0);
4527*4882a593Smuzhiyun 		if (rc != 0) {
4528*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4529*4882a593Smuzhiyun 			goto rw_error;
4530*4882a593Smuzhiyun 		}
4531*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_WD__A, 0, 0);
4532*4882a593Smuzhiyun 		if (rc != 0) {
4533*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4534*4882a593Smuzhiyun 			goto rw_error;
4535*4882a593Smuzhiyun 		}
4536*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_STP__A, 1, 0);
4537*4882a593Smuzhiyun 		if (rc != 0) {
4538*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4539*4882a593Smuzhiyun 			goto rw_error;
4540*4882a593Smuzhiyun 		}
4541*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM__A, 0, 0);
4542*4882a593Smuzhiyun 		if (rc != 0) {
4543*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4544*4882a593Smuzhiyun 			goto rw_error;
4545*4882a593Smuzhiyun 		}
4546*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_CYCCNT__A, 0, 0);
4547*4882a593Smuzhiyun 		if (rc != 0) {
4548*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4549*4882a593Smuzhiyun 			goto rw_error;
4550*4882a593Smuzhiyun 		}
4551*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_WD__A, 0, 0);
4552*4882a593Smuzhiyun 		if (rc != 0) {
4553*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4554*4882a593Smuzhiyun 			goto rw_error;
4555*4882a593Smuzhiyun 		}
4556*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_STP__A, 1, 0);
4557*4882a593Smuzhiyun 		if (rc != 0) {
4558*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4559*4882a593Smuzhiyun 			goto rw_error;
4560*4882a593Smuzhiyun 		}
4561*4882a593Smuzhiyun 		p_agc_if_settings = &(ext_attr->qam_if_agc_cfg);
4562*4882a593Smuzhiyun 		p_agc_rf_settings = &(ext_attr->qam_rf_agc_cfg);
4563*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT__A, p_agc_if_settings->top, 0);
4564*4882a593Smuzhiyun 		if (rc != 0) {
4565*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4566*4882a593Smuzhiyun 			goto rw_error;
4567*4882a593Smuzhiyun 		}
4568*4882a593Smuzhiyun 
4569*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_AGC_KI__A, &agc_ki, 0);
4570*4882a593Smuzhiyun 		if (rc != 0) {
4571*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4572*4882a593Smuzhiyun 			goto rw_error;
4573*4882a593Smuzhiyun 		}
4574*4882a593Smuzhiyun 		agc_ki &= 0xf000;
4575*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI__A, agc_ki, 0);
4576*4882a593Smuzhiyun 		if (rc != 0) {
4577*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
4578*4882a593Smuzhiyun 			goto rw_error;
4579*4882a593Smuzhiyun 		}
4580*4882a593Smuzhiyun 		break;
4581*4882a593Smuzhiyun #endif
4582*4882a593Smuzhiyun 	default:
4583*4882a593Smuzhiyun 		return -EINVAL;
4584*4882a593Smuzhiyun 	}
4585*4882a593Smuzhiyun 
4586*4882a593Smuzhiyun 	/* for new AGC interface */
4587*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT_MIN__A, p_agc_if_settings->top, 0);
4588*4882a593Smuzhiyun 	if (rc != 0) {
4589*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4590*4882a593Smuzhiyun 		goto rw_error;
4591*4882a593Smuzhiyun 	}
4592*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN__A, p_agc_if_settings->top, 0);
4593*4882a593Smuzhiyun 	if (rc != 0) {
4594*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4595*4882a593Smuzhiyun 		goto rw_error;
4596*4882a593Smuzhiyun 	}	/* Gain fed from inner to outer AGC */
4597*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT_MAX__A, ingain_tgt_max, 0);
4598*4882a593Smuzhiyun 	if (rc != 0) {
4599*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4600*4882a593Smuzhiyun 		goto rw_error;
4601*4882a593Smuzhiyun 	}
4602*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MIN__A, if_iaccu_hi_tgt_min, 0);
4603*4882a593Smuzhiyun 	if (rc != 0) {
4604*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4605*4882a593Smuzhiyun 		goto rw_error;
4606*4882a593Smuzhiyun 	}
4607*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_IF_IACCU_HI__A, 0, 0);
4608*4882a593Smuzhiyun 	if (rc != 0) {
4609*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4610*4882a593Smuzhiyun 		goto rw_error;
4611*4882a593Smuzhiyun 	}	/* set to p_agc_settings->top before */
4612*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_IF_IACCU_LO__A, 0, 0);
4613*4882a593Smuzhiyun 	if (rc != 0) {
4614*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4615*4882a593Smuzhiyun 		goto rw_error;
4616*4882a593Smuzhiyun 	}
4617*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_RF_IACCU_HI__A, 0, 0);
4618*4882a593Smuzhiyun 	if (rc != 0) {
4619*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4620*4882a593Smuzhiyun 		goto rw_error;
4621*4882a593Smuzhiyun 	}
4622*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_RF_IACCU_LO__A, 0, 0);
4623*4882a593Smuzhiyun 	if (rc != 0) {
4624*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4625*4882a593Smuzhiyun 		goto rw_error;
4626*4882a593Smuzhiyun 	}
4627*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_RF_MAX__A, 32767, 0);
4628*4882a593Smuzhiyun 	if (rc != 0) {
4629*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4630*4882a593Smuzhiyun 		goto rw_error;
4631*4882a593Smuzhiyun 	}
4632*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM_MAX__A, clp_sum_max, 0);
4633*4882a593Smuzhiyun 	if (rc != 0) {
4634*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4635*4882a593Smuzhiyun 		goto rw_error;
4636*4882a593Smuzhiyun 	}
4637*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM_MAX__A, sns_sum_max, 0);
4638*4882a593Smuzhiyun 	if (rc != 0) {
4639*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4640*4882a593Smuzhiyun 		goto rw_error;
4641*4882a593Smuzhiyun 	}
4642*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_INNERGAIN_MIN__A, ki_innergain_min, 0);
4643*4882a593Smuzhiyun 	if (rc != 0) {
4644*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4645*4882a593Smuzhiyun 		goto rw_error;
4646*4882a593Smuzhiyun 	}
4647*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_FAST_SNS_CTRL_DELAY__A, 50, 0);
4648*4882a593Smuzhiyun 	if (rc != 0) {
4649*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4650*4882a593Smuzhiyun 		goto rw_error;
4651*4882a593Smuzhiyun 	}
4652*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_CYCLEN__A, 500, 0);
4653*4882a593Smuzhiyun 	if (rc != 0) {
4654*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4655*4882a593Smuzhiyun 		goto rw_error;
4656*4882a593Smuzhiyun 	}
4657*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_CYCLEN__A, 500, 0);
4658*4882a593Smuzhiyun 	if (rc != 0) {
4659*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4660*4882a593Smuzhiyun 		goto rw_error;
4661*4882a593Smuzhiyun 	}
4662*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAXMINGAIN_TH__A, 20, 0);
4663*4882a593Smuzhiyun 	if (rc != 0) {
4664*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4665*4882a593Smuzhiyun 		goto rw_error;
4666*4882a593Smuzhiyun 	}
4667*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MIN__A, ki_min, 0);
4668*4882a593Smuzhiyun 	if (rc != 0) {
4669*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4670*4882a593Smuzhiyun 		goto rw_error;
4671*4882a593Smuzhiyun 	}
4672*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAX__A, ki_max, 0);
4673*4882a593Smuzhiyun 	if (rc != 0) {
4674*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4675*4882a593Smuzhiyun 		goto rw_error;
4676*4882a593Smuzhiyun 	}
4677*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_RED__A, 0, 0);
4678*4882a593Smuzhiyun 	if (rc != 0) {
4679*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4680*4882a593Smuzhiyun 		goto rw_error;
4681*4882a593Smuzhiyun 	}
4682*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM_MIN__A, 8, 0);
4683*4882a593Smuzhiyun 	if (rc != 0) {
4684*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4685*4882a593Smuzhiyun 		goto rw_error;
4686*4882a593Smuzhiyun 	}
4687*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CYCLEN__A, 500, 0);
4688*4882a593Smuzhiyun 	if (rc != 0) {
4689*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4690*4882a593Smuzhiyun 		goto rw_error;
4691*4882a593Smuzhiyun 	}
4692*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_TO__A, clp_dir_to, 0);
4693*4882a593Smuzhiyun 	if (rc != 0) {
4694*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4695*4882a593Smuzhiyun 		goto rw_error;
4696*4882a593Smuzhiyun 	}
4697*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM_MIN__A, 8, 0);
4698*4882a593Smuzhiyun 	if (rc != 0) {
4699*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4700*4882a593Smuzhiyun 		goto rw_error;
4701*4882a593Smuzhiyun 	}
4702*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_TO__A, sns_dir_to, 0);
4703*4882a593Smuzhiyun 	if (rc != 0) {
4704*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4705*4882a593Smuzhiyun 		goto rw_error;
4706*4882a593Smuzhiyun 	}
4707*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_FAST_CLP_CTRL_DELAY__A, 50, 0);
4708*4882a593Smuzhiyun 	if (rc != 0) {
4709*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4710*4882a593Smuzhiyun 		goto rw_error;
4711*4882a593Smuzhiyun 	}
4712*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CTRL_MODE__A, clp_ctrl_mode, 0);
4713*4882a593Smuzhiyun 	if (rc != 0) {
4714*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4715*4882a593Smuzhiyun 		goto rw_error;
4716*4882a593Smuzhiyun 	}
4717*4882a593Smuzhiyun 
4718*4882a593Smuzhiyun 	agc_rf = 0x800 + p_agc_rf_settings->cut_off_current;
4719*4882a593Smuzhiyun 	if (common_attr->tuner_rf_agc_pol == true)
4720*4882a593Smuzhiyun 		agc_rf = 0x87ff - agc_rf;
4721*4882a593Smuzhiyun 
4722*4882a593Smuzhiyun 	agc_if = 0x800;
4723*4882a593Smuzhiyun 	if (common_attr->tuner_if_agc_pol == true)
4724*4882a593Smuzhiyun 		agc_rf = 0x87ff - agc_rf;
4725*4882a593Smuzhiyun 
4726*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AGC_RF__A, agc_rf, 0);
4727*4882a593Smuzhiyun 	if (rc != 0) {
4728*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4729*4882a593Smuzhiyun 		goto rw_error;
4730*4882a593Smuzhiyun 	}
4731*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AGC_IF__A, agc_if, 0);
4732*4882a593Smuzhiyun 	if (rc != 0) {
4733*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4734*4882a593Smuzhiyun 		goto rw_error;
4735*4882a593Smuzhiyun 	}
4736*4882a593Smuzhiyun 
4737*4882a593Smuzhiyun 	/* Set/restore Ki DGAIN factor */
4738*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
4739*4882a593Smuzhiyun 	if (rc != 0) {
4740*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4741*4882a593Smuzhiyun 		goto rw_error;
4742*4882a593Smuzhiyun 	}
4743*4882a593Smuzhiyun 	data &= ~SCU_RAM_AGC_KI_DGAIN__M;
4744*4882a593Smuzhiyun 	data |= (agc_ki_dgain << SCU_RAM_AGC_KI_DGAIN__B);
4745*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
4746*4882a593Smuzhiyun 	if (rc != 0) {
4747*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4748*4882a593Smuzhiyun 		goto rw_error;
4749*4882a593Smuzhiyun 	}
4750*4882a593Smuzhiyun 
4751*4882a593Smuzhiyun 	return 0;
4752*4882a593Smuzhiyun rw_error:
4753*4882a593Smuzhiyun 	return rc;
4754*4882a593Smuzhiyun }
4755*4882a593Smuzhiyun 
4756*4882a593Smuzhiyun /*
4757*4882a593Smuzhiyun * \fn int set_frequency ()
4758*4882a593Smuzhiyun * \brief Set frequency shift.
4759*4882a593Smuzhiyun * \param demod instance of demodulator.
4760*4882a593Smuzhiyun * \param channel pointer to channel data.
4761*4882a593Smuzhiyun * \param tuner_freq_offset residual frequency from tuner.
4762*4882a593Smuzhiyun * \return int.
4763*4882a593Smuzhiyun */
4764*4882a593Smuzhiyun static int
set_frequency(struct drx_demod_instance * demod,struct drx_channel * channel,s32 tuner_freq_offset)4765*4882a593Smuzhiyun set_frequency(struct drx_demod_instance *demod,
4766*4882a593Smuzhiyun 	      struct drx_channel *channel, s32 tuner_freq_offset)
4767*4882a593Smuzhiyun {
4768*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
4769*4882a593Smuzhiyun 	struct drxj_data *ext_attr = demod->my_ext_attr;
4770*4882a593Smuzhiyun 	int rc;
4771*4882a593Smuzhiyun 	s32 sampling_frequency = 0;
4772*4882a593Smuzhiyun 	s32 frequency_shift = 0;
4773*4882a593Smuzhiyun 	s32 if_freq_actual = 0;
4774*4882a593Smuzhiyun 	s32 rf_freq_residual = -1 * tuner_freq_offset;
4775*4882a593Smuzhiyun 	s32 adc_freq = 0;
4776*4882a593Smuzhiyun 	s32 intermediate_freq = 0;
4777*4882a593Smuzhiyun 	u32 iqm_fs_rate_ofs = 0;
4778*4882a593Smuzhiyun 	bool adc_flip = true;
4779*4882a593Smuzhiyun 	bool select_pos_image = false;
4780*4882a593Smuzhiyun 	bool rf_mirror;
4781*4882a593Smuzhiyun 	bool tuner_mirror;
4782*4882a593Smuzhiyun 	bool image_to_select = true;
4783*4882a593Smuzhiyun 	s32 fm_frequency_shift = 0;
4784*4882a593Smuzhiyun 
4785*4882a593Smuzhiyun 	rf_mirror = (ext_attr->mirror == DRX_MIRROR_YES) ? true : false;
4786*4882a593Smuzhiyun 	tuner_mirror = demod->my_common_attr->mirror_freq_spect ? false : true;
4787*4882a593Smuzhiyun 	/*
4788*4882a593Smuzhiyun 	   Program frequency shifter
4789*4882a593Smuzhiyun 	   No need to account for mirroring on RF
4790*4882a593Smuzhiyun 	 */
4791*4882a593Smuzhiyun 	switch (ext_attr->standard) {
4792*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
4793*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
4794*4882a593Smuzhiyun 	case DRX_STANDARD_PAL_SECAM_LP:
4795*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
4796*4882a593Smuzhiyun 		select_pos_image = true;
4797*4882a593Smuzhiyun 		break;
4798*4882a593Smuzhiyun 	case DRX_STANDARD_FM:
4799*4882a593Smuzhiyun 		/* After IQM FS sound carrier must appear at 4 Mhz in spect.
4800*4882a593Smuzhiyun 		   Sound carrier is already 3Mhz above centre frequency due
4801*4882a593Smuzhiyun 		   to tuner setting so now add an extra shift of 1MHz... */
4802*4882a593Smuzhiyun 		fm_frequency_shift = 1000;
4803*4882a593Smuzhiyun 		fallthrough;
4804*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
4805*4882a593Smuzhiyun 	case DRX_STANDARD_NTSC:
4806*4882a593Smuzhiyun 	case DRX_STANDARD_PAL_SECAM_BG:
4807*4882a593Smuzhiyun 	case DRX_STANDARD_PAL_SECAM_DK:
4808*4882a593Smuzhiyun 	case DRX_STANDARD_PAL_SECAM_I:
4809*4882a593Smuzhiyun 	case DRX_STANDARD_PAL_SECAM_L:
4810*4882a593Smuzhiyun 		select_pos_image = false;
4811*4882a593Smuzhiyun 		break;
4812*4882a593Smuzhiyun 	default:
4813*4882a593Smuzhiyun 		return -EINVAL;
4814*4882a593Smuzhiyun 	}
4815*4882a593Smuzhiyun 	intermediate_freq = demod->my_common_attr->intermediate_freq;
4816*4882a593Smuzhiyun 	sampling_frequency = demod->my_common_attr->sys_clock_freq / 3;
4817*4882a593Smuzhiyun 	if (tuner_mirror)
4818*4882a593Smuzhiyun 		if_freq_actual = intermediate_freq + rf_freq_residual + fm_frequency_shift;
4819*4882a593Smuzhiyun 	else
4820*4882a593Smuzhiyun 		if_freq_actual = intermediate_freq - rf_freq_residual - fm_frequency_shift;
4821*4882a593Smuzhiyun 	if (if_freq_actual > sampling_frequency / 2) {
4822*4882a593Smuzhiyun 		/* adc mirrors */
4823*4882a593Smuzhiyun 		adc_freq = sampling_frequency - if_freq_actual;
4824*4882a593Smuzhiyun 		adc_flip = true;
4825*4882a593Smuzhiyun 	} else {
4826*4882a593Smuzhiyun 		/* adc doesn't mirror */
4827*4882a593Smuzhiyun 		adc_freq = if_freq_actual;
4828*4882a593Smuzhiyun 		adc_flip = false;
4829*4882a593Smuzhiyun 	}
4830*4882a593Smuzhiyun 
4831*4882a593Smuzhiyun 	frequency_shift = adc_freq;
4832*4882a593Smuzhiyun 	image_to_select =
4833*4882a593Smuzhiyun 	    (bool) (rf_mirror ^ tuner_mirror ^ adc_flip ^ select_pos_image);
4834*4882a593Smuzhiyun 	iqm_fs_rate_ofs = frac28(frequency_shift, sampling_frequency);
4835*4882a593Smuzhiyun 
4836*4882a593Smuzhiyun 	if (image_to_select)
4837*4882a593Smuzhiyun 		iqm_fs_rate_ofs = ~iqm_fs_rate_ofs + 1;
4838*4882a593Smuzhiyun 
4839*4882a593Smuzhiyun 	/* Program frequency shifter with tuner offset compensation */
4840*4882a593Smuzhiyun 	/* frequency_shift += tuner_freq_offset; TODO */
4841*4882a593Smuzhiyun 	rc = drxdap_fasi_write_reg32(dev_addr, IQM_FS_RATE_OFS_LO__A, iqm_fs_rate_ofs, 0);
4842*4882a593Smuzhiyun 	if (rc != 0) {
4843*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4844*4882a593Smuzhiyun 		goto rw_error;
4845*4882a593Smuzhiyun 	}
4846*4882a593Smuzhiyun 	ext_attr->iqm_fs_rate_ofs = iqm_fs_rate_ofs;
4847*4882a593Smuzhiyun 	ext_attr->pos_image = (bool) (rf_mirror ^ tuner_mirror ^ select_pos_image);
4848*4882a593Smuzhiyun 
4849*4882a593Smuzhiyun 	return 0;
4850*4882a593Smuzhiyun rw_error:
4851*4882a593Smuzhiyun 	return rc;
4852*4882a593Smuzhiyun }
4853*4882a593Smuzhiyun 
4854*4882a593Smuzhiyun /*
4855*4882a593Smuzhiyun * \fn int get_acc_pkt_err()
4856*4882a593Smuzhiyun * \brief Retrieve signal strength for VSB and QAM.
4857*4882a593Smuzhiyun * \param demod Pointer to demod instance
4858*4882a593Smuzhiyun * \param packet_err Pointer to packet error
4859*4882a593Smuzhiyun * \return int.
4860*4882a593Smuzhiyun * \retval 0 sig_strength contains valid data.
4861*4882a593Smuzhiyun * \retval -EINVAL sig_strength is NULL.
4862*4882a593Smuzhiyun * \retval -EIO Erroneous data, sig_strength contains invalid data.
4863*4882a593Smuzhiyun */
4864*4882a593Smuzhiyun #ifdef DRXJ_SIGNAL_ACCUM_ERR
get_acc_pkt_err(struct drx_demod_instance * demod,u16 * packet_err)4865*4882a593Smuzhiyun static int get_acc_pkt_err(struct drx_demod_instance *demod, u16 *packet_err)
4866*4882a593Smuzhiyun {
4867*4882a593Smuzhiyun 	int rc;
4868*4882a593Smuzhiyun 	static u16 pkt_err;
4869*4882a593Smuzhiyun 	static u16 last_pkt_err;
4870*4882a593Smuzhiyun 	u16 data = 0;
4871*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
4872*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
4873*4882a593Smuzhiyun 
4874*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
4875*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
4876*4882a593Smuzhiyun 
4877*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_FEC_ACCUM_PKT_FAILURES__A, &data, 0);
4878*4882a593Smuzhiyun 	if (rc != 0) {
4879*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
4880*4882a593Smuzhiyun 		goto rw_error;
4881*4882a593Smuzhiyun 	}
4882*4882a593Smuzhiyun 	if (ext_attr->reset_pkt_err_acc) {
4883*4882a593Smuzhiyun 		last_pkt_err = data;
4884*4882a593Smuzhiyun 		pkt_err = 0;
4885*4882a593Smuzhiyun 		ext_attr->reset_pkt_err_acc = false;
4886*4882a593Smuzhiyun 	}
4887*4882a593Smuzhiyun 
4888*4882a593Smuzhiyun 	if (data < last_pkt_err) {
4889*4882a593Smuzhiyun 		pkt_err += 0xffff - last_pkt_err;
4890*4882a593Smuzhiyun 		pkt_err += data;
4891*4882a593Smuzhiyun 	} else {
4892*4882a593Smuzhiyun 		pkt_err += (data - last_pkt_err);
4893*4882a593Smuzhiyun 	}
4894*4882a593Smuzhiyun 	*packet_err = pkt_err;
4895*4882a593Smuzhiyun 	last_pkt_err = data;
4896*4882a593Smuzhiyun 
4897*4882a593Smuzhiyun 	return 0;
4898*4882a593Smuzhiyun rw_error:
4899*4882a593Smuzhiyun 	return rc;
4900*4882a593Smuzhiyun }
4901*4882a593Smuzhiyun #endif
4902*4882a593Smuzhiyun 
4903*4882a593Smuzhiyun 
4904*4882a593Smuzhiyun /*============================================================================*/
4905*4882a593Smuzhiyun 
4906*4882a593Smuzhiyun /*
4907*4882a593Smuzhiyun * \fn int set_agc_rf ()
4908*4882a593Smuzhiyun * \brief Configure RF AGC
4909*4882a593Smuzhiyun * \param demod instance of demodulator.
4910*4882a593Smuzhiyun * \param agc_settings AGC configuration structure
4911*4882a593Smuzhiyun * \return int.
4912*4882a593Smuzhiyun */
4913*4882a593Smuzhiyun static int
set_agc_rf(struct drx_demod_instance * demod,struct drxj_cfg_agc * agc_settings,bool atomic)4914*4882a593Smuzhiyun set_agc_rf(struct drx_demod_instance *demod, struct drxj_cfg_agc *agc_settings, bool atomic)
4915*4882a593Smuzhiyun {
4916*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
4917*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
4918*4882a593Smuzhiyun 	struct drxj_cfg_agc *p_agc_settings = NULL;
4919*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = NULL;
4920*4882a593Smuzhiyun 	int rc;
4921*4882a593Smuzhiyun 	drx_write_reg16func_t scu_wr16 = NULL;
4922*4882a593Smuzhiyun 	drx_read_reg16func_t scu_rr16 = NULL;
4923*4882a593Smuzhiyun 
4924*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
4925*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
4926*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
4927*4882a593Smuzhiyun 
4928*4882a593Smuzhiyun 	if (atomic) {
4929*4882a593Smuzhiyun 		scu_rr16 = drxj_dap_scu_atomic_read_reg16;
4930*4882a593Smuzhiyun 		scu_wr16 = drxj_dap_scu_atomic_write_reg16;
4931*4882a593Smuzhiyun 	} else {
4932*4882a593Smuzhiyun 		scu_rr16 = drxj_dap_read_reg16;
4933*4882a593Smuzhiyun 		scu_wr16 = drxj_dap_write_reg16;
4934*4882a593Smuzhiyun 	}
4935*4882a593Smuzhiyun 
4936*4882a593Smuzhiyun 	/* Configure AGC only if standard is currently active */
4937*4882a593Smuzhiyun 	if ((ext_attr->standard == agc_settings->standard) ||
4938*4882a593Smuzhiyun 	    (DRXJ_ISQAMSTD(ext_attr->standard) &&
4939*4882a593Smuzhiyun 	     DRXJ_ISQAMSTD(agc_settings->standard)) ||
4940*4882a593Smuzhiyun 	    (DRXJ_ISATVSTD(ext_attr->standard) &&
4941*4882a593Smuzhiyun 	     DRXJ_ISATVSTD(agc_settings->standard))) {
4942*4882a593Smuzhiyun 		u16 data = 0;
4943*4882a593Smuzhiyun 
4944*4882a593Smuzhiyun 		switch (agc_settings->ctrl_mode) {
4945*4882a593Smuzhiyun 		case DRX_AGC_CTRL_AUTO:
4946*4882a593Smuzhiyun 
4947*4882a593Smuzhiyun 			/* Enable RF AGC DAC */
4948*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
4949*4882a593Smuzhiyun 			if (rc != 0) {
4950*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4951*4882a593Smuzhiyun 				goto rw_error;
4952*4882a593Smuzhiyun 			}
4953*4882a593Smuzhiyun 			data |= IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE;
4954*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
4955*4882a593Smuzhiyun 			if (rc != 0) {
4956*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4957*4882a593Smuzhiyun 				goto rw_error;
4958*4882a593Smuzhiyun 			}
4959*4882a593Smuzhiyun 
4960*4882a593Smuzhiyun 			/* Enable SCU RF AGC loop */
4961*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
4962*4882a593Smuzhiyun 			if (rc != 0) {
4963*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4964*4882a593Smuzhiyun 				goto rw_error;
4965*4882a593Smuzhiyun 			}
4966*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_RF__M;
4967*4882a593Smuzhiyun 			if (ext_attr->standard == DRX_STANDARD_8VSB)
4968*4882a593Smuzhiyun 				data |= (2 << SCU_RAM_AGC_KI_RF__B);
4969*4882a593Smuzhiyun 			else if (DRXJ_ISQAMSTD(ext_attr->standard))
4970*4882a593Smuzhiyun 				data |= (5 << SCU_RAM_AGC_KI_RF__B);
4971*4882a593Smuzhiyun 			else
4972*4882a593Smuzhiyun 				data |= (4 << SCU_RAM_AGC_KI_RF__B);
4973*4882a593Smuzhiyun 
4974*4882a593Smuzhiyun 			if (common_attr->tuner_rf_agc_pol)
4975*4882a593Smuzhiyun 				data |= SCU_RAM_AGC_KI_INV_RF_POL__M;
4976*4882a593Smuzhiyun 			else
4977*4882a593Smuzhiyun 				data &= ~SCU_RAM_AGC_KI_INV_RF_POL__M;
4978*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
4979*4882a593Smuzhiyun 			if (rc != 0) {
4980*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4981*4882a593Smuzhiyun 				goto rw_error;
4982*4882a593Smuzhiyun 			}
4983*4882a593Smuzhiyun 
4984*4882a593Smuzhiyun 			/* Set speed ( using complementary reduction value ) */
4985*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI_RED__A, &data, 0);
4986*4882a593Smuzhiyun 			if (rc != 0) {
4987*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4988*4882a593Smuzhiyun 				goto rw_error;
4989*4882a593Smuzhiyun 			}
4990*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_RED_RAGC_RED__M;
4991*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI_RED__A, (~(agc_settings->speed << SCU_RAM_AGC_KI_RED_RAGC_RED__B) & SCU_RAM_AGC_KI_RED_RAGC_RED__M) | data, 0);
4992*4882a593Smuzhiyun 			if (rc != 0) {
4993*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
4994*4882a593Smuzhiyun 				goto rw_error;
4995*4882a593Smuzhiyun 			}
4996*4882a593Smuzhiyun 
4997*4882a593Smuzhiyun 			if (agc_settings->standard == DRX_STANDARD_8VSB)
4998*4882a593Smuzhiyun 				p_agc_settings = &(ext_attr->vsb_if_agc_cfg);
4999*4882a593Smuzhiyun 			else if (DRXJ_ISQAMSTD(agc_settings->standard))
5000*4882a593Smuzhiyun 				p_agc_settings = &(ext_attr->qam_if_agc_cfg);
5001*4882a593Smuzhiyun 			else if (DRXJ_ISATVSTD(agc_settings->standard))
5002*4882a593Smuzhiyun 				p_agc_settings = &(ext_attr->atv_if_agc_cfg);
5003*4882a593Smuzhiyun 			else
5004*4882a593Smuzhiyun 				return -EINVAL;
5005*4882a593Smuzhiyun 
5006*4882a593Smuzhiyun 			/* Set TOP, only if IF-AGC is in AUTO mode */
5007*4882a593Smuzhiyun 			if (p_agc_settings->ctrl_mode == DRX_AGC_CTRL_AUTO) {
5008*4882a593Smuzhiyun 				rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, agc_settings->top, 0);
5009*4882a593Smuzhiyun 				if (rc != 0) {
5010*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
5011*4882a593Smuzhiyun 					goto rw_error;
5012*4882a593Smuzhiyun 				}
5013*4882a593Smuzhiyun 				rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT__A, agc_settings->top, 0);
5014*4882a593Smuzhiyun 				if (rc != 0) {
5015*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
5016*4882a593Smuzhiyun 					goto rw_error;
5017*4882a593Smuzhiyun 				}
5018*4882a593Smuzhiyun 			}
5019*4882a593Smuzhiyun 
5020*4882a593Smuzhiyun 			/* Cut-Off current */
5021*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_RF_IACCU_HI_CO__A, agc_settings->cut_off_current, 0);
5022*4882a593Smuzhiyun 			if (rc != 0) {
5023*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5024*4882a593Smuzhiyun 				goto rw_error;
5025*4882a593Smuzhiyun 			}
5026*4882a593Smuzhiyun 			break;
5027*4882a593Smuzhiyun 		case DRX_AGC_CTRL_USER:
5028*4882a593Smuzhiyun 
5029*4882a593Smuzhiyun 			/* Enable RF AGC DAC */
5030*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
5031*4882a593Smuzhiyun 			if (rc != 0) {
5032*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5033*4882a593Smuzhiyun 				goto rw_error;
5034*4882a593Smuzhiyun 			}
5035*4882a593Smuzhiyun 			data |= IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE;
5036*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
5037*4882a593Smuzhiyun 			if (rc != 0) {
5038*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5039*4882a593Smuzhiyun 				goto rw_error;
5040*4882a593Smuzhiyun 			}
5041*4882a593Smuzhiyun 
5042*4882a593Smuzhiyun 			/* Disable SCU RF AGC loop */
5043*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
5044*4882a593Smuzhiyun 			if (rc != 0) {
5045*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5046*4882a593Smuzhiyun 				goto rw_error;
5047*4882a593Smuzhiyun 			}
5048*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_RF__M;
5049*4882a593Smuzhiyun 			if (common_attr->tuner_rf_agc_pol)
5050*4882a593Smuzhiyun 				data |= SCU_RAM_AGC_KI_INV_RF_POL__M;
5051*4882a593Smuzhiyun 			else
5052*4882a593Smuzhiyun 				data &= ~SCU_RAM_AGC_KI_INV_RF_POL__M;
5053*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
5054*4882a593Smuzhiyun 			if (rc != 0) {
5055*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5056*4882a593Smuzhiyun 				goto rw_error;
5057*4882a593Smuzhiyun 			}
5058*4882a593Smuzhiyun 
5059*4882a593Smuzhiyun 			/* Write value to output pin */
5060*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_RF_IACCU_HI__A, agc_settings->output_level, 0);
5061*4882a593Smuzhiyun 			if (rc != 0) {
5062*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5063*4882a593Smuzhiyun 				goto rw_error;
5064*4882a593Smuzhiyun 			}
5065*4882a593Smuzhiyun 			break;
5066*4882a593Smuzhiyun 		case DRX_AGC_CTRL_OFF:
5067*4882a593Smuzhiyun 
5068*4882a593Smuzhiyun 			/* Disable RF AGC DAC */
5069*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
5070*4882a593Smuzhiyun 			if (rc != 0) {
5071*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5072*4882a593Smuzhiyun 				goto rw_error;
5073*4882a593Smuzhiyun 			}
5074*4882a593Smuzhiyun 			data &= (~IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE);
5075*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
5076*4882a593Smuzhiyun 			if (rc != 0) {
5077*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5078*4882a593Smuzhiyun 				goto rw_error;
5079*4882a593Smuzhiyun 			}
5080*4882a593Smuzhiyun 
5081*4882a593Smuzhiyun 			/* Disable SCU RF AGC loop */
5082*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
5083*4882a593Smuzhiyun 			if (rc != 0) {
5084*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5085*4882a593Smuzhiyun 				goto rw_error;
5086*4882a593Smuzhiyun 			}
5087*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_RF__M;
5088*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
5089*4882a593Smuzhiyun 			if (rc != 0) {
5090*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5091*4882a593Smuzhiyun 				goto rw_error;
5092*4882a593Smuzhiyun 			}
5093*4882a593Smuzhiyun 			break;
5094*4882a593Smuzhiyun 		default:
5095*4882a593Smuzhiyun 			return -EINVAL;
5096*4882a593Smuzhiyun 		}		/* switch ( agcsettings->ctrl_mode ) */
5097*4882a593Smuzhiyun 	}
5098*4882a593Smuzhiyun 
5099*4882a593Smuzhiyun 	/* Store rf agc settings */
5100*4882a593Smuzhiyun 	switch (agc_settings->standard) {
5101*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
5102*4882a593Smuzhiyun 		ext_attr->vsb_rf_agc_cfg = *agc_settings;
5103*4882a593Smuzhiyun 		break;
5104*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
5105*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
5106*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
5107*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
5108*4882a593Smuzhiyun 		ext_attr->qam_rf_agc_cfg = *agc_settings;
5109*4882a593Smuzhiyun 		break;
5110*4882a593Smuzhiyun #endif
5111*4882a593Smuzhiyun 	default:
5112*4882a593Smuzhiyun 		return -EIO;
5113*4882a593Smuzhiyun 	}
5114*4882a593Smuzhiyun 
5115*4882a593Smuzhiyun 	return 0;
5116*4882a593Smuzhiyun rw_error:
5117*4882a593Smuzhiyun 	return rc;
5118*4882a593Smuzhiyun }
5119*4882a593Smuzhiyun 
5120*4882a593Smuzhiyun /*
5121*4882a593Smuzhiyun * \fn int set_agc_if ()
5122*4882a593Smuzhiyun * \brief Configure If AGC
5123*4882a593Smuzhiyun * \param demod instance of demodulator.
5124*4882a593Smuzhiyun * \param agc_settings AGC configuration structure
5125*4882a593Smuzhiyun * \return int.
5126*4882a593Smuzhiyun */
5127*4882a593Smuzhiyun static int
set_agc_if(struct drx_demod_instance * demod,struct drxj_cfg_agc * agc_settings,bool atomic)5128*4882a593Smuzhiyun set_agc_if(struct drx_demod_instance *demod, struct drxj_cfg_agc *agc_settings, bool atomic)
5129*4882a593Smuzhiyun {
5130*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
5131*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
5132*4882a593Smuzhiyun 	struct drxj_cfg_agc *p_agc_settings = NULL;
5133*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = NULL;
5134*4882a593Smuzhiyun 	drx_write_reg16func_t scu_wr16 = NULL;
5135*4882a593Smuzhiyun 	drx_read_reg16func_t scu_rr16 = NULL;
5136*4882a593Smuzhiyun 	int rc;
5137*4882a593Smuzhiyun 
5138*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
5139*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
5140*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
5141*4882a593Smuzhiyun 
5142*4882a593Smuzhiyun 	if (atomic) {
5143*4882a593Smuzhiyun 		scu_rr16 = drxj_dap_scu_atomic_read_reg16;
5144*4882a593Smuzhiyun 		scu_wr16 = drxj_dap_scu_atomic_write_reg16;
5145*4882a593Smuzhiyun 	} else {
5146*4882a593Smuzhiyun 		scu_rr16 = drxj_dap_read_reg16;
5147*4882a593Smuzhiyun 		scu_wr16 = drxj_dap_write_reg16;
5148*4882a593Smuzhiyun 	}
5149*4882a593Smuzhiyun 
5150*4882a593Smuzhiyun 	/* Configure AGC only if standard is currently active */
5151*4882a593Smuzhiyun 	if ((ext_attr->standard == agc_settings->standard) ||
5152*4882a593Smuzhiyun 	    (DRXJ_ISQAMSTD(ext_attr->standard) &&
5153*4882a593Smuzhiyun 	     DRXJ_ISQAMSTD(agc_settings->standard)) ||
5154*4882a593Smuzhiyun 	    (DRXJ_ISATVSTD(ext_attr->standard) &&
5155*4882a593Smuzhiyun 	     DRXJ_ISATVSTD(agc_settings->standard))) {
5156*4882a593Smuzhiyun 		u16 data = 0;
5157*4882a593Smuzhiyun 
5158*4882a593Smuzhiyun 		switch (agc_settings->ctrl_mode) {
5159*4882a593Smuzhiyun 		case DRX_AGC_CTRL_AUTO:
5160*4882a593Smuzhiyun 			/* Enable IF AGC DAC */
5161*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
5162*4882a593Smuzhiyun 			if (rc != 0) {
5163*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5164*4882a593Smuzhiyun 				goto rw_error;
5165*4882a593Smuzhiyun 			}
5166*4882a593Smuzhiyun 			data |= IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE;
5167*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
5168*4882a593Smuzhiyun 			if (rc != 0) {
5169*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5170*4882a593Smuzhiyun 				goto rw_error;
5171*4882a593Smuzhiyun 			}
5172*4882a593Smuzhiyun 
5173*4882a593Smuzhiyun 			/* Enable SCU IF AGC loop */
5174*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
5175*4882a593Smuzhiyun 			if (rc != 0) {
5176*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5177*4882a593Smuzhiyun 				goto rw_error;
5178*4882a593Smuzhiyun 			}
5179*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
5180*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_IF__M;
5181*4882a593Smuzhiyun 			if (ext_attr->standard == DRX_STANDARD_8VSB)
5182*4882a593Smuzhiyun 				data |= (3 << SCU_RAM_AGC_KI_IF__B);
5183*4882a593Smuzhiyun 			else if (DRXJ_ISQAMSTD(ext_attr->standard))
5184*4882a593Smuzhiyun 				data |= (6 << SCU_RAM_AGC_KI_IF__B);
5185*4882a593Smuzhiyun 			else
5186*4882a593Smuzhiyun 				data |= (5 << SCU_RAM_AGC_KI_IF__B);
5187*4882a593Smuzhiyun 
5188*4882a593Smuzhiyun 			if (common_attr->tuner_if_agc_pol)
5189*4882a593Smuzhiyun 				data |= SCU_RAM_AGC_KI_INV_IF_POL__M;
5190*4882a593Smuzhiyun 			else
5191*4882a593Smuzhiyun 				data &= ~SCU_RAM_AGC_KI_INV_IF_POL__M;
5192*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
5193*4882a593Smuzhiyun 			if (rc != 0) {
5194*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5195*4882a593Smuzhiyun 				goto rw_error;
5196*4882a593Smuzhiyun 			}
5197*4882a593Smuzhiyun 
5198*4882a593Smuzhiyun 			/* Set speed (using complementary reduction value) */
5199*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI_RED__A, &data, 0);
5200*4882a593Smuzhiyun 			if (rc != 0) {
5201*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5202*4882a593Smuzhiyun 				goto rw_error;
5203*4882a593Smuzhiyun 			}
5204*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_RED_IAGC_RED__M;
5205*4882a593Smuzhiyun 			rc = (*scu_wr16) (dev_addr, SCU_RAM_AGC_KI_RED__A, (~(agc_settings->speed << SCU_RAM_AGC_KI_RED_IAGC_RED__B) & SCU_RAM_AGC_KI_RED_IAGC_RED__M) | data, 0);
5206*4882a593Smuzhiyun 			if (rc != 0) {
5207*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5208*4882a593Smuzhiyun 				goto rw_error;
5209*4882a593Smuzhiyun 			}
5210*4882a593Smuzhiyun 
5211*4882a593Smuzhiyun 			if (agc_settings->standard == DRX_STANDARD_8VSB)
5212*4882a593Smuzhiyun 				p_agc_settings = &(ext_attr->vsb_rf_agc_cfg);
5213*4882a593Smuzhiyun 			else if (DRXJ_ISQAMSTD(agc_settings->standard))
5214*4882a593Smuzhiyun 				p_agc_settings = &(ext_attr->qam_rf_agc_cfg);
5215*4882a593Smuzhiyun 			else if (DRXJ_ISATVSTD(agc_settings->standard))
5216*4882a593Smuzhiyun 				p_agc_settings = &(ext_attr->atv_rf_agc_cfg);
5217*4882a593Smuzhiyun 			else
5218*4882a593Smuzhiyun 				return -EINVAL;
5219*4882a593Smuzhiyun 
5220*4882a593Smuzhiyun 			/* Restore TOP */
5221*4882a593Smuzhiyun 			if (p_agc_settings->ctrl_mode == DRX_AGC_CTRL_AUTO) {
5222*4882a593Smuzhiyun 				rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, p_agc_settings->top, 0);
5223*4882a593Smuzhiyun 				if (rc != 0) {
5224*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
5225*4882a593Smuzhiyun 					goto rw_error;
5226*4882a593Smuzhiyun 				}
5227*4882a593Smuzhiyun 				rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT__A, p_agc_settings->top, 0);
5228*4882a593Smuzhiyun 				if (rc != 0) {
5229*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
5230*4882a593Smuzhiyun 					goto rw_error;
5231*4882a593Smuzhiyun 				}
5232*4882a593Smuzhiyun 			} else {
5233*4882a593Smuzhiyun 				rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, 0, 0);
5234*4882a593Smuzhiyun 				if (rc != 0) {
5235*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
5236*4882a593Smuzhiyun 					goto rw_error;
5237*4882a593Smuzhiyun 				}
5238*4882a593Smuzhiyun 				rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT__A, 0, 0);
5239*4882a593Smuzhiyun 				if (rc != 0) {
5240*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
5241*4882a593Smuzhiyun 					goto rw_error;
5242*4882a593Smuzhiyun 				}
5243*4882a593Smuzhiyun 			}
5244*4882a593Smuzhiyun 			break;
5245*4882a593Smuzhiyun 
5246*4882a593Smuzhiyun 		case DRX_AGC_CTRL_USER:
5247*4882a593Smuzhiyun 
5248*4882a593Smuzhiyun 			/* Enable IF AGC DAC */
5249*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
5250*4882a593Smuzhiyun 			if (rc != 0) {
5251*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5252*4882a593Smuzhiyun 				goto rw_error;
5253*4882a593Smuzhiyun 			}
5254*4882a593Smuzhiyun 			data |= IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE;
5255*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
5256*4882a593Smuzhiyun 			if (rc != 0) {
5257*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5258*4882a593Smuzhiyun 				goto rw_error;
5259*4882a593Smuzhiyun 			}
5260*4882a593Smuzhiyun 
5261*4882a593Smuzhiyun 			/* Disable SCU IF AGC loop */
5262*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
5263*4882a593Smuzhiyun 			if (rc != 0) {
5264*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5265*4882a593Smuzhiyun 				goto rw_error;
5266*4882a593Smuzhiyun 			}
5267*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
5268*4882a593Smuzhiyun 			data |= SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
5269*4882a593Smuzhiyun 			if (common_attr->tuner_if_agc_pol)
5270*4882a593Smuzhiyun 				data |= SCU_RAM_AGC_KI_INV_IF_POL__M;
5271*4882a593Smuzhiyun 			else
5272*4882a593Smuzhiyun 				data &= ~SCU_RAM_AGC_KI_INV_IF_POL__M;
5273*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
5274*4882a593Smuzhiyun 			if (rc != 0) {
5275*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5276*4882a593Smuzhiyun 				goto rw_error;
5277*4882a593Smuzhiyun 			}
5278*4882a593Smuzhiyun 
5279*4882a593Smuzhiyun 			/* Write value to output pin */
5280*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, agc_settings->output_level, 0);
5281*4882a593Smuzhiyun 			if (rc != 0) {
5282*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5283*4882a593Smuzhiyun 				goto rw_error;
5284*4882a593Smuzhiyun 			}
5285*4882a593Smuzhiyun 			break;
5286*4882a593Smuzhiyun 
5287*4882a593Smuzhiyun 		case DRX_AGC_CTRL_OFF:
5288*4882a593Smuzhiyun 
5289*4882a593Smuzhiyun 			/* Disable If AGC DAC */
5290*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
5291*4882a593Smuzhiyun 			if (rc != 0) {
5292*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5293*4882a593Smuzhiyun 				goto rw_error;
5294*4882a593Smuzhiyun 			}
5295*4882a593Smuzhiyun 			data &= (~IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE);
5296*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
5297*4882a593Smuzhiyun 			if (rc != 0) {
5298*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5299*4882a593Smuzhiyun 				goto rw_error;
5300*4882a593Smuzhiyun 			}
5301*4882a593Smuzhiyun 
5302*4882a593Smuzhiyun 			/* Disable SCU IF AGC loop */
5303*4882a593Smuzhiyun 			rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
5304*4882a593Smuzhiyun 			if (rc != 0) {
5305*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5306*4882a593Smuzhiyun 				goto rw_error;
5307*4882a593Smuzhiyun 			}
5308*4882a593Smuzhiyun 			data &= ~SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
5309*4882a593Smuzhiyun 			data |= SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
5310*4882a593Smuzhiyun 			rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
5311*4882a593Smuzhiyun 			if (rc != 0) {
5312*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
5313*4882a593Smuzhiyun 				goto rw_error;
5314*4882a593Smuzhiyun 			}
5315*4882a593Smuzhiyun 			break;
5316*4882a593Smuzhiyun 		default:
5317*4882a593Smuzhiyun 			return -EINVAL;
5318*4882a593Smuzhiyun 		}		/* switch ( agcsettings->ctrl_mode ) */
5319*4882a593Smuzhiyun 
5320*4882a593Smuzhiyun 		/* always set the top to support configurations without if-loop */
5321*4882a593Smuzhiyun 		rc = (*scu_wr16) (dev_addr, SCU_RAM_AGC_INGAIN_TGT_MIN__A, agc_settings->top, 0);
5322*4882a593Smuzhiyun 		if (rc != 0) {
5323*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5324*4882a593Smuzhiyun 			goto rw_error;
5325*4882a593Smuzhiyun 		}
5326*4882a593Smuzhiyun 	}
5327*4882a593Smuzhiyun 
5328*4882a593Smuzhiyun 	/* Store if agc settings */
5329*4882a593Smuzhiyun 	switch (agc_settings->standard) {
5330*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
5331*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg = *agc_settings;
5332*4882a593Smuzhiyun 		break;
5333*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
5334*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
5335*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
5336*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
5337*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg = *agc_settings;
5338*4882a593Smuzhiyun 		break;
5339*4882a593Smuzhiyun #endif
5340*4882a593Smuzhiyun 	default:
5341*4882a593Smuzhiyun 		return -EIO;
5342*4882a593Smuzhiyun 	}
5343*4882a593Smuzhiyun 
5344*4882a593Smuzhiyun 	return 0;
5345*4882a593Smuzhiyun rw_error:
5346*4882a593Smuzhiyun 	return rc;
5347*4882a593Smuzhiyun }
5348*4882a593Smuzhiyun 
5349*4882a593Smuzhiyun /*
5350*4882a593Smuzhiyun * \fn int set_iqm_af ()
5351*4882a593Smuzhiyun * \brief Configure IQM AF registers
5352*4882a593Smuzhiyun * \param demod instance of demodulator.
5353*4882a593Smuzhiyun * \param active
5354*4882a593Smuzhiyun * \return int.
5355*4882a593Smuzhiyun */
set_iqm_af(struct drx_demod_instance * demod,bool active)5356*4882a593Smuzhiyun static int set_iqm_af(struct drx_demod_instance *demod, bool active)
5357*4882a593Smuzhiyun {
5358*4882a593Smuzhiyun 	u16 data = 0;
5359*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
5360*4882a593Smuzhiyun 	int rc;
5361*4882a593Smuzhiyun 
5362*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
5363*4882a593Smuzhiyun 
5364*4882a593Smuzhiyun 	/* Configure IQM */
5365*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
5366*4882a593Smuzhiyun 	if (rc != 0) {
5367*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5368*4882a593Smuzhiyun 		goto rw_error;
5369*4882a593Smuzhiyun 	}
5370*4882a593Smuzhiyun 	if (!active)
5371*4882a593Smuzhiyun 		data &= ((~IQM_AF_STDBY_STDBY_ADC_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_AMP_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_PD_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE));
5372*4882a593Smuzhiyun 	else
5373*4882a593Smuzhiyun 		data |= (IQM_AF_STDBY_STDBY_ADC_A2_ACTIVE | IQM_AF_STDBY_STDBY_AMP_A2_ACTIVE | IQM_AF_STDBY_STDBY_PD_A2_ACTIVE | IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE | IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE);
5374*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
5375*4882a593Smuzhiyun 	if (rc != 0) {
5376*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5377*4882a593Smuzhiyun 		goto rw_error;
5378*4882a593Smuzhiyun 	}
5379*4882a593Smuzhiyun 
5380*4882a593Smuzhiyun 	return 0;
5381*4882a593Smuzhiyun rw_error:
5382*4882a593Smuzhiyun 	return rc;
5383*4882a593Smuzhiyun }
5384*4882a593Smuzhiyun 
5385*4882a593Smuzhiyun /*============================================================================*/
5386*4882a593Smuzhiyun /*==              END 8VSB & QAM COMMON DATAPATH FUNCTIONS                  ==*/
5387*4882a593Smuzhiyun /*============================================================================*/
5388*4882a593Smuzhiyun 
5389*4882a593Smuzhiyun /*============================================================================*/
5390*4882a593Smuzhiyun /*============================================================================*/
5391*4882a593Smuzhiyun /*==                       8VSB DATAPATH FUNCTIONS                          ==*/
5392*4882a593Smuzhiyun /*============================================================================*/
5393*4882a593Smuzhiyun /*============================================================================*/
5394*4882a593Smuzhiyun 
5395*4882a593Smuzhiyun /*
5396*4882a593Smuzhiyun * \fn int power_down_vsb ()
5397*4882a593Smuzhiyun * \brief Powr down QAM related blocks.
5398*4882a593Smuzhiyun * \param demod instance of demodulator.
5399*4882a593Smuzhiyun * \param channel pointer to channel data.
5400*4882a593Smuzhiyun * \return int.
5401*4882a593Smuzhiyun */
power_down_vsb(struct drx_demod_instance * demod,bool primary)5402*4882a593Smuzhiyun static int power_down_vsb(struct drx_demod_instance *demod, bool primary)
5403*4882a593Smuzhiyun {
5404*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
5405*4882a593Smuzhiyun 	struct drxjscu_cmd cmd_scu = { /* command     */ 0,
5406*4882a593Smuzhiyun 		/* parameter_len */ 0,
5407*4882a593Smuzhiyun 		/* result_len    */ 0,
5408*4882a593Smuzhiyun 		/* *parameter   */ NULL,
5409*4882a593Smuzhiyun 		/* *result      */ NULL
5410*4882a593Smuzhiyun 	};
5411*4882a593Smuzhiyun 	struct drx_cfg_mpeg_output cfg_mpeg_output;
5412*4882a593Smuzhiyun 	int rc;
5413*4882a593Smuzhiyun 	u16 cmd_result = 0;
5414*4882a593Smuzhiyun 
5415*4882a593Smuzhiyun 	/*
5416*4882a593Smuzhiyun 	   STOP demodulator
5417*4882a593Smuzhiyun 	   reset of FEC and VSB HW
5418*4882a593Smuzhiyun 	 */
5419*4882a593Smuzhiyun 	cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB |
5420*4882a593Smuzhiyun 	    SCU_RAM_COMMAND_CMD_DEMOD_STOP;
5421*4882a593Smuzhiyun 	cmd_scu.parameter_len = 0;
5422*4882a593Smuzhiyun 	cmd_scu.result_len = 1;
5423*4882a593Smuzhiyun 	cmd_scu.parameter = NULL;
5424*4882a593Smuzhiyun 	cmd_scu.result = &cmd_result;
5425*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &cmd_scu);
5426*4882a593Smuzhiyun 	if (rc != 0) {
5427*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5428*4882a593Smuzhiyun 		goto rw_error;
5429*4882a593Smuzhiyun 	}
5430*4882a593Smuzhiyun 
5431*4882a593Smuzhiyun 	/* stop all comm_exec */
5432*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
5433*4882a593Smuzhiyun 	if (rc != 0) {
5434*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5435*4882a593Smuzhiyun 		goto rw_error;
5436*4882a593Smuzhiyun 	}
5437*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_COMM_EXEC__A, VSB_COMM_EXEC_STOP, 0);
5438*4882a593Smuzhiyun 	if (rc != 0) {
5439*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5440*4882a593Smuzhiyun 		goto rw_error;
5441*4882a593Smuzhiyun 	}
5442*4882a593Smuzhiyun 	if (primary) {
5443*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_STOP, 0);
5444*4882a593Smuzhiyun 		if (rc != 0) {
5445*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5446*4882a593Smuzhiyun 			goto rw_error;
5447*4882a593Smuzhiyun 		}
5448*4882a593Smuzhiyun 		rc = set_iqm_af(demod, false);
5449*4882a593Smuzhiyun 		if (rc != 0) {
5450*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5451*4882a593Smuzhiyun 			goto rw_error;
5452*4882a593Smuzhiyun 		}
5453*4882a593Smuzhiyun 	} else {
5454*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
5455*4882a593Smuzhiyun 		if (rc != 0) {
5456*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5457*4882a593Smuzhiyun 			goto rw_error;
5458*4882a593Smuzhiyun 		}
5459*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
5460*4882a593Smuzhiyun 		if (rc != 0) {
5461*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5462*4882a593Smuzhiyun 			goto rw_error;
5463*4882a593Smuzhiyun 		}
5464*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
5465*4882a593Smuzhiyun 		if (rc != 0) {
5466*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5467*4882a593Smuzhiyun 			goto rw_error;
5468*4882a593Smuzhiyun 		}
5469*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
5470*4882a593Smuzhiyun 		if (rc != 0) {
5471*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5472*4882a593Smuzhiyun 			goto rw_error;
5473*4882a593Smuzhiyun 		}
5474*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
5475*4882a593Smuzhiyun 		if (rc != 0) {
5476*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5477*4882a593Smuzhiyun 			goto rw_error;
5478*4882a593Smuzhiyun 		}
5479*4882a593Smuzhiyun 	}
5480*4882a593Smuzhiyun 
5481*4882a593Smuzhiyun 	cfg_mpeg_output.enable_mpeg_output = false;
5482*4882a593Smuzhiyun 	rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
5483*4882a593Smuzhiyun 	if (rc != 0) {
5484*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5485*4882a593Smuzhiyun 		goto rw_error;
5486*4882a593Smuzhiyun 	}
5487*4882a593Smuzhiyun 
5488*4882a593Smuzhiyun 	return 0;
5489*4882a593Smuzhiyun rw_error:
5490*4882a593Smuzhiyun 	return rc;
5491*4882a593Smuzhiyun }
5492*4882a593Smuzhiyun 
5493*4882a593Smuzhiyun /*
5494*4882a593Smuzhiyun * \fn int set_vsb_leak_n_gain ()
5495*4882a593Smuzhiyun * \brief Set ATSC demod.
5496*4882a593Smuzhiyun * \param demod instance of demodulator.
5497*4882a593Smuzhiyun * \return int.
5498*4882a593Smuzhiyun */
set_vsb_leak_n_gain(struct drx_demod_instance * demod)5499*4882a593Smuzhiyun static int set_vsb_leak_n_gain(struct drx_demod_instance *demod)
5500*4882a593Smuzhiyun {
5501*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
5502*4882a593Smuzhiyun 	int rc;
5503*4882a593Smuzhiyun 
5504*4882a593Smuzhiyun 	static const u8 vsb_ffe_leak_gain_ram0[] = {
5505*4882a593Smuzhiyun 		DRXJ_16TO8(0x8),	/* FFETRAINLKRATIO1  */
5506*4882a593Smuzhiyun 		DRXJ_16TO8(0x8),	/* FFETRAINLKRATIO2  */
5507*4882a593Smuzhiyun 		DRXJ_16TO8(0x8),	/* FFETRAINLKRATIO3  */
5508*4882a593Smuzhiyun 		DRXJ_16TO8(0xf),	/* FFETRAINLKRATIO4  */
5509*4882a593Smuzhiyun 		DRXJ_16TO8(0xf),	/* FFETRAINLKRATIO5  */
5510*4882a593Smuzhiyun 		DRXJ_16TO8(0xf),	/* FFETRAINLKRATIO6  */
5511*4882a593Smuzhiyun 		DRXJ_16TO8(0xf),	/* FFETRAINLKRATIO7  */
5512*4882a593Smuzhiyun 		DRXJ_16TO8(0xf),	/* FFETRAINLKRATIO8  */
5513*4882a593Smuzhiyun 		DRXJ_16TO8(0xf),	/* FFETRAINLKRATIO9  */
5514*4882a593Smuzhiyun 		DRXJ_16TO8(0x8),	/* FFETRAINLKRATIO10  */
5515*4882a593Smuzhiyun 		DRXJ_16TO8(0x8),	/* FFETRAINLKRATIO11 */
5516*4882a593Smuzhiyun 		DRXJ_16TO8(0x8),	/* FFETRAINLKRATIO12 */
5517*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1TRAINLKRATIO1 */
5518*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1TRAINLKRATIO2 */
5519*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1TRAINLKRATIO3 */
5520*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1TRAINLKRATIO4 */
5521*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1TRAINLKRATIO5 */
5522*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1TRAINLKRATIO6 */
5523*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1TRAINLKRATIO7 */
5524*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1TRAINLKRATIO8 */
5525*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1TRAINLKRATIO9 */
5526*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1TRAINLKRATIO10 */
5527*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1TRAINLKRATIO11 */
5528*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1TRAINLKRATIO12 */
5529*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1DATALKRATIO1 */
5530*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1DATALKRATIO2 */
5531*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1DATALKRATIO3 */
5532*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1DATALKRATIO4 */
5533*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1DATALKRATIO5 */
5534*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1DATALKRATIO6 */
5535*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1DATALKRATIO7 */
5536*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1DATALKRATIO8 */
5537*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA1DATALKRATIO9 */
5538*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1DATALKRATIO10 */
5539*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1DATALKRATIO11 */
5540*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA1DATALKRATIO12 */
5541*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2TRAINLKRATIO1 */
5542*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2TRAINLKRATIO2 */
5543*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2TRAINLKRATIO3 */
5544*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2TRAINLKRATIO4 */
5545*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2TRAINLKRATIO5 */
5546*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2TRAINLKRATIO6 */
5547*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2TRAINLKRATIO7 */
5548*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2TRAINLKRATIO8 */
5549*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2TRAINLKRATIO9 */
5550*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2TRAINLKRATIO10 */
5551*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2TRAINLKRATIO11 */
5552*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2TRAINLKRATIO12 */
5553*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2DATALKRATIO1 */
5554*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2DATALKRATIO2 */
5555*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2DATALKRATIO3 */
5556*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2DATALKRATIO4 */
5557*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2DATALKRATIO5 */
5558*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2DATALKRATIO6 */
5559*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2DATALKRATIO7 */
5560*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2DATALKRATIO8 */
5561*4882a593Smuzhiyun 		DRXJ_16TO8(0x20),	/* FFERCA2DATALKRATIO9 */
5562*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2DATALKRATIO10 */
5563*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2DATALKRATIO11 */
5564*4882a593Smuzhiyun 		DRXJ_16TO8(0x10),	/* FFERCA2DATALKRATIO12 */
5565*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1TRAINLKRATIO1 */
5566*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1TRAINLKRATIO2 */
5567*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1TRAINLKRATIO3 */
5568*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1TRAINLKRATIO4 */
5569*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1TRAINLKRATIO5 */
5570*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1TRAINLKRATIO6 */
5571*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1TRAINLKRATIO7 */
5572*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1TRAINLKRATIO8 */
5573*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1TRAINLKRATIO9 */
5574*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1TRAINLKRATIO10 */
5575*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1TRAINLKRATIO11 */
5576*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1TRAINLKRATIO12 */
5577*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1DATALKRATIO1 */
5578*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1DATALKRATIO2 */
5579*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1DATALKRATIO3 */
5580*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1DATALKRATIO4 */
5581*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1DATALKRATIO5 */
5582*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1DATALKRATIO6 */
5583*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1DATALKRATIO7 */
5584*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1DATALKRATIO8 */
5585*4882a593Smuzhiyun 		DRXJ_16TO8(0x0e),	/* FFEDDM1DATALKRATIO9 */
5586*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1DATALKRATIO10 */
5587*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1DATALKRATIO11 */
5588*4882a593Smuzhiyun 		DRXJ_16TO8(0x07),	/* FFEDDM1DATALKRATIO12 */
5589*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2TRAINLKRATIO1 */
5590*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2TRAINLKRATIO2 */
5591*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2TRAINLKRATIO3 */
5592*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2TRAINLKRATIO4 */
5593*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2TRAINLKRATIO5 */
5594*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2TRAINLKRATIO6 */
5595*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2TRAINLKRATIO7 */
5596*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2TRAINLKRATIO8 */
5597*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2TRAINLKRATIO9 */
5598*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2TRAINLKRATIO10 */
5599*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2TRAINLKRATIO11 */
5600*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2TRAINLKRATIO12 */
5601*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2DATALKRATIO1 */
5602*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2DATALKRATIO2 */
5603*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2DATALKRATIO3 */
5604*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2DATALKRATIO4 */
5605*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2DATALKRATIO5 */
5606*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2DATALKRATIO6 */
5607*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2DATALKRATIO7 */
5608*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2DATALKRATIO8 */
5609*4882a593Smuzhiyun 		DRXJ_16TO8(0x0c),	/* FFEDDM2DATALKRATIO9 */
5610*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2DATALKRATIO10 */
5611*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2DATALKRATIO11 */
5612*4882a593Smuzhiyun 		DRXJ_16TO8(0x06),	/* FFEDDM2DATALKRATIO12 */
5613*4882a593Smuzhiyun 		DRXJ_16TO8(0x2020),	/* FIRTRAINGAIN1 */
5614*4882a593Smuzhiyun 		DRXJ_16TO8(0x2020),	/* FIRTRAINGAIN2 */
5615*4882a593Smuzhiyun 		DRXJ_16TO8(0x2020),	/* FIRTRAINGAIN3 */
5616*4882a593Smuzhiyun 		DRXJ_16TO8(0x4040),	/* FIRTRAINGAIN4 */
5617*4882a593Smuzhiyun 		DRXJ_16TO8(0x4040),	/* FIRTRAINGAIN5 */
5618*4882a593Smuzhiyun 		DRXJ_16TO8(0x4040),	/* FIRTRAINGAIN6 */
5619*4882a593Smuzhiyun 		DRXJ_16TO8(0x4040),	/* FIRTRAINGAIN7 */
5620*4882a593Smuzhiyun 		DRXJ_16TO8(0x4040),	/* FIRTRAINGAIN8 */
5621*4882a593Smuzhiyun 		DRXJ_16TO8(0x4040),	/* FIRTRAINGAIN9 */
5622*4882a593Smuzhiyun 		DRXJ_16TO8(0x2020),	/* FIRTRAINGAIN10 */
5623*4882a593Smuzhiyun 		DRXJ_16TO8(0x2020),	/* FIRTRAINGAIN11 */
5624*4882a593Smuzhiyun 		DRXJ_16TO8(0x2020),	/* FIRTRAINGAIN12 */
5625*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA1GAIN1 */
5626*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA1GAIN2 */
5627*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA1GAIN3 */
5628*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA1GAIN4 */
5629*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA1GAIN5 */
5630*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA1GAIN6 */
5631*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA1GAIN7 */
5632*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010)	/* FIRRCA1GAIN8 */
5633*4882a593Smuzhiyun 	};
5634*4882a593Smuzhiyun 
5635*4882a593Smuzhiyun 	static const u8 vsb_ffe_leak_gain_ram1[] = {
5636*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA1GAIN9 */
5637*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA1GAIN10 */
5638*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA1GAIN11 */
5639*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA1GAIN12 */
5640*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA2GAIN1 */
5641*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA2GAIN2 */
5642*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA2GAIN3 */
5643*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA2GAIN4 */
5644*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA2GAIN5 */
5645*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA2GAIN6 */
5646*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA2GAIN7 */
5647*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA2GAIN8 */
5648*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* FIRRCA2GAIN9 */
5649*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA2GAIN10 */
5650*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA2GAIN11 */
5651*4882a593Smuzhiyun 		DRXJ_16TO8(0x0808),	/* FIRRCA2GAIN12 */
5652*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM1GAIN1 */
5653*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM1GAIN2 */
5654*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM1GAIN3 */
5655*4882a593Smuzhiyun 		DRXJ_16TO8(0x0606),	/* FIRDDM1GAIN4 */
5656*4882a593Smuzhiyun 		DRXJ_16TO8(0x0606),	/* FIRDDM1GAIN5 */
5657*4882a593Smuzhiyun 		DRXJ_16TO8(0x0606),	/* FIRDDM1GAIN6 */
5658*4882a593Smuzhiyun 		DRXJ_16TO8(0x0606),	/* FIRDDM1GAIN7 */
5659*4882a593Smuzhiyun 		DRXJ_16TO8(0x0606),	/* FIRDDM1GAIN8 */
5660*4882a593Smuzhiyun 		DRXJ_16TO8(0x0606),	/* FIRDDM1GAIN9 */
5661*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM1GAIN10 */
5662*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM1GAIN11 */
5663*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM1GAIN12 */
5664*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM2GAIN1 */
5665*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM2GAIN2 */
5666*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM2GAIN3 */
5667*4882a593Smuzhiyun 		DRXJ_16TO8(0x0505),	/* FIRDDM2GAIN4 */
5668*4882a593Smuzhiyun 		DRXJ_16TO8(0x0505),	/* FIRDDM2GAIN5 */
5669*4882a593Smuzhiyun 		DRXJ_16TO8(0x0505),	/* FIRDDM2GAIN6 */
5670*4882a593Smuzhiyun 		DRXJ_16TO8(0x0505),	/* FIRDDM2GAIN7 */
5671*4882a593Smuzhiyun 		DRXJ_16TO8(0x0505),	/* FIRDDM2GAIN8 */
5672*4882a593Smuzhiyun 		DRXJ_16TO8(0x0505),	/* FIRDDM2GAIN9 */
5673*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM2GAIN10 */
5674*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM2GAIN11 */
5675*4882a593Smuzhiyun 		DRXJ_16TO8(0x0303),	/* FIRDDM2GAIN12 */
5676*4882a593Smuzhiyun 		DRXJ_16TO8(0x001f),	/* DFETRAINLKRATIO */
5677*4882a593Smuzhiyun 		DRXJ_16TO8(0x01ff),	/* DFERCA1TRAINLKRATIO */
5678*4882a593Smuzhiyun 		DRXJ_16TO8(0x01ff),	/* DFERCA1DATALKRATIO */
5679*4882a593Smuzhiyun 		DRXJ_16TO8(0x004f),	/* DFERCA2TRAINLKRATIO */
5680*4882a593Smuzhiyun 		DRXJ_16TO8(0x004f),	/* DFERCA2DATALKRATIO */
5681*4882a593Smuzhiyun 		DRXJ_16TO8(0x01ff),	/* DFEDDM1TRAINLKRATIO */
5682*4882a593Smuzhiyun 		DRXJ_16TO8(0x01ff),	/* DFEDDM1DATALKRATIO */
5683*4882a593Smuzhiyun 		DRXJ_16TO8(0x0352),	/* DFEDDM2TRAINLKRATIO */
5684*4882a593Smuzhiyun 		DRXJ_16TO8(0x0352),	/* DFEDDM2DATALKRATIO */
5685*4882a593Smuzhiyun 		DRXJ_16TO8(0x0000),	/* DFETRAINGAIN */
5686*4882a593Smuzhiyun 		DRXJ_16TO8(0x2020),	/* DFERCA1GAIN */
5687*4882a593Smuzhiyun 		DRXJ_16TO8(0x1010),	/* DFERCA2GAIN */
5688*4882a593Smuzhiyun 		DRXJ_16TO8(0x1818),	/* DFEDDM1GAIN */
5689*4882a593Smuzhiyun 		DRXJ_16TO8(0x1212)	/* DFEDDM2GAIN */
5690*4882a593Smuzhiyun 	};
5691*4882a593Smuzhiyun 
5692*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
5693*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, VSB_SYSCTRL_RAM0_FFETRAINLKRATIO1__A, sizeof(vsb_ffe_leak_gain_ram0), ((u8 *)vsb_ffe_leak_gain_ram0), 0);
5694*4882a593Smuzhiyun 	if (rc != 0) {
5695*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5696*4882a593Smuzhiyun 		goto rw_error;
5697*4882a593Smuzhiyun 	}
5698*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, VSB_SYSCTRL_RAM1_FIRRCA1GAIN9__A, sizeof(vsb_ffe_leak_gain_ram1), ((u8 *)vsb_ffe_leak_gain_ram1), 0);
5699*4882a593Smuzhiyun 	if (rc != 0) {
5700*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5701*4882a593Smuzhiyun 		goto rw_error;
5702*4882a593Smuzhiyun 	}
5703*4882a593Smuzhiyun 
5704*4882a593Smuzhiyun 	return 0;
5705*4882a593Smuzhiyun rw_error:
5706*4882a593Smuzhiyun 	return rc;
5707*4882a593Smuzhiyun }
5708*4882a593Smuzhiyun 
5709*4882a593Smuzhiyun /*
5710*4882a593Smuzhiyun * \fn int set_vsb()
5711*4882a593Smuzhiyun * \brief Set 8VSB demod.
5712*4882a593Smuzhiyun * \param demod instance of demodulator.
5713*4882a593Smuzhiyun * \return int.
5714*4882a593Smuzhiyun *
5715*4882a593Smuzhiyun */
set_vsb(struct drx_demod_instance * demod)5716*4882a593Smuzhiyun static int set_vsb(struct drx_demod_instance *demod)
5717*4882a593Smuzhiyun {
5718*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
5719*4882a593Smuzhiyun 	int rc;
5720*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = NULL;
5721*4882a593Smuzhiyun 	struct drxjscu_cmd cmd_scu;
5722*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
5723*4882a593Smuzhiyun 	u16 cmd_result = 0;
5724*4882a593Smuzhiyun 	u16 cmd_param = 0;
5725*4882a593Smuzhiyun 	static const u8 vsb_taps_re[] = {
5726*4882a593Smuzhiyun 		DRXJ_16TO8(-2),	/* re0  */
5727*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re1  */
5728*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re2  */
5729*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re3  */
5730*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re4  */
5731*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re5  */
5732*4882a593Smuzhiyun 		DRXJ_16TO8(-3),	/* re6  */
5733*4882a593Smuzhiyun 		DRXJ_16TO8(-3),	/* re7  */
5734*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* re8  */
5735*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re9  */
5736*4882a593Smuzhiyun 		DRXJ_16TO8(-9),	/* re10 */
5737*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* re11 */
5738*4882a593Smuzhiyun 		DRXJ_16TO8(12),	/* re12 */
5739*4882a593Smuzhiyun 		DRXJ_16TO8(-9),	/* re13 */
5740*4882a593Smuzhiyun 		DRXJ_16TO8(-15),	/* re14 */
5741*4882a593Smuzhiyun 		DRXJ_16TO8(17),	/* re15 */
5742*4882a593Smuzhiyun 		DRXJ_16TO8(19),	/* re16 */
5743*4882a593Smuzhiyun 		DRXJ_16TO8(-29),	/* re17 */
5744*4882a593Smuzhiyun 		DRXJ_16TO8(-22),	/* re18 */
5745*4882a593Smuzhiyun 		DRXJ_16TO8(45),	/* re19 */
5746*4882a593Smuzhiyun 		DRXJ_16TO8(25),	/* re20 */
5747*4882a593Smuzhiyun 		DRXJ_16TO8(-70),	/* re21 */
5748*4882a593Smuzhiyun 		DRXJ_16TO8(-28),	/* re22 */
5749*4882a593Smuzhiyun 		DRXJ_16TO8(111),	/* re23 */
5750*4882a593Smuzhiyun 		DRXJ_16TO8(30),	/* re24 */
5751*4882a593Smuzhiyun 		DRXJ_16TO8(-201),	/* re25 */
5752*4882a593Smuzhiyun 		DRXJ_16TO8(-31),	/* re26 */
5753*4882a593Smuzhiyun 		DRXJ_16TO8(629)	/* re27 */
5754*4882a593Smuzhiyun 	};
5755*4882a593Smuzhiyun 
5756*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
5757*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
5758*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
5759*4882a593Smuzhiyun 
5760*4882a593Smuzhiyun 	/* stop all comm_exec */
5761*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
5762*4882a593Smuzhiyun 	if (rc != 0) {
5763*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5764*4882a593Smuzhiyun 		goto rw_error;
5765*4882a593Smuzhiyun 	}
5766*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_COMM_EXEC__A, VSB_COMM_EXEC_STOP, 0);
5767*4882a593Smuzhiyun 	if (rc != 0) {
5768*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5769*4882a593Smuzhiyun 		goto rw_error;
5770*4882a593Smuzhiyun 	}
5771*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
5772*4882a593Smuzhiyun 	if (rc != 0) {
5773*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5774*4882a593Smuzhiyun 		goto rw_error;
5775*4882a593Smuzhiyun 	}
5776*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
5777*4882a593Smuzhiyun 	if (rc != 0) {
5778*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5779*4882a593Smuzhiyun 		goto rw_error;
5780*4882a593Smuzhiyun 	}
5781*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
5782*4882a593Smuzhiyun 	if (rc != 0) {
5783*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5784*4882a593Smuzhiyun 		goto rw_error;
5785*4882a593Smuzhiyun 	}
5786*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
5787*4882a593Smuzhiyun 	if (rc != 0) {
5788*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5789*4882a593Smuzhiyun 		goto rw_error;
5790*4882a593Smuzhiyun 	}
5791*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
5792*4882a593Smuzhiyun 	if (rc != 0) {
5793*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5794*4882a593Smuzhiyun 		goto rw_error;
5795*4882a593Smuzhiyun 	}
5796*4882a593Smuzhiyun 
5797*4882a593Smuzhiyun 	/* reset demodulator */
5798*4882a593Smuzhiyun 	cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB
5799*4882a593Smuzhiyun 	    | SCU_RAM_COMMAND_CMD_DEMOD_RESET;
5800*4882a593Smuzhiyun 	cmd_scu.parameter_len = 0;
5801*4882a593Smuzhiyun 	cmd_scu.result_len = 1;
5802*4882a593Smuzhiyun 	cmd_scu.parameter = NULL;
5803*4882a593Smuzhiyun 	cmd_scu.result = &cmd_result;
5804*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &cmd_scu);
5805*4882a593Smuzhiyun 	if (rc != 0) {
5806*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5807*4882a593Smuzhiyun 		goto rw_error;
5808*4882a593Smuzhiyun 	}
5809*4882a593Smuzhiyun 
5810*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_DCF_BYPASS__A, 1, 0);
5811*4882a593Smuzhiyun 	if (rc != 0) {
5812*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5813*4882a593Smuzhiyun 		goto rw_error;
5814*4882a593Smuzhiyun 	}
5815*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_FS_ADJ_SEL__A, IQM_FS_ADJ_SEL_B_VSB, 0);
5816*4882a593Smuzhiyun 	if (rc != 0) {
5817*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5818*4882a593Smuzhiyun 		goto rw_error;
5819*4882a593Smuzhiyun 	}
5820*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_RC_ADJ_SEL__A, IQM_RC_ADJ_SEL_B_VSB, 0);
5821*4882a593Smuzhiyun 	if (rc != 0) {
5822*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5823*4882a593Smuzhiyun 		goto rw_error;
5824*4882a593Smuzhiyun 	}
5825*4882a593Smuzhiyun 	ext_attr->iqm_rc_rate_ofs = 0x00AD0D79;
5826*4882a593Smuzhiyun 	rc = drxdap_fasi_write_reg32(dev_addr, IQM_RC_RATE_OFS_LO__A, ext_attr->iqm_rc_rate_ofs, 0);
5827*4882a593Smuzhiyun 	if (rc != 0) {
5828*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5829*4882a593Smuzhiyun 		goto rw_error;
5830*4882a593Smuzhiyun 	}
5831*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CFAGC_GAINSHIFT__A, 4, 0);
5832*4882a593Smuzhiyun 	if (rc != 0) {
5833*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5834*4882a593Smuzhiyun 		goto rw_error;
5835*4882a593Smuzhiyun 	}
5836*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN1TRK__A, 1, 0);
5837*4882a593Smuzhiyun 	if (rc != 0) {
5838*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5839*4882a593Smuzhiyun 		goto rw_error;
5840*4882a593Smuzhiyun 	}
5841*4882a593Smuzhiyun 
5842*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_RC_CROUT_ENA__A, 1, 0);
5843*4882a593Smuzhiyun 	if (rc != 0) {
5844*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5845*4882a593Smuzhiyun 		goto rw_error;
5846*4882a593Smuzhiyun 	}
5847*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_RC_STRETCH__A, 28, 0);
5848*4882a593Smuzhiyun 	if (rc != 0) {
5849*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5850*4882a593Smuzhiyun 		goto rw_error;
5851*4882a593Smuzhiyun 	}
5852*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_RT_ACTIVE__A, 0, 0);
5853*4882a593Smuzhiyun 	if (rc != 0) {
5854*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5855*4882a593Smuzhiyun 		goto rw_error;
5856*4882a593Smuzhiyun 	}
5857*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SYMMETRIC__A, 0, 0);
5858*4882a593Smuzhiyun 	if (rc != 0) {
5859*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5860*4882a593Smuzhiyun 		goto rw_error;
5861*4882a593Smuzhiyun 	}
5862*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_CF_MIDTAP__A, 3, 0);
5863*4882a593Smuzhiyun 	if (rc != 0) {
5864*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5865*4882a593Smuzhiyun 		goto rw_error;
5866*4882a593Smuzhiyun 	}
5867*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_CF_OUT_ENA__A, IQM_CF_OUT_ENA_VSB__M, 0);
5868*4882a593Smuzhiyun 	if (rc != 0) {
5869*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5870*4882a593Smuzhiyun 		goto rw_error;
5871*4882a593Smuzhiyun 	}
5872*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE__A, 1393, 0);
5873*4882a593Smuzhiyun 	if (rc != 0) {
5874*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5875*4882a593Smuzhiyun 		goto rw_error;
5876*4882a593Smuzhiyun 	}
5877*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE_SH__A, 0, 0);
5878*4882a593Smuzhiyun 	if (rc != 0) {
5879*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5880*4882a593Smuzhiyun 		goto rw_error;
5881*4882a593Smuzhiyun 	}
5882*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_CF_POW_MEAS_LEN__A, 1, 0);
5883*4882a593Smuzhiyun 	if (rc != 0) {
5884*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5885*4882a593Smuzhiyun 		goto rw_error;
5886*4882a593Smuzhiyun 	}
5887*4882a593Smuzhiyun 
5888*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(vsb_taps_re), ((u8 *)vsb_taps_re), 0);
5889*4882a593Smuzhiyun 	if (rc != 0) {
5890*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5891*4882a593Smuzhiyun 		goto rw_error;
5892*4882a593Smuzhiyun 	}
5893*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(vsb_taps_re), ((u8 *)vsb_taps_re), 0);
5894*4882a593Smuzhiyun 	if (rc != 0) {
5895*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5896*4882a593Smuzhiyun 		goto rw_error;
5897*4882a593Smuzhiyun 	}
5898*4882a593Smuzhiyun 
5899*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_BNTHRESH__A, 330, 0);
5900*4882a593Smuzhiyun 	if (rc != 0) {
5901*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5902*4882a593Smuzhiyun 		goto rw_error;
5903*4882a593Smuzhiyun 	}	/* set higher threshold */
5904*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CLPLASTNUM__A, 90, 0);
5905*4882a593Smuzhiyun 	if (rc != 0) {
5906*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5907*4882a593Smuzhiyun 		goto rw_error;
5908*4882a593Smuzhiyun 	}	/* burst detection on   */
5909*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_RCA1__A, 0x0042, 0);
5910*4882a593Smuzhiyun 	if (rc != 0) {
5911*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5912*4882a593Smuzhiyun 		goto rw_error;
5913*4882a593Smuzhiyun 	}	/* drop thresholds by 1 dB */
5914*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_RCA2__A, 0x0053, 0);
5915*4882a593Smuzhiyun 	if (rc != 0) {
5916*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5917*4882a593Smuzhiyun 		goto rw_error;
5918*4882a593Smuzhiyun 	}	/* drop thresholds by 2 dB */
5919*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_EQCTRL__A, 0x1, 0);
5920*4882a593Smuzhiyun 	if (rc != 0) {
5921*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5922*4882a593Smuzhiyun 		goto rw_error;
5923*4882a593Smuzhiyun 	}	/* cma on               */
5924*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_GPIO__A, 0, 0);
5925*4882a593Smuzhiyun 	if (rc != 0) {
5926*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5927*4882a593Smuzhiyun 		goto rw_error;
5928*4882a593Smuzhiyun 	}	/* GPIO               */
5929*4882a593Smuzhiyun 
5930*4882a593Smuzhiyun 	/* Initialize the FEC Subsystem */
5931*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_TOP_ANNEX__A, FEC_TOP_ANNEX_D, 0);
5932*4882a593Smuzhiyun 	if (rc != 0) {
5933*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5934*4882a593Smuzhiyun 		goto rw_error;
5935*4882a593Smuzhiyun 	}
5936*4882a593Smuzhiyun 	{
5937*4882a593Smuzhiyun 		u16 fec_oc_snc_mode = 0;
5938*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_MODE__A, &fec_oc_snc_mode, 0);
5939*4882a593Smuzhiyun 		if (rc != 0) {
5940*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5941*4882a593Smuzhiyun 			goto rw_error;
5942*4882a593Smuzhiyun 		}
5943*4882a593Smuzhiyun 		/* output data even when not locked */
5944*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_MODE__A, fec_oc_snc_mode | FEC_OC_SNC_MODE_UNLOCK_ENABLE__M, 0);
5945*4882a593Smuzhiyun 		if (rc != 0) {
5946*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5947*4882a593Smuzhiyun 			goto rw_error;
5948*4882a593Smuzhiyun 		}
5949*4882a593Smuzhiyun 	}
5950*4882a593Smuzhiyun 
5951*4882a593Smuzhiyun 	/* set clip */
5952*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_LEN__A, 0, 0);
5953*4882a593Smuzhiyun 	if (rc != 0) {
5954*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5955*4882a593Smuzhiyun 		goto rw_error;
5956*4882a593Smuzhiyun 	}
5957*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_TH__A, 470, 0);
5958*4882a593Smuzhiyun 	if (rc != 0) {
5959*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5960*4882a593Smuzhiyun 		goto rw_error;
5961*4882a593Smuzhiyun 	}
5962*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_AF_SNS_LEN__A, 0, 0);
5963*4882a593Smuzhiyun 	if (rc != 0) {
5964*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5965*4882a593Smuzhiyun 		goto rw_error;
5966*4882a593Smuzhiyun 	}
5967*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_PT__A, 0xD4, 0);
5968*4882a593Smuzhiyun 	if (rc != 0) {
5969*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5970*4882a593Smuzhiyun 		goto rw_error;
5971*4882a593Smuzhiyun 	}
5972*4882a593Smuzhiyun 	/* no transparent, no A&C framing; parity is set in mpegoutput */
5973*4882a593Smuzhiyun 	{
5974*4882a593Smuzhiyun 		u16 fec_oc_reg_mode = 0;
5975*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, FEC_OC_MODE__A, &fec_oc_reg_mode, 0);
5976*4882a593Smuzhiyun 		if (rc != 0) {
5977*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5978*4882a593Smuzhiyun 			goto rw_error;
5979*4882a593Smuzhiyun 		}
5980*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_OC_MODE__A, fec_oc_reg_mode & (~(FEC_OC_MODE_TRANSPARENT__M | FEC_OC_MODE_CLEAR__M | FEC_OC_MODE_RETAIN_FRAMING__M)), 0);
5981*4882a593Smuzhiyun 		if (rc != 0) {
5982*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
5983*4882a593Smuzhiyun 			goto rw_error;
5984*4882a593Smuzhiyun 		}
5985*4882a593Smuzhiyun 	}
5986*4882a593Smuzhiyun 
5987*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_DI_TIMEOUT_LO__A, 0, 0);
5988*4882a593Smuzhiyun 	if (rc != 0) {
5989*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5990*4882a593Smuzhiyun 		goto rw_error;
5991*4882a593Smuzhiyun 	}	/* timeout counter for restarting */
5992*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_DI_TIMEOUT_HI__A, 3, 0);
5993*4882a593Smuzhiyun 	if (rc != 0) {
5994*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
5995*4882a593Smuzhiyun 		goto rw_error;
5996*4882a593Smuzhiyun 	}
5997*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MODE__A, 0, 0);
5998*4882a593Smuzhiyun 	if (rc != 0) {
5999*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6000*4882a593Smuzhiyun 		goto rw_error;
6001*4882a593Smuzhiyun 	}	/* bypass disabled */
6002*4882a593Smuzhiyun 	/* initialize RS packet error measurement parameters */
6003*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PERIOD__A, FEC_RS_MEASUREMENT_PERIOD, 0);
6004*4882a593Smuzhiyun 	if (rc != 0) {
6005*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6006*4882a593Smuzhiyun 		goto rw_error;
6007*4882a593Smuzhiyun 	}
6008*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PRESCALE__A, FEC_RS_MEASUREMENT_PRESCALE, 0);
6009*4882a593Smuzhiyun 	if (rc != 0) {
6010*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6011*4882a593Smuzhiyun 		goto rw_error;
6012*4882a593Smuzhiyun 	}
6013*4882a593Smuzhiyun 
6014*4882a593Smuzhiyun 	/* init measurement period of MER/SER */
6015*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_MEASUREMENT_PERIOD__A, VSB_TOP_MEASUREMENT_PERIOD, 0);
6016*4882a593Smuzhiyun 	if (rc != 0) {
6017*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6018*4882a593Smuzhiyun 		goto rw_error;
6019*4882a593Smuzhiyun 	}
6020*4882a593Smuzhiyun 	rc = drxdap_fasi_write_reg32(dev_addr, SCU_RAM_FEC_ACCUM_CW_CORRECTED_LO__A, 0, 0);
6021*4882a593Smuzhiyun 	if (rc != 0) {
6022*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6023*4882a593Smuzhiyun 		goto rw_error;
6024*4882a593Smuzhiyun 	}
6025*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_MEAS_COUNT__A, 0, 0);
6026*4882a593Smuzhiyun 	if (rc != 0) {
6027*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6028*4882a593Smuzhiyun 		goto rw_error;
6029*4882a593Smuzhiyun 	}
6030*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_ACCUM_PKT_FAILURES__A, 0, 0);
6031*4882a593Smuzhiyun 	if (rc != 0) {
6032*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6033*4882a593Smuzhiyun 		goto rw_error;
6034*4882a593Smuzhiyun 	}
6035*4882a593Smuzhiyun 
6036*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CKGN1TRK__A, 128, 0);
6037*4882a593Smuzhiyun 	if (rc != 0) {
6038*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6039*4882a593Smuzhiyun 		goto rw_error;
6040*4882a593Smuzhiyun 	}
6041*4882a593Smuzhiyun 	/* B-Input to ADC, PGA+filter in standby */
6042*4882a593Smuzhiyun 	if (!ext_attr->has_lna) {
6043*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AMUX__A, 0x02, 0);
6044*4882a593Smuzhiyun 		if (rc != 0) {
6045*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6046*4882a593Smuzhiyun 			goto rw_error;
6047*4882a593Smuzhiyun 		}
6048*4882a593Smuzhiyun 	}
6049*4882a593Smuzhiyun 
6050*4882a593Smuzhiyun 	/* turn on IQMAF. It has to be in front of setAgc**() */
6051*4882a593Smuzhiyun 	rc = set_iqm_af(demod, true);
6052*4882a593Smuzhiyun 	if (rc != 0) {
6053*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6054*4882a593Smuzhiyun 		goto rw_error;
6055*4882a593Smuzhiyun 	}
6056*4882a593Smuzhiyun 	rc = adc_synchronization(demod);
6057*4882a593Smuzhiyun 	if (rc != 0) {
6058*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6059*4882a593Smuzhiyun 		goto rw_error;
6060*4882a593Smuzhiyun 	}
6061*4882a593Smuzhiyun 
6062*4882a593Smuzhiyun 	rc = init_agc(demod);
6063*4882a593Smuzhiyun 	if (rc != 0) {
6064*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6065*4882a593Smuzhiyun 		goto rw_error;
6066*4882a593Smuzhiyun 	}
6067*4882a593Smuzhiyun 	rc = set_agc_if(demod, &(ext_attr->vsb_if_agc_cfg), false);
6068*4882a593Smuzhiyun 	if (rc != 0) {
6069*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6070*4882a593Smuzhiyun 		goto rw_error;
6071*4882a593Smuzhiyun 	}
6072*4882a593Smuzhiyun 	rc = set_agc_rf(demod, &(ext_attr->vsb_rf_agc_cfg), false);
6073*4882a593Smuzhiyun 	if (rc != 0) {
6074*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6075*4882a593Smuzhiyun 		goto rw_error;
6076*4882a593Smuzhiyun 	}
6077*4882a593Smuzhiyun 	{
6078*4882a593Smuzhiyun 		/* TODO fix this, store a struct drxj_cfg_afe_gain structure in struct drxj_data instead
6079*4882a593Smuzhiyun 		   of only the gain */
6080*4882a593Smuzhiyun 		struct drxj_cfg_afe_gain vsb_pga_cfg = { DRX_STANDARD_8VSB, 0 };
6081*4882a593Smuzhiyun 
6082*4882a593Smuzhiyun 		vsb_pga_cfg.gain = ext_attr->vsb_pga_cfg;
6083*4882a593Smuzhiyun 		rc = ctrl_set_cfg_afe_gain(demod, &vsb_pga_cfg);
6084*4882a593Smuzhiyun 		if (rc != 0) {
6085*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6086*4882a593Smuzhiyun 			goto rw_error;
6087*4882a593Smuzhiyun 		}
6088*4882a593Smuzhiyun 	}
6089*4882a593Smuzhiyun 	rc = ctrl_set_cfg_pre_saw(demod, &(ext_attr->vsb_pre_saw_cfg));
6090*4882a593Smuzhiyun 	if (rc != 0) {
6091*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6092*4882a593Smuzhiyun 		goto rw_error;
6093*4882a593Smuzhiyun 	}
6094*4882a593Smuzhiyun 
6095*4882a593Smuzhiyun 	/* Mpeg output has to be in front of FEC active */
6096*4882a593Smuzhiyun 	rc = set_mpegtei_handling(demod);
6097*4882a593Smuzhiyun 	if (rc != 0) {
6098*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6099*4882a593Smuzhiyun 		goto rw_error;
6100*4882a593Smuzhiyun 	}
6101*4882a593Smuzhiyun 	rc = bit_reverse_mpeg_output(demod);
6102*4882a593Smuzhiyun 	if (rc != 0) {
6103*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6104*4882a593Smuzhiyun 		goto rw_error;
6105*4882a593Smuzhiyun 	}
6106*4882a593Smuzhiyun 	rc = set_mpeg_start_width(demod);
6107*4882a593Smuzhiyun 	if (rc != 0) {
6108*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6109*4882a593Smuzhiyun 		goto rw_error;
6110*4882a593Smuzhiyun 	}
6111*4882a593Smuzhiyun 	{
6112*4882a593Smuzhiyun 		/* TODO: move to set_standard after hardware reset value problem is solved */
6113*4882a593Smuzhiyun 		/* Configure initial MPEG output */
6114*4882a593Smuzhiyun 		struct drx_cfg_mpeg_output cfg_mpeg_output;
6115*4882a593Smuzhiyun 
6116*4882a593Smuzhiyun 		memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
6117*4882a593Smuzhiyun 		cfg_mpeg_output.enable_mpeg_output = true;
6118*4882a593Smuzhiyun 
6119*4882a593Smuzhiyun 		rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
6120*4882a593Smuzhiyun 		if (rc != 0) {
6121*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6122*4882a593Smuzhiyun 			goto rw_error;
6123*4882a593Smuzhiyun 		}
6124*4882a593Smuzhiyun 	}
6125*4882a593Smuzhiyun 
6126*4882a593Smuzhiyun 	/* TBD: what parameters should be set */
6127*4882a593Smuzhiyun 	cmd_param = 0x00;	/* Default mode AGC on, etc */
6128*4882a593Smuzhiyun 	cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB
6129*4882a593Smuzhiyun 	    | SCU_RAM_COMMAND_CMD_DEMOD_SET_PARAM;
6130*4882a593Smuzhiyun 	cmd_scu.parameter_len = 1;
6131*4882a593Smuzhiyun 	cmd_scu.result_len = 1;
6132*4882a593Smuzhiyun 	cmd_scu.parameter = &cmd_param;
6133*4882a593Smuzhiyun 	cmd_scu.result = &cmd_result;
6134*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &cmd_scu);
6135*4882a593Smuzhiyun 	if (rc != 0) {
6136*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6137*4882a593Smuzhiyun 		goto rw_error;
6138*4882a593Smuzhiyun 	}
6139*4882a593Smuzhiyun 
6140*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_BEAGC_GAINSHIFT__A, 0x0004, 0);
6141*4882a593Smuzhiyun 	if (rc != 0) {
6142*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6143*4882a593Smuzhiyun 		goto rw_error;
6144*4882a593Smuzhiyun 	}
6145*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_PT__A, 0x00D2, 0);
6146*4882a593Smuzhiyun 	if (rc != 0) {
6147*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6148*4882a593Smuzhiyun 		goto rw_error;
6149*4882a593Smuzhiyun 	}
6150*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SYSSMTRNCTRL__A, VSB_TOP_SYSSMTRNCTRL__PRE | VSB_TOP_SYSSMTRNCTRL_NCOTIMEOUTCNTEN__M, 0);
6151*4882a593Smuzhiyun 	if (rc != 0) {
6152*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6153*4882a593Smuzhiyun 		goto rw_error;
6154*4882a593Smuzhiyun 	}
6155*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_BEDETCTRL__A, 0x142, 0);
6156*4882a593Smuzhiyun 	if (rc != 0) {
6157*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6158*4882a593Smuzhiyun 		goto rw_error;
6159*4882a593Smuzhiyun 	}
6160*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_LBAGCREFLVL__A, 640, 0);
6161*4882a593Smuzhiyun 	if (rc != 0) {
6162*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6163*4882a593Smuzhiyun 		goto rw_error;
6164*4882a593Smuzhiyun 	}
6165*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN1ACQ__A, 4, 0);
6166*4882a593Smuzhiyun 	if (rc != 0) {
6167*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6168*4882a593Smuzhiyun 		goto rw_error;
6169*4882a593Smuzhiyun 	}
6170*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN1TRK__A, 2, 0);
6171*4882a593Smuzhiyun 	if (rc != 0) {
6172*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6173*4882a593Smuzhiyun 		goto rw_error;
6174*4882a593Smuzhiyun 	}
6175*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN2TRK__A, 3, 0);
6176*4882a593Smuzhiyun 	if (rc != 0) {
6177*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6178*4882a593Smuzhiyun 		goto rw_error;
6179*4882a593Smuzhiyun 	}
6180*4882a593Smuzhiyun 
6181*4882a593Smuzhiyun 	/* start demodulator */
6182*4882a593Smuzhiyun 	cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB
6183*4882a593Smuzhiyun 	    | SCU_RAM_COMMAND_CMD_DEMOD_START;
6184*4882a593Smuzhiyun 	cmd_scu.parameter_len = 0;
6185*4882a593Smuzhiyun 	cmd_scu.result_len = 1;
6186*4882a593Smuzhiyun 	cmd_scu.parameter = NULL;
6187*4882a593Smuzhiyun 	cmd_scu.result = &cmd_result;
6188*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &cmd_scu);
6189*4882a593Smuzhiyun 	if (rc != 0) {
6190*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6191*4882a593Smuzhiyun 		goto rw_error;
6192*4882a593Smuzhiyun 	}
6193*4882a593Smuzhiyun 
6194*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_ACTIVE, 0);
6195*4882a593Smuzhiyun 	if (rc != 0) {
6196*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6197*4882a593Smuzhiyun 		goto rw_error;
6198*4882a593Smuzhiyun 	}
6199*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, VSB_COMM_EXEC__A, VSB_COMM_EXEC_ACTIVE, 0);
6200*4882a593Smuzhiyun 	if (rc != 0) {
6201*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6202*4882a593Smuzhiyun 		goto rw_error;
6203*4882a593Smuzhiyun 	}
6204*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_ACTIVE, 0);
6205*4882a593Smuzhiyun 	if (rc != 0) {
6206*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6207*4882a593Smuzhiyun 		goto rw_error;
6208*4882a593Smuzhiyun 	}
6209*4882a593Smuzhiyun 
6210*4882a593Smuzhiyun 	return 0;
6211*4882a593Smuzhiyun rw_error:
6212*4882a593Smuzhiyun 	return rc;
6213*4882a593Smuzhiyun }
6214*4882a593Smuzhiyun 
6215*4882a593Smuzhiyun /*
6216*4882a593Smuzhiyun * \fn static short get_vsb_post_rs_pck_err(struct i2c_device_addr *dev_addr, u16 *PckErrs)
6217*4882a593Smuzhiyun * \brief Get the values of packet error in 8VSB mode
6218*4882a593Smuzhiyun * \return Error code
6219*4882a593Smuzhiyun */
get_vsb_post_rs_pck_err(struct i2c_device_addr * dev_addr,u32 * pck_errs,u32 * pck_count)6220*4882a593Smuzhiyun static int get_vsb_post_rs_pck_err(struct i2c_device_addr *dev_addr,
6221*4882a593Smuzhiyun 				   u32 *pck_errs, u32 *pck_count)
6222*4882a593Smuzhiyun {
6223*4882a593Smuzhiyun 	int rc;
6224*4882a593Smuzhiyun 	u16 data = 0;
6225*4882a593Smuzhiyun 	u16 period = 0;
6226*4882a593Smuzhiyun 	u16 prescale = 0;
6227*4882a593Smuzhiyun 	u16 packet_errors_mant = 0;
6228*4882a593Smuzhiyun 	u16 packet_errors_exp = 0;
6229*4882a593Smuzhiyun 
6230*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_FAILURES__A, &data, 0);
6231*4882a593Smuzhiyun 	if (rc != 0) {
6232*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6233*4882a593Smuzhiyun 		goto rw_error;
6234*4882a593Smuzhiyun 	}
6235*4882a593Smuzhiyun 	packet_errors_mant = data & FEC_RS_NR_FAILURES_FIXED_MANT__M;
6236*4882a593Smuzhiyun 	packet_errors_exp = (data & FEC_RS_NR_FAILURES_EXP__M)
6237*4882a593Smuzhiyun 	    >> FEC_RS_NR_FAILURES_EXP__B;
6238*4882a593Smuzhiyun 	period = FEC_RS_MEASUREMENT_PERIOD;
6239*4882a593Smuzhiyun 	prescale = FEC_RS_MEASUREMENT_PRESCALE;
6240*4882a593Smuzhiyun 	/* packet error rate = (error packet number) per second */
6241*4882a593Smuzhiyun 	/* 77.3 us is time for per packet */
6242*4882a593Smuzhiyun 	if (period * prescale == 0) {
6243*4882a593Smuzhiyun 		pr_err("error: period and/or prescale is zero!\n");
6244*4882a593Smuzhiyun 		return -EIO;
6245*4882a593Smuzhiyun 	}
6246*4882a593Smuzhiyun 	*pck_errs = packet_errors_mant * (1 << packet_errors_exp);
6247*4882a593Smuzhiyun 	*pck_count = period * prescale * 77;
6248*4882a593Smuzhiyun 
6249*4882a593Smuzhiyun 	return 0;
6250*4882a593Smuzhiyun rw_error:
6251*4882a593Smuzhiyun 	return rc;
6252*4882a593Smuzhiyun }
6253*4882a593Smuzhiyun 
6254*4882a593Smuzhiyun /*
6255*4882a593Smuzhiyun * \fn static short GetVSBBer(struct i2c_device_addr *dev_addr, u32 *ber)
6256*4882a593Smuzhiyun * \brief Get the values of ber in VSB mode
6257*4882a593Smuzhiyun * \return Error code
6258*4882a593Smuzhiyun */
get_vs_bpost_viterbi_ber(struct i2c_device_addr * dev_addr,u32 * ber,u32 * cnt)6259*4882a593Smuzhiyun static int get_vs_bpost_viterbi_ber(struct i2c_device_addr *dev_addr,
6260*4882a593Smuzhiyun 				    u32 *ber, u32 *cnt)
6261*4882a593Smuzhiyun {
6262*4882a593Smuzhiyun 	int rc;
6263*4882a593Smuzhiyun 	u16 data = 0;
6264*4882a593Smuzhiyun 	u16 period = 0;
6265*4882a593Smuzhiyun 	u16 prescale = 0;
6266*4882a593Smuzhiyun 	u16 bit_errors_mant = 0;
6267*4882a593Smuzhiyun 	u16 bit_errors_exp = 0;
6268*4882a593Smuzhiyun 
6269*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_BIT_ERRORS__A, &data, 0);
6270*4882a593Smuzhiyun 	if (rc != 0) {
6271*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6272*4882a593Smuzhiyun 		goto rw_error;
6273*4882a593Smuzhiyun 	}
6274*4882a593Smuzhiyun 	period = FEC_RS_MEASUREMENT_PERIOD;
6275*4882a593Smuzhiyun 	prescale = FEC_RS_MEASUREMENT_PRESCALE;
6276*4882a593Smuzhiyun 
6277*4882a593Smuzhiyun 	bit_errors_mant = data & FEC_RS_NR_BIT_ERRORS_FIXED_MANT__M;
6278*4882a593Smuzhiyun 	bit_errors_exp = (data & FEC_RS_NR_BIT_ERRORS_EXP__M)
6279*4882a593Smuzhiyun 	    >> FEC_RS_NR_BIT_ERRORS_EXP__B;
6280*4882a593Smuzhiyun 
6281*4882a593Smuzhiyun 	*cnt = period * prescale * 207 * ((bit_errors_exp > 2) ? 1 : 8);
6282*4882a593Smuzhiyun 
6283*4882a593Smuzhiyun 	if (((bit_errors_mant << bit_errors_exp) >> 3) > 68700)
6284*4882a593Smuzhiyun 		*ber = (*cnt) * 26570;
6285*4882a593Smuzhiyun 	else {
6286*4882a593Smuzhiyun 		if (period * prescale == 0) {
6287*4882a593Smuzhiyun 			pr_err("error: period and/or prescale is zero!\n");
6288*4882a593Smuzhiyun 			return -EIO;
6289*4882a593Smuzhiyun 		}
6290*4882a593Smuzhiyun 		*ber = bit_errors_mant << ((bit_errors_exp > 2) ?
6291*4882a593Smuzhiyun 			(bit_errors_exp - 3) : bit_errors_exp);
6292*4882a593Smuzhiyun 	}
6293*4882a593Smuzhiyun 
6294*4882a593Smuzhiyun 	return 0;
6295*4882a593Smuzhiyun rw_error:
6296*4882a593Smuzhiyun 	return rc;
6297*4882a593Smuzhiyun }
6298*4882a593Smuzhiyun 
6299*4882a593Smuzhiyun /*
6300*4882a593Smuzhiyun * \fn static short get_vs_bpre_viterbi_ber(struct i2c_device_addr *dev_addr, u32 *ber)
6301*4882a593Smuzhiyun * \brief Get the values of ber in VSB mode
6302*4882a593Smuzhiyun * \return Error code
6303*4882a593Smuzhiyun */
get_vs_bpre_viterbi_ber(struct i2c_device_addr * dev_addr,u32 * ber,u32 * cnt)6304*4882a593Smuzhiyun static int get_vs_bpre_viterbi_ber(struct i2c_device_addr *dev_addr,
6305*4882a593Smuzhiyun 				   u32 *ber, u32 *cnt)
6306*4882a593Smuzhiyun {
6307*4882a593Smuzhiyun 	u16 data = 0;
6308*4882a593Smuzhiyun 	int rc;
6309*4882a593Smuzhiyun 
6310*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, VSB_TOP_NR_SYM_ERRS__A, &data, 0);
6311*4882a593Smuzhiyun 	if (rc != 0) {
6312*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6313*4882a593Smuzhiyun 		return -EIO;
6314*4882a593Smuzhiyun 	}
6315*4882a593Smuzhiyun 	*ber = data;
6316*4882a593Smuzhiyun 	*cnt = VSB_TOP_MEASUREMENT_PERIOD * SYMBOLS_PER_SEGMENT;
6317*4882a593Smuzhiyun 
6318*4882a593Smuzhiyun 	return 0;
6319*4882a593Smuzhiyun }
6320*4882a593Smuzhiyun 
6321*4882a593Smuzhiyun /*
6322*4882a593Smuzhiyun * \fn static int get_vsbmer(struct i2c_device_addr *dev_addr, u16 *mer)
6323*4882a593Smuzhiyun * \brief Get the values of MER
6324*4882a593Smuzhiyun * \return Error code
6325*4882a593Smuzhiyun */
get_vsbmer(struct i2c_device_addr * dev_addr,u16 * mer)6326*4882a593Smuzhiyun static int get_vsbmer(struct i2c_device_addr *dev_addr, u16 *mer)
6327*4882a593Smuzhiyun {
6328*4882a593Smuzhiyun 	int rc;
6329*4882a593Smuzhiyun 	u16 data_hi = 0;
6330*4882a593Smuzhiyun 
6331*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, VSB_TOP_ERR_ENERGY_H__A, &data_hi, 0);
6332*4882a593Smuzhiyun 	if (rc != 0) {
6333*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6334*4882a593Smuzhiyun 		goto rw_error;
6335*4882a593Smuzhiyun 	}
6336*4882a593Smuzhiyun 	*mer =
6337*4882a593Smuzhiyun 	    (u16) (log1_times100(21504) - log1_times100((data_hi << 6) / 52));
6338*4882a593Smuzhiyun 
6339*4882a593Smuzhiyun 	return 0;
6340*4882a593Smuzhiyun rw_error:
6341*4882a593Smuzhiyun 	return rc;
6342*4882a593Smuzhiyun }
6343*4882a593Smuzhiyun 
6344*4882a593Smuzhiyun 
6345*4882a593Smuzhiyun /*============================================================================*/
6346*4882a593Smuzhiyun /*==                     END 8VSB DATAPATH FUNCTIONS                        ==*/
6347*4882a593Smuzhiyun /*============================================================================*/
6348*4882a593Smuzhiyun 
6349*4882a593Smuzhiyun /*============================================================================*/
6350*4882a593Smuzhiyun /*============================================================================*/
6351*4882a593Smuzhiyun /*==                       QAM DATAPATH FUNCTIONS                           ==*/
6352*4882a593Smuzhiyun /*============================================================================*/
6353*4882a593Smuzhiyun /*============================================================================*/
6354*4882a593Smuzhiyun 
6355*4882a593Smuzhiyun /*
6356*4882a593Smuzhiyun * \fn int power_down_qam ()
6357*4882a593Smuzhiyun * \brief Powr down QAM related blocks.
6358*4882a593Smuzhiyun * \param demod instance of demodulator.
6359*4882a593Smuzhiyun * \param channel pointer to channel data.
6360*4882a593Smuzhiyun * \return int.
6361*4882a593Smuzhiyun */
power_down_qam(struct drx_demod_instance * demod,bool primary)6362*4882a593Smuzhiyun static int power_down_qam(struct drx_demod_instance *demod, bool primary)
6363*4882a593Smuzhiyun {
6364*4882a593Smuzhiyun 	struct drxjscu_cmd cmd_scu = { /* command      */ 0,
6365*4882a593Smuzhiyun 		/* parameter_len */ 0,
6366*4882a593Smuzhiyun 		/* result_len    */ 0,
6367*4882a593Smuzhiyun 		/* *parameter   */ NULL,
6368*4882a593Smuzhiyun 		/* *result      */ NULL
6369*4882a593Smuzhiyun 	};
6370*4882a593Smuzhiyun 	int rc;
6371*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
6372*4882a593Smuzhiyun 	struct drx_cfg_mpeg_output cfg_mpeg_output;
6373*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = demod->my_common_attr;
6374*4882a593Smuzhiyun 	u16 cmd_result = 0;
6375*4882a593Smuzhiyun 
6376*4882a593Smuzhiyun 	/*
6377*4882a593Smuzhiyun 	   STOP demodulator
6378*4882a593Smuzhiyun 	   resets IQM, QAM and FEC HW blocks
6379*4882a593Smuzhiyun 	 */
6380*4882a593Smuzhiyun 	/* stop all comm_exec */
6381*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
6382*4882a593Smuzhiyun 	if (rc != 0) {
6383*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6384*4882a593Smuzhiyun 		goto rw_error;
6385*4882a593Smuzhiyun 	}
6386*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_COMM_EXEC__A, QAM_COMM_EXEC_STOP, 0);
6387*4882a593Smuzhiyun 	if (rc != 0) {
6388*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6389*4882a593Smuzhiyun 		goto rw_error;
6390*4882a593Smuzhiyun 	}
6391*4882a593Smuzhiyun 
6392*4882a593Smuzhiyun 	cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
6393*4882a593Smuzhiyun 	    SCU_RAM_COMMAND_CMD_DEMOD_STOP;
6394*4882a593Smuzhiyun 	cmd_scu.parameter_len = 0;
6395*4882a593Smuzhiyun 	cmd_scu.result_len = 1;
6396*4882a593Smuzhiyun 	cmd_scu.parameter = NULL;
6397*4882a593Smuzhiyun 	cmd_scu.result = &cmd_result;
6398*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &cmd_scu);
6399*4882a593Smuzhiyun 	if (rc != 0) {
6400*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6401*4882a593Smuzhiyun 		goto rw_error;
6402*4882a593Smuzhiyun 	}
6403*4882a593Smuzhiyun 
6404*4882a593Smuzhiyun 	if (primary) {
6405*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_STOP, 0);
6406*4882a593Smuzhiyun 		if (rc != 0) {
6407*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6408*4882a593Smuzhiyun 			goto rw_error;
6409*4882a593Smuzhiyun 		}
6410*4882a593Smuzhiyun 		rc = set_iqm_af(demod, false);
6411*4882a593Smuzhiyun 		if (rc != 0) {
6412*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6413*4882a593Smuzhiyun 			goto rw_error;
6414*4882a593Smuzhiyun 		}
6415*4882a593Smuzhiyun 	} else {
6416*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
6417*4882a593Smuzhiyun 		if (rc != 0) {
6418*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6419*4882a593Smuzhiyun 			goto rw_error;
6420*4882a593Smuzhiyun 		}
6421*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
6422*4882a593Smuzhiyun 		if (rc != 0) {
6423*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6424*4882a593Smuzhiyun 			goto rw_error;
6425*4882a593Smuzhiyun 		}
6426*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
6427*4882a593Smuzhiyun 		if (rc != 0) {
6428*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6429*4882a593Smuzhiyun 			goto rw_error;
6430*4882a593Smuzhiyun 		}
6431*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
6432*4882a593Smuzhiyun 		if (rc != 0) {
6433*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6434*4882a593Smuzhiyun 			goto rw_error;
6435*4882a593Smuzhiyun 		}
6436*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
6437*4882a593Smuzhiyun 		if (rc != 0) {
6438*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6439*4882a593Smuzhiyun 			goto rw_error;
6440*4882a593Smuzhiyun 		}
6441*4882a593Smuzhiyun 	}
6442*4882a593Smuzhiyun 
6443*4882a593Smuzhiyun 	memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
6444*4882a593Smuzhiyun 	cfg_mpeg_output.enable_mpeg_output = false;
6445*4882a593Smuzhiyun 
6446*4882a593Smuzhiyun 	rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
6447*4882a593Smuzhiyun 	if (rc != 0) {
6448*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6449*4882a593Smuzhiyun 		goto rw_error;
6450*4882a593Smuzhiyun 	}
6451*4882a593Smuzhiyun 
6452*4882a593Smuzhiyun 	return 0;
6453*4882a593Smuzhiyun rw_error:
6454*4882a593Smuzhiyun 	return rc;
6455*4882a593Smuzhiyun }
6456*4882a593Smuzhiyun 
6457*4882a593Smuzhiyun /*============================================================================*/
6458*4882a593Smuzhiyun 
6459*4882a593Smuzhiyun /*
6460*4882a593Smuzhiyun * \fn int set_qam_measurement ()
6461*4882a593Smuzhiyun * \brief Setup of the QAM Measuremnt intervals for signal quality
6462*4882a593Smuzhiyun * \param demod instance of demod.
6463*4882a593Smuzhiyun * \param constellation current constellation.
6464*4882a593Smuzhiyun * \return int.
6465*4882a593Smuzhiyun *
6466*4882a593Smuzhiyun *  NOTE:
6467*4882a593Smuzhiyun *  Take into account that for certain settings the errorcounters can overflow.
6468*4882a593Smuzhiyun *  The implementation does not check this.
6469*4882a593Smuzhiyun *
6470*4882a593Smuzhiyun *  TODO: overriding the ext_attr->fec_bits_desired by constellation dependent
6471*4882a593Smuzhiyun *  constants to get a measurement period of approx. 1 sec. Remove fec_bits_desired
6472*4882a593Smuzhiyun *  field ?
6473*4882a593Smuzhiyun *
6474*4882a593Smuzhiyun */
6475*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
6476*4882a593Smuzhiyun static int
set_qam_measurement(struct drx_demod_instance * demod,enum drx_modulation constellation,u32 symbol_rate)6477*4882a593Smuzhiyun set_qam_measurement(struct drx_demod_instance *demod,
6478*4882a593Smuzhiyun 		    enum drx_modulation constellation, u32 symbol_rate)
6479*4882a593Smuzhiyun {
6480*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;	/* device address for I2C writes */
6481*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;	/* Global data container for DRXJ specific data */
6482*4882a593Smuzhiyun 	int rc;
6483*4882a593Smuzhiyun 	u32 fec_bits_desired = 0;	/* BER accounting period */
6484*4882a593Smuzhiyun 	u16 fec_rs_plen = 0;	/* defines RS BER measurement period */
6485*4882a593Smuzhiyun 	u16 fec_rs_prescale = 0;	/* ReedSolomon Measurement Prescale */
6486*4882a593Smuzhiyun 	u32 fec_rs_period = 0;	/* Value for corresponding I2C register */
6487*4882a593Smuzhiyun 	u32 fec_rs_bit_cnt = 0;	/* Actual precise amount of bits */
6488*4882a593Smuzhiyun 	u32 fec_oc_snc_fail_period = 0;	/* Value for corresponding I2C register */
6489*4882a593Smuzhiyun 	u32 qam_vd_period = 0;	/* Value for corresponding I2C register */
6490*4882a593Smuzhiyun 	u32 qam_vd_bit_cnt = 0;	/* Actual precise amount of bits */
6491*4882a593Smuzhiyun 	u16 fec_vd_plen = 0;	/* no of trellis symbols: VD SER measur period */
6492*4882a593Smuzhiyun 	u16 qam_vd_prescale = 0;	/* Viterbi Measurement Prescale */
6493*4882a593Smuzhiyun 
6494*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
6495*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
6496*4882a593Smuzhiyun 
6497*4882a593Smuzhiyun 	fec_bits_desired = ext_attr->fec_bits_desired;
6498*4882a593Smuzhiyun 	fec_rs_prescale = ext_attr->fec_rs_prescale;
6499*4882a593Smuzhiyun 
6500*4882a593Smuzhiyun 	switch (constellation) {
6501*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM16:
6502*4882a593Smuzhiyun 		fec_bits_desired = 4 * symbol_rate;
6503*4882a593Smuzhiyun 		break;
6504*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM32:
6505*4882a593Smuzhiyun 		fec_bits_desired = 5 * symbol_rate;
6506*4882a593Smuzhiyun 		break;
6507*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM64:
6508*4882a593Smuzhiyun 		fec_bits_desired = 6 * symbol_rate;
6509*4882a593Smuzhiyun 		break;
6510*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM128:
6511*4882a593Smuzhiyun 		fec_bits_desired = 7 * symbol_rate;
6512*4882a593Smuzhiyun 		break;
6513*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM256:
6514*4882a593Smuzhiyun 		fec_bits_desired = 8 * symbol_rate;
6515*4882a593Smuzhiyun 		break;
6516*4882a593Smuzhiyun 	default:
6517*4882a593Smuzhiyun 		return -EINVAL;
6518*4882a593Smuzhiyun 	}
6519*4882a593Smuzhiyun 
6520*4882a593Smuzhiyun 	/* Parameters for Reed-Solomon Decoder */
6521*4882a593Smuzhiyun 	/* fecrs_period = (int)ceil(FEC_BITS_DESIRED/(fecrs_prescale*plen)) */
6522*4882a593Smuzhiyun 	/* rs_bit_cnt   = fecrs_period*fecrs_prescale*plen                  */
6523*4882a593Smuzhiyun 	/*     result is within 32 bit arithmetic ->                        */
6524*4882a593Smuzhiyun 	/*     no need for mult or frac functions                           */
6525*4882a593Smuzhiyun 
6526*4882a593Smuzhiyun 	/* TODO: use constant instead of calculation and remove the fec_rs_plen in ext_attr */
6527*4882a593Smuzhiyun 	switch (ext_attr->standard) {
6528*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
6529*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
6530*4882a593Smuzhiyun 		fec_rs_plen = 204 * 8;
6531*4882a593Smuzhiyun 		break;
6532*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
6533*4882a593Smuzhiyun 		fec_rs_plen = 128 * 7;
6534*4882a593Smuzhiyun 		break;
6535*4882a593Smuzhiyun 	default:
6536*4882a593Smuzhiyun 		return -EINVAL;
6537*4882a593Smuzhiyun 	}
6538*4882a593Smuzhiyun 
6539*4882a593Smuzhiyun 	ext_attr->fec_rs_plen = fec_rs_plen;	/* for getSigQual */
6540*4882a593Smuzhiyun 	fec_rs_bit_cnt = fec_rs_prescale * fec_rs_plen;	/* temp storage   */
6541*4882a593Smuzhiyun 	if (fec_rs_bit_cnt == 0) {
6542*4882a593Smuzhiyun 		pr_err("error: fec_rs_bit_cnt is zero!\n");
6543*4882a593Smuzhiyun 		return -EIO;
6544*4882a593Smuzhiyun 	}
6545*4882a593Smuzhiyun 	fec_rs_period = fec_bits_desired / fec_rs_bit_cnt + 1;	/* ceil */
6546*4882a593Smuzhiyun 	if (ext_attr->standard != DRX_STANDARD_ITU_B)
6547*4882a593Smuzhiyun 		fec_oc_snc_fail_period = fec_rs_period;
6548*4882a593Smuzhiyun 
6549*4882a593Smuzhiyun 	/* limit to max 16 bit value (I2C register width) if needed */
6550*4882a593Smuzhiyun 	if (fec_rs_period > 0xFFFF)
6551*4882a593Smuzhiyun 		fec_rs_period = 0xFFFF;
6552*4882a593Smuzhiyun 
6553*4882a593Smuzhiyun 	/* write corresponding registers */
6554*4882a593Smuzhiyun 	switch (ext_attr->standard) {
6555*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
6556*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
6557*4882a593Smuzhiyun 		break;
6558*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
6559*4882a593Smuzhiyun 		switch (constellation) {
6560*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM64:
6561*4882a593Smuzhiyun 			fec_rs_period = 31581;
6562*4882a593Smuzhiyun 			fec_oc_snc_fail_period = 17932;
6563*4882a593Smuzhiyun 			break;
6564*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM256:
6565*4882a593Smuzhiyun 			fec_rs_period = 45446;
6566*4882a593Smuzhiyun 			fec_oc_snc_fail_period = 25805;
6567*4882a593Smuzhiyun 			break;
6568*4882a593Smuzhiyun 		default:
6569*4882a593Smuzhiyun 			return -EINVAL;
6570*4882a593Smuzhiyun 		}
6571*4882a593Smuzhiyun 		break;
6572*4882a593Smuzhiyun 	default:
6573*4882a593Smuzhiyun 		return -EINVAL;
6574*4882a593Smuzhiyun 	}
6575*4882a593Smuzhiyun 
6576*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_FAIL_PERIOD__A, (u16)fec_oc_snc_fail_period, 0);
6577*4882a593Smuzhiyun 	if (rc != 0) {
6578*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6579*4882a593Smuzhiyun 		goto rw_error;
6580*4882a593Smuzhiyun 	}
6581*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PERIOD__A, (u16)fec_rs_period, 0);
6582*4882a593Smuzhiyun 	if (rc != 0) {
6583*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6584*4882a593Smuzhiyun 		goto rw_error;
6585*4882a593Smuzhiyun 	}
6586*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PRESCALE__A, fec_rs_prescale, 0);
6587*4882a593Smuzhiyun 	if (rc != 0) {
6588*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6589*4882a593Smuzhiyun 		goto rw_error;
6590*4882a593Smuzhiyun 	}
6591*4882a593Smuzhiyun 	ext_attr->fec_rs_period = (u16) fec_rs_period;
6592*4882a593Smuzhiyun 	ext_attr->fec_rs_prescale = fec_rs_prescale;
6593*4882a593Smuzhiyun 	rc = drxdap_fasi_write_reg32(dev_addr, SCU_RAM_FEC_ACCUM_CW_CORRECTED_LO__A, 0, 0);
6594*4882a593Smuzhiyun 	if (rc != 0) {
6595*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6596*4882a593Smuzhiyun 		goto rw_error;
6597*4882a593Smuzhiyun 	}
6598*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_MEAS_COUNT__A, 0, 0);
6599*4882a593Smuzhiyun 	if (rc != 0) {
6600*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6601*4882a593Smuzhiyun 		goto rw_error;
6602*4882a593Smuzhiyun 	}
6603*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_ACCUM_PKT_FAILURES__A, 0, 0);
6604*4882a593Smuzhiyun 	if (rc != 0) {
6605*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6606*4882a593Smuzhiyun 		goto rw_error;
6607*4882a593Smuzhiyun 	}
6608*4882a593Smuzhiyun 
6609*4882a593Smuzhiyun 	if (ext_attr->standard == DRX_STANDARD_ITU_B) {
6610*4882a593Smuzhiyun 		/* Parameters for Viterbi Decoder */
6611*4882a593Smuzhiyun 		/* qamvd_period = (int)ceil(FEC_BITS_DESIRED/                      */
6612*4882a593Smuzhiyun 		/*                    (qamvd_prescale*plen*(qam_constellation+1))) */
6613*4882a593Smuzhiyun 		/* vd_bit_cnt   = qamvd_period*qamvd_prescale*plen                 */
6614*4882a593Smuzhiyun 		/*     result is within 32 bit arithmetic ->                       */
6615*4882a593Smuzhiyun 		/*     no need for mult or frac functions                          */
6616*4882a593Smuzhiyun 
6617*4882a593Smuzhiyun 		/* a(8 bit) * b(8 bit) = 16 bit result => mult32 not needed */
6618*4882a593Smuzhiyun 		fec_vd_plen = ext_attr->fec_vd_plen;
6619*4882a593Smuzhiyun 		qam_vd_prescale = ext_attr->qam_vd_prescale;
6620*4882a593Smuzhiyun 		qam_vd_bit_cnt = qam_vd_prescale * fec_vd_plen;	/* temp storage */
6621*4882a593Smuzhiyun 
6622*4882a593Smuzhiyun 		switch (constellation) {
6623*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM64:
6624*4882a593Smuzhiyun 			/* a(16 bit) * b(4 bit) = 20 bit result => mult32 not needed */
6625*4882a593Smuzhiyun 			qam_vd_period =
6626*4882a593Smuzhiyun 			    qam_vd_bit_cnt * (QAM_TOP_CONSTELLATION_QAM64 + 1)
6627*4882a593Smuzhiyun 			    * (QAM_TOP_CONSTELLATION_QAM64 + 1);
6628*4882a593Smuzhiyun 			break;
6629*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM256:
6630*4882a593Smuzhiyun 			/* a(16 bit) * b(5 bit) = 21 bit result => mult32 not needed */
6631*4882a593Smuzhiyun 			qam_vd_period =
6632*4882a593Smuzhiyun 			    qam_vd_bit_cnt * (QAM_TOP_CONSTELLATION_QAM256 + 1)
6633*4882a593Smuzhiyun 			    * (QAM_TOP_CONSTELLATION_QAM256 + 1);
6634*4882a593Smuzhiyun 			break;
6635*4882a593Smuzhiyun 		default:
6636*4882a593Smuzhiyun 			return -EINVAL;
6637*4882a593Smuzhiyun 		}
6638*4882a593Smuzhiyun 		if (qam_vd_period == 0) {
6639*4882a593Smuzhiyun 			pr_err("error: qam_vd_period is zero!\n");
6640*4882a593Smuzhiyun 			return -EIO;
6641*4882a593Smuzhiyun 		}
6642*4882a593Smuzhiyun 		qam_vd_period = fec_bits_desired / qam_vd_period;
6643*4882a593Smuzhiyun 		/* limit to max 16 bit value (I2C register width) if needed */
6644*4882a593Smuzhiyun 		if (qam_vd_period > 0xFFFF)
6645*4882a593Smuzhiyun 			qam_vd_period = 0xFFFF;
6646*4882a593Smuzhiyun 
6647*4882a593Smuzhiyun 		/* a(16 bit) * b(16 bit) = 32 bit result => mult32 not needed */
6648*4882a593Smuzhiyun 		qam_vd_bit_cnt *= qam_vd_period;
6649*4882a593Smuzhiyun 
6650*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_VD_MEASUREMENT_PERIOD__A, (u16)qam_vd_period, 0);
6651*4882a593Smuzhiyun 		if (rc != 0) {
6652*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6653*4882a593Smuzhiyun 			goto rw_error;
6654*4882a593Smuzhiyun 		}
6655*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_VD_MEASUREMENT_PRESCALE__A, qam_vd_prescale, 0);
6656*4882a593Smuzhiyun 		if (rc != 0) {
6657*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
6658*4882a593Smuzhiyun 			goto rw_error;
6659*4882a593Smuzhiyun 		}
6660*4882a593Smuzhiyun 		ext_attr->qam_vd_period = (u16) qam_vd_period;
6661*4882a593Smuzhiyun 		ext_attr->qam_vd_prescale = qam_vd_prescale;
6662*4882a593Smuzhiyun 	}
6663*4882a593Smuzhiyun 
6664*4882a593Smuzhiyun 	return 0;
6665*4882a593Smuzhiyun rw_error:
6666*4882a593Smuzhiyun 	return rc;
6667*4882a593Smuzhiyun }
6668*4882a593Smuzhiyun 
6669*4882a593Smuzhiyun /*============================================================================*/
6670*4882a593Smuzhiyun 
6671*4882a593Smuzhiyun /*
6672*4882a593Smuzhiyun * \fn int set_qam16 ()
6673*4882a593Smuzhiyun * \brief QAM16 specific setup
6674*4882a593Smuzhiyun * \param demod instance of demod.
6675*4882a593Smuzhiyun * \return int.
6676*4882a593Smuzhiyun */
set_qam16(struct drx_demod_instance * demod)6677*4882a593Smuzhiyun static int set_qam16(struct drx_demod_instance *demod)
6678*4882a593Smuzhiyun {
6679*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
6680*4882a593Smuzhiyun 	int rc;
6681*4882a593Smuzhiyun 	static const u8 qam_dq_qual_fun[] = {
6682*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* fun0  */
6683*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* fun1  */
6684*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* fun2  */
6685*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* fun3  */
6686*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* fun4  */
6687*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* fun5  */
6688*4882a593Smuzhiyun 	};
6689*4882a593Smuzhiyun 	static const u8 qam_eq_cma_rad[] = {
6690*4882a593Smuzhiyun 		DRXJ_16TO8(13517),	/* RAD0  */
6691*4882a593Smuzhiyun 		DRXJ_16TO8(13517),	/* RAD1  */
6692*4882a593Smuzhiyun 		DRXJ_16TO8(13517),	/* RAD2  */
6693*4882a593Smuzhiyun 		DRXJ_16TO8(13517),	/* RAD3  */
6694*4882a593Smuzhiyun 		DRXJ_16TO8(13517),	/* RAD4  */
6695*4882a593Smuzhiyun 		DRXJ_16TO8(13517),	/* RAD5  */
6696*4882a593Smuzhiyun 	};
6697*4882a593Smuzhiyun 
6698*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
6699*4882a593Smuzhiyun 	if (rc != 0) {
6700*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6701*4882a593Smuzhiyun 		goto rw_error;
6702*4882a593Smuzhiyun 	}
6703*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
6704*4882a593Smuzhiyun 	if (rc != 0) {
6705*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6706*4882a593Smuzhiyun 		goto rw_error;
6707*4882a593Smuzhiyun 	}
6708*4882a593Smuzhiyun 
6709*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 140, 0);
6710*4882a593Smuzhiyun 	if (rc != 0) {
6711*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6712*4882a593Smuzhiyun 		goto rw_error;
6713*4882a593Smuzhiyun 	}
6714*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 50, 0);
6715*4882a593Smuzhiyun 	if (rc != 0) {
6716*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6717*4882a593Smuzhiyun 		goto rw_error;
6718*4882a593Smuzhiyun 	}
6719*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 120, 0);
6720*4882a593Smuzhiyun 	if (rc != 0) {
6721*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6722*4882a593Smuzhiyun 		goto rw_error;
6723*4882a593Smuzhiyun 	}
6724*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 230, 0);
6725*4882a593Smuzhiyun 	if (rc != 0) {
6726*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6727*4882a593Smuzhiyun 		goto rw_error;
6728*4882a593Smuzhiyun 	}
6729*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 95, 0);
6730*4882a593Smuzhiyun 	if (rc != 0) {
6731*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6732*4882a593Smuzhiyun 		goto rw_error;
6733*4882a593Smuzhiyun 	}
6734*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 105, 0);
6735*4882a593Smuzhiyun 	if (rc != 0) {
6736*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6737*4882a593Smuzhiyun 		goto rw_error;
6738*4882a593Smuzhiyun 	}
6739*4882a593Smuzhiyun 
6740*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
6741*4882a593Smuzhiyun 	if (rc != 0) {
6742*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6743*4882a593Smuzhiyun 		goto rw_error;
6744*4882a593Smuzhiyun 	}
6745*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 56, 0);
6746*4882a593Smuzhiyun 	if (rc != 0) {
6747*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6748*4882a593Smuzhiyun 		goto rw_error;
6749*4882a593Smuzhiyun 	}
6750*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
6751*4882a593Smuzhiyun 	if (rc != 0) {
6752*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6753*4882a593Smuzhiyun 		goto rw_error;
6754*4882a593Smuzhiyun 	}
6755*4882a593Smuzhiyun 
6756*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 16, 0);
6757*4882a593Smuzhiyun 	if (rc != 0) {
6758*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6759*4882a593Smuzhiyun 		goto rw_error;
6760*4882a593Smuzhiyun 	}
6761*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 220, 0);
6762*4882a593Smuzhiyun 	if (rc != 0) {
6763*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6764*4882a593Smuzhiyun 		goto rw_error;
6765*4882a593Smuzhiyun 	}
6766*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 25, 0);
6767*4882a593Smuzhiyun 	if (rc != 0) {
6768*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6769*4882a593Smuzhiyun 		goto rw_error;
6770*4882a593Smuzhiyun 	}
6771*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 6, 0);
6772*4882a593Smuzhiyun 	if (rc != 0) {
6773*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6774*4882a593Smuzhiyun 		goto rw_error;
6775*4882a593Smuzhiyun 	}
6776*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-24), 0);
6777*4882a593Smuzhiyun 	if (rc != 0) {
6778*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6779*4882a593Smuzhiyun 		goto rw_error;
6780*4882a593Smuzhiyun 	}
6781*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, (u16)(-65), 0);
6782*4882a593Smuzhiyun 	if (rc != 0) {
6783*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6784*4882a593Smuzhiyun 		goto rw_error;
6785*4882a593Smuzhiyun 	}
6786*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-127), 0);
6787*4882a593Smuzhiyun 	if (rc != 0) {
6788*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6789*4882a593Smuzhiyun 		goto rw_error;
6790*4882a593Smuzhiyun 	}
6791*4882a593Smuzhiyun 
6792*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
6793*4882a593Smuzhiyun 	if (rc != 0) {
6794*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6795*4882a593Smuzhiyun 		goto rw_error;
6796*4882a593Smuzhiyun 	}
6797*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
6798*4882a593Smuzhiyun 	if (rc != 0) {
6799*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6800*4882a593Smuzhiyun 		goto rw_error;
6801*4882a593Smuzhiyun 	}
6802*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
6803*4882a593Smuzhiyun 	if (rc != 0) {
6804*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6805*4882a593Smuzhiyun 		goto rw_error;
6806*4882a593Smuzhiyun 	}
6807*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 20, 0);
6808*4882a593Smuzhiyun 	if (rc != 0) {
6809*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6810*4882a593Smuzhiyun 		goto rw_error;
6811*4882a593Smuzhiyun 	}
6812*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
6813*4882a593Smuzhiyun 	if (rc != 0) {
6814*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6815*4882a593Smuzhiyun 		goto rw_error;
6816*4882a593Smuzhiyun 	}
6817*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
6818*4882a593Smuzhiyun 	if (rc != 0) {
6819*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6820*4882a593Smuzhiyun 		goto rw_error;
6821*4882a593Smuzhiyun 	}
6822*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 10, 0);
6823*4882a593Smuzhiyun 	if (rc != 0) {
6824*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6825*4882a593Smuzhiyun 		goto rw_error;
6826*4882a593Smuzhiyun 	}
6827*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 50, 0);
6828*4882a593Smuzhiyun 	if (rc != 0) {
6829*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6830*4882a593Smuzhiyun 		goto rw_error;
6831*4882a593Smuzhiyun 	}
6832*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
6833*4882a593Smuzhiyun 	if (rc != 0) {
6834*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6835*4882a593Smuzhiyun 		goto rw_error;
6836*4882a593Smuzhiyun 	}
6837*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
6838*4882a593Smuzhiyun 	if (rc != 0) {
6839*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6840*4882a593Smuzhiyun 		goto rw_error;
6841*4882a593Smuzhiyun 	}
6842*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
6843*4882a593Smuzhiyun 	if (rc != 0) {
6844*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6845*4882a593Smuzhiyun 		goto rw_error;
6846*4882a593Smuzhiyun 	}
6847*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
6848*4882a593Smuzhiyun 	if (rc != 0) {
6849*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6850*4882a593Smuzhiyun 		goto rw_error;
6851*4882a593Smuzhiyun 	}
6852*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
6853*4882a593Smuzhiyun 	if (rc != 0) {
6854*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6855*4882a593Smuzhiyun 		goto rw_error;
6856*4882a593Smuzhiyun 	}
6857*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
6858*4882a593Smuzhiyun 	if (rc != 0) {
6859*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6860*4882a593Smuzhiyun 		goto rw_error;
6861*4882a593Smuzhiyun 	}
6862*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
6863*4882a593Smuzhiyun 	if (rc != 0) {
6864*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6865*4882a593Smuzhiyun 		goto rw_error;
6866*4882a593Smuzhiyun 	}
6867*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 32, 0);
6868*4882a593Smuzhiyun 	if (rc != 0) {
6869*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6870*4882a593Smuzhiyun 		goto rw_error;
6871*4882a593Smuzhiyun 	}
6872*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 240, 0);
6873*4882a593Smuzhiyun 	if (rc != 0) {
6874*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6875*4882a593Smuzhiyun 		goto rw_error;
6876*4882a593Smuzhiyun 	}
6877*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
6878*4882a593Smuzhiyun 	if (rc != 0) {
6879*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6880*4882a593Smuzhiyun 		goto rw_error;
6881*4882a593Smuzhiyun 	}
6882*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
6883*4882a593Smuzhiyun 	if (rc != 0) {
6884*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6885*4882a593Smuzhiyun 		goto rw_error;
6886*4882a593Smuzhiyun 	}
6887*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 32, 0);
6888*4882a593Smuzhiyun 	if (rc != 0) {
6889*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6890*4882a593Smuzhiyun 		goto rw_error;
6891*4882a593Smuzhiyun 	}
6892*4882a593Smuzhiyun 
6893*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 40960, 0);
6894*4882a593Smuzhiyun 	if (rc != 0) {
6895*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6896*4882a593Smuzhiyun 		goto rw_error;
6897*4882a593Smuzhiyun 	}
6898*4882a593Smuzhiyun 
6899*4882a593Smuzhiyun 	return 0;
6900*4882a593Smuzhiyun rw_error:
6901*4882a593Smuzhiyun 	return rc;
6902*4882a593Smuzhiyun }
6903*4882a593Smuzhiyun 
6904*4882a593Smuzhiyun /*============================================================================*/
6905*4882a593Smuzhiyun 
6906*4882a593Smuzhiyun /*
6907*4882a593Smuzhiyun * \fn int set_qam32 ()
6908*4882a593Smuzhiyun * \brief QAM32 specific setup
6909*4882a593Smuzhiyun * \param demod instance of demod.
6910*4882a593Smuzhiyun * \return int.
6911*4882a593Smuzhiyun */
set_qam32(struct drx_demod_instance * demod)6912*4882a593Smuzhiyun static int set_qam32(struct drx_demod_instance *demod)
6913*4882a593Smuzhiyun {
6914*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
6915*4882a593Smuzhiyun 	int rc;
6916*4882a593Smuzhiyun 	static const u8 qam_dq_qual_fun[] = {
6917*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* fun0  */
6918*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* fun1  */
6919*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* fun2  */
6920*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* fun3  */
6921*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* fun4  */
6922*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* fun5  */
6923*4882a593Smuzhiyun 	};
6924*4882a593Smuzhiyun 	static const u8 qam_eq_cma_rad[] = {
6925*4882a593Smuzhiyun 		DRXJ_16TO8(6707),	/* RAD0  */
6926*4882a593Smuzhiyun 		DRXJ_16TO8(6707),	/* RAD1  */
6927*4882a593Smuzhiyun 		DRXJ_16TO8(6707),	/* RAD2  */
6928*4882a593Smuzhiyun 		DRXJ_16TO8(6707),	/* RAD3  */
6929*4882a593Smuzhiyun 		DRXJ_16TO8(6707),	/* RAD4  */
6930*4882a593Smuzhiyun 		DRXJ_16TO8(6707),	/* RAD5  */
6931*4882a593Smuzhiyun 	};
6932*4882a593Smuzhiyun 
6933*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
6934*4882a593Smuzhiyun 	if (rc != 0) {
6935*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6936*4882a593Smuzhiyun 		goto rw_error;
6937*4882a593Smuzhiyun 	}
6938*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
6939*4882a593Smuzhiyun 	if (rc != 0) {
6940*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6941*4882a593Smuzhiyun 		goto rw_error;
6942*4882a593Smuzhiyun 	}
6943*4882a593Smuzhiyun 
6944*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 90, 0);
6945*4882a593Smuzhiyun 	if (rc != 0) {
6946*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6947*4882a593Smuzhiyun 		goto rw_error;
6948*4882a593Smuzhiyun 	}
6949*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 50, 0);
6950*4882a593Smuzhiyun 	if (rc != 0) {
6951*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6952*4882a593Smuzhiyun 		goto rw_error;
6953*4882a593Smuzhiyun 	}
6954*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
6955*4882a593Smuzhiyun 	if (rc != 0) {
6956*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6957*4882a593Smuzhiyun 		goto rw_error;
6958*4882a593Smuzhiyun 	}
6959*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 170, 0);
6960*4882a593Smuzhiyun 	if (rc != 0) {
6961*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6962*4882a593Smuzhiyun 		goto rw_error;
6963*4882a593Smuzhiyun 	}
6964*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
6965*4882a593Smuzhiyun 	if (rc != 0) {
6966*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6967*4882a593Smuzhiyun 		goto rw_error;
6968*4882a593Smuzhiyun 	}
6969*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 100, 0);
6970*4882a593Smuzhiyun 	if (rc != 0) {
6971*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6972*4882a593Smuzhiyun 		goto rw_error;
6973*4882a593Smuzhiyun 	}
6974*4882a593Smuzhiyun 
6975*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
6976*4882a593Smuzhiyun 	if (rc != 0) {
6977*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6978*4882a593Smuzhiyun 		goto rw_error;
6979*4882a593Smuzhiyun 	}
6980*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 56, 0);
6981*4882a593Smuzhiyun 	if (rc != 0) {
6982*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6983*4882a593Smuzhiyun 		goto rw_error;
6984*4882a593Smuzhiyun 	}
6985*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
6986*4882a593Smuzhiyun 	if (rc != 0) {
6987*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6988*4882a593Smuzhiyun 		goto rw_error;
6989*4882a593Smuzhiyun 	}
6990*4882a593Smuzhiyun 
6991*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 12, 0);
6992*4882a593Smuzhiyun 	if (rc != 0) {
6993*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6994*4882a593Smuzhiyun 		goto rw_error;
6995*4882a593Smuzhiyun 	}
6996*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 140, 0);
6997*4882a593Smuzhiyun 	if (rc != 0) {
6998*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
6999*4882a593Smuzhiyun 		goto rw_error;
7000*4882a593Smuzhiyun 	}
7001*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, (u16)(-8), 0);
7002*4882a593Smuzhiyun 	if (rc != 0) {
7003*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7004*4882a593Smuzhiyun 		goto rw_error;
7005*4882a593Smuzhiyun 	}
7006*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, (u16)(-16), 0);
7007*4882a593Smuzhiyun 	if (rc != 0) {
7008*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7009*4882a593Smuzhiyun 		goto rw_error;
7010*4882a593Smuzhiyun 	}
7011*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-26), 0);
7012*4882a593Smuzhiyun 	if (rc != 0) {
7013*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7014*4882a593Smuzhiyun 		goto rw_error;
7015*4882a593Smuzhiyun 	}
7016*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, (u16)(-56), 0);
7017*4882a593Smuzhiyun 	if (rc != 0) {
7018*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7019*4882a593Smuzhiyun 		goto rw_error;
7020*4882a593Smuzhiyun 	}
7021*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-86), 0);
7022*4882a593Smuzhiyun 	if (rc != 0) {
7023*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7024*4882a593Smuzhiyun 		goto rw_error;
7025*4882a593Smuzhiyun 	}
7026*4882a593Smuzhiyun 
7027*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
7028*4882a593Smuzhiyun 	if (rc != 0) {
7029*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7030*4882a593Smuzhiyun 		goto rw_error;
7031*4882a593Smuzhiyun 	}
7032*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
7033*4882a593Smuzhiyun 	if (rc != 0) {
7034*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7035*4882a593Smuzhiyun 		goto rw_error;
7036*4882a593Smuzhiyun 	}
7037*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
7038*4882a593Smuzhiyun 	if (rc != 0) {
7039*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7040*4882a593Smuzhiyun 		goto rw_error;
7041*4882a593Smuzhiyun 	}
7042*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 20, 0);
7043*4882a593Smuzhiyun 	if (rc != 0) {
7044*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7045*4882a593Smuzhiyun 		goto rw_error;
7046*4882a593Smuzhiyun 	}
7047*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
7048*4882a593Smuzhiyun 	if (rc != 0) {
7049*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7050*4882a593Smuzhiyun 		goto rw_error;
7051*4882a593Smuzhiyun 	}
7052*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
7053*4882a593Smuzhiyun 	if (rc != 0) {
7054*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7055*4882a593Smuzhiyun 		goto rw_error;
7056*4882a593Smuzhiyun 	}
7057*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 10, 0);
7058*4882a593Smuzhiyun 	if (rc != 0) {
7059*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7060*4882a593Smuzhiyun 		goto rw_error;
7061*4882a593Smuzhiyun 	}
7062*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 50, 0);
7063*4882a593Smuzhiyun 	if (rc != 0) {
7064*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7065*4882a593Smuzhiyun 		goto rw_error;
7066*4882a593Smuzhiyun 	}
7067*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
7068*4882a593Smuzhiyun 	if (rc != 0) {
7069*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7070*4882a593Smuzhiyun 		goto rw_error;
7071*4882a593Smuzhiyun 	}
7072*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
7073*4882a593Smuzhiyun 	if (rc != 0) {
7074*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7075*4882a593Smuzhiyun 		goto rw_error;
7076*4882a593Smuzhiyun 	}
7077*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
7078*4882a593Smuzhiyun 	if (rc != 0) {
7079*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7080*4882a593Smuzhiyun 		goto rw_error;
7081*4882a593Smuzhiyun 	}
7082*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
7083*4882a593Smuzhiyun 	if (rc != 0) {
7084*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7085*4882a593Smuzhiyun 		goto rw_error;
7086*4882a593Smuzhiyun 	}
7087*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
7088*4882a593Smuzhiyun 	if (rc != 0) {
7089*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7090*4882a593Smuzhiyun 		goto rw_error;
7091*4882a593Smuzhiyun 	}
7092*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
7093*4882a593Smuzhiyun 	if (rc != 0) {
7094*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7095*4882a593Smuzhiyun 		goto rw_error;
7096*4882a593Smuzhiyun 	}
7097*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
7098*4882a593Smuzhiyun 	if (rc != 0) {
7099*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7100*4882a593Smuzhiyun 		goto rw_error;
7101*4882a593Smuzhiyun 	}
7102*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 32, 0);
7103*4882a593Smuzhiyun 	if (rc != 0) {
7104*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7105*4882a593Smuzhiyun 		goto rw_error;
7106*4882a593Smuzhiyun 	}
7107*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 176, 0);
7108*4882a593Smuzhiyun 	if (rc != 0) {
7109*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7110*4882a593Smuzhiyun 		goto rw_error;
7111*4882a593Smuzhiyun 	}
7112*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
7113*4882a593Smuzhiyun 	if (rc != 0) {
7114*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7115*4882a593Smuzhiyun 		goto rw_error;
7116*4882a593Smuzhiyun 	}
7117*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
7118*4882a593Smuzhiyun 	if (rc != 0) {
7119*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7120*4882a593Smuzhiyun 		goto rw_error;
7121*4882a593Smuzhiyun 	}
7122*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 8, 0);
7123*4882a593Smuzhiyun 	if (rc != 0) {
7124*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7125*4882a593Smuzhiyun 		goto rw_error;
7126*4882a593Smuzhiyun 	}
7127*4882a593Smuzhiyun 
7128*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 20480, 0);
7129*4882a593Smuzhiyun 	if (rc != 0) {
7130*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7131*4882a593Smuzhiyun 		goto rw_error;
7132*4882a593Smuzhiyun 	}
7133*4882a593Smuzhiyun 
7134*4882a593Smuzhiyun 	return 0;
7135*4882a593Smuzhiyun rw_error:
7136*4882a593Smuzhiyun 	return rc;
7137*4882a593Smuzhiyun }
7138*4882a593Smuzhiyun 
7139*4882a593Smuzhiyun /*============================================================================*/
7140*4882a593Smuzhiyun 
7141*4882a593Smuzhiyun /*
7142*4882a593Smuzhiyun * \fn int set_qam64 ()
7143*4882a593Smuzhiyun * \brief QAM64 specific setup
7144*4882a593Smuzhiyun * \param demod instance of demod.
7145*4882a593Smuzhiyun * \return int.
7146*4882a593Smuzhiyun */
set_qam64(struct drx_demod_instance * demod)7147*4882a593Smuzhiyun static int set_qam64(struct drx_demod_instance *demod)
7148*4882a593Smuzhiyun {
7149*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
7150*4882a593Smuzhiyun 	int rc;
7151*4882a593Smuzhiyun 	static const u8 qam_dq_qual_fun[] = {
7152*4882a593Smuzhiyun 		/* this is hw reset value. no necessary to re-write */
7153*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* fun0  */
7154*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* fun1  */
7155*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* fun2  */
7156*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* fun3  */
7157*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* fun4  */
7158*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* fun5  */
7159*4882a593Smuzhiyun 	};
7160*4882a593Smuzhiyun 	static const u8 qam_eq_cma_rad[] = {
7161*4882a593Smuzhiyun 		DRXJ_16TO8(13336),	/* RAD0  */
7162*4882a593Smuzhiyun 		DRXJ_16TO8(12618),	/* RAD1  */
7163*4882a593Smuzhiyun 		DRXJ_16TO8(11988),	/* RAD2  */
7164*4882a593Smuzhiyun 		DRXJ_16TO8(13809),	/* RAD3  */
7165*4882a593Smuzhiyun 		DRXJ_16TO8(13809),	/* RAD4  */
7166*4882a593Smuzhiyun 		DRXJ_16TO8(15609),	/* RAD5  */
7167*4882a593Smuzhiyun 	};
7168*4882a593Smuzhiyun 
7169*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
7170*4882a593Smuzhiyun 	if (rc != 0) {
7171*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7172*4882a593Smuzhiyun 		goto rw_error;
7173*4882a593Smuzhiyun 	}
7174*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
7175*4882a593Smuzhiyun 	if (rc != 0) {
7176*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7177*4882a593Smuzhiyun 		goto rw_error;
7178*4882a593Smuzhiyun 	}
7179*4882a593Smuzhiyun 
7180*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 105, 0);
7181*4882a593Smuzhiyun 	if (rc != 0) {
7182*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7183*4882a593Smuzhiyun 		goto rw_error;
7184*4882a593Smuzhiyun 	}
7185*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 60, 0);
7186*4882a593Smuzhiyun 	if (rc != 0) {
7187*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7188*4882a593Smuzhiyun 		goto rw_error;
7189*4882a593Smuzhiyun 	}
7190*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
7191*4882a593Smuzhiyun 	if (rc != 0) {
7192*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7193*4882a593Smuzhiyun 		goto rw_error;
7194*4882a593Smuzhiyun 	}
7195*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 195, 0);
7196*4882a593Smuzhiyun 	if (rc != 0) {
7197*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7198*4882a593Smuzhiyun 		goto rw_error;
7199*4882a593Smuzhiyun 	}
7200*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
7201*4882a593Smuzhiyun 	if (rc != 0) {
7202*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7203*4882a593Smuzhiyun 		goto rw_error;
7204*4882a593Smuzhiyun 	}
7205*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 84, 0);
7206*4882a593Smuzhiyun 	if (rc != 0) {
7207*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7208*4882a593Smuzhiyun 		goto rw_error;
7209*4882a593Smuzhiyun 	}
7210*4882a593Smuzhiyun 
7211*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
7212*4882a593Smuzhiyun 	if (rc != 0) {
7213*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7214*4882a593Smuzhiyun 		goto rw_error;
7215*4882a593Smuzhiyun 	}
7216*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 32, 0);
7217*4882a593Smuzhiyun 	if (rc != 0) {
7218*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7219*4882a593Smuzhiyun 		goto rw_error;
7220*4882a593Smuzhiyun 	}
7221*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
7222*4882a593Smuzhiyun 	if (rc != 0) {
7223*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7224*4882a593Smuzhiyun 		goto rw_error;
7225*4882a593Smuzhiyun 	}
7226*4882a593Smuzhiyun 
7227*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 12, 0);
7228*4882a593Smuzhiyun 	if (rc != 0) {
7229*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7230*4882a593Smuzhiyun 		goto rw_error;
7231*4882a593Smuzhiyun 	}
7232*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 141, 0);
7233*4882a593Smuzhiyun 	if (rc != 0) {
7234*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7235*4882a593Smuzhiyun 		goto rw_error;
7236*4882a593Smuzhiyun 	}
7237*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 7, 0);
7238*4882a593Smuzhiyun 	if (rc != 0) {
7239*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7240*4882a593Smuzhiyun 		goto rw_error;
7241*4882a593Smuzhiyun 	}
7242*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 0, 0);
7243*4882a593Smuzhiyun 	if (rc != 0) {
7244*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7245*4882a593Smuzhiyun 		goto rw_error;
7246*4882a593Smuzhiyun 	}
7247*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-15), 0);
7248*4882a593Smuzhiyun 	if (rc != 0) {
7249*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7250*4882a593Smuzhiyun 		goto rw_error;
7251*4882a593Smuzhiyun 	}
7252*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, (u16)(-45), 0);
7253*4882a593Smuzhiyun 	if (rc != 0) {
7254*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7255*4882a593Smuzhiyun 		goto rw_error;
7256*4882a593Smuzhiyun 	}
7257*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-80), 0);
7258*4882a593Smuzhiyun 	if (rc != 0) {
7259*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7260*4882a593Smuzhiyun 		goto rw_error;
7261*4882a593Smuzhiyun 	}
7262*4882a593Smuzhiyun 
7263*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
7264*4882a593Smuzhiyun 	if (rc != 0) {
7265*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7266*4882a593Smuzhiyun 		goto rw_error;
7267*4882a593Smuzhiyun 	}
7268*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
7269*4882a593Smuzhiyun 	if (rc != 0) {
7270*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7271*4882a593Smuzhiyun 		goto rw_error;
7272*4882a593Smuzhiyun 	}
7273*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
7274*4882a593Smuzhiyun 	if (rc != 0) {
7275*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7276*4882a593Smuzhiyun 		goto rw_error;
7277*4882a593Smuzhiyun 	}
7278*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 30, 0);
7279*4882a593Smuzhiyun 	if (rc != 0) {
7280*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7281*4882a593Smuzhiyun 		goto rw_error;
7282*4882a593Smuzhiyun 	}
7283*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
7284*4882a593Smuzhiyun 	if (rc != 0) {
7285*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7286*4882a593Smuzhiyun 		goto rw_error;
7287*4882a593Smuzhiyun 	}
7288*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
7289*4882a593Smuzhiyun 	if (rc != 0) {
7290*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7291*4882a593Smuzhiyun 		goto rw_error;
7292*4882a593Smuzhiyun 	}
7293*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 15, 0);
7294*4882a593Smuzhiyun 	if (rc != 0) {
7295*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7296*4882a593Smuzhiyun 		goto rw_error;
7297*4882a593Smuzhiyun 	}
7298*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 80, 0);
7299*4882a593Smuzhiyun 	if (rc != 0) {
7300*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7301*4882a593Smuzhiyun 		goto rw_error;
7302*4882a593Smuzhiyun 	}
7303*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
7304*4882a593Smuzhiyun 	if (rc != 0) {
7305*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7306*4882a593Smuzhiyun 		goto rw_error;
7307*4882a593Smuzhiyun 	}
7308*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
7309*4882a593Smuzhiyun 	if (rc != 0) {
7310*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7311*4882a593Smuzhiyun 		goto rw_error;
7312*4882a593Smuzhiyun 	}
7313*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
7314*4882a593Smuzhiyun 	if (rc != 0) {
7315*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7316*4882a593Smuzhiyun 		goto rw_error;
7317*4882a593Smuzhiyun 	}
7318*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
7319*4882a593Smuzhiyun 	if (rc != 0) {
7320*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7321*4882a593Smuzhiyun 		goto rw_error;
7322*4882a593Smuzhiyun 	}
7323*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
7324*4882a593Smuzhiyun 	if (rc != 0) {
7325*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7326*4882a593Smuzhiyun 		goto rw_error;
7327*4882a593Smuzhiyun 	}
7328*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
7329*4882a593Smuzhiyun 	if (rc != 0) {
7330*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7331*4882a593Smuzhiyun 		goto rw_error;
7332*4882a593Smuzhiyun 	}
7333*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
7334*4882a593Smuzhiyun 	if (rc != 0) {
7335*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7336*4882a593Smuzhiyun 		goto rw_error;
7337*4882a593Smuzhiyun 	}
7338*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 48, 0);
7339*4882a593Smuzhiyun 	if (rc != 0) {
7340*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7341*4882a593Smuzhiyun 		goto rw_error;
7342*4882a593Smuzhiyun 	}
7343*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 160, 0);
7344*4882a593Smuzhiyun 	if (rc != 0) {
7345*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7346*4882a593Smuzhiyun 		goto rw_error;
7347*4882a593Smuzhiyun 	}
7348*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
7349*4882a593Smuzhiyun 	if (rc != 0) {
7350*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7351*4882a593Smuzhiyun 		goto rw_error;
7352*4882a593Smuzhiyun 	}
7353*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
7354*4882a593Smuzhiyun 	if (rc != 0) {
7355*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7356*4882a593Smuzhiyun 		goto rw_error;
7357*4882a593Smuzhiyun 	}
7358*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 32, 0);
7359*4882a593Smuzhiyun 	if (rc != 0) {
7360*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7361*4882a593Smuzhiyun 		goto rw_error;
7362*4882a593Smuzhiyun 	}
7363*4882a593Smuzhiyun 
7364*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 43008, 0);
7365*4882a593Smuzhiyun 	if (rc != 0) {
7366*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7367*4882a593Smuzhiyun 		goto rw_error;
7368*4882a593Smuzhiyun 	}
7369*4882a593Smuzhiyun 
7370*4882a593Smuzhiyun 	return 0;
7371*4882a593Smuzhiyun rw_error:
7372*4882a593Smuzhiyun 	return rc;
7373*4882a593Smuzhiyun }
7374*4882a593Smuzhiyun 
7375*4882a593Smuzhiyun /*============================================================================*/
7376*4882a593Smuzhiyun 
7377*4882a593Smuzhiyun /*
7378*4882a593Smuzhiyun * \fn int set_qam128 ()
7379*4882a593Smuzhiyun * \brief QAM128 specific setup
7380*4882a593Smuzhiyun * \param demod: instance of demod.
7381*4882a593Smuzhiyun * \return int.
7382*4882a593Smuzhiyun */
set_qam128(struct drx_demod_instance * demod)7383*4882a593Smuzhiyun static int set_qam128(struct drx_demod_instance *demod)
7384*4882a593Smuzhiyun {
7385*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
7386*4882a593Smuzhiyun 	int rc;
7387*4882a593Smuzhiyun 	static const u8 qam_dq_qual_fun[] = {
7388*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* fun0  */
7389*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* fun1  */
7390*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* fun2  */
7391*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* fun3  */
7392*4882a593Smuzhiyun 		DRXJ_16TO8(9),	/* fun4  */
7393*4882a593Smuzhiyun 		DRXJ_16TO8(9),	/* fun5  */
7394*4882a593Smuzhiyun 	};
7395*4882a593Smuzhiyun 	static const u8 qam_eq_cma_rad[] = {
7396*4882a593Smuzhiyun 		DRXJ_16TO8(6164),	/* RAD0  */
7397*4882a593Smuzhiyun 		DRXJ_16TO8(6598),	/* RAD1  */
7398*4882a593Smuzhiyun 		DRXJ_16TO8(6394),	/* RAD2  */
7399*4882a593Smuzhiyun 		DRXJ_16TO8(6409),	/* RAD3  */
7400*4882a593Smuzhiyun 		DRXJ_16TO8(6656),	/* RAD4  */
7401*4882a593Smuzhiyun 		DRXJ_16TO8(7238),	/* RAD5  */
7402*4882a593Smuzhiyun 	};
7403*4882a593Smuzhiyun 
7404*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
7405*4882a593Smuzhiyun 	if (rc != 0) {
7406*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7407*4882a593Smuzhiyun 		goto rw_error;
7408*4882a593Smuzhiyun 	}
7409*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
7410*4882a593Smuzhiyun 	if (rc != 0) {
7411*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7412*4882a593Smuzhiyun 		goto rw_error;
7413*4882a593Smuzhiyun 	}
7414*4882a593Smuzhiyun 
7415*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 50, 0);
7416*4882a593Smuzhiyun 	if (rc != 0) {
7417*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7418*4882a593Smuzhiyun 		goto rw_error;
7419*4882a593Smuzhiyun 	}
7420*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 60, 0);
7421*4882a593Smuzhiyun 	if (rc != 0) {
7422*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7423*4882a593Smuzhiyun 		goto rw_error;
7424*4882a593Smuzhiyun 	}
7425*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
7426*4882a593Smuzhiyun 	if (rc != 0) {
7427*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7428*4882a593Smuzhiyun 		goto rw_error;
7429*4882a593Smuzhiyun 	}
7430*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 140, 0);
7431*4882a593Smuzhiyun 	if (rc != 0) {
7432*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7433*4882a593Smuzhiyun 		goto rw_error;
7434*4882a593Smuzhiyun 	}
7435*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
7436*4882a593Smuzhiyun 	if (rc != 0) {
7437*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7438*4882a593Smuzhiyun 		goto rw_error;
7439*4882a593Smuzhiyun 	}
7440*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 100, 0);
7441*4882a593Smuzhiyun 	if (rc != 0) {
7442*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7443*4882a593Smuzhiyun 		goto rw_error;
7444*4882a593Smuzhiyun 	}
7445*4882a593Smuzhiyun 
7446*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
7447*4882a593Smuzhiyun 	if (rc != 0) {
7448*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7449*4882a593Smuzhiyun 		goto rw_error;
7450*4882a593Smuzhiyun 	}
7451*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 32, 0);
7452*4882a593Smuzhiyun 	if (rc != 0) {
7453*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7454*4882a593Smuzhiyun 		goto rw_error;
7455*4882a593Smuzhiyun 	}
7456*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
7457*4882a593Smuzhiyun 	if (rc != 0) {
7458*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7459*4882a593Smuzhiyun 		goto rw_error;
7460*4882a593Smuzhiyun 	}
7461*4882a593Smuzhiyun 
7462*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 8, 0);
7463*4882a593Smuzhiyun 	if (rc != 0) {
7464*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7465*4882a593Smuzhiyun 		goto rw_error;
7466*4882a593Smuzhiyun 	}
7467*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 65, 0);
7468*4882a593Smuzhiyun 	if (rc != 0) {
7469*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7470*4882a593Smuzhiyun 		goto rw_error;
7471*4882a593Smuzhiyun 	}
7472*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 5, 0);
7473*4882a593Smuzhiyun 	if (rc != 0) {
7474*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7475*4882a593Smuzhiyun 		goto rw_error;
7476*4882a593Smuzhiyun 	}
7477*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 3, 0);
7478*4882a593Smuzhiyun 	if (rc != 0) {
7479*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7480*4882a593Smuzhiyun 		goto rw_error;
7481*4882a593Smuzhiyun 	}
7482*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-1), 0);
7483*4882a593Smuzhiyun 	if (rc != 0) {
7484*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7485*4882a593Smuzhiyun 		goto rw_error;
7486*4882a593Smuzhiyun 	}
7487*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, 12, 0);
7488*4882a593Smuzhiyun 	if (rc != 0) {
7489*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7490*4882a593Smuzhiyun 		goto rw_error;
7491*4882a593Smuzhiyun 	}
7492*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-23), 0);
7493*4882a593Smuzhiyun 	if (rc != 0) {
7494*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7495*4882a593Smuzhiyun 		goto rw_error;
7496*4882a593Smuzhiyun 	}
7497*4882a593Smuzhiyun 
7498*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
7499*4882a593Smuzhiyun 	if (rc != 0) {
7500*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7501*4882a593Smuzhiyun 		goto rw_error;
7502*4882a593Smuzhiyun 	}
7503*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
7504*4882a593Smuzhiyun 	if (rc != 0) {
7505*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7506*4882a593Smuzhiyun 		goto rw_error;
7507*4882a593Smuzhiyun 	}
7508*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
7509*4882a593Smuzhiyun 	if (rc != 0) {
7510*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7511*4882a593Smuzhiyun 		goto rw_error;
7512*4882a593Smuzhiyun 	}
7513*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 40, 0);
7514*4882a593Smuzhiyun 	if (rc != 0) {
7515*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7516*4882a593Smuzhiyun 		goto rw_error;
7517*4882a593Smuzhiyun 	}
7518*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
7519*4882a593Smuzhiyun 	if (rc != 0) {
7520*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7521*4882a593Smuzhiyun 		goto rw_error;
7522*4882a593Smuzhiyun 	}
7523*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
7524*4882a593Smuzhiyun 	if (rc != 0) {
7525*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7526*4882a593Smuzhiyun 		goto rw_error;
7527*4882a593Smuzhiyun 	}
7528*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 20, 0);
7529*4882a593Smuzhiyun 	if (rc != 0) {
7530*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7531*4882a593Smuzhiyun 		goto rw_error;
7532*4882a593Smuzhiyun 	}
7533*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 80, 0);
7534*4882a593Smuzhiyun 	if (rc != 0) {
7535*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7536*4882a593Smuzhiyun 		goto rw_error;
7537*4882a593Smuzhiyun 	}
7538*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
7539*4882a593Smuzhiyun 	if (rc != 0) {
7540*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7541*4882a593Smuzhiyun 		goto rw_error;
7542*4882a593Smuzhiyun 	}
7543*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
7544*4882a593Smuzhiyun 	if (rc != 0) {
7545*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7546*4882a593Smuzhiyun 		goto rw_error;
7547*4882a593Smuzhiyun 	}
7548*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
7549*4882a593Smuzhiyun 	if (rc != 0) {
7550*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7551*4882a593Smuzhiyun 		goto rw_error;
7552*4882a593Smuzhiyun 	}
7553*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
7554*4882a593Smuzhiyun 	if (rc != 0) {
7555*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7556*4882a593Smuzhiyun 		goto rw_error;
7557*4882a593Smuzhiyun 	}
7558*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
7559*4882a593Smuzhiyun 	if (rc != 0) {
7560*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7561*4882a593Smuzhiyun 		goto rw_error;
7562*4882a593Smuzhiyun 	}
7563*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
7564*4882a593Smuzhiyun 	if (rc != 0) {
7565*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7566*4882a593Smuzhiyun 		goto rw_error;
7567*4882a593Smuzhiyun 	}
7568*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
7569*4882a593Smuzhiyun 	if (rc != 0) {
7570*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7571*4882a593Smuzhiyun 		goto rw_error;
7572*4882a593Smuzhiyun 	}
7573*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 32, 0);
7574*4882a593Smuzhiyun 	if (rc != 0) {
7575*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7576*4882a593Smuzhiyun 		goto rw_error;
7577*4882a593Smuzhiyun 	}
7578*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 144, 0);
7579*4882a593Smuzhiyun 	if (rc != 0) {
7580*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7581*4882a593Smuzhiyun 		goto rw_error;
7582*4882a593Smuzhiyun 	}
7583*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
7584*4882a593Smuzhiyun 	if (rc != 0) {
7585*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7586*4882a593Smuzhiyun 		goto rw_error;
7587*4882a593Smuzhiyun 	}
7588*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
7589*4882a593Smuzhiyun 	if (rc != 0) {
7590*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7591*4882a593Smuzhiyun 		goto rw_error;
7592*4882a593Smuzhiyun 	}
7593*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 16, 0);
7594*4882a593Smuzhiyun 	if (rc != 0) {
7595*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7596*4882a593Smuzhiyun 		goto rw_error;
7597*4882a593Smuzhiyun 	}
7598*4882a593Smuzhiyun 
7599*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 20992, 0);
7600*4882a593Smuzhiyun 	if (rc != 0) {
7601*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7602*4882a593Smuzhiyun 		goto rw_error;
7603*4882a593Smuzhiyun 	}
7604*4882a593Smuzhiyun 
7605*4882a593Smuzhiyun 	return 0;
7606*4882a593Smuzhiyun rw_error:
7607*4882a593Smuzhiyun 	return rc;
7608*4882a593Smuzhiyun }
7609*4882a593Smuzhiyun 
7610*4882a593Smuzhiyun /*============================================================================*/
7611*4882a593Smuzhiyun 
7612*4882a593Smuzhiyun /*
7613*4882a593Smuzhiyun * \fn int set_qam256 ()
7614*4882a593Smuzhiyun * \brief QAM256 specific setup
7615*4882a593Smuzhiyun * \param demod: instance of demod.
7616*4882a593Smuzhiyun * \return int.
7617*4882a593Smuzhiyun */
set_qam256(struct drx_demod_instance * demod)7618*4882a593Smuzhiyun static int set_qam256(struct drx_demod_instance *demod)
7619*4882a593Smuzhiyun {
7620*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
7621*4882a593Smuzhiyun 	int rc;
7622*4882a593Smuzhiyun 	static const u8 qam_dq_qual_fun[] = {
7623*4882a593Smuzhiyun 		DRXJ_16TO8(8),	/* fun0  */
7624*4882a593Smuzhiyun 		DRXJ_16TO8(8),	/* fun1  */
7625*4882a593Smuzhiyun 		DRXJ_16TO8(8),	/* fun2  */
7626*4882a593Smuzhiyun 		DRXJ_16TO8(8),	/* fun3  */
7627*4882a593Smuzhiyun 		DRXJ_16TO8(12),	/* fun4  */
7628*4882a593Smuzhiyun 		DRXJ_16TO8(12),	/* fun5  */
7629*4882a593Smuzhiyun 	};
7630*4882a593Smuzhiyun 	static const u8 qam_eq_cma_rad[] = {
7631*4882a593Smuzhiyun 		DRXJ_16TO8(12345),	/* RAD0  */
7632*4882a593Smuzhiyun 		DRXJ_16TO8(12345),	/* RAD1  */
7633*4882a593Smuzhiyun 		DRXJ_16TO8(13626),	/* RAD2  */
7634*4882a593Smuzhiyun 		DRXJ_16TO8(12931),	/* RAD3  */
7635*4882a593Smuzhiyun 		DRXJ_16TO8(14719),	/* RAD4  */
7636*4882a593Smuzhiyun 		DRXJ_16TO8(15356),	/* RAD5  */
7637*4882a593Smuzhiyun 	};
7638*4882a593Smuzhiyun 
7639*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
7640*4882a593Smuzhiyun 	if (rc != 0) {
7641*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7642*4882a593Smuzhiyun 		goto rw_error;
7643*4882a593Smuzhiyun 	}
7644*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
7645*4882a593Smuzhiyun 	if (rc != 0) {
7646*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7647*4882a593Smuzhiyun 		goto rw_error;
7648*4882a593Smuzhiyun 	}
7649*4882a593Smuzhiyun 
7650*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 50, 0);
7651*4882a593Smuzhiyun 	if (rc != 0) {
7652*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7653*4882a593Smuzhiyun 		goto rw_error;
7654*4882a593Smuzhiyun 	}
7655*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 60, 0);
7656*4882a593Smuzhiyun 	if (rc != 0) {
7657*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7658*4882a593Smuzhiyun 		goto rw_error;
7659*4882a593Smuzhiyun 	}
7660*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
7661*4882a593Smuzhiyun 	if (rc != 0) {
7662*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7663*4882a593Smuzhiyun 		goto rw_error;
7664*4882a593Smuzhiyun 	}
7665*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 150, 0);
7666*4882a593Smuzhiyun 	if (rc != 0) {
7667*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7668*4882a593Smuzhiyun 		goto rw_error;
7669*4882a593Smuzhiyun 	}
7670*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
7671*4882a593Smuzhiyun 	if (rc != 0) {
7672*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7673*4882a593Smuzhiyun 		goto rw_error;
7674*4882a593Smuzhiyun 	}
7675*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 110, 0);
7676*4882a593Smuzhiyun 	if (rc != 0) {
7677*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7678*4882a593Smuzhiyun 		goto rw_error;
7679*4882a593Smuzhiyun 	}
7680*4882a593Smuzhiyun 
7681*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
7682*4882a593Smuzhiyun 	if (rc != 0) {
7683*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7684*4882a593Smuzhiyun 		goto rw_error;
7685*4882a593Smuzhiyun 	}
7686*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 16, 0);
7687*4882a593Smuzhiyun 	if (rc != 0) {
7688*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7689*4882a593Smuzhiyun 		goto rw_error;
7690*4882a593Smuzhiyun 	}
7691*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
7692*4882a593Smuzhiyun 	if (rc != 0) {
7693*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7694*4882a593Smuzhiyun 		goto rw_error;
7695*4882a593Smuzhiyun 	}
7696*4882a593Smuzhiyun 
7697*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 8, 0);
7698*4882a593Smuzhiyun 	if (rc != 0) {
7699*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7700*4882a593Smuzhiyun 		goto rw_error;
7701*4882a593Smuzhiyun 	}
7702*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 74, 0);
7703*4882a593Smuzhiyun 	if (rc != 0) {
7704*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7705*4882a593Smuzhiyun 		goto rw_error;
7706*4882a593Smuzhiyun 	}
7707*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 18, 0);
7708*4882a593Smuzhiyun 	if (rc != 0) {
7709*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7710*4882a593Smuzhiyun 		goto rw_error;
7711*4882a593Smuzhiyun 	}
7712*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 13, 0);
7713*4882a593Smuzhiyun 	if (rc != 0) {
7714*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7715*4882a593Smuzhiyun 		goto rw_error;
7716*4882a593Smuzhiyun 	}
7717*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, 7, 0);
7718*4882a593Smuzhiyun 	if (rc != 0) {
7719*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7720*4882a593Smuzhiyun 		goto rw_error;
7721*4882a593Smuzhiyun 	}
7722*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, 0, 0);
7723*4882a593Smuzhiyun 	if (rc != 0) {
7724*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7725*4882a593Smuzhiyun 		goto rw_error;
7726*4882a593Smuzhiyun 	}
7727*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-8), 0);
7728*4882a593Smuzhiyun 	if (rc != 0) {
7729*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7730*4882a593Smuzhiyun 		goto rw_error;
7731*4882a593Smuzhiyun 	}
7732*4882a593Smuzhiyun 
7733*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
7734*4882a593Smuzhiyun 	if (rc != 0) {
7735*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7736*4882a593Smuzhiyun 		goto rw_error;
7737*4882a593Smuzhiyun 	}
7738*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
7739*4882a593Smuzhiyun 	if (rc != 0) {
7740*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7741*4882a593Smuzhiyun 		goto rw_error;
7742*4882a593Smuzhiyun 	}
7743*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
7744*4882a593Smuzhiyun 	if (rc != 0) {
7745*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7746*4882a593Smuzhiyun 		goto rw_error;
7747*4882a593Smuzhiyun 	}
7748*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 50, 0);
7749*4882a593Smuzhiyun 	if (rc != 0) {
7750*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7751*4882a593Smuzhiyun 		goto rw_error;
7752*4882a593Smuzhiyun 	}
7753*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
7754*4882a593Smuzhiyun 	if (rc != 0) {
7755*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7756*4882a593Smuzhiyun 		goto rw_error;
7757*4882a593Smuzhiyun 	}
7758*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
7759*4882a593Smuzhiyun 	if (rc != 0) {
7760*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7761*4882a593Smuzhiyun 		goto rw_error;
7762*4882a593Smuzhiyun 	}
7763*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 25, 0);
7764*4882a593Smuzhiyun 	if (rc != 0) {
7765*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7766*4882a593Smuzhiyun 		goto rw_error;
7767*4882a593Smuzhiyun 	}
7768*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 80, 0);
7769*4882a593Smuzhiyun 	if (rc != 0) {
7770*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7771*4882a593Smuzhiyun 		goto rw_error;
7772*4882a593Smuzhiyun 	}
7773*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
7774*4882a593Smuzhiyun 	if (rc != 0) {
7775*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7776*4882a593Smuzhiyun 		goto rw_error;
7777*4882a593Smuzhiyun 	}
7778*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
7779*4882a593Smuzhiyun 	if (rc != 0) {
7780*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7781*4882a593Smuzhiyun 		goto rw_error;
7782*4882a593Smuzhiyun 	}
7783*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
7784*4882a593Smuzhiyun 	if (rc != 0) {
7785*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7786*4882a593Smuzhiyun 		goto rw_error;
7787*4882a593Smuzhiyun 	}
7788*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
7789*4882a593Smuzhiyun 	if (rc != 0) {
7790*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7791*4882a593Smuzhiyun 		goto rw_error;
7792*4882a593Smuzhiyun 	}
7793*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
7794*4882a593Smuzhiyun 	if (rc != 0) {
7795*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7796*4882a593Smuzhiyun 		goto rw_error;
7797*4882a593Smuzhiyun 	}
7798*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
7799*4882a593Smuzhiyun 	if (rc != 0) {
7800*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7801*4882a593Smuzhiyun 		goto rw_error;
7802*4882a593Smuzhiyun 	}
7803*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
7804*4882a593Smuzhiyun 	if (rc != 0) {
7805*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7806*4882a593Smuzhiyun 		goto rw_error;
7807*4882a593Smuzhiyun 	}
7808*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 48, 0);
7809*4882a593Smuzhiyun 	if (rc != 0) {
7810*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7811*4882a593Smuzhiyun 		goto rw_error;
7812*4882a593Smuzhiyun 	}
7813*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 80, 0);
7814*4882a593Smuzhiyun 	if (rc != 0) {
7815*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7816*4882a593Smuzhiyun 		goto rw_error;
7817*4882a593Smuzhiyun 	}
7818*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
7819*4882a593Smuzhiyun 	if (rc != 0) {
7820*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7821*4882a593Smuzhiyun 		goto rw_error;
7822*4882a593Smuzhiyun 	}
7823*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
7824*4882a593Smuzhiyun 	if (rc != 0) {
7825*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7826*4882a593Smuzhiyun 		goto rw_error;
7827*4882a593Smuzhiyun 	}
7828*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 16, 0);
7829*4882a593Smuzhiyun 	if (rc != 0) {
7830*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7831*4882a593Smuzhiyun 		goto rw_error;
7832*4882a593Smuzhiyun 	}
7833*4882a593Smuzhiyun 
7834*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 43520, 0);
7835*4882a593Smuzhiyun 	if (rc != 0) {
7836*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
7837*4882a593Smuzhiyun 		goto rw_error;
7838*4882a593Smuzhiyun 	}
7839*4882a593Smuzhiyun 
7840*4882a593Smuzhiyun 	return 0;
7841*4882a593Smuzhiyun rw_error:
7842*4882a593Smuzhiyun 	return rc;
7843*4882a593Smuzhiyun }
7844*4882a593Smuzhiyun 
7845*4882a593Smuzhiyun /*============================================================================*/
7846*4882a593Smuzhiyun #define QAM_SET_OP_ALL 0x1
7847*4882a593Smuzhiyun #define QAM_SET_OP_CONSTELLATION 0x2
7848*4882a593Smuzhiyun #define QAM_SET_OP_SPECTRUM 0X4
7849*4882a593Smuzhiyun 
7850*4882a593Smuzhiyun /*
7851*4882a593Smuzhiyun * \fn int set_qam ()
7852*4882a593Smuzhiyun * \brief Set QAM demod.
7853*4882a593Smuzhiyun * \param demod:   instance of demod.
7854*4882a593Smuzhiyun * \param channel: pointer to channel data.
7855*4882a593Smuzhiyun * \return int.
7856*4882a593Smuzhiyun */
7857*4882a593Smuzhiyun static int
set_qam(struct drx_demod_instance * demod,struct drx_channel * channel,s32 tuner_freq_offset,u32 op)7858*4882a593Smuzhiyun set_qam(struct drx_demod_instance *demod,
7859*4882a593Smuzhiyun 	struct drx_channel *channel, s32 tuner_freq_offset, u32 op)
7860*4882a593Smuzhiyun {
7861*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
7862*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
7863*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = NULL;
7864*4882a593Smuzhiyun 	int rc;
7865*4882a593Smuzhiyun 	u32 adc_frequency = 0;
7866*4882a593Smuzhiyun 	u32 iqm_rc_rate = 0;
7867*4882a593Smuzhiyun 	u16 cmd_result = 0;
7868*4882a593Smuzhiyun 	u16 lc_symbol_freq = 0;
7869*4882a593Smuzhiyun 	u16 iqm_rc_stretch = 0;
7870*4882a593Smuzhiyun 	u16 set_env_parameters = 0;
7871*4882a593Smuzhiyun 	u16 set_param_parameters[2] = { 0 };
7872*4882a593Smuzhiyun 	struct drxjscu_cmd cmd_scu = { /* command      */ 0,
7873*4882a593Smuzhiyun 		/* parameter_len */ 0,
7874*4882a593Smuzhiyun 		/* result_len    */ 0,
7875*4882a593Smuzhiyun 		/* parameter    */ NULL,
7876*4882a593Smuzhiyun 		/* result       */ NULL
7877*4882a593Smuzhiyun 	};
7878*4882a593Smuzhiyun 	static const u8 qam_a_taps[] = {
7879*4882a593Smuzhiyun 		DRXJ_16TO8(-1),	/* re0  */
7880*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re1  */
7881*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re2  */
7882*4882a593Smuzhiyun 		DRXJ_16TO8(-1),	/* re3  */
7883*4882a593Smuzhiyun 		DRXJ_16TO8(-1),	/* re4  */
7884*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* re5  */
7885*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re6  */
7886*4882a593Smuzhiyun 		DRXJ_16TO8(-2),	/* re7  */
7887*4882a593Smuzhiyun 		DRXJ_16TO8(0),	/* re8  */
7888*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* re9  */
7889*4882a593Smuzhiyun 		DRXJ_16TO8(-1),	/* re10 */
7890*4882a593Smuzhiyun 		DRXJ_16TO8(-3),	/* re11 */
7891*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re12 */
7892*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re13 */
7893*4882a593Smuzhiyun 		DRXJ_16TO8(-8),	/* re14 */
7894*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re15 */
7895*4882a593Smuzhiyun 		DRXJ_16TO8(13),	/* re16 */
7896*4882a593Smuzhiyun 		DRXJ_16TO8(-13),	/* re17 */
7897*4882a593Smuzhiyun 		DRXJ_16TO8(-19),	/* re18 */
7898*4882a593Smuzhiyun 		DRXJ_16TO8(28),	/* re19 */
7899*4882a593Smuzhiyun 		DRXJ_16TO8(25),	/* re20 */
7900*4882a593Smuzhiyun 		DRXJ_16TO8(-53),	/* re21 */
7901*4882a593Smuzhiyun 		DRXJ_16TO8(-31),	/* re22 */
7902*4882a593Smuzhiyun 		DRXJ_16TO8(96),	/* re23 */
7903*4882a593Smuzhiyun 		DRXJ_16TO8(37),	/* re24 */
7904*4882a593Smuzhiyun 		DRXJ_16TO8(-190),	/* re25 */
7905*4882a593Smuzhiyun 		DRXJ_16TO8(-40),	/* re26 */
7906*4882a593Smuzhiyun 		DRXJ_16TO8(619)	/* re27 */
7907*4882a593Smuzhiyun 	};
7908*4882a593Smuzhiyun 	static const u8 qam_b64_taps[] = {
7909*4882a593Smuzhiyun 		DRXJ_16TO8(0),	/* re0  */
7910*4882a593Smuzhiyun 		DRXJ_16TO8(-2),	/* re1  */
7911*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re2  */
7912*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* re3  */
7913*4882a593Smuzhiyun 		DRXJ_16TO8(-2),	/* re4  */
7914*4882a593Smuzhiyun 		DRXJ_16TO8(0),	/* re5  */
7915*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re6  */
7916*4882a593Smuzhiyun 		DRXJ_16TO8(-2),	/* re7  */
7917*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re8  */
7918*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re9  */
7919*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* re10 */
7920*4882a593Smuzhiyun 		DRXJ_16TO8(-6),	/* re11 */
7921*4882a593Smuzhiyun 		DRXJ_16TO8(0),	/* re12 */
7922*4882a593Smuzhiyun 		DRXJ_16TO8(6),	/* re13 */
7923*4882a593Smuzhiyun 		DRXJ_16TO8(-5),	/* re14 */
7924*4882a593Smuzhiyun 		DRXJ_16TO8(-3),	/* re15 */
7925*4882a593Smuzhiyun 		DRXJ_16TO8(11),	/* re16 */
7926*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re17 */
7927*4882a593Smuzhiyun 		DRXJ_16TO8(-19),	/* re18 */
7928*4882a593Smuzhiyun 		DRXJ_16TO8(19),	/* re19 */
7929*4882a593Smuzhiyun 		DRXJ_16TO8(28),	/* re20 */
7930*4882a593Smuzhiyun 		DRXJ_16TO8(-45),	/* re21 */
7931*4882a593Smuzhiyun 		DRXJ_16TO8(-36),	/* re22 */
7932*4882a593Smuzhiyun 		DRXJ_16TO8(90),	/* re23 */
7933*4882a593Smuzhiyun 		DRXJ_16TO8(42),	/* re24 */
7934*4882a593Smuzhiyun 		DRXJ_16TO8(-185),	/* re25 */
7935*4882a593Smuzhiyun 		DRXJ_16TO8(-46),	/* re26 */
7936*4882a593Smuzhiyun 		DRXJ_16TO8(614)	/* re27 */
7937*4882a593Smuzhiyun 	};
7938*4882a593Smuzhiyun 	static const u8 qam_b256_taps[] = {
7939*4882a593Smuzhiyun 		DRXJ_16TO8(-2),	/* re0  */
7940*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re1  */
7941*4882a593Smuzhiyun 		DRXJ_16TO8(1),	/* re2  */
7942*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re3  */
7943*4882a593Smuzhiyun 		DRXJ_16TO8(0),	/* re4  */
7944*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re5  */
7945*4882a593Smuzhiyun 		DRXJ_16TO8(-2),	/* re6  */
7946*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re7  */
7947*4882a593Smuzhiyun 		DRXJ_16TO8(5),	/* re8  */
7948*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* re9  */
7949*4882a593Smuzhiyun 		DRXJ_16TO8(-8),	/* re10 */
7950*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* re11 */
7951*4882a593Smuzhiyun 		DRXJ_16TO8(11),	/* re12 */
7952*4882a593Smuzhiyun 		DRXJ_16TO8(-8),	/* re13 */
7953*4882a593Smuzhiyun 		DRXJ_16TO8(-15),	/* re14 */
7954*4882a593Smuzhiyun 		DRXJ_16TO8(16),	/* re15 */
7955*4882a593Smuzhiyun 		DRXJ_16TO8(19),	/* re16 */
7956*4882a593Smuzhiyun 		DRXJ_16TO8(-27),	/* re17 */
7957*4882a593Smuzhiyun 		DRXJ_16TO8(-22),	/* re18 */
7958*4882a593Smuzhiyun 		DRXJ_16TO8(44),	/* re19 */
7959*4882a593Smuzhiyun 		DRXJ_16TO8(26),	/* re20 */
7960*4882a593Smuzhiyun 		DRXJ_16TO8(-69),	/* re21 */
7961*4882a593Smuzhiyun 		DRXJ_16TO8(-28),	/* re22 */
7962*4882a593Smuzhiyun 		DRXJ_16TO8(110),	/* re23 */
7963*4882a593Smuzhiyun 		DRXJ_16TO8(31),	/* re24 */
7964*4882a593Smuzhiyun 		DRXJ_16TO8(-201),	/* re25 */
7965*4882a593Smuzhiyun 		DRXJ_16TO8(-32),	/* re26 */
7966*4882a593Smuzhiyun 		DRXJ_16TO8(628)	/* re27 */
7967*4882a593Smuzhiyun 	};
7968*4882a593Smuzhiyun 	static const u8 qam_c_taps[] = {
7969*4882a593Smuzhiyun 		DRXJ_16TO8(-3),	/* re0  */
7970*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* re1  */
7971*4882a593Smuzhiyun 		DRXJ_16TO8(2),	/* re2  */
7972*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re3  */
7973*4882a593Smuzhiyun 		DRXJ_16TO8(0),	/* re4  */
7974*4882a593Smuzhiyun 		DRXJ_16TO8(4),	/* re5  */
7975*4882a593Smuzhiyun 		DRXJ_16TO8(-1),	/* re6  */
7976*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re7  */
7977*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* re8  */
7978*4882a593Smuzhiyun 		DRXJ_16TO8(3),	/* re9  */
7979*4882a593Smuzhiyun 		DRXJ_16TO8(-5),	/* re10 */
7980*4882a593Smuzhiyun 		DRXJ_16TO8(0),	/* re11 */
7981*4882a593Smuzhiyun 		DRXJ_16TO8(9),	/* re12 */
7982*4882a593Smuzhiyun 		DRXJ_16TO8(-4),	/* re13 */
7983*4882a593Smuzhiyun 		DRXJ_16TO8(-12),	/* re14 */
7984*4882a593Smuzhiyun 		DRXJ_16TO8(10),	/* re15 */
7985*4882a593Smuzhiyun 		DRXJ_16TO8(16),	/* re16 */
7986*4882a593Smuzhiyun 		DRXJ_16TO8(-21),	/* re17 */
7987*4882a593Smuzhiyun 		DRXJ_16TO8(-20),	/* re18 */
7988*4882a593Smuzhiyun 		DRXJ_16TO8(37),	/* re19 */
7989*4882a593Smuzhiyun 		DRXJ_16TO8(25),	/* re20 */
7990*4882a593Smuzhiyun 		DRXJ_16TO8(-62),	/* re21 */
7991*4882a593Smuzhiyun 		DRXJ_16TO8(-28),	/* re22 */
7992*4882a593Smuzhiyun 		DRXJ_16TO8(105),	/* re23 */
7993*4882a593Smuzhiyun 		DRXJ_16TO8(31),	/* re24 */
7994*4882a593Smuzhiyun 		DRXJ_16TO8(-197),	/* re25 */
7995*4882a593Smuzhiyun 		DRXJ_16TO8(-33),	/* re26 */
7996*4882a593Smuzhiyun 		DRXJ_16TO8(626)	/* re27 */
7997*4882a593Smuzhiyun 	};
7998*4882a593Smuzhiyun 
7999*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
8000*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
8001*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
8002*4882a593Smuzhiyun 
8003*4882a593Smuzhiyun 	if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
8004*4882a593Smuzhiyun 		if (ext_attr->standard == DRX_STANDARD_ITU_B) {
8005*4882a593Smuzhiyun 			switch (channel->constellation) {
8006*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM256:
8007*4882a593Smuzhiyun 				iqm_rc_rate = 0x00AE3562;
8008*4882a593Smuzhiyun 				lc_symbol_freq =
8009*4882a593Smuzhiyun 				    QAM_LC_SYMBOL_FREQ_FREQ_QAM_B_256;
8010*4882a593Smuzhiyun 				channel->symbolrate = 5360537;
8011*4882a593Smuzhiyun 				iqm_rc_stretch = IQM_RC_STRETCH_QAM_B_256;
8012*4882a593Smuzhiyun 				break;
8013*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM64:
8014*4882a593Smuzhiyun 				iqm_rc_rate = 0x00C05A0E;
8015*4882a593Smuzhiyun 				lc_symbol_freq = 409;
8016*4882a593Smuzhiyun 				channel->symbolrate = 5056941;
8017*4882a593Smuzhiyun 				iqm_rc_stretch = IQM_RC_STRETCH_QAM_B_64;
8018*4882a593Smuzhiyun 				break;
8019*4882a593Smuzhiyun 			default:
8020*4882a593Smuzhiyun 				return -EINVAL;
8021*4882a593Smuzhiyun 			}
8022*4882a593Smuzhiyun 		} else {
8023*4882a593Smuzhiyun 			adc_frequency = (common_attr->sys_clock_freq * 1000) / 3;
8024*4882a593Smuzhiyun 			if (channel->symbolrate == 0) {
8025*4882a593Smuzhiyun 				pr_err("error: channel symbolrate is zero!\n");
8026*4882a593Smuzhiyun 				return -EIO;
8027*4882a593Smuzhiyun 			}
8028*4882a593Smuzhiyun 			iqm_rc_rate =
8029*4882a593Smuzhiyun 			    (adc_frequency / channel->symbolrate) * (1 << 21) +
8030*4882a593Smuzhiyun 			    (frac28
8031*4882a593Smuzhiyun 			     ((adc_frequency % channel->symbolrate),
8032*4882a593Smuzhiyun 			      channel->symbolrate) >> 7) - (1 << 23);
8033*4882a593Smuzhiyun 			lc_symbol_freq =
8034*4882a593Smuzhiyun 			    (u16) (frac28
8035*4882a593Smuzhiyun 				     (channel->symbolrate +
8036*4882a593Smuzhiyun 				      (adc_frequency >> 13),
8037*4882a593Smuzhiyun 				      adc_frequency) >> 16);
8038*4882a593Smuzhiyun 			if (lc_symbol_freq > 511)
8039*4882a593Smuzhiyun 				lc_symbol_freq = 511;
8040*4882a593Smuzhiyun 
8041*4882a593Smuzhiyun 			iqm_rc_stretch = 21;
8042*4882a593Smuzhiyun 		}
8043*4882a593Smuzhiyun 
8044*4882a593Smuzhiyun 		if (ext_attr->standard == DRX_STANDARD_ITU_A) {
8045*4882a593Smuzhiyun 			set_env_parameters = QAM_TOP_ANNEX_A;	/* annex             */
8046*4882a593Smuzhiyun 			set_param_parameters[0] = channel->constellation;	/* constellation     */
8047*4882a593Smuzhiyun 			set_param_parameters[1] = DRX_INTERLEAVEMODE_I12_J17;	/* interleave mode   */
8048*4882a593Smuzhiyun 		} else if (ext_attr->standard == DRX_STANDARD_ITU_B) {
8049*4882a593Smuzhiyun 			set_env_parameters = QAM_TOP_ANNEX_B;	/* annex             */
8050*4882a593Smuzhiyun 			set_param_parameters[0] = channel->constellation;	/* constellation     */
8051*4882a593Smuzhiyun 			set_param_parameters[1] = channel->interleavemode;	/* interleave mode   */
8052*4882a593Smuzhiyun 		} else if (ext_attr->standard == DRX_STANDARD_ITU_C) {
8053*4882a593Smuzhiyun 			set_env_parameters = QAM_TOP_ANNEX_C;	/* annex             */
8054*4882a593Smuzhiyun 			set_param_parameters[0] = channel->constellation;	/* constellation     */
8055*4882a593Smuzhiyun 			set_param_parameters[1] = DRX_INTERLEAVEMODE_I12_J17;	/* interleave mode   */
8056*4882a593Smuzhiyun 		} else {
8057*4882a593Smuzhiyun 			return -EINVAL;
8058*4882a593Smuzhiyun 		}
8059*4882a593Smuzhiyun 	}
8060*4882a593Smuzhiyun 
8061*4882a593Smuzhiyun 	if (op & QAM_SET_OP_ALL) {
8062*4882a593Smuzhiyun 		/*
8063*4882a593Smuzhiyun 		   STEP 1: reset demodulator
8064*4882a593Smuzhiyun 		   resets IQM, QAM and FEC HW blocks
8065*4882a593Smuzhiyun 		   resets SCU variables
8066*4882a593Smuzhiyun 		 */
8067*4882a593Smuzhiyun 		/* stop all comm_exec */
8068*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
8069*4882a593Smuzhiyun 		if (rc != 0) {
8070*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8071*4882a593Smuzhiyun 			goto rw_error;
8072*4882a593Smuzhiyun 		}
8073*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_COMM_EXEC__A, QAM_COMM_EXEC_STOP, 0);
8074*4882a593Smuzhiyun 		if (rc != 0) {
8075*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8076*4882a593Smuzhiyun 			goto rw_error;
8077*4882a593Smuzhiyun 		}
8078*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
8079*4882a593Smuzhiyun 		if (rc != 0) {
8080*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8081*4882a593Smuzhiyun 			goto rw_error;
8082*4882a593Smuzhiyun 		}
8083*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
8084*4882a593Smuzhiyun 		if (rc != 0) {
8085*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8086*4882a593Smuzhiyun 			goto rw_error;
8087*4882a593Smuzhiyun 		}
8088*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
8089*4882a593Smuzhiyun 		if (rc != 0) {
8090*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8091*4882a593Smuzhiyun 			goto rw_error;
8092*4882a593Smuzhiyun 		}
8093*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
8094*4882a593Smuzhiyun 		if (rc != 0) {
8095*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8096*4882a593Smuzhiyun 			goto rw_error;
8097*4882a593Smuzhiyun 		}
8098*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
8099*4882a593Smuzhiyun 		if (rc != 0) {
8100*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8101*4882a593Smuzhiyun 			goto rw_error;
8102*4882a593Smuzhiyun 		}
8103*4882a593Smuzhiyun 
8104*4882a593Smuzhiyun 		cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
8105*4882a593Smuzhiyun 		    SCU_RAM_COMMAND_CMD_DEMOD_RESET;
8106*4882a593Smuzhiyun 		cmd_scu.parameter_len = 0;
8107*4882a593Smuzhiyun 		cmd_scu.result_len = 1;
8108*4882a593Smuzhiyun 		cmd_scu.parameter = NULL;
8109*4882a593Smuzhiyun 		cmd_scu.result = &cmd_result;
8110*4882a593Smuzhiyun 		rc = scu_command(dev_addr, &cmd_scu);
8111*4882a593Smuzhiyun 		if (rc != 0) {
8112*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8113*4882a593Smuzhiyun 			goto rw_error;
8114*4882a593Smuzhiyun 		}
8115*4882a593Smuzhiyun 	}
8116*4882a593Smuzhiyun 
8117*4882a593Smuzhiyun 	if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
8118*4882a593Smuzhiyun 		/*
8119*4882a593Smuzhiyun 		   STEP 2: configure demodulator
8120*4882a593Smuzhiyun 		   -set env
8121*4882a593Smuzhiyun 		   -set params (resets IQM,QAM,FEC HW; initializes some SCU variables )
8122*4882a593Smuzhiyun 		 */
8123*4882a593Smuzhiyun 		cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
8124*4882a593Smuzhiyun 		    SCU_RAM_COMMAND_CMD_DEMOD_SET_ENV;
8125*4882a593Smuzhiyun 		cmd_scu.parameter_len = 1;
8126*4882a593Smuzhiyun 		cmd_scu.result_len = 1;
8127*4882a593Smuzhiyun 		cmd_scu.parameter = &set_env_parameters;
8128*4882a593Smuzhiyun 		cmd_scu.result = &cmd_result;
8129*4882a593Smuzhiyun 		rc = scu_command(dev_addr, &cmd_scu);
8130*4882a593Smuzhiyun 		if (rc != 0) {
8131*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8132*4882a593Smuzhiyun 			goto rw_error;
8133*4882a593Smuzhiyun 		}
8134*4882a593Smuzhiyun 
8135*4882a593Smuzhiyun 		cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
8136*4882a593Smuzhiyun 		    SCU_RAM_COMMAND_CMD_DEMOD_SET_PARAM;
8137*4882a593Smuzhiyun 		cmd_scu.parameter_len = 2;
8138*4882a593Smuzhiyun 		cmd_scu.result_len = 1;
8139*4882a593Smuzhiyun 		cmd_scu.parameter = set_param_parameters;
8140*4882a593Smuzhiyun 		cmd_scu.result = &cmd_result;
8141*4882a593Smuzhiyun 		rc = scu_command(dev_addr, &cmd_scu);
8142*4882a593Smuzhiyun 		if (rc != 0) {
8143*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8144*4882a593Smuzhiyun 			goto rw_error;
8145*4882a593Smuzhiyun 		}
8146*4882a593Smuzhiyun 		/* set symbol rate */
8147*4882a593Smuzhiyun 		rc = drxdap_fasi_write_reg32(dev_addr, IQM_RC_RATE_OFS_LO__A, iqm_rc_rate, 0);
8148*4882a593Smuzhiyun 		if (rc != 0) {
8149*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8150*4882a593Smuzhiyun 			goto rw_error;
8151*4882a593Smuzhiyun 		}
8152*4882a593Smuzhiyun 		ext_attr->iqm_rc_rate_ofs = iqm_rc_rate;
8153*4882a593Smuzhiyun 		rc = set_qam_measurement(demod, channel->constellation, channel->symbolrate);
8154*4882a593Smuzhiyun 		if (rc != 0) {
8155*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8156*4882a593Smuzhiyun 			goto rw_error;
8157*4882a593Smuzhiyun 		}
8158*4882a593Smuzhiyun 	}
8159*4882a593Smuzhiyun 	/* STEP 3: enable the system in a mode where the ADC provides valid signal
8160*4882a593Smuzhiyun 	   setup constellation independent registers */
8161*4882a593Smuzhiyun 	/* from qam_cmd.py script (qam_driver_b) */
8162*4882a593Smuzhiyun 	/* TODO: remove re-writes of HW reset values */
8163*4882a593Smuzhiyun 	if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_SPECTRUM)) {
8164*4882a593Smuzhiyun 		rc = set_frequency(demod, channel, tuner_freq_offset);
8165*4882a593Smuzhiyun 		if (rc != 0) {
8166*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8167*4882a593Smuzhiyun 			goto rw_error;
8168*4882a593Smuzhiyun 		}
8169*4882a593Smuzhiyun 	}
8170*4882a593Smuzhiyun 
8171*4882a593Smuzhiyun 	if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
8172*4882a593Smuzhiyun 
8173*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_SYMBOL_FREQ__A, lc_symbol_freq, 0);
8174*4882a593Smuzhiyun 		if (rc != 0) {
8175*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8176*4882a593Smuzhiyun 			goto rw_error;
8177*4882a593Smuzhiyun 		}
8178*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RC_STRETCH__A, iqm_rc_stretch, 0);
8179*4882a593Smuzhiyun 		if (rc != 0) {
8180*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8181*4882a593Smuzhiyun 			goto rw_error;
8182*4882a593Smuzhiyun 		}
8183*4882a593Smuzhiyun 	}
8184*4882a593Smuzhiyun 
8185*4882a593Smuzhiyun 	if (op & QAM_SET_OP_ALL) {
8186*4882a593Smuzhiyun 		if (!ext_attr->has_lna) {
8187*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AMUX__A, 0x02, 0);
8188*4882a593Smuzhiyun 			if (rc != 0) {
8189*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8190*4882a593Smuzhiyun 				goto rw_error;
8191*4882a593Smuzhiyun 			}
8192*4882a593Smuzhiyun 		}
8193*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SYMMETRIC__A, 0, 0);
8194*4882a593Smuzhiyun 		if (rc != 0) {
8195*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8196*4882a593Smuzhiyun 			goto rw_error;
8197*4882a593Smuzhiyun 		}
8198*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_MIDTAP__A, 3, 0);
8199*4882a593Smuzhiyun 		if (rc != 0) {
8200*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8201*4882a593Smuzhiyun 			goto rw_error;
8202*4882a593Smuzhiyun 		}
8203*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_OUT_ENA__A, IQM_CF_OUT_ENA_QAM__M, 0);
8204*4882a593Smuzhiyun 		if (rc != 0) {
8205*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8206*4882a593Smuzhiyun 			goto rw_error;
8207*4882a593Smuzhiyun 		}
8208*4882a593Smuzhiyun 
8209*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_WR_RSV_0__A, 0x5f, 0);
8210*4882a593Smuzhiyun 		if (rc != 0) {
8211*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8212*4882a593Smuzhiyun 			goto rw_error;
8213*4882a593Smuzhiyun 		}	/* scu temporary shut down agc */
8214*4882a593Smuzhiyun 
8215*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_SYNC_SEL__A, 3, 0);
8216*4882a593Smuzhiyun 		if (rc != 0) {
8217*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8218*4882a593Smuzhiyun 			goto rw_error;
8219*4882a593Smuzhiyun 		}
8220*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_LEN__A, 0, 0);
8221*4882a593Smuzhiyun 		if (rc != 0) {
8222*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8223*4882a593Smuzhiyun 			goto rw_error;
8224*4882a593Smuzhiyun 		}
8225*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_TH__A, 448, 0);
8226*4882a593Smuzhiyun 		if (rc != 0) {
8227*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8228*4882a593Smuzhiyun 			goto rw_error;
8229*4882a593Smuzhiyun 		}
8230*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_SNS_LEN__A, 0, 0);
8231*4882a593Smuzhiyun 		if (rc != 0) {
8232*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8233*4882a593Smuzhiyun 			goto rw_error;
8234*4882a593Smuzhiyun 		}
8235*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PDREF__A, 4, 0);
8236*4882a593Smuzhiyun 		if (rc != 0) {
8237*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8238*4882a593Smuzhiyun 			goto rw_error;
8239*4882a593Smuzhiyun 		}
8240*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, 0x10, 0);
8241*4882a593Smuzhiyun 		if (rc != 0) {
8242*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8243*4882a593Smuzhiyun 			goto rw_error;
8244*4882a593Smuzhiyun 		}
8245*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PGA_GAIN__A, 11, 0);
8246*4882a593Smuzhiyun 		if (rc != 0) {
8247*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8248*4882a593Smuzhiyun 			goto rw_error;
8249*4882a593Smuzhiyun 		}
8250*4882a593Smuzhiyun 
8251*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_POW_MEAS_LEN__A, 1, 0);
8252*4882a593Smuzhiyun 		if (rc != 0) {
8253*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8254*4882a593Smuzhiyun 			goto rw_error;
8255*4882a593Smuzhiyun 		}
8256*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE_SH__A, IQM_CF_SCALE_SH__PRE, 0);
8257*4882a593Smuzhiyun 		if (rc != 0) {
8258*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8259*4882a593Smuzhiyun 			goto rw_error;
8260*4882a593Smuzhiyun 		}	/*! reset default val ! */
8261*4882a593Smuzhiyun 
8262*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_SY_TIMEOUT__A, QAM_SY_TIMEOUT__PRE, 0);
8263*4882a593Smuzhiyun 		if (rc != 0) {
8264*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8265*4882a593Smuzhiyun 			goto rw_error;
8266*4882a593Smuzhiyun 		}	/*! reset default val ! */
8267*4882a593Smuzhiyun 		if (ext_attr->standard == DRX_STANDARD_ITU_B) {
8268*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_LWM__A, QAM_SY_SYNC_LWM__PRE, 0);
8269*4882a593Smuzhiyun 			if (rc != 0) {
8270*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8271*4882a593Smuzhiyun 				goto rw_error;
8272*4882a593Smuzhiyun 			}	/*! reset default val ! */
8273*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_AWM__A, QAM_SY_SYNC_AWM__PRE, 0);
8274*4882a593Smuzhiyun 			if (rc != 0) {
8275*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8276*4882a593Smuzhiyun 				goto rw_error;
8277*4882a593Smuzhiyun 			}	/*! reset default val ! */
8278*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_HWM__A, QAM_SY_SYNC_HWM__PRE, 0);
8279*4882a593Smuzhiyun 			if (rc != 0) {
8280*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8281*4882a593Smuzhiyun 				goto rw_error;
8282*4882a593Smuzhiyun 			}	/*! reset default val ! */
8283*4882a593Smuzhiyun 		} else {
8284*4882a593Smuzhiyun 			switch (channel->constellation) {
8285*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM16:
8286*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM64:
8287*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM256:
8288*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_LWM__A, 0x03, 0);
8289*4882a593Smuzhiyun 				if (rc != 0) {
8290*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8291*4882a593Smuzhiyun 					goto rw_error;
8292*4882a593Smuzhiyun 				}
8293*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_AWM__A, 0x04, 0);
8294*4882a593Smuzhiyun 				if (rc != 0) {
8295*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8296*4882a593Smuzhiyun 					goto rw_error;
8297*4882a593Smuzhiyun 				}
8298*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_HWM__A, QAM_SY_SYNC_HWM__PRE, 0);
8299*4882a593Smuzhiyun 				if (rc != 0) {
8300*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8301*4882a593Smuzhiyun 					goto rw_error;
8302*4882a593Smuzhiyun 				}	/*! reset default val ! */
8303*4882a593Smuzhiyun 				break;
8304*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM32:
8305*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM128:
8306*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_LWM__A, 0x03, 0);
8307*4882a593Smuzhiyun 				if (rc != 0) {
8308*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8309*4882a593Smuzhiyun 					goto rw_error;
8310*4882a593Smuzhiyun 				}
8311*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_AWM__A, 0x05, 0);
8312*4882a593Smuzhiyun 				if (rc != 0) {
8313*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8314*4882a593Smuzhiyun 					goto rw_error;
8315*4882a593Smuzhiyun 				}
8316*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_HWM__A, 0x06, 0);
8317*4882a593Smuzhiyun 				if (rc != 0) {
8318*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8319*4882a593Smuzhiyun 					goto rw_error;
8320*4882a593Smuzhiyun 				}
8321*4882a593Smuzhiyun 				break;
8322*4882a593Smuzhiyun 			default:
8323*4882a593Smuzhiyun 				return -EIO;
8324*4882a593Smuzhiyun 			}	/* switch */
8325*4882a593Smuzhiyun 		}
8326*4882a593Smuzhiyun 
8327*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_MODE__A, QAM_LC_MODE__PRE, 0);
8328*4882a593Smuzhiyun 		if (rc != 0) {
8329*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8330*4882a593Smuzhiyun 			goto rw_error;
8331*4882a593Smuzhiyun 		}	/*! reset default val ! */
8332*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_RATE_LIMIT__A, 3, 0);
8333*4882a593Smuzhiyun 		if (rc != 0) {
8334*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8335*4882a593Smuzhiyun 			goto rw_error;
8336*4882a593Smuzhiyun 		}
8337*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_LPF_FACTORP__A, 4, 0);
8338*4882a593Smuzhiyun 		if (rc != 0) {
8339*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8340*4882a593Smuzhiyun 			goto rw_error;
8341*4882a593Smuzhiyun 		}
8342*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_LPF_FACTORI__A, 4, 0);
8343*4882a593Smuzhiyun 		if (rc != 0) {
8344*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8345*4882a593Smuzhiyun 			goto rw_error;
8346*4882a593Smuzhiyun 		}
8347*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_MODE__A, 7, 0);
8348*4882a593Smuzhiyun 		if (rc != 0) {
8349*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8350*4882a593Smuzhiyun 			goto rw_error;
8351*4882a593Smuzhiyun 		}
8352*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB0__A, 1, 0);
8353*4882a593Smuzhiyun 		if (rc != 0) {
8354*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8355*4882a593Smuzhiyun 			goto rw_error;
8356*4882a593Smuzhiyun 		}
8357*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB1__A, 1, 0);
8358*4882a593Smuzhiyun 		if (rc != 0) {
8359*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8360*4882a593Smuzhiyun 			goto rw_error;
8361*4882a593Smuzhiyun 		}
8362*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB2__A, 1, 0);
8363*4882a593Smuzhiyun 		if (rc != 0) {
8364*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8365*4882a593Smuzhiyun 			goto rw_error;
8366*4882a593Smuzhiyun 		}
8367*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB3__A, 1, 0);
8368*4882a593Smuzhiyun 		if (rc != 0) {
8369*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8370*4882a593Smuzhiyun 			goto rw_error;
8371*4882a593Smuzhiyun 		}
8372*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB4__A, 2, 0);
8373*4882a593Smuzhiyun 		if (rc != 0) {
8374*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8375*4882a593Smuzhiyun 			goto rw_error;
8376*4882a593Smuzhiyun 		}
8377*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB5__A, 2, 0);
8378*4882a593Smuzhiyun 		if (rc != 0) {
8379*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8380*4882a593Smuzhiyun 			goto rw_error;
8381*4882a593Smuzhiyun 		}
8382*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB6__A, 2, 0);
8383*4882a593Smuzhiyun 		if (rc != 0) {
8384*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8385*4882a593Smuzhiyun 			goto rw_error;
8386*4882a593Smuzhiyun 		}
8387*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB8__A, 2, 0);
8388*4882a593Smuzhiyun 		if (rc != 0) {
8389*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8390*4882a593Smuzhiyun 			goto rw_error;
8391*4882a593Smuzhiyun 		}
8392*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB9__A, 2, 0);
8393*4882a593Smuzhiyun 		if (rc != 0) {
8394*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8395*4882a593Smuzhiyun 			goto rw_error;
8396*4882a593Smuzhiyun 		}
8397*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB10__A, 2, 0);
8398*4882a593Smuzhiyun 		if (rc != 0) {
8399*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8400*4882a593Smuzhiyun 			goto rw_error;
8401*4882a593Smuzhiyun 		}
8402*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB12__A, 2, 0);
8403*4882a593Smuzhiyun 		if (rc != 0) {
8404*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8405*4882a593Smuzhiyun 			goto rw_error;
8406*4882a593Smuzhiyun 		}
8407*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB15__A, 3, 0);
8408*4882a593Smuzhiyun 		if (rc != 0) {
8409*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8410*4882a593Smuzhiyun 			goto rw_error;
8411*4882a593Smuzhiyun 		}
8412*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB16__A, 3, 0);
8413*4882a593Smuzhiyun 		if (rc != 0) {
8414*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8415*4882a593Smuzhiyun 			goto rw_error;
8416*4882a593Smuzhiyun 		}
8417*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB20__A, 4, 0);
8418*4882a593Smuzhiyun 		if (rc != 0) {
8419*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8420*4882a593Smuzhiyun 			goto rw_error;
8421*4882a593Smuzhiyun 		}
8422*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB25__A, 4, 0);
8423*4882a593Smuzhiyun 		if (rc != 0) {
8424*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8425*4882a593Smuzhiyun 			goto rw_error;
8426*4882a593Smuzhiyun 		}
8427*4882a593Smuzhiyun 
8428*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FS_ADJ_SEL__A, 1, 0);
8429*4882a593Smuzhiyun 		if (rc != 0) {
8430*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8431*4882a593Smuzhiyun 			goto rw_error;
8432*4882a593Smuzhiyun 		}
8433*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RC_ADJ_SEL__A, 1, 0);
8434*4882a593Smuzhiyun 		if (rc != 0) {
8435*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8436*4882a593Smuzhiyun 			goto rw_error;
8437*4882a593Smuzhiyun 		}
8438*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_ADJ_SEL__A, 1, 0);
8439*4882a593Smuzhiyun 		if (rc != 0) {
8440*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8441*4882a593Smuzhiyun 			goto rw_error;
8442*4882a593Smuzhiyun 		}
8443*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_POW_MEAS_LEN__A, 0, 0);
8444*4882a593Smuzhiyun 		if (rc != 0) {
8445*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8446*4882a593Smuzhiyun 			goto rw_error;
8447*4882a593Smuzhiyun 		}
8448*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_GPIO__A, 0, 0);
8449*4882a593Smuzhiyun 		if (rc != 0) {
8450*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8451*4882a593Smuzhiyun 			goto rw_error;
8452*4882a593Smuzhiyun 		}
8453*4882a593Smuzhiyun 
8454*4882a593Smuzhiyun 		/* No more resets of the IQM, current standard correctly set =>
8455*4882a593Smuzhiyun 		   now AGCs can be configured. */
8456*4882a593Smuzhiyun 		/* turn on IQMAF. It has to be in front of setAgc**() */
8457*4882a593Smuzhiyun 		rc = set_iqm_af(demod, true);
8458*4882a593Smuzhiyun 		if (rc != 0) {
8459*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8460*4882a593Smuzhiyun 			goto rw_error;
8461*4882a593Smuzhiyun 		}
8462*4882a593Smuzhiyun 		rc = adc_synchronization(demod);
8463*4882a593Smuzhiyun 		if (rc != 0) {
8464*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8465*4882a593Smuzhiyun 			goto rw_error;
8466*4882a593Smuzhiyun 		}
8467*4882a593Smuzhiyun 
8468*4882a593Smuzhiyun 		rc = init_agc(demod);
8469*4882a593Smuzhiyun 		if (rc != 0) {
8470*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8471*4882a593Smuzhiyun 			goto rw_error;
8472*4882a593Smuzhiyun 		}
8473*4882a593Smuzhiyun 		rc = set_agc_if(demod, &(ext_attr->qam_if_agc_cfg), false);
8474*4882a593Smuzhiyun 		if (rc != 0) {
8475*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8476*4882a593Smuzhiyun 			goto rw_error;
8477*4882a593Smuzhiyun 		}
8478*4882a593Smuzhiyun 		rc = set_agc_rf(demod, &(ext_attr->qam_rf_agc_cfg), false);
8479*4882a593Smuzhiyun 		if (rc != 0) {
8480*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8481*4882a593Smuzhiyun 			goto rw_error;
8482*4882a593Smuzhiyun 		}
8483*4882a593Smuzhiyun 		{
8484*4882a593Smuzhiyun 			/* TODO fix this, store a struct drxj_cfg_afe_gain structure in struct drxj_data instead
8485*4882a593Smuzhiyun 			   of only the gain */
8486*4882a593Smuzhiyun 			struct drxj_cfg_afe_gain qam_pga_cfg = { DRX_STANDARD_ITU_B, 0 };
8487*4882a593Smuzhiyun 
8488*4882a593Smuzhiyun 			qam_pga_cfg.gain = ext_attr->qam_pga_cfg;
8489*4882a593Smuzhiyun 			rc = ctrl_set_cfg_afe_gain(demod, &qam_pga_cfg);
8490*4882a593Smuzhiyun 			if (rc != 0) {
8491*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8492*4882a593Smuzhiyun 				goto rw_error;
8493*4882a593Smuzhiyun 			}
8494*4882a593Smuzhiyun 		}
8495*4882a593Smuzhiyun 		rc = ctrl_set_cfg_pre_saw(demod, &(ext_attr->qam_pre_saw_cfg));
8496*4882a593Smuzhiyun 		if (rc != 0) {
8497*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8498*4882a593Smuzhiyun 			goto rw_error;
8499*4882a593Smuzhiyun 		}
8500*4882a593Smuzhiyun 	}
8501*4882a593Smuzhiyun 
8502*4882a593Smuzhiyun 	if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
8503*4882a593Smuzhiyun 		if (ext_attr->standard == DRX_STANDARD_ITU_A) {
8504*4882a593Smuzhiyun 			rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_a_taps), ((u8 *)qam_a_taps), 0);
8505*4882a593Smuzhiyun 			if (rc != 0) {
8506*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8507*4882a593Smuzhiyun 				goto rw_error;
8508*4882a593Smuzhiyun 			}
8509*4882a593Smuzhiyun 			rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_a_taps), ((u8 *)qam_a_taps), 0);
8510*4882a593Smuzhiyun 			if (rc != 0) {
8511*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8512*4882a593Smuzhiyun 				goto rw_error;
8513*4882a593Smuzhiyun 			}
8514*4882a593Smuzhiyun 		} else if (ext_attr->standard == DRX_STANDARD_ITU_B) {
8515*4882a593Smuzhiyun 			switch (channel->constellation) {
8516*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM64:
8517*4882a593Smuzhiyun 				rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_b64_taps), ((u8 *)qam_b64_taps), 0);
8518*4882a593Smuzhiyun 				if (rc != 0) {
8519*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8520*4882a593Smuzhiyun 					goto rw_error;
8521*4882a593Smuzhiyun 				}
8522*4882a593Smuzhiyun 				rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_b64_taps), ((u8 *)qam_b64_taps), 0);
8523*4882a593Smuzhiyun 				if (rc != 0) {
8524*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8525*4882a593Smuzhiyun 					goto rw_error;
8526*4882a593Smuzhiyun 				}
8527*4882a593Smuzhiyun 				break;
8528*4882a593Smuzhiyun 			case DRX_CONSTELLATION_QAM256:
8529*4882a593Smuzhiyun 				rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_b256_taps), ((u8 *)qam_b256_taps), 0);
8530*4882a593Smuzhiyun 				if (rc != 0) {
8531*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8532*4882a593Smuzhiyun 					goto rw_error;
8533*4882a593Smuzhiyun 				}
8534*4882a593Smuzhiyun 				rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_b256_taps), ((u8 *)qam_b256_taps), 0);
8535*4882a593Smuzhiyun 				if (rc != 0) {
8536*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8537*4882a593Smuzhiyun 					goto rw_error;
8538*4882a593Smuzhiyun 				}
8539*4882a593Smuzhiyun 				break;
8540*4882a593Smuzhiyun 			default:
8541*4882a593Smuzhiyun 				return -EIO;
8542*4882a593Smuzhiyun 			}
8543*4882a593Smuzhiyun 		} else if (ext_attr->standard == DRX_STANDARD_ITU_C) {
8544*4882a593Smuzhiyun 			rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_c_taps), ((u8 *)qam_c_taps), 0);
8545*4882a593Smuzhiyun 			if (rc != 0) {
8546*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8547*4882a593Smuzhiyun 				goto rw_error;
8548*4882a593Smuzhiyun 			}
8549*4882a593Smuzhiyun 			rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_c_taps), ((u8 *)qam_c_taps), 0);
8550*4882a593Smuzhiyun 			if (rc != 0) {
8551*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8552*4882a593Smuzhiyun 				goto rw_error;
8553*4882a593Smuzhiyun 			}
8554*4882a593Smuzhiyun 		}
8555*4882a593Smuzhiyun 
8556*4882a593Smuzhiyun 		/* SETP 4: constellation specific setup */
8557*4882a593Smuzhiyun 		switch (channel->constellation) {
8558*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM16:
8559*4882a593Smuzhiyun 			rc = set_qam16(demod);
8560*4882a593Smuzhiyun 			if (rc != 0) {
8561*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8562*4882a593Smuzhiyun 				goto rw_error;
8563*4882a593Smuzhiyun 			}
8564*4882a593Smuzhiyun 			break;
8565*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM32:
8566*4882a593Smuzhiyun 			rc = set_qam32(demod);
8567*4882a593Smuzhiyun 			if (rc != 0) {
8568*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8569*4882a593Smuzhiyun 				goto rw_error;
8570*4882a593Smuzhiyun 			}
8571*4882a593Smuzhiyun 			break;
8572*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM64:
8573*4882a593Smuzhiyun 			rc = set_qam64(demod);
8574*4882a593Smuzhiyun 			if (rc != 0) {
8575*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8576*4882a593Smuzhiyun 				goto rw_error;
8577*4882a593Smuzhiyun 			}
8578*4882a593Smuzhiyun 			break;
8579*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM128:
8580*4882a593Smuzhiyun 			rc = set_qam128(demod);
8581*4882a593Smuzhiyun 			if (rc != 0) {
8582*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8583*4882a593Smuzhiyun 				goto rw_error;
8584*4882a593Smuzhiyun 			}
8585*4882a593Smuzhiyun 			break;
8586*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM256:
8587*4882a593Smuzhiyun 			rc = set_qam256(demod);
8588*4882a593Smuzhiyun 			if (rc != 0) {
8589*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8590*4882a593Smuzhiyun 				goto rw_error;
8591*4882a593Smuzhiyun 			}
8592*4882a593Smuzhiyun 			break;
8593*4882a593Smuzhiyun 		default:
8594*4882a593Smuzhiyun 			return -EIO;
8595*4882a593Smuzhiyun 		}		/* switch */
8596*4882a593Smuzhiyun 	}
8597*4882a593Smuzhiyun 
8598*4882a593Smuzhiyun 	if ((op & QAM_SET_OP_ALL)) {
8599*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE_SH__A, 0, 0);
8600*4882a593Smuzhiyun 		if (rc != 0) {
8601*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8602*4882a593Smuzhiyun 			goto rw_error;
8603*4882a593Smuzhiyun 		}
8604*4882a593Smuzhiyun 
8605*4882a593Smuzhiyun 		/* Mpeg output has to be in front of FEC active */
8606*4882a593Smuzhiyun 		rc = set_mpegtei_handling(demod);
8607*4882a593Smuzhiyun 		if (rc != 0) {
8608*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8609*4882a593Smuzhiyun 			goto rw_error;
8610*4882a593Smuzhiyun 		}
8611*4882a593Smuzhiyun 		rc = bit_reverse_mpeg_output(demod);
8612*4882a593Smuzhiyun 		if (rc != 0) {
8613*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8614*4882a593Smuzhiyun 			goto rw_error;
8615*4882a593Smuzhiyun 		}
8616*4882a593Smuzhiyun 		rc = set_mpeg_start_width(demod);
8617*4882a593Smuzhiyun 		if (rc != 0) {
8618*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8619*4882a593Smuzhiyun 			goto rw_error;
8620*4882a593Smuzhiyun 		}
8621*4882a593Smuzhiyun 		{
8622*4882a593Smuzhiyun 			/* TODO: move to set_standard after hardware reset value problem is solved */
8623*4882a593Smuzhiyun 			/* Configure initial MPEG output */
8624*4882a593Smuzhiyun 			struct drx_cfg_mpeg_output cfg_mpeg_output;
8625*4882a593Smuzhiyun 
8626*4882a593Smuzhiyun 			memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
8627*4882a593Smuzhiyun 			cfg_mpeg_output.enable_mpeg_output = true;
8628*4882a593Smuzhiyun 
8629*4882a593Smuzhiyun 			rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
8630*4882a593Smuzhiyun 			if (rc != 0) {
8631*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
8632*4882a593Smuzhiyun 				goto rw_error;
8633*4882a593Smuzhiyun 			}
8634*4882a593Smuzhiyun 		}
8635*4882a593Smuzhiyun 	}
8636*4882a593Smuzhiyun 
8637*4882a593Smuzhiyun 	if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
8638*4882a593Smuzhiyun 
8639*4882a593Smuzhiyun 		/* STEP 5: start QAM demodulator (starts FEC, QAM and IQM HW) */
8640*4882a593Smuzhiyun 		cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
8641*4882a593Smuzhiyun 		    SCU_RAM_COMMAND_CMD_DEMOD_START;
8642*4882a593Smuzhiyun 		cmd_scu.parameter_len = 0;
8643*4882a593Smuzhiyun 		cmd_scu.result_len = 1;
8644*4882a593Smuzhiyun 		cmd_scu.parameter = NULL;
8645*4882a593Smuzhiyun 		cmd_scu.result = &cmd_result;
8646*4882a593Smuzhiyun 		rc = scu_command(dev_addr, &cmd_scu);
8647*4882a593Smuzhiyun 		if (rc != 0) {
8648*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8649*4882a593Smuzhiyun 			goto rw_error;
8650*4882a593Smuzhiyun 		}
8651*4882a593Smuzhiyun 	}
8652*4882a593Smuzhiyun 
8653*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_ACTIVE, 0);
8654*4882a593Smuzhiyun 	if (rc != 0) {
8655*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8656*4882a593Smuzhiyun 		goto rw_error;
8657*4882a593Smuzhiyun 	}
8658*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_COMM_EXEC__A, QAM_COMM_EXEC_ACTIVE, 0);
8659*4882a593Smuzhiyun 	if (rc != 0) {
8660*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8661*4882a593Smuzhiyun 		goto rw_error;
8662*4882a593Smuzhiyun 	}
8663*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_ACTIVE, 0);
8664*4882a593Smuzhiyun 	if (rc != 0) {
8665*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8666*4882a593Smuzhiyun 		goto rw_error;
8667*4882a593Smuzhiyun 	}
8668*4882a593Smuzhiyun 
8669*4882a593Smuzhiyun 	return 0;
8670*4882a593Smuzhiyun rw_error:
8671*4882a593Smuzhiyun 	return rc;
8672*4882a593Smuzhiyun }
8673*4882a593Smuzhiyun 
8674*4882a593Smuzhiyun /*============================================================================*/
8675*4882a593Smuzhiyun static int ctrl_get_qam_sig_quality(struct drx_demod_instance *demod);
8676*4882a593Smuzhiyun 
qam_flip_spec(struct drx_demod_instance * demod,struct drx_channel * channel)8677*4882a593Smuzhiyun static int qam_flip_spec(struct drx_demod_instance *demod, struct drx_channel *channel)
8678*4882a593Smuzhiyun {
8679*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
8680*4882a593Smuzhiyun 	struct drxj_data *ext_attr = demod->my_ext_attr;
8681*4882a593Smuzhiyun 	int rc;
8682*4882a593Smuzhiyun 	u32 iqm_fs_rate_ofs = 0;
8683*4882a593Smuzhiyun 	u32 iqm_fs_rate_lo = 0;
8684*4882a593Smuzhiyun 	u16 qam_ctl_ena = 0;
8685*4882a593Smuzhiyun 	u16 data = 0;
8686*4882a593Smuzhiyun 	u16 equ_mode = 0;
8687*4882a593Smuzhiyun 	u16 fsm_state = 0;
8688*4882a593Smuzhiyun 	int i = 0;
8689*4882a593Smuzhiyun 	int ofsofs = 0;
8690*4882a593Smuzhiyun 
8691*4882a593Smuzhiyun 	/* Silence the controlling of lc, equ, and the acquisition state machine */
8692*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_QAM_CTL_ENA__A, &qam_ctl_ena, 0);
8693*4882a593Smuzhiyun 	if (rc != 0) {
8694*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8695*4882a593Smuzhiyun 		goto rw_error;
8696*4882a593Smuzhiyun 	}
8697*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_CTL_ENA__A, qam_ctl_ena & ~(SCU_RAM_QAM_CTL_ENA_ACQ__M | SCU_RAM_QAM_CTL_ENA_EQU__M | SCU_RAM_QAM_CTL_ENA_LC__M), 0);
8698*4882a593Smuzhiyun 	if (rc != 0) {
8699*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8700*4882a593Smuzhiyun 		goto rw_error;
8701*4882a593Smuzhiyun 	}
8702*4882a593Smuzhiyun 
8703*4882a593Smuzhiyun 	/* freeze the frequency control loop */
8704*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_LC_CF__A, 0, 0);
8705*4882a593Smuzhiyun 	if (rc != 0) {
8706*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8707*4882a593Smuzhiyun 		goto rw_error;
8708*4882a593Smuzhiyun 	}
8709*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_LC_CF1__A, 0, 0);
8710*4882a593Smuzhiyun 	if (rc != 0) {
8711*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8712*4882a593Smuzhiyun 		goto rw_error;
8713*4882a593Smuzhiyun 	}
8714*4882a593Smuzhiyun 
8715*4882a593Smuzhiyun 	rc = drxj_dap_atomic_read_reg32(dev_addr, IQM_FS_RATE_OFS_LO__A, &iqm_fs_rate_ofs, 0);
8716*4882a593Smuzhiyun 	if (rc != 0) {
8717*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8718*4882a593Smuzhiyun 		goto rw_error;
8719*4882a593Smuzhiyun 	}
8720*4882a593Smuzhiyun 	rc = drxj_dap_atomic_read_reg32(dev_addr, IQM_FS_RATE_LO__A, &iqm_fs_rate_lo, 0);
8721*4882a593Smuzhiyun 	if (rc != 0) {
8722*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8723*4882a593Smuzhiyun 		goto rw_error;
8724*4882a593Smuzhiyun 	}
8725*4882a593Smuzhiyun 	ofsofs = iqm_fs_rate_lo - iqm_fs_rate_ofs;
8726*4882a593Smuzhiyun 	iqm_fs_rate_ofs = ~iqm_fs_rate_ofs + 1;
8727*4882a593Smuzhiyun 	iqm_fs_rate_ofs -= 2 * ofsofs;
8728*4882a593Smuzhiyun 
8729*4882a593Smuzhiyun 	/* freeze dq/fq updating */
8730*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, QAM_DQ_MODE__A, &data, 0);
8731*4882a593Smuzhiyun 	if (rc != 0) {
8732*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8733*4882a593Smuzhiyun 		goto rw_error;
8734*4882a593Smuzhiyun 	}
8735*4882a593Smuzhiyun 	data = (data & 0xfff9);
8736*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_MODE__A, data, 0);
8737*4882a593Smuzhiyun 	if (rc != 0) {
8738*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8739*4882a593Smuzhiyun 		goto rw_error;
8740*4882a593Smuzhiyun 	}
8741*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_MODE__A, data, 0);
8742*4882a593Smuzhiyun 	if (rc != 0) {
8743*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8744*4882a593Smuzhiyun 		goto rw_error;
8745*4882a593Smuzhiyun 	}
8746*4882a593Smuzhiyun 
8747*4882a593Smuzhiyun 	/* lc_cp / _ci / _ca */
8748*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_LC_CI__A, 0, 0);
8749*4882a593Smuzhiyun 	if (rc != 0) {
8750*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8751*4882a593Smuzhiyun 		goto rw_error;
8752*4882a593Smuzhiyun 	}
8753*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_LC_EP__A, 0, 0);
8754*4882a593Smuzhiyun 	if (rc != 0) {
8755*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8756*4882a593Smuzhiyun 		goto rw_error;
8757*4882a593Smuzhiyun 	}
8758*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_LA_FACTOR__A, 0, 0);
8759*4882a593Smuzhiyun 	if (rc != 0) {
8760*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8761*4882a593Smuzhiyun 		goto rw_error;
8762*4882a593Smuzhiyun 	}
8763*4882a593Smuzhiyun 
8764*4882a593Smuzhiyun 	/* flip the spec */
8765*4882a593Smuzhiyun 	rc = drxdap_fasi_write_reg32(dev_addr, IQM_FS_RATE_OFS_LO__A, iqm_fs_rate_ofs, 0);
8766*4882a593Smuzhiyun 	if (rc != 0) {
8767*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8768*4882a593Smuzhiyun 		goto rw_error;
8769*4882a593Smuzhiyun 	}
8770*4882a593Smuzhiyun 	ext_attr->iqm_fs_rate_ofs = iqm_fs_rate_ofs;
8771*4882a593Smuzhiyun 	ext_attr->pos_image = (ext_attr->pos_image) ? false : true;
8772*4882a593Smuzhiyun 
8773*4882a593Smuzhiyun 	/* freeze dq/fq updating */
8774*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, QAM_DQ_MODE__A, &data, 0);
8775*4882a593Smuzhiyun 	if (rc != 0) {
8776*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8777*4882a593Smuzhiyun 		goto rw_error;
8778*4882a593Smuzhiyun 	}
8779*4882a593Smuzhiyun 	equ_mode = data;
8780*4882a593Smuzhiyun 	data = (data & 0xfff9);
8781*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_MODE__A, data, 0);
8782*4882a593Smuzhiyun 	if (rc != 0) {
8783*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8784*4882a593Smuzhiyun 		goto rw_error;
8785*4882a593Smuzhiyun 	}
8786*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_MODE__A, data, 0);
8787*4882a593Smuzhiyun 	if (rc != 0) {
8788*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8789*4882a593Smuzhiyun 		goto rw_error;
8790*4882a593Smuzhiyun 	}
8791*4882a593Smuzhiyun 
8792*4882a593Smuzhiyun 	for (i = 0; i < 28; i++) {
8793*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, QAM_DQ_TAP_IM_EL0__A + (2 * i), &data, 0);
8794*4882a593Smuzhiyun 		if (rc != 0) {
8795*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8796*4882a593Smuzhiyun 			goto rw_error;
8797*4882a593Smuzhiyun 		}
8798*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_TAP_IM_EL0__A + (2 * i), -data, 0);
8799*4882a593Smuzhiyun 		if (rc != 0) {
8800*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8801*4882a593Smuzhiyun 			goto rw_error;
8802*4882a593Smuzhiyun 		}
8803*4882a593Smuzhiyun 	}
8804*4882a593Smuzhiyun 
8805*4882a593Smuzhiyun 	for (i = 0; i < 24; i++) {
8806*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, QAM_FQ_TAP_IM_EL0__A + (2 * i), &data, 0);
8807*4882a593Smuzhiyun 		if (rc != 0) {
8808*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8809*4882a593Smuzhiyun 			goto rw_error;
8810*4882a593Smuzhiyun 		}
8811*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_TAP_IM_EL0__A + (2 * i), -data, 0);
8812*4882a593Smuzhiyun 		if (rc != 0) {
8813*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8814*4882a593Smuzhiyun 			goto rw_error;
8815*4882a593Smuzhiyun 		}
8816*4882a593Smuzhiyun 	}
8817*4882a593Smuzhiyun 
8818*4882a593Smuzhiyun 	data = equ_mode;
8819*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_MODE__A, data, 0);
8820*4882a593Smuzhiyun 	if (rc != 0) {
8821*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8822*4882a593Smuzhiyun 		goto rw_error;
8823*4882a593Smuzhiyun 	}
8824*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_MODE__A, data, 0);
8825*4882a593Smuzhiyun 	if (rc != 0) {
8826*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8827*4882a593Smuzhiyun 		goto rw_error;
8828*4882a593Smuzhiyun 	}
8829*4882a593Smuzhiyun 
8830*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_STATE_TGT__A, 4, 0);
8831*4882a593Smuzhiyun 	if (rc != 0) {
8832*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8833*4882a593Smuzhiyun 		goto rw_error;
8834*4882a593Smuzhiyun 	}
8835*4882a593Smuzhiyun 
8836*4882a593Smuzhiyun 	i = 0;
8837*4882a593Smuzhiyun 	while ((fsm_state != 4) && (i++ < 100)) {
8838*4882a593Smuzhiyun 		rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_QAM_FSM_STATE__A, &fsm_state, 0);
8839*4882a593Smuzhiyun 		if (rc != 0) {
8840*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8841*4882a593Smuzhiyun 			goto rw_error;
8842*4882a593Smuzhiyun 		}
8843*4882a593Smuzhiyun 	}
8844*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_CTL_ENA__A, (qam_ctl_ena | 0x0016), 0);
8845*4882a593Smuzhiyun 	if (rc != 0) {
8846*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
8847*4882a593Smuzhiyun 		goto rw_error;
8848*4882a593Smuzhiyun 	}
8849*4882a593Smuzhiyun 
8850*4882a593Smuzhiyun 	return 0;
8851*4882a593Smuzhiyun rw_error:
8852*4882a593Smuzhiyun 	return rc;
8853*4882a593Smuzhiyun 
8854*4882a593Smuzhiyun }
8855*4882a593Smuzhiyun 
8856*4882a593Smuzhiyun #define  NO_LOCK        0x0
8857*4882a593Smuzhiyun #define  DEMOD_LOCKED   0x1
8858*4882a593Smuzhiyun #define  SYNC_FLIPPED   0x2
8859*4882a593Smuzhiyun #define  SPEC_MIRRORED  0x4
8860*4882a593Smuzhiyun /*
8861*4882a593Smuzhiyun * \fn int qam64auto ()
8862*4882a593Smuzhiyun * \brief auto do sync pattern switching and mirroring.
8863*4882a593Smuzhiyun * \param demod:   instance of demod.
8864*4882a593Smuzhiyun * \param channel: pointer to channel data.
8865*4882a593Smuzhiyun * \param tuner_freq_offset: tuner frequency offset.
8866*4882a593Smuzhiyun * \param lock_status: pointer to lock status.
8867*4882a593Smuzhiyun * \return int.
8868*4882a593Smuzhiyun */
8869*4882a593Smuzhiyun static int
qam64auto(struct drx_demod_instance * demod,struct drx_channel * channel,s32 tuner_freq_offset,enum drx_lock_status * lock_status)8870*4882a593Smuzhiyun qam64auto(struct drx_demod_instance *demod,
8871*4882a593Smuzhiyun 	  struct drx_channel *channel,
8872*4882a593Smuzhiyun 	  s32 tuner_freq_offset, enum drx_lock_status *lock_status)
8873*4882a593Smuzhiyun {
8874*4882a593Smuzhiyun 	struct drxj_data *ext_attr = demod->my_ext_attr;
8875*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
8876*4882a593Smuzhiyun 	struct drx39xxj_state *state = dev_addr->user_data;
8877*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
8878*4882a593Smuzhiyun 	int rc;
8879*4882a593Smuzhiyun 	u32 lck_state = NO_LOCK;
8880*4882a593Smuzhiyun 	u32 start_time = 0;
8881*4882a593Smuzhiyun 	u32 d_locked_time = 0;
8882*4882a593Smuzhiyun 	u32 timeout_ofs = 0;
8883*4882a593Smuzhiyun 	u16 data = 0;
8884*4882a593Smuzhiyun 
8885*4882a593Smuzhiyun 	/* external attributes for storing acquired channel constellation */
8886*4882a593Smuzhiyun 	*lock_status = DRX_NOT_LOCKED;
8887*4882a593Smuzhiyun 	start_time = jiffies_to_msecs(jiffies);
8888*4882a593Smuzhiyun 	lck_state = NO_LOCK;
8889*4882a593Smuzhiyun 	do {
8890*4882a593Smuzhiyun 		rc = ctrl_lock_status(demod, lock_status);
8891*4882a593Smuzhiyun 		if (rc != 0) {
8892*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
8893*4882a593Smuzhiyun 			goto rw_error;
8894*4882a593Smuzhiyun 		}
8895*4882a593Smuzhiyun 
8896*4882a593Smuzhiyun 		switch (lck_state) {
8897*4882a593Smuzhiyun 		case NO_LOCK:
8898*4882a593Smuzhiyun 			if (*lock_status == DRXJ_DEMOD_LOCK) {
8899*4882a593Smuzhiyun 				rc = ctrl_get_qam_sig_quality(demod);
8900*4882a593Smuzhiyun 				if (rc != 0) {
8901*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8902*4882a593Smuzhiyun 					goto rw_error;
8903*4882a593Smuzhiyun 				}
8904*4882a593Smuzhiyun 				if (p->cnr.stat[0].svalue > 20800) {
8905*4882a593Smuzhiyun 					lck_state = DEMOD_LOCKED;
8906*4882a593Smuzhiyun 					/* some delay to see if fec_lock possible TODO find the right value */
8907*4882a593Smuzhiyun 					timeout_ofs += DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME;	/* see something, waiting longer */
8908*4882a593Smuzhiyun 					d_locked_time = jiffies_to_msecs(jiffies);
8909*4882a593Smuzhiyun 				}
8910*4882a593Smuzhiyun 			}
8911*4882a593Smuzhiyun 			break;
8912*4882a593Smuzhiyun 		case DEMOD_LOCKED:
8913*4882a593Smuzhiyun 			if ((*lock_status == DRXJ_DEMOD_LOCK) &&	/* still demod_lock in 150ms */
8914*4882a593Smuzhiyun 			    ((jiffies_to_msecs(jiffies) - d_locked_time) >
8915*4882a593Smuzhiyun 			     DRXJ_QAM_FEC_LOCK_WAITTIME)) {
8916*4882a593Smuzhiyun 				rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, &data, 0);
8917*4882a593Smuzhiyun 				if (rc != 0) {
8918*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8919*4882a593Smuzhiyun 					goto rw_error;
8920*4882a593Smuzhiyun 				}
8921*4882a593Smuzhiyun 				rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, data | 0x1, 0);
8922*4882a593Smuzhiyun 				if (rc != 0) {
8923*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8924*4882a593Smuzhiyun 					goto rw_error;
8925*4882a593Smuzhiyun 				}
8926*4882a593Smuzhiyun 				lck_state = SYNC_FLIPPED;
8927*4882a593Smuzhiyun 				msleep(10);
8928*4882a593Smuzhiyun 			}
8929*4882a593Smuzhiyun 			break;
8930*4882a593Smuzhiyun 		case SYNC_FLIPPED:
8931*4882a593Smuzhiyun 			if (*lock_status == DRXJ_DEMOD_LOCK) {
8932*4882a593Smuzhiyun 				if (channel->mirror == DRX_MIRROR_AUTO) {
8933*4882a593Smuzhiyun 					/* flip sync pattern back */
8934*4882a593Smuzhiyun 					rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, &data, 0);
8935*4882a593Smuzhiyun 					if (rc != 0) {
8936*4882a593Smuzhiyun 						pr_err("error %d\n", rc);
8937*4882a593Smuzhiyun 						goto rw_error;
8938*4882a593Smuzhiyun 					}
8939*4882a593Smuzhiyun 					rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, data & 0xFFFE, 0);
8940*4882a593Smuzhiyun 					if (rc != 0) {
8941*4882a593Smuzhiyun 						pr_err("error %d\n", rc);
8942*4882a593Smuzhiyun 						goto rw_error;
8943*4882a593Smuzhiyun 					}
8944*4882a593Smuzhiyun 					/* flip spectrum */
8945*4882a593Smuzhiyun 					ext_attr->mirror = DRX_MIRROR_YES;
8946*4882a593Smuzhiyun 					rc = qam_flip_spec(demod, channel);
8947*4882a593Smuzhiyun 					if (rc != 0) {
8948*4882a593Smuzhiyun 						pr_err("error %d\n", rc);
8949*4882a593Smuzhiyun 						goto rw_error;
8950*4882a593Smuzhiyun 					}
8951*4882a593Smuzhiyun 					lck_state = SPEC_MIRRORED;
8952*4882a593Smuzhiyun 					/* reset timer TODO: still need 500ms? */
8953*4882a593Smuzhiyun 					start_time = d_locked_time =
8954*4882a593Smuzhiyun 					    jiffies_to_msecs(jiffies);
8955*4882a593Smuzhiyun 					timeout_ofs = 0;
8956*4882a593Smuzhiyun 				} else {	/* no need to wait lock */
8957*4882a593Smuzhiyun 
8958*4882a593Smuzhiyun 					start_time =
8959*4882a593Smuzhiyun 					    jiffies_to_msecs(jiffies) -
8960*4882a593Smuzhiyun 					    DRXJ_QAM_MAX_WAITTIME - timeout_ofs;
8961*4882a593Smuzhiyun 				}
8962*4882a593Smuzhiyun 			}
8963*4882a593Smuzhiyun 			break;
8964*4882a593Smuzhiyun 		case SPEC_MIRRORED:
8965*4882a593Smuzhiyun 			if ((*lock_status == DRXJ_DEMOD_LOCK) &&	/* still demod_lock in 150ms */
8966*4882a593Smuzhiyun 			    ((jiffies_to_msecs(jiffies) - d_locked_time) >
8967*4882a593Smuzhiyun 			     DRXJ_QAM_FEC_LOCK_WAITTIME)) {
8968*4882a593Smuzhiyun 				rc = ctrl_get_qam_sig_quality(demod);
8969*4882a593Smuzhiyun 				if (rc != 0) {
8970*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
8971*4882a593Smuzhiyun 					goto rw_error;
8972*4882a593Smuzhiyun 				}
8973*4882a593Smuzhiyun 				if (p->cnr.stat[0].svalue > 20800) {
8974*4882a593Smuzhiyun 					rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, &data, 0);
8975*4882a593Smuzhiyun 					if (rc != 0) {
8976*4882a593Smuzhiyun 						pr_err("error %d\n", rc);
8977*4882a593Smuzhiyun 						goto rw_error;
8978*4882a593Smuzhiyun 					}
8979*4882a593Smuzhiyun 					rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, data | 0x1, 0);
8980*4882a593Smuzhiyun 					if (rc != 0) {
8981*4882a593Smuzhiyun 						pr_err("error %d\n", rc);
8982*4882a593Smuzhiyun 						goto rw_error;
8983*4882a593Smuzhiyun 					}
8984*4882a593Smuzhiyun 					/* no need to wait lock */
8985*4882a593Smuzhiyun 					start_time =
8986*4882a593Smuzhiyun 					    jiffies_to_msecs(jiffies) -
8987*4882a593Smuzhiyun 					    DRXJ_QAM_MAX_WAITTIME - timeout_ofs;
8988*4882a593Smuzhiyun 				}
8989*4882a593Smuzhiyun 			}
8990*4882a593Smuzhiyun 			break;
8991*4882a593Smuzhiyun 		default:
8992*4882a593Smuzhiyun 			break;
8993*4882a593Smuzhiyun 		}
8994*4882a593Smuzhiyun 		msleep(10);
8995*4882a593Smuzhiyun 	} while
8996*4882a593Smuzhiyun 	    ((*lock_status != DRX_LOCKED) &&
8997*4882a593Smuzhiyun 	     (*lock_status != DRX_NEVER_LOCK) &&
8998*4882a593Smuzhiyun 	     ((jiffies_to_msecs(jiffies) - start_time) <
8999*4882a593Smuzhiyun 	      (DRXJ_QAM_MAX_WAITTIME + timeout_ofs))
9000*4882a593Smuzhiyun 	    );
9001*4882a593Smuzhiyun 	/* Returning control to application ... */
9002*4882a593Smuzhiyun 
9003*4882a593Smuzhiyun 	return 0;
9004*4882a593Smuzhiyun rw_error:
9005*4882a593Smuzhiyun 	return rc;
9006*4882a593Smuzhiyun }
9007*4882a593Smuzhiyun 
9008*4882a593Smuzhiyun /*
9009*4882a593Smuzhiyun * \fn int qam256auto ()
9010*4882a593Smuzhiyun * \brief auto do sync pattern switching and mirroring.
9011*4882a593Smuzhiyun * \param demod:   instance of demod.
9012*4882a593Smuzhiyun * \param channel: pointer to channel data.
9013*4882a593Smuzhiyun * \param tuner_freq_offset: tuner frequency offset.
9014*4882a593Smuzhiyun * \param lock_status: pointer to lock status.
9015*4882a593Smuzhiyun * \return int.
9016*4882a593Smuzhiyun */
9017*4882a593Smuzhiyun static int
qam256auto(struct drx_demod_instance * demod,struct drx_channel * channel,s32 tuner_freq_offset,enum drx_lock_status * lock_status)9018*4882a593Smuzhiyun qam256auto(struct drx_demod_instance *demod,
9019*4882a593Smuzhiyun 	   struct drx_channel *channel,
9020*4882a593Smuzhiyun 	   s32 tuner_freq_offset, enum drx_lock_status *lock_status)
9021*4882a593Smuzhiyun {
9022*4882a593Smuzhiyun 	struct drxj_data *ext_attr = demod->my_ext_attr;
9023*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
9024*4882a593Smuzhiyun 	struct drx39xxj_state *state = dev_addr->user_data;
9025*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
9026*4882a593Smuzhiyun 	int rc;
9027*4882a593Smuzhiyun 	u32 lck_state = NO_LOCK;
9028*4882a593Smuzhiyun 	u32 start_time = 0;
9029*4882a593Smuzhiyun 	u32 d_locked_time = 0;
9030*4882a593Smuzhiyun 	u32 timeout_ofs = DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME;
9031*4882a593Smuzhiyun 
9032*4882a593Smuzhiyun 	/* external attributes for storing acquired channel constellation */
9033*4882a593Smuzhiyun 	*lock_status = DRX_NOT_LOCKED;
9034*4882a593Smuzhiyun 	start_time = jiffies_to_msecs(jiffies);
9035*4882a593Smuzhiyun 	lck_state = NO_LOCK;
9036*4882a593Smuzhiyun 	do {
9037*4882a593Smuzhiyun 		rc = ctrl_lock_status(demod, lock_status);
9038*4882a593Smuzhiyun 		if (rc != 0) {
9039*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9040*4882a593Smuzhiyun 			goto rw_error;
9041*4882a593Smuzhiyun 		}
9042*4882a593Smuzhiyun 		switch (lck_state) {
9043*4882a593Smuzhiyun 		case NO_LOCK:
9044*4882a593Smuzhiyun 			if (*lock_status == DRXJ_DEMOD_LOCK) {
9045*4882a593Smuzhiyun 				rc = ctrl_get_qam_sig_quality(demod);
9046*4882a593Smuzhiyun 				if (rc != 0) {
9047*4882a593Smuzhiyun 					pr_err("error %d\n", rc);
9048*4882a593Smuzhiyun 					goto rw_error;
9049*4882a593Smuzhiyun 				}
9050*4882a593Smuzhiyun 				if (p->cnr.stat[0].svalue > 26800) {
9051*4882a593Smuzhiyun 					lck_state = DEMOD_LOCKED;
9052*4882a593Smuzhiyun 					timeout_ofs += DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME;	/* see something, wait longer */
9053*4882a593Smuzhiyun 					d_locked_time = jiffies_to_msecs(jiffies);
9054*4882a593Smuzhiyun 				}
9055*4882a593Smuzhiyun 			}
9056*4882a593Smuzhiyun 			break;
9057*4882a593Smuzhiyun 		case DEMOD_LOCKED:
9058*4882a593Smuzhiyun 			if (*lock_status == DRXJ_DEMOD_LOCK) {
9059*4882a593Smuzhiyun 				if ((channel->mirror == DRX_MIRROR_AUTO) &&
9060*4882a593Smuzhiyun 				    ((jiffies_to_msecs(jiffies) - d_locked_time) >
9061*4882a593Smuzhiyun 				     DRXJ_QAM_FEC_LOCK_WAITTIME)) {
9062*4882a593Smuzhiyun 					ext_attr->mirror = DRX_MIRROR_YES;
9063*4882a593Smuzhiyun 					rc = qam_flip_spec(demod, channel);
9064*4882a593Smuzhiyun 					if (rc != 0) {
9065*4882a593Smuzhiyun 						pr_err("error %d\n", rc);
9066*4882a593Smuzhiyun 						goto rw_error;
9067*4882a593Smuzhiyun 					}
9068*4882a593Smuzhiyun 					lck_state = SPEC_MIRRORED;
9069*4882a593Smuzhiyun 					/* reset timer TODO: still need 300ms? */
9070*4882a593Smuzhiyun 					start_time = jiffies_to_msecs(jiffies);
9071*4882a593Smuzhiyun 					timeout_ofs = -DRXJ_QAM_MAX_WAITTIME / 2;
9072*4882a593Smuzhiyun 				}
9073*4882a593Smuzhiyun 			}
9074*4882a593Smuzhiyun 			break;
9075*4882a593Smuzhiyun 		case SPEC_MIRRORED:
9076*4882a593Smuzhiyun 			break;
9077*4882a593Smuzhiyun 		default:
9078*4882a593Smuzhiyun 			break;
9079*4882a593Smuzhiyun 		}
9080*4882a593Smuzhiyun 		msleep(10);
9081*4882a593Smuzhiyun 	} while
9082*4882a593Smuzhiyun 	    ((*lock_status < DRX_LOCKED) &&
9083*4882a593Smuzhiyun 	     (*lock_status != DRX_NEVER_LOCK) &&
9084*4882a593Smuzhiyun 	     ((jiffies_to_msecs(jiffies) - start_time) <
9085*4882a593Smuzhiyun 	      (DRXJ_QAM_MAX_WAITTIME + timeout_ofs)));
9086*4882a593Smuzhiyun 
9087*4882a593Smuzhiyun 	return 0;
9088*4882a593Smuzhiyun rw_error:
9089*4882a593Smuzhiyun 	return rc;
9090*4882a593Smuzhiyun }
9091*4882a593Smuzhiyun 
9092*4882a593Smuzhiyun /*
9093*4882a593Smuzhiyun * \fn int set_qam_channel ()
9094*4882a593Smuzhiyun * \brief Set QAM channel according to the requested constellation.
9095*4882a593Smuzhiyun * \param demod:   instance of demod.
9096*4882a593Smuzhiyun * \param channel: pointer to channel data.
9097*4882a593Smuzhiyun * \return int.
9098*4882a593Smuzhiyun */
9099*4882a593Smuzhiyun static int
set_qam_channel(struct drx_demod_instance * demod,struct drx_channel * channel,s32 tuner_freq_offset)9100*4882a593Smuzhiyun set_qam_channel(struct drx_demod_instance *demod,
9101*4882a593Smuzhiyun 	       struct drx_channel *channel, s32 tuner_freq_offset)
9102*4882a593Smuzhiyun {
9103*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
9104*4882a593Smuzhiyun 	int rc;
9105*4882a593Smuzhiyun 	enum drx_lock_status lock_status = DRX_NOT_LOCKED;
9106*4882a593Smuzhiyun 	bool auto_flag = false;
9107*4882a593Smuzhiyun 
9108*4882a593Smuzhiyun 	/* external attributes for storing acquired channel constellation */
9109*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
9110*4882a593Smuzhiyun 
9111*4882a593Smuzhiyun 	/* set QAM channel constellation */
9112*4882a593Smuzhiyun 	switch (channel->constellation) {
9113*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM16:
9114*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM32:
9115*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM128:
9116*4882a593Smuzhiyun 		return -EINVAL;
9117*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM64:
9118*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM256:
9119*4882a593Smuzhiyun 		if (ext_attr->standard != DRX_STANDARD_ITU_B)
9120*4882a593Smuzhiyun 			return -EINVAL;
9121*4882a593Smuzhiyun 
9122*4882a593Smuzhiyun 		ext_attr->constellation = channel->constellation;
9123*4882a593Smuzhiyun 		if (channel->mirror == DRX_MIRROR_AUTO)
9124*4882a593Smuzhiyun 			ext_attr->mirror = DRX_MIRROR_NO;
9125*4882a593Smuzhiyun 		else
9126*4882a593Smuzhiyun 			ext_attr->mirror = channel->mirror;
9127*4882a593Smuzhiyun 
9128*4882a593Smuzhiyun 		rc = set_qam(demod, channel, tuner_freq_offset, QAM_SET_OP_ALL);
9129*4882a593Smuzhiyun 		if (rc != 0) {
9130*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9131*4882a593Smuzhiyun 			goto rw_error;
9132*4882a593Smuzhiyun 		}
9133*4882a593Smuzhiyun 
9134*4882a593Smuzhiyun 		if (channel->constellation == DRX_CONSTELLATION_QAM64)
9135*4882a593Smuzhiyun 			rc = qam64auto(demod, channel, tuner_freq_offset,
9136*4882a593Smuzhiyun 				       &lock_status);
9137*4882a593Smuzhiyun 		else
9138*4882a593Smuzhiyun 			rc = qam256auto(demod, channel, tuner_freq_offset,
9139*4882a593Smuzhiyun 					&lock_status);
9140*4882a593Smuzhiyun 		if (rc != 0) {
9141*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9142*4882a593Smuzhiyun 			goto rw_error;
9143*4882a593Smuzhiyun 		}
9144*4882a593Smuzhiyun 		break;
9145*4882a593Smuzhiyun 	case DRX_CONSTELLATION_AUTO:	/* for channel scan */
9146*4882a593Smuzhiyun 		if (ext_attr->standard == DRX_STANDARD_ITU_B) {
9147*4882a593Smuzhiyun 			u16 qam_ctl_ena = 0;
9148*4882a593Smuzhiyun 
9149*4882a593Smuzhiyun 			auto_flag = true;
9150*4882a593Smuzhiyun 
9151*4882a593Smuzhiyun 			/* try to lock default QAM constellation: QAM256 */
9152*4882a593Smuzhiyun 			channel->constellation = DRX_CONSTELLATION_QAM256;
9153*4882a593Smuzhiyun 			ext_attr->constellation = DRX_CONSTELLATION_QAM256;
9154*4882a593Smuzhiyun 			if (channel->mirror == DRX_MIRROR_AUTO)
9155*4882a593Smuzhiyun 				ext_attr->mirror = DRX_MIRROR_NO;
9156*4882a593Smuzhiyun 			else
9157*4882a593Smuzhiyun 				ext_attr->mirror = channel->mirror;
9158*4882a593Smuzhiyun 			rc = set_qam(demod, channel, tuner_freq_offset,
9159*4882a593Smuzhiyun 				     QAM_SET_OP_ALL);
9160*4882a593Smuzhiyun 			if (rc != 0) {
9161*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9162*4882a593Smuzhiyun 				goto rw_error;
9163*4882a593Smuzhiyun 			}
9164*4882a593Smuzhiyun 			rc = qam256auto(demod, channel, tuner_freq_offset,
9165*4882a593Smuzhiyun 					&lock_status);
9166*4882a593Smuzhiyun 			if (rc != 0) {
9167*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9168*4882a593Smuzhiyun 				goto rw_error;
9169*4882a593Smuzhiyun 			}
9170*4882a593Smuzhiyun 
9171*4882a593Smuzhiyun 			if (lock_status >= DRX_LOCKED) {
9172*4882a593Smuzhiyun 				channel->constellation = DRX_CONSTELLATION_AUTO;
9173*4882a593Smuzhiyun 				break;
9174*4882a593Smuzhiyun 			}
9175*4882a593Smuzhiyun 
9176*4882a593Smuzhiyun 			/* QAM254 not locked. Try QAM64 constellation */
9177*4882a593Smuzhiyun 			channel->constellation = DRX_CONSTELLATION_QAM64;
9178*4882a593Smuzhiyun 			ext_attr->constellation = DRX_CONSTELLATION_QAM64;
9179*4882a593Smuzhiyun 			if (channel->mirror == DRX_MIRROR_AUTO)
9180*4882a593Smuzhiyun 				ext_attr->mirror = DRX_MIRROR_NO;
9181*4882a593Smuzhiyun 			else
9182*4882a593Smuzhiyun 				ext_attr->mirror = channel->mirror;
9183*4882a593Smuzhiyun 
9184*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr,
9185*4882a593Smuzhiyun 						     SCU_RAM_QAM_CTL_ENA__A,
9186*4882a593Smuzhiyun 						     &qam_ctl_ena, 0);
9187*4882a593Smuzhiyun 			if (rc != 0) {
9188*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9189*4882a593Smuzhiyun 				goto rw_error;
9190*4882a593Smuzhiyun 			}
9191*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
9192*4882a593Smuzhiyun 						      SCU_RAM_QAM_CTL_ENA__A,
9193*4882a593Smuzhiyun 						      qam_ctl_ena & ~SCU_RAM_QAM_CTL_ENA_ACQ__M, 0);
9194*4882a593Smuzhiyun 			if (rc != 0) {
9195*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9196*4882a593Smuzhiyun 				goto rw_error;
9197*4882a593Smuzhiyun 			}
9198*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
9199*4882a593Smuzhiyun 						      SCU_RAM_QAM_FSM_STATE_TGT__A,
9200*4882a593Smuzhiyun 						      0x2, 0);
9201*4882a593Smuzhiyun 			if (rc != 0) {
9202*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9203*4882a593Smuzhiyun 				goto rw_error;
9204*4882a593Smuzhiyun 			}	/* force to rate hunting */
9205*4882a593Smuzhiyun 
9206*4882a593Smuzhiyun 			rc = set_qam(demod, channel, tuner_freq_offset,
9207*4882a593Smuzhiyun 				     QAM_SET_OP_CONSTELLATION);
9208*4882a593Smuzhiyun 			if (rc != 0) {
9209*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9210*4882a593Smuzhiyun 				goto rw_error;
9211*4882a593Smuzhiyun 			}
9212*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
9213*4882a593Smuzhiyun 						      SCU_RAM_QAM_CTL_ENA__A,
9214*4882a593Smuzhiyun 						      qam_ctl_ena, 0);
9215*4882a593Smuzhiyun 			if (rc != 0) {
9216*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9217*4882a593Smuzhiyun 				goto rw_error;
9218*4882a593Smuzhiyun 			}
9219*4882a593Smuzhiyun 
9220*4882a593Smuzhiyun 			rc = qam64auto(demod, channel, tuner_freq_offset,
9221*4882a593Smuzhiyun 				       &lock_status);
9222*4882a593Smuzhiyun 			if (rc != 0) {
9223*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9224*4882a593Smuzhiyun 				goto rw_error;
9225*4882a593Smuzhiyun 			}
9226*4882a593Smuzhiyun 
9227*4882a593Smuzhiyun 			channel->constellation = DRX_CONSTELLATION_AUTO;
9228*4882a593Smuzhiyun 		} else if (ext_attr->standard == DRX_STANDARD_ITU_C) {
9229*4882a593Smuzhiyun 			u16 qam_ctl_ena = 0;
9230*4882a593Smuzhiyun 
9231*4882a593Smuzhiyun 			channel->constellation = DRX_CONSTELLATION_QAM64;
9232*4882a593Smuzhiyun 			ext_attr->constellation = DRX_CONSTELLATION_QAM64;
9233*4882a593Smuzhiyun 			auto_flag = true;
9234*4882a593Smuzhiyun 
9235*4882a593Smuzhiyun 			if (channel->mirror == DRX_MIRROR_AUTO)
9236*4882a593Smuzhiyun 				ext_attr->mirror = DRX_MIRROR_NO;
9237*4882a593Smuzhiyun 			else
9238*4882a593Smuzhiyun 				ext_attr->mirror = channel->mirror;
9239*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr,
9240*4882a593Smuzhiyun 						     SCU_RAM_QAM_CTL_ENA__A,
9241*4882a593Smuzhiyun 						     &qam_ctl_ena, 0);
9242*4882a593Smuzhiyun 			if (rc != 0) {
9243*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9244*4882a593Smuzhiyun 				goto rw_error;
9245*4882a593Smuzhiyun 			}
9246*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
9247*4882a593Smuzhiyun 						      SCU_RAM_QAM_CTL_ENA__A,
9248*4882a593Smuzhiyun 						      qam_ctl_ena & ~SCU_RAM_QAM_CTL_ENA_ACQ__M, 0);
9249*4882a593Smuzhiyun 			if (rc != 0) {
9250*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9251*4882a593Smuzhiyun 				goto rw_error;
9252*4882a593Smuzhiyun 			}
9253*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
9254*4882a593Smuzhiyun 						      SCU_RAM_QAM_FSM_STATE_TGT__A,
9255*4882a593Smuzhiyun 						      0x2, 0);
9256*4882a593Smuzhiyun 			if (rc != 0) {
9257*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9258*4882a593Smuzhiyun 				goto rw_error;
9259*4882a593Smuzhiyun 			}	/* force to rate hunting */
9260*4882a593Smuzhiyun 
9261*4882a593Smuzhiyun 			rc = set_qam(demod, channel, tuner_freq_offset,
9262*4882a593Smuzhiyun 				     QAM_SET_OP_CONSTELLATION);
9263*4882a593Smuzhiyun 			if (rc != 0) {
9264*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9265*4882a593Smuzhiyun 				goto rw_error;
9266*4882a593Smuzhiyun 			}
9267*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
9268*4882a593Smuzhiyun 						      SCU_RAM_QAM_CTL_ENA__A,
9269*4882a593Smuzhiyun 						      qam_ctl_ena, 0);
9270*4882a593Smuzhiyun 			if (rc != 0) {
9271*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9272*4882a593Smuzhiyun 				goto rw_error;
9273*4882a593Smuzhiyun 			}
9274*4882a593Smuzhiyun 			rc = qam64auto(demod, channel, tuner_freq_offset,
9275*4882a593Smuzhiyun 				       &lock_status);
9276*4882a593Smuzhiyun 			if (rc != 0) {
9277*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
9278*4882a593Smuzhiyun 				goto rw_error;
9279*4882a593Smuzhiyun 			}
9280*4882a593Smuzhiyun 			channel->constellation = DRX_CONSTELLATION_AUTO;
9281*4882a593Smuzhiyun 		} else {
9282*4882a593Smuzhiyun 			return -EINVAL;
9283*4882a593Smuzhiyun 		}
9284*4882a593Smuzhiyun 		break;
9285*4882a593Smuzhiyun 	default:
9286*4882a593Smuzhiyun 		return -EINVAL;
9287*4882a593Smuzhiyun 	}
9288*4882a593Smuzhiyun 
9289*4882a593Smuzhiyun 	return 0;
9290*4882a593Smuzhiyun rw_error:
9291*4882a593Smuzhiyun 	/* restore starting value */
9292*4882a593Smuzhiyun 	if (auto_flag)
9293*4882a593Smuzhiyun 		channel->constellation = DRX_CONSTELLATION_AUTO;
9294*4882a593Smuzhiyun 	return rc;
9295*4882a593Smuzhiyun }
9296*4882a593Smuzhiyun 
9297*4882a593Smuzhiyun /*============================================================================*/
9298*4882a593Smuzhiyun 
9299*4882a593Smuzhiyun /*
9300*4882a593Smuzhiyun * \fn static short get_qamrs_err_count(struct i2c_device_addr *dev_addr)
9301*4882a593Smuzhiyun * \brief Get RS error count in QAM mode (used for post RS BER calculation)
9302*4882a593Smuzhiyun * \return Error code
9303*4882a593Smuzhiyun *
9304*4882a593Smuzhiyun * precondition: measurement period & measurement prescale must be set
9305*4882a593Smuzhiyun *
9306*4882a593Smuzhiyun */
9307*4882a593Smuzhiyun static int
get_qamrs_err_count(struct i2c_device_addr * dev_addr,struct drxjrs_errors * rs_errors)9308*4882a593Smuzhiyun get_qamrs_err_count(struct i2c_device_addr *dev_addr,
9309*4882a593Smuzhiyun 		    struct drxjrs_errors *rs_errors)
9310*4882a593Smuzhiyun {
9311*4882a593Smuzhiyun 	int rc;
9312*4882a593Smuzhiyun 	u16 nr_bit_errors = 0,
9313*4882a593Smuzhiyun 	    nr_symbol_errors = 0,
9314*4882a593Smuzhiyun 	    nr_packet_errors = 0, nr_failures = 0, nr_snc_par_fail_count = 0;
9315*4882a593Smuzhiyun 
9316*4882a593Smuzhiyun 	/* check arguments */
9317*4882a593Smuzhiyun 	if (dev_addr == NULL)
9318*4882a593Smuzhiyun 		return -EINVAL;
9319*4882a593Smuzhiyun 
9320*4882a593Smuzhiyun 	/* all reported errors are received in the  */
9321*4882a593Smuzhiyun 	/* most recently finished measurement period */
9322*4882a593Smuzhiyun 	/*   no of pre RS bit errors */
9323*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_BIT_ERRORS__A, &nr_bit_errors, 0);
9324*4882a593Smuzhiyun 	if (rc != 0) {
9325*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9326*4882a593Smuzhiyun 		goto rw_error;
9327*4882a593Smuzhiyun 	}
9328*4882a593Smuzhiyun 	/*   no of symbol errors      */
9329*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_SYMBOL_ERRORS__A, &nr_symbol_errors, 0);
9330*4882a593Smuzhiyun 	if (rc != 0) {
9331*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9332*4882a593Smuzhiyun 		goto rw_error;
9333*4882a593Smuzhiyun 	}
9334*4882a593Smuzhiyun 	/*   no of packet errors      */
9335*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_PACKET_ERRORS__A, &nr_packet_errors, 0);
9336*4882a593Smuzhiyun 	if (rc != 0) {
9337*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9338*4882a593Smuzhiyun 		goto rw_error;
9339*4882a593Smuzhiyun 	}
9340*4882a593Smuzhiyun 	/*   no of failures to decode */
9341*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_FAILURES__A, &nr_failures, 0);
9342*4882a593Smuzhiyun 	if (rc != 0) {
9343*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9344*4882a593Smuzhiyun 		goto rw_error;
9345*4882a593Smuzhiyun 	}
9346*4882a593Smuzhiyun 	/*   no of post RS bit erros  */
9347*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_FAIL_COUNT__A, &nr_snc_par_fail_count, 0);
9348*4882a593Smuzhiyun 	if (rc != 0) {
9349*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9350*4882a593Smuzhiyun 		goto rw_error;
9351*4882a593Smuzhiyun 	}
9352*4882a593Smuzhiyun 	/* TODO: NOTE */
9353*4882a593Smuzhiyun 	/* These register values are fetched in non-atomic fashion           */
9354*4882a593Smuzhiyun 	/* It is possible that the read values contain unrelated information */
9355*4882a593Smuzhiyun 
9356*4882a593Smuzhiyun 	rs_errors->nr_bit_errors = nr_bit_errors & FEC_RS_NR_BIT_ERRORS__M;
9357*4882a593Smuzhiyun 	rs_errors->nr_symbol_errors = nr_symbol_errors & FEC_RS_NR_SYMBOL_ERRORS__M;
9358*4882a593Smuzhiyun 	rs_errors->nr_packet_errors = nr_packet_errors & FEC_RS_NR_PACKET_ERRORS__M;
9359*4882a593Smuzhiyun 	rs_errors->nr_failures = nr_failures & FEC_RS_NR_FAILURES__M;
9360*4882a593Smuzhiyun 	rs_errors->nr_snc_par_fail_count =
9361*4882a593Smuzhiyun 	    nr_snc_par_fail_count & FEC_OC_SNC_FAIL_COUNT__M;
9362*4882a593Smuzhiyun 
9363*4882a593Smuzhiyun 	return 0;
9364*4882a593Smuzhiyun rw_error:
9365*4882a593Smuzhiyun 	return rc;
9366*4882a593Smuzhiyun }
9367*4882a593Smuzhiyun 
9368*4882a593Smuzhiyun /*============================================================================*/
9369*4882a593Smuzhiyun 
9370*4882a593Smuzhiyun /*
9371*4882a593Smuzhiyun  * \fn int get_sig_strength()
9372*4882a593Smuzhiyun  * \brief Retrieve signal strength for VSB and QAM.
9373*4882a593Smuzhiyun  * \param demod Pointer to demod instance
9374*4882a593Smuzhiyun  * \param u16-t Pointer to signal strength data; range 0, .. , 100.
9375*4882a593Smuzhiyun  * \return int.
9376*4882a593Smuzhiyun  * \retval 0 sig_strength contains valid data.
9377*4882a593Smuzhiyun  * \retval -EINVAL sig_strength is NULL.
9378*4882a593Smuzhiyun  * \retval -EIO Erroneous data, sig_strength contains invalid data.
9379*4882a593Smuzhiyun  */
9380*4882a593Smuzhiyun #define DRXJ_AGC_TOP    0x2800
9381*4882a593Smuzhiyun #define DRXJ_AGC_SNS    0x1600
9382*4882a593Smuzhiyun #define DRXJ_RFAGC_MAX  0x3fff
9383*4882a593Smuzhiyun #define DRXJ_RFAGC_MIN  0x800
9384*4882a593Smuzhiyun 
get_sig_strength(struct drx_demod_instance * demod,u16 * sig_strength)9385*4882a593Smuzhiyun static int get_sig_strength(struct drx_demod_instance *demod, u16 *sig_strength)
9386*4882a593Smuzhiyun {
9387*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
9388*4882a593Smuzhiyun 	int rc;
9389*4882a593Smuzhiyun 	u16 rf_gain = 0;
9390*4882a593Smuzhiyun 	u16 if_gain = 0;
9391*4882a593Smuzhiyun 	u16 if_agc_sns = 0;
9392*4882a593Smuzhiyun 	u16 if_agc_top = 0;
9393*4882a593Smuzhiyun 	u16 rf_agc_max = 0;
9394*4882a593Smuzhiyun 	u16 rf_agc_min = 0;
9395*4882a593Smuzhiyun 
9396*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, IQM_AF_AGC_IF__A, &if_gain, 0);
9397*4882a593Smuzhiyun 	if (rc != 0) {
9398*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9399*4882a593Smuzhiyun 		goto rw_error;
9400*4882a593Smuzhiyun 	}
9401*4882a593Smuzhiyun 	if_gain &= IQM_AF_AGC_IF__M;
9402*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, IQM_AF_AGC_RF__A, &rf_gain, 0);
9403*4882a593Smuzhiyun 	if (rc != 0) {
9404*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9405*4882a593Smuzhiyun 		goto rw_error;
9406*4882a593Smuzhiyun 	}
9407*4882a593Smuzhiyun 	rf_gain &= IQM_AF_AGC_RF__M;
9408*4882a593Smuzhiyun 
9409*4882a593Smuzhiyun 	if_agc_sns = DRXJ_AGC_SNS;
9410*4882a593Smuzhiyun 	if_agc_top = DRXJ_AGC_TOP;
9411*4882a593Smuzhiyun 	rf_agc_max = DRXJ_RFAGC_MAX;
9412*4882a593Smuzhiyun 	rf_agc_min = DRXJ_RFAGC_MIN;
9413*4882a593Smuzhiyun 
9414*4882a593Smuzhiyun 	if (if_gain > if_agc_top) {
9415*4882a593Smuzhiyun 		if (rf_gain > rf_agc_max)
9416*4882a593Smuzhiyun 			*sig_strength = 100;
9417*4882a593Smuzhiyun 		else if (rf_gain > rf_agc_min) {
9418*4882a593Smuzhiyun 			if (rf_agc_max == rf_agc_min) {
9419*4882a593Smuzhiyun 				pr_err("error: rf_agc_max == rf_agc_min\n");
9420*4882a593Smuzhiyun 				return -EIO;
9421*4882a593Smuzhiyun 			}
9422*4882a593Smuzhiyun 			*sig_strength =
9423*4882a593Smuzhiyun 			75 + 25 * (rf_gain - rf_agc_min) / (rf_agc_max -
9424*4882a593Smuzhiyun 								rf_agc_min);
9425*4882a593Smuzhiyun 		} else
9426*4882a593Smuzhiyun 			*sig_strength = 75;
9427*4882a593Smuzhiyun 	} else if (if_gain > if_agc_sns) {
9428*4882a593Smuzhiyun 		if (if_agc_top == if_agc_sns) {
9429*4882a593Smuzhiyun 			pr_err("error: if_agc_top == if_agc_sns\n");
9430*4882a593Smuzhiyun 			return -EIO;
9431*4882a593Smuzhiyun 		}
9432*4882a593Smuzhiyun 		*sig_strength =
9433*4882a593Smuzhiyun 		20 + 55 * (if_gain - if_agc_sns) / (if_agc_top - if_agc_sns);
9434*4882a593Smuzhiyun 	} else {
9435*4882a593Smuzhiyun 		if (!if_agc_sns) {
9436*4882a593Smuzhiyun 			pr_err("error: if_agc_sns is zero!\n");
9437*4882a593Smuzhiyun 			return -EIO;
9438*4882a593Smuzhiyun 		}
9439*4882a593Smuzhiyun 		*sig_strength = (20 * if_gain / if_agc_sns);
9440*4882a593Smuzhiyun 	}
9441*4882a593Smuzhiyun 
9442*4882a593Smuzhiyun 	if (*sig_strength <= 7)
9443*4882a593Smuzhiyun 		*sig_strength = 0;
9444*4882a593Smuzhiyun 
9445*4882a593Smuzhiyun 	return 0;
9446*4882a593Smuzhiyun rw_error:
9447*4882a593Smuzhiyun 	return rc;
9448*4882a593Smuzhiyun }
9449*4882a593Smuzhiyun 
9450*4882a593Smuzhiyun /*
9451*4882a593Smuzhiyun * \fn int ctrl_get_qam_sig_quality()
9452*4882a593Smuzhiyun * \brief Retrieve QAM signal quality from device.
9453*4882a593Smuzhiyun * \param devmod Pointer to demodulator instance.
9454*4882a593Smuzhiyun * \param sig_quality Pointer to signal quality data.
9455*4882a593Smuzhiyun * \return int.
9456*4882a593Smuzhiyun * \retval 0 sig_quality contains valid data.
9457*4882a593Smuzhiyun * \retval -EINVAL sig_quality is NULL.
9458*4882a593Smuzhiyun * \retval -EIO Erroneous data, sig_quality contains invalid data.
9459*4882a593Smuzhiyun 
9460*4882a593Smuzhiyun *  Pre-condition: Device must be started and in lock.
9461*4882a593Smuzhiyun */
9462*4882a593Smuzhiyun static int
ctrl_get_qam_sig_quality(struct drx_demod_instance * demod)9463*4882a593Smuzhiyun ctrl_get_qam_sig_quality(struct drx_demod_instance *demod)
9464*4882a593Smuzhiyun {
9465*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
9466*4882a593Smuzhiyun 	struct drxj_data *ext_attr = demod->my_ext_attr;
9467*4882a593Smuzhiyun 	struct drx39xxj_state *state = dev_addr->user_data;
9468*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
9469*4882a593Smuzhiyun 	struct drxjrs_errors measuredrs_errors = { 0, 0, 0, 0, 0 };
9470*4882a593Smuzhiyun 	enum drx_modulation constellation = ext_attr->constellation;
9471*4882a593Smuzhiyun 	int rc;
9472*4882a593Smuzhiyun 
9473*4882a593Smuzhiyun 	u32 pre_bit_err_rs = 0;	/* pre RedSolomon Bit Error Rate */
9474*4882a593Smuzhiyun 	u32 post_bit_err_rs = 0;	/* post RedSolomon Bit Error Rate */
9475*4882a593Smuzhiyun 	u32 pkt_errs = 0;	/* no of packet errors in RS */
9476*4882a593Smuzhiyun 	u16 qam_sl_err_power = 0;	/* accumulated error between raw and sliced symbols */
9477*4882a593Smuzhiyun 	u16 qsym_err_vd = 0;	/* quadrature symbol errors in QAM_VD */
9478*4882a593Smuzhiyun 	u16 fec_oc_period = 0;	/* SNC sync failure measurement period */
9479*4882a593Smuzhiyun 	u16 fec_rs_prescale = 0;	/* ReedSolomon Measurement Prescale */
9480*4882a593Smuzhiyun 	u16 fec_rs_period = 0;	/* Value for corresponding I2C register */
9481*4882a593Smuzhiyun 	/* calculation constants */
9482*4882a593Smuzhiyun 	u32 rs_bit_cnt = 0;	/* RedSolomon Bit Count */
9483*4882a593Smuzhiyun 	u32 qam_sl_sig_power = 0;	/* used for MER, depends of QAM constellation */
9484*4882a593Smuzhiyun 	/* intermediate results */
9485*4882a593Smuzhiyun 	u32 e = 0;		/* exponent value used for QAM BER/SER */
9486*4882a593Smuzhiyun 	u32 m = 0;		/* mantisa value used for QAM BER/SER */
9487*4882a593Smuzhiyun 	u32 ber_cnt = 0;	/* BER count */
9488*4882a593Smuzhiyun 	/* signal quality info */
9489*4882a593Smuzhiyun 	u32 qam_sl_mer = 0;	/* QAM MER */
9490*4882a593Smuzhiyun 	u32 qam_pre_rs_ber = 0;	/* Pre RedSolomon BER */
9491*4882a593Smuzhiyun 	u32 qam_post_rs_ber = 0;	/* Post RedSolomon BER */
9492*4882a593Smuzhiyun 	u32 qam_vd_ser = 0;	/* ViterbiDecoder SER */
9493*4882a593Smuzhiyun 	u16 qam_vd_prescale = 0;	/* Viterbi Measurement Prescale */
9494*4882a593Smuzhiyun 	u16 qam_vd_period = 0;	/* Viterbi Measurement period */
9495*4882a593Smuzhiyun 	u32 vd_bit_cnt = 0;	/* ViterbiDecoder Bit Count */
9496*4882a593Smuzhiyun 
9497*4882a593Smuzhiyun 	p->block_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
9498*4882a593Smuzhiyun 
9499*4882a593Smuzhiyun 	/* read the physical registers */
9500*4882a593Smuzhiyun 	/*   Get the RS error data */
9501*4882a593Smuzhiyun 	rc = get_qamrs_err_count(dev_addr, &measuredrs_errors);
9502*4882a593Smuzhiyun 	if (rc != 0) {
9503*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9504*4882a593Smuzhiyun 		goto rw_error;
9505*4882a593Smuzhiyun 	}
9506*4882a593Smuzhiyun 	/* get the register value needed for MER */
9507*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, QAM_SL_ERR_POWER__A, &qam_sl_err_power, 0);
9508*4882a593Smuzhiyun 	if (rc != 0) {
9509*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9510*4882a593Smuzhiyun 		goto rw_error;
9511*4882a593Smuzhiyun 	}
9512*4882a593Smuzhiyun 	/* get the register value needed for post RS BER */
9513*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_FAIL_PERIOD__A, &fec_oc_period, 0);
9514*4882a593Smuzhiyun 	if (rc != 0) {
9515*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9516*4882a593Smuzhiyun 		goto rw_error;
9517*4882a593Smuzhiyun 	}
9518*4882a593Smuzhiyun 
9519*4882a593Smuzhiyun 	/* get constants needed for signal quality calculation */
9520*4882a593Smuzhiyun 	fec_rs_period = ext_attr->fec_rs_period;
9521*4882a593Smuzhiyun 	fec_rs_prescale = ext_attr->fec_rs_prescale;
9522*4882a593Smuzhiyun 	rs_bit_cnt = fec_rs_period * fec_rs_prescale * ext_attr->fec_rs_plen;
9523*4882a593Smuzhiyun 	qam_vd_period = ext_attr->qam_vd_period;
9524*4882a593Smuzhiyun 	qam_vd_prescale = ext_attr->qam_vd_prescale;
9525*4882a593Smuzhiyun 	vd_bit_cnt = qam_vd_period * qam_vd_prescale * ext_attr->fec_vd_plen;
9526*4882a593Smuzhiyun 
9527*4882a593Smuzhiyun 	/* DRXJ_QAM_SL_SIG_POWER_QAMxxx  * 4     */
9528*4882a593Smuzhiyun 	switch (constellation) {
9529*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM16:
9530*4882a593Smuzhiyun 		qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM16 << 2;
9531*4882a593Smuzhiyun 		break;
9532*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM32:
9533*4882a593Smuzhiyun 		qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM32 << 2;
9534*4882a593Smuzhiyun 		break;
9535*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM64:
9536*4882a593Smuzhiyun 		qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM64 << 2;
9537*4882a593Smuzhiyun 		break;
9538*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM128:
9539*4882a593Smuzhiyun 		qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM128 << 2;
9540*4882a593Smuzhiyun 		break;
9541*4882a593Smuzhiyun 	case DRX_CONSTELLATION_QAM256:
9542*4882a593Smuzhiyun 		qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM256 << 2;
9543*4882a593Smuzhiyun 		break;
9544*4882a593Smuzhiyun 	default:
9545*4882a593Smuzhiyun 		return -EIO;
9546*4882a593Smuzhiyun 	}
9547*4882a593Smuzhiyun 
9548*4882a593Smuzhiyun 	/* ------------------------------ */
9549*4882a593Smuzhiyun 	/* MER Calculation                */
9550*4882a593Smuzhiyun 	/* ------------------------------ */
9551*4882a593Smuzhiyun 	/* MER is good if it is above 27.5 for QAM256 or 21.5 for QAM64 */
9552*4882a593Smuzhiyun 
9553*4882a593Smuzhiyun 	/* 10.0*log10(qam_sl_sig_power * 4.0 / qam_sl_err_power); */
9554*4882a593Smuzhiyun 	if (qam_sl_err_power == 0)
9555*4882a593Smuzhiyun 		qam_sl_mer = 0;
9556*4882a593Smuzhiyun 	else
9557*4882a593Smuzhiyun 		qam_sl_mer = log1_times100(qam_sl_sig_power) - log1_times100((u32)qam_sl_err_power);
9558*4882a593Smuzhiyun 
9559*4882a593Smuzhiyun 	/* ----------------------------------------- */
9560*4882a593Smuzhiyun 	/* Pre Viterbi Symbol Error Rate Calculation */
9561*4882a593Smuzhiyun 	/* ----------------------------------------- */
9562*4882a593Smuzhiyun 	/* pre viterbi SER is good if it is below 0.025 */
9563*4882a593Smuzhiyun 
9564*4882a593Smuzhiyun 	/* get the register value */
9565*4882a593Smuzhiyun 	/*   no of quadrature symbol errors */
9566*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, QAM_VD_NR_QSYM_ERRORS__A, &qsym_err_vd, 0);
9567*4882a593Smuzhiyun 	if (rc != 0) {
9568*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9569*4882a593Smuzhiyun 		goto rw_error;
9570*4882a593Smuzhiyun 	}
9571*4882a593Smuzhiyun 	/* Extract the Exponent and the Mantisa  */
9572*4882a593Smuzhiyun 	/* of number of quadrature symbol errors */
9573*4882a593Smuzhiyun 	e = (qsym_err_vd & QAM_VD_NR_QSYM_ERRORS_EXP__M) >>
9574*4882a593Smuzhiyun 	    QAM_VD_NR_QSYM_ERRORS_EXP__B;
9575*4882a593Smuzhiyun 	m = (qsym_err_vd & QAM_VD_NR_SYMBOL_ERRORS_FIXED_MANT__M) >>
9576*4882a593Smuzhiyun 	    QAM_VD_NR_SYMBOL_ERRORS_FIXED_MANT__B;
9577*4882a593Smuzhiyun 
9578*4882a593Smuzhiyun 	if ((m << e) >> 3 > 549752)
9579*4882a593Smuzhiyun 		qam_vd_ser = 500000 * vd_bit_cnt * ((e > 2) ? 1 : 8) / 8;
9580*4882a593Smuzhiyun 	else
9581*4882a593Smuzhiyun 		qam_vd_ser = m << ((e > 2) ? (e - 3) : e);
9582*4882a593Smuzhiyun 
9583*4882a593Smuzhiyun 	/* --------------------------------------- */
9584*4882a593Smuzhiyun 	/* pre and post RedSolomon BER Calculation */
9585*4882a593Smuzhiyun 	/* --------------------------------------- */
9586*4882a593Smuzhiyun 	/* pre RS BER is good if it is below 3.5e-4 */
9587*4882a593Smuzhiyun 
9588*4882a593Smuzhiyun 	/* get the register values */
9589*4882a593Smuzhiyun 	pre_bit_err_rs = (u32) measuredrs_errors.nr_bit_errors;
9590*4882a593Smuzhiyun 	pkt_errs = post_bit_err_rs = (u32) measuredrs_errors.nr_snc_par_fail_count;
9591*4882a593Smuzhiyun 
9592*4882a593Smuzhiyun 	/* Extract the Exponent and the Mantisa of the */
9593*4882a593Smuzhiyun 	/* pre Reed-Solomon bit error count            */
9594*4882a593Smuzhiyun 	e = (pre_bit_err_rs & FEC_RS_NR_BIT_ERRORS_EXP__M) >>
9595*4882a593Smuzhiyun 	    FEC_RS_NR_BIT_ERRORS_EXP__B;
9596*4882a593Smuzhiyun 	m = (pre_bit_err_rs & FEC_RS_NR_BIT_ERRORS_FIXED_MANT__M) >>
9597*4882a593Smuzhiyun 	    FEC_RS_NR_BIT_ERRORS_FIXED_MANT__B;
9598*4882a593Smuzhiyun 
9599*4882a593Smuzhiyun 	ber_cnt = m << e;
9600*4882a593Smuzhiyun 
9601*4882a593Smuzhiyun 	/*qam_pre_rs_ber = frac_times1e6( ber_cnt, rs_bit_cnt ); */
9602*4882a593Smuzhiyun 	if (m > (rs_bit_cnt >> (e + 1)) || (rs_bit_cnt >> e) == 0)
9603*4882a593Smuzhiyun 		qam_pre_rs_ber = 500000 * rs_bit_cnt >> e;
9604*4882a593Smuzhiyun 	else
9605*4882a593Smuzhiyun 		qam_pre_rs_ber = ber_cnt;
9606*4882a593Smuzhiyun 
9607*4882a593Smuzhiyun 	/* post RS BER = 1000000* (11.17 * FEC_OC_SNC_FAIL_COUNT__A) /  */
9608*4882a593Smuzhiyun 	/*               (1504.0 * FEC_OC_SNC_FAIL_PERIOD__A)  */
9609*4882a593Smuzhiyun 	/*
9610*4882a593Smuzhiyun 	   => c = (1000000*100*11.17)/1504 =
9611*4882a593Smuzhiyun 	   post RS BER = (( c* FEC_OC_SNC_FAIL_COUNT__A) /
9612*4882a593Smuzhiyun 	   (100 * FEC_OC_SNC_FAIL_PERIOD__A)
9613*4882a593Smuzhiyun 	   *100 and /100 is for more precision.
9614*4882a593Smuzhiyun 	   => (20 bits * 12 bits) /(16 bits * 7 bits)  => safe in 32 bits computation
9615*4882a593Smuzhiyun 
9616*4882a593Smuzhiyun 	   Precision errors still possible.
9617*4882a593Smuzhiyun 	 */
9618*4882a593Smuzhiyun 	if (!fec_oc_period) {
9619*4882a593Smuzhiyun 		qam_post_rs_ber = 0xFFFFFFFF;
9620*4882a593Smuzhiyun 	} else {
9621*4882a593Smuzhiyun 		e = post_bit_err_rs * 742686;
9622*4882a593Smuzhiyun 		m = fec_oc_period * 100;
9623*4882a593Smuzhiyun 		qam_post_rs_ber = e / m;
9624*4882a593Smuzhiyun 	}
9625*4882a593Smuzhiyun 
9626*4882a593Smuzhiyun 	/* fill signal quality data structure */
9627*4882a593Smuzhiyun 	p->pre_bit_count.stat[0].scale = FE_SCALE_COUNTER;
9628*4882a593Smuzhiyun 	p->post_bit_count.stat[0].scale = FE_SCALE_COUNTER;
9629*4882a593Smuzhiyun 	p->pre_bit_error.stat[0].scale = FE_SCALE_COUNTER;
9630*4882a593Smuzhiyun 	p->post_bit_error.stat[0].scale = FE_SCALE_COUNTER;
9631*4882a593Smuzhiyun 	p->block_error.stat[0].scale = FE_SCALE_COUNTER;
9632*4882a593Smuzhiyun 	p->cnr.stat[0].scale = FE_SCALE_DECIBEL;
9633*4882a593Smuzhiyun 
9634*4882a593Smuzhiyun 	p->cnr.stat[0].svalue = ((u16) qam_sl_mer) * 100;
9635*4882a593Smuzhiyun 	if (ext_attr->standard == DRX_STANDARD_ITU_B) {
9636*4882a593Smuzhiyun 		p->pre_bit_error.stat[0].uvalue += qam_vd_ser;
9637*4882a593Smuzhiyun 		p->pre_bit_count.stat[0].uvalue += vd_bit_cnt * ((e > 2) ? 1 : 8) / 8;
9638*4882a593Smuzhiyun 	} else {
9639*4882a593Smuzhiyun 		p->pre_bit_error.stat[0].uvalue += qam_pre_rs_ber;
9640*4882a593Smuzhiyun 		p->pre_bit_count.stat[0].uvalue += rs_bit_cnt >> e;
9641*4882a593Smuzhiyun 	}
9642*4882a593Smuzhiyun 
9643*4882a593Smuzhiyun 	p->post_bit_error.stat[0].uvalue += qam_post_rs_ber;
9644*4882a593Smuzhiyun 	p->post_bit_count.stat[0].uvalue += rs_bit_cnt >> e;
9645*4882a593Smuzhiyun 
9646*4882a593Smuzhiyun 	p->block_error.stat[0].uvalue += pkt_errs;
9647*4882a593Smuzhiyun 
9648*4882a593Smuzhiyun #ifdef DRXJ_SIGNAL_ACCUM_ERR
9649*4882a593Smuzhiyun 	rc = get_acc_pkt_err(demod, &sig_quality->packet_error);
9650*4882a593Smuzhiyun 	if (rc != 0) {
9651*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9652*4882a593Smuzhiyun 		goto rw_error;
9653*4882a593Smuzhiyun 	}
9654*4882a593Smuzhiyun #endif
9655*4882a593Smuzhiyun 
9656*4882a593Smuzhiyun 	return 0;
9657*4882a593Smuzhiyun rw_error:
9658*4882a593Smuzhiyun 	p->pre_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
9659*4882a593Smuzhiyun 	p->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
9660*4882a593Smuzhiyun 	p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
9661*4882a593Smuzhiyun 	p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
9662*4882a593Smuzhiyun 	p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
9663*4882a593Smuzhiyun 	p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
9664*4882a593Smuzhiyun 
9665*4882a593Smuzhiyun 	return rc;
9666*4882a593Smuzhiyun }
9667*4882a593Smuzhiyun 
9668*4882a593Smuzhiyun #endif /* #ifndef DRXJ_VSB_ONLY */
9669*4882a593Smuzhiyun 
9670*4882a593Smuzhiyun /*============================================================================*/
9671*4882a593Smuzhiyun /*==                     END QAM DATAPATH FUNCTIONS                         ==*/
9672*4882a593Smuzhiyun /*============================================================================*/
9673*4882a593Smuzhiyun 
9674*4882a593Smuzhiyun /*============================================================================*/
9675*4882a593Smuzhiyun /*============================================================================*/
9676*4882a593Smuzhiyun /*==                       ATV DATAPATH FUNCTIONS                           ==*/
9677*4882a593Smuzhiyun /*============================================================================*/
9678*4882a593Smuzhiyun /*============================================================================*/
9679*4882a593Smuzhiyun 
9680*4882a593Smuzhiyun /*
9681*4882a593Smuzhiyun    Implementation notes.
9682*4882a593Smuzhiyun 
9683*4882a593Smuzhiyun    NTSC/FM AGCs
9684*4882a593Smuzhiyun 
9685*4882a593Smuzhiyun       Four AGCs are used for NTSC:
9686*4882a593Smuzhiyun       (1) RF (used to attenuate the input signal in case of to much power)
9687*4882a593Smuzhiyun       (2) IF (used to attenuate the input signal in case of to much power)
9688*4882a593Smuzhiyun       (3) Video AGC (used to amplify the output signal in case input to low)
9689*4882a593Smuzhiyun       (4) SIF AGC (used to amplify the output signal in case input to low)
9690*4882a593Smuzhiyun 
9691*4882a593Smuzhiyun       Video AGC is coupled to RF and IF. SIF AGC is not coupled. It is assumed
9692*4882a593Smuzhiyun       that the coupling between Video AGC and the RF and IF AGCs also works in
9693*4882a593Smuzhiyun       favor of the SIF AGC.
9694*4882a593Smuzhiyun 
9695*4882a593Smuzhiyun       Three AGCs are used for FM:
9696*4882a593Smuzhiyun       (1) RF (used to attenuate the input signal in case of to much power)
9697*4882a593Smuzhiyun       (2) IF (used to attenuate the input signal in case of to much power)
9698*4882a593Smuzhiyun       (3) SIF AGC (used to amplify the output signal in case input to low)
9699*4882a593Smuzhiyun 
9700*4882a593Smuzhiyun       The SIF AGC is now coupled to the RF/IF AGCs.
9701*4882a593Smuzhiyun       The SIF AGC is needed for both SIF output and the internal SIF signal to
9702*4882a593Smuzhiyun       the AUD block.
9703*4882a593Smuzhiyun 
9704*4882a593Smuzhiyun       RF and IF AGCs DACs are part of AFE, Video and SIF AGC DACs are part of
9705*4882a593Smuzhiyun       the ATV block. The AGC control algorithms are all implemented in
9706*4882a593Smuzhiyun       microcode.
9707*4882a593Smuzhiyun 
9708*4882a593Smuzhiyun    ATV SETTINGS
9709*4882a593Smuzhiyun 
9710*4882a593Smuzhiyun       (Shadow settings will not be used for now, they will be implemented
9711*4882a593Smuzhiyun        later on because of the schedule)
9712*4882a593Smuzhiyun 
9713*4882a593Smuzhiyun       Several HW/SCU "settings" can be used for ATV. The standard selection
9714*4882a593Smuzhiyun       will reset most of these settings. To avoid that the end user application
9715*4882a593Smuzhiyun       has to perform these settings each time the ATV or FM standards is
9716*4882a593Smuzhiyun       selected the driver will shadow these settings. This enables the end user
9717*4882a593Smuzhiyun       to perform the settings only once after a drx_open(). The driver must
9718*4882a593Smuzhiyun       write the shadow settings to HW/SCU in case:
9719*4882a593Smuzhiyun 	 ( setstandard FM/ATV) ||
9720*4882a593Smuzhiyun 	 ( settings have changed && FM/ATV standard is active)
9721*4882a593Smuzhiyun       The shadow settings will be stored in the device specific data container.
9722*4882a593Smuzhiyun       A set of flags will be defined to flag changes in shadow settings.
9723*4882a593Smuzhiyun       A routine will be implemented to write all changed shadow settings to
9724*4882a593Smuzhiyun       HW/SCU.
9725*4882a593Smuzhiyun 
9726*4882a593Smuzhiyun       The "settings" will consist of: AGC settings, filter settings etc.
9727*4882a593Smuzhiyun 
9728*4882a593Smuzhiyun       Disadvantage of use of shadow settings:
9729*4882a593Smuzhiyun       Direct changes in HW/SCU registers will not be reflected in the
9730*4882a593Smuzhiyun       shadow settings and these changes will be overwritten during a next
9731*4882a593Smuzhiyun       update. This can happen during evaluation. This will not be a problem
9732*4882a593Smuzhiyun       for normal customer usage.
9733*4882a593Smuzhiyun */
9734*4882a593Smuzhiyun /* -------------------------------------------------------------------------- */
9735*4882a593Smuzhiyun 
9736*4882a593Smuzhiyun /*
9737*4882a593Smuzhiyun * \fn int power_down_atv ()
9738*4882a593Smuzhiyun * \brief Power down ATV.
9739*4882a593Smuzhiyun * \param demod instance of demodulator
9740*4882a593Smuzhiyun * \param standard either NTSC or FM (sub strandard for ATV )
9741*4882a593Smuzhiyun * \return int.
9742*4882a593Smuzhiyun *
9743*4882a593Smuzhiyun *  Stops and thus resets ATV and IQM block
9744*4882a593Smuzhiyun *  SIF and CVBS ADC are powered down
9745*4882a593Smuzhiyun *  Calls audio power down
9746*4882a593Smuzhiyun */
9747*4882a593Smuzhiyun static int
power_down_atv(struct drx_demod_instance * demod,enum drx_standard standard,bool primary)9748*4882a593Smuzhiyun power_down_atv(struct drx_demod_instance *demod, enum drx_standard standard, bool primary)
9749*4882a593Smuzhiyun {
9750*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
9751*4882a593Smuzhiyun 	struct drxjscu_cmd cmd_scu = { /* command      */ 0,
9752*4882a593Smuzhiyun 		/* parameter_len */ 0,
9753*4882a593Smuzhiyun 		/* result_len    */ 0,
9754*4882a593Smuzhiyun 		/* *parameter   */ NULL,
9755*4882a593Smuzhiyun 		/* *result      */ NULL
9756*4882a593Smuzhiyun 	};
9757*4882a593Smuzhiyun 	int rc;
9758*4882a593Smuzhiyun 	u16 cmd_result = 0;
9759*4882a593Smuzhiyun 
9760*4882a593Smuzhiyun 	/* ATV NTSC */
9761*4882a593Smuzhiyun 
9762*4882a593Smuzhiyun 	/* Stop ATV SCU (will reset ATV and IQM hardware */
9763*4882a593Smuzhiyun 	cmd_scu.command = SCU_RAM_COMMAND_STANDARD_ATV |
9764*4882a593Smuzhiyun 	    SCU_RAM_COMMAND_CMD_DEMOD_STOP;
9765*4882a593Smuzhiyun 	cmd_scu.parameter_len = 0;
9766*4882a593Smuzhiyun 	cmd_scu.result_len = 1;
9767*4882a593Smuzhiyun 	cmd_scu.parameter = NULL;
9768*4882a593Smuzhiyun 	cmd_scu.result = &cmd_result;
9769*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &cmd_scu);
9770*4882a593Smuzhiyun 	if (rc != 0) {
9771*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9772*4882a593Smuzhiyun 		goto rw_error;
9773*4882a593Smuzhiyun 	}
9774*4882a593Smuzhiyun 	/* Disable ATV outputs (ATV reset enables CVBS, undo this) */
9775*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ATV_TOP_STDBY__A, (ATV_TOP_STDBY_SIF_STDBY_STANDBY & (~ATV_TOP_STDBY_CVBS_STDBY_A2_ACTIVE)), 0);
9776*4882a593Smuzhiyun 	if (rc != 0) {
9777*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9778*4882a593Smuzhiyun 		goto rw_error;
9779*4882a593Smuzhiyun 	}
9780*4882a593Smuzhiyun 
9781*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ATV_COMM_EXEC__A, ATV_COMM_EXEC_STOP, 0);
9782*4882a593Smuzhiyun 	if (rc != 0) {
9783*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9784*4882a593Smuzhiyun 		goto rw_error;
9785*4882a593Smuzhiyun 	}
9786*4882a593Smuzhiyun 	if (primary) {
9787*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_STOP, 0);
9788*4882a593Smuzhiyun 		if (rc != 0) {
9789*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9790*4882a593Smuzhiyun 			goto rw_error;
9791*4882a593Smuzhiyun 		}
9792*4882a593Smuzhiyun 		rc = set_iqm_af(demod, false);
9793*4882a593Smuzhiyun 		if (rc != 0) {
9794*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9795*4882a593Smuzhiyun 			goto rw_error;
9796*4882a593Smuzhiyun 		}
9797*4882a593Smuzhiyun 	} else {
9798*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
9799*4882a593Smuzhiyun 		if (rc != 0) {
9800*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9801*4882a593Smuzhiyun 			goto rw_error;
9802*4882a593Smuzhiyun 		}
9803*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
9804*4882a593Smuzhiyun 		if (rc != 0) {
9805*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9806*4882a593Smuzhiyun 			goto rw_error;
9807*4882a593Smuzhiyun 		}
9808*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
9809*4882a593Smuzhiyun 		if (rc != 0) {
9810*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9811*4882a593Smuzhiyun 			goto rw_error;
9812*4882a593Smuzhiyun 		}
9813*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
9814*4882a593Smuzhiyun 		if (rc != 0) {
9815*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9816*4882a593Smuzhiyun 			goto rw_error;
9817*4882a593Smuzhiyun 		}
9818*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
9819*4882a593Smuzhiyun 		if (rc != 0) {
9820*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9821*4882a593Smuzhiyun 			goto rw_error;
9822*4882a593Smuzhiyun 		}
9823*4882a593Smuzhiyun 	}
9824*4882a593Smuzhiyun 	rc = power_down_aud(demod);
9825*4882a593Smuzhiyun 	if (rc != 0) {
9826*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9827*4882a593Smuzhiyun 		goto rw_error;
9828*4882a593Smuzhiyun 	}
9829*4882a593Smuzhiyun 
9830*4882a593Smuzhiyun 	return 0;
9831*4882a593Smuzhiyun rw_error:
9832*4882a593Smuzhiyun 	return rc;
9833*4882a593Smuzhiyun }
9834*4882a593Smuzhiyun 
9835*4882a593Smuzhiyun /*============================================================================*/
9836*4882a593Smuzhiyun 
9837*4882a593Smuzhiyun /*
9838*4882a593Smuzhiyun * \brief Power up AUD.
9839*4882a593Smuzhiyun * \param demod instance of demodulator
9840*4882a593Smuzhiyun * \return int.
9841*4882a593Smuzhiyun *
9842*4882a593Smuzhiyun */
power_down_aud(struct drx_demod_instance * demod)9843*4882a593Smuzhiyun static int power_down_aud(struct drx_demod_instance *demod)
9844*4882a593Smuzhiyun {
9845*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
9846*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
9847*4882a593Smuzhiyun 	int rc;
9848*4882a593Smuzhiyun 
9849*4882a593Smuzhiyun 	dev_addr = (struct i2c_device_addr *)demod->my_i2c_dev_addr;
9850*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
9851*4882a593Smuzhiyun 
9852*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, AUD_COMM_EXEC__A, AUD_COMM_EXEC_STOP, 0);
9853*4882a593Smuzhiyun 	if (rc != 0) {
9854*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9855*4882a593Smuzhiyun 		goto rw_error;
9856*4882a593Smuzhiyun 	}
9857*4882a593Smuzhiyun 
9858*4882a593Smuzhiyun 	ext_attr->aud_data.audio_is_active = false;
9859*4882a593Smuzhiyun 
9860*4882a593Smuzhiyun 	return 0;
9861*4882a593Smuzhiyun rw_error:
9862*4882a593Smuzhiyun 	return rc;
9863*4882a593Smuzhiyun }
9864*4882a593Smuzhiyun 
9865*4882a593Smuzhiyun /*
9866*4882a593Smuzhiyun * \fn int set_orx_nsu_aox()
9867*4882a593Smuzhiyun * \brief Configure OrxNsuAox for OOB
9868*4882a593Smuzhiyun * \param demod instance of demodulator.
9869*4882a593Smuzhiyun * \param active
9870*4882a593Smuzhiyun * \return int.
9871*4882a593Smuzhiyun */
set_orx_nsu_aox(struct drx_demod_instance * demod,bool active)9872*4882a593Smuzhiyun static int set_orx_nsu_aox(struct drx_demod_instance *demod, bool active)
9873*4882a593Smuzhiyun {
9874*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
9875*4882a593Smuzhiyun 	int rc;
9876*4882a593Smuzhiyun 	u16 data = 0;
9877*4882a593Smuzhiyun 
9878*4882a593Smuzhiyun 	/* Configure NSU_AOX */
9879*4882a593Smuzhiyun 	rc = drxj_dap_read_reg16(dev_addr, ORX_NSU_AOX_STDBY_W__A, &data, 0);
9880*4882a593Smuzhiyun 	if (rc != 0) {
9881*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9882*4882a593Smuzhiyun 		goto rw_error;
9883*4882a593Smuzhiyun 	}
9884*4882a593Smuzhiyun 	if (!active)
9885*4882a593Smuzhiyun 		data &= ((~ORX_NSU_AOX_STDBY_W_STDBYADC_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYAMP_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYBIAS_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYPLL_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYPD_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYTAGC_IF_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYTAGC_RF_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYFLT_A2_ON));
9886*4882a593Smuzhiyun 	else
9887*4882a593Smuzhiyun 		data |= (ORX_NSU_AOX_STDBY_W_STDBYADC_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYAMP_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYBIAS_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYPLL_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYPD_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYTAGC_IF_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYTAGC_RF_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYFLT_A2_ON);
9888*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_NSU_AOX_STDBY_W__A, data, 0);
9889*4882a593Smuzhiyun 	if (rc != 0) {
9890*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
9891*4882a593Smuzhiyun 		goto rw_error;
9892*4882a593Smuzhiyun 	}
9893*4882a593Smuzhiyun 
9894*4882a593Smuzhiyun 	return 0;
9895*4882a593Smuzhiyun rw_error:
9896*4882a593Smuzhiyun 	return rc;
9897*4882a593Smuzhiyun }
9898*4882a593Smuzhiyun 
9899*4882a593Smuzhiyun /*
9900*4882a593Smuzhiyun * \fn int ctrl_set_oob()
9901*4882a593Smuzhiyun * \brief Set OOB channel to be used.
9902*4882a593Smuzhiyun * \param demod instance of demodulator
9903*4882a593Smuzhiyun * \param oob_param OOB parameters for channel setting.
9904*4882a593Smuzhiyun * \frequency should be in KHz
9905*4882a593Smuzhiyun * \return int.
9906*4882a593Smuzhiyun *
9907*4882a593Smuzhiyun * Accepts  only. Returns error otherwise.
9908*4882a593Smuzhiyun * Demapper value is written after scu_command START
9909*4882a593Smuzhiyun * because START command causes COMM_EXEC transition
9910*4882a593Smuzhiyun * from 0 to 1 which causes all registers to be
9911*4882a593Smuzhiyun * overwritten with initial value
9912*4882a593Smuzhiyun *
9913*4882a593Smuzhiyun */
9914*4882a593Smuzhiyun 
9915*4882a593Smuzhiyun /* Nyquist filter impulse response */
9916*4882a593Smuzhiyun #define IMPULSE_COSINE_ALPHA_0_3    {-3, -4, -1, 6, 10, 7, -5, -20, -25, -10, 29, 79, 123, 140}	/*sqrt raised-cosine filter with alpha=0.3 */
9917*4882a593Smuzhiyun #define IMPULSE_COSINE_ALPHA_0_5    { 2, 0, -2, -2, 2, 5, 2, -10, -20, -14, 20, 74, 125, 145}	/*sqrt raised-cosine filter with alpha=0.5 */
9918*4882a593Smuzhiyun #define IMPULSE_COSINE_ALPHA_RO_0_5 { 0, 0, 1, 2, 3, 0, -7, -15, -16,  0, 34, 77, 114, 128}	/*full raised-cosine filter with alpha=0.5 (receiver only) */
9919*4882a593Smuzhiyun 
9920*4882a593Smuzhiyun /* Coefficients for the nyquist filter (total: 27 taps) */
9921*4882a593Smuzhiyun #define NYQFILTERLEN 27
9922*4882a593Smuzhiyun 
ctrl_set_oob(struct drx_demod_instance * demod,struct drxoob * oob_param)9923*4882a593Smuzhiyun static int ctrl_set_oob(struct drx_demod_instance *demod, struct drxoob *oob_param)
9924*4882a593Smuzhiyun {
9925*4882a593Smuzhiyun 	int rc;
9926*4882a593Smuzhiyun 	s32 freq = 0;	/* KHz */
9927*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
9928*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
9929*4882a593Smuzhiyun 	u16 i = 0;
9930*4882a593Smuzhiyun 	bool mirror_freq_spect_oob = false;
9931*4882a593Smuzhiyun 	u16 trk_filter_value = 0;
9932*4882a593Smuzhiyun 	struct drxjscu_cmd scu_cmd;
9933*4882a593Smuzhiyun 	u16 set_param_parameters[3];
9934*4882a593Smuzhiyun 	u16 cmd_result[2] = { 0, 0 };
9935*4882a593Smuzhiyun 	s16 nyquist_coeffs[4][(NYQFILTERLEN + 1) / 2] = {
9936*4882a593Smuzhiyun 		IMPULSE_COSINE_ALPHA_0_3,	/* Target Mode 0 */
9937*4882a593Smuzhiyun 		IMPULSE_COSINE_ALPHA_0_3,	/* Target Mode 1 */
9938*4882a593Smuzhiyun 		IMPULSE_COSINE_ALPHA_0_5,	/* Target Mode 2 */
9939*4882a593Smuzhiyun 		IMPULSE_COSINE_ALPHA_RO_0_5	/* Target Mode 3 */
9940*4882a593Smuzhiyun 	};
9941*4882a593Smuzhiyun 	u8 mode_val[4] = { 2, 2, 0, 1 };
9942*4882a593Smuzhiyun 	u8 pfi_coeffs[4][6] = {
9943*4882a593Smuzhiyun 		{DRXJ_16TO8(-92), DRXJ_16TO8(-108), DRXJ_16TO8(100)},	/* TARGET_MODE = 0:     PFI_A = -23/32; PFI_B = -54/32;  PFI_C = 25/32; fg = 0.5 MHz (Att=26dB) */
9944*4882a593Smuzhiyun 		{DRXJ_16TO8(-64), DRXJ_16TO8(-80), DRXJ_16TO8(80)},	/* TARGET_MODE = 1:     PFI_A = -16/32; PFI_B = -40/32;  PFI_C = 20/32; fg = 1.0 MHz (Att=28dB) */
9945*4882a593Smuzhiyun 		{DRXJ_16TO8(-80), DRXJ_16TO8(-98), DRXJ_16TO8(92)},	/* TARGET_MODE = 2, 3:  PFI_A = -20/32; PFI_B = -49/32;  PFI_C = 23/32; fg = 0.8 MHz (Att=25dB) */
9946*4882a593Smuzhiyun 		{DRXJ_16TO8(-80), DRXJ_16TO8(-98), DRXJ_16TO8(92)}	/* TARGET_MODE = 2, 3:  PFI_A = -20/32; PFI_B = -49/32;  PFI_C = 23/32; fg = 0.8 MHz (Att=25dB) */
9947*4882a593Smuzhiyun 	};
9948*4882a593Smuzhiyun 	u16 mode_index;
9949*4882a593Smuzhiyun 
9950*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
9951*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
9952*4882a593Smuzhiyun 	mirror_freq_spect_oob = ext_attr->mirror_freq_spect_oob;
9953*4882a593Smuzhiyun 
9954*4882a593Smuzhiyun 	/* Check parameters */
9955*4882a593Smuzhiyun 	if (oob_param == NULL) {
9956*4882a593Smuzhiyun 		/* power off oob module  */
9957*4882a593Smuzhiyun 		scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
9958*4882a593Smuzhiyun 		    | SCU_RAM_COMMAND_CMD_DEMOD_STOP;
9959*4882a593Smuzhiyun 		scu_cmd.parameter_len = 0;
9960*4882a593Smuzhiyun 		scu_cmd.result_len = 1;
9961*4882a593Smuzhiyun 		scu_cmd.result = cmd_result;
9962*4882a593Smuzhiyun 		rc = scu_command(dev_addr, &scu_cmd);
9963*4882a593Smuzhiyun 		if (rc != 0) {
9964*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9965*4882a593Smuzhiyun 			goto rw_error;
9966*4882a593Smuzhiyun 		}
9967*4882a593Smuzhiyun 		rc = set_orx_nsu_aox(demod, false);
9968*4882a593Smuzhiyun 		if (rc != 0) {
9969*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9970*4882a593Smuzhiyun 			goto rw_error;
9971*4882a593Smuzhiyun 		}
9972*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, ORX_COMM_EXEC__A, ORX_COMM_EXEC_STOP, 0);
9973*4882a593Smuzhiyun 		if (rc != 0) {
9974*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
9975*4882a593Smuzhiyun 			goto rw_error;
9976*4882a593Smuzhiyun 		}
9977*4882a593Smuzhiyun 
9978*4882a593Smuzhiyun 		ext_attr->oob_power_on = false;
9979*4882a593Smuzhiyun 		return 0;
9980*4882a593Smuzhiyun 	}
9981*4882a593Smuzhiyun 
9982*4882a593Smuzhiyun 	freq = oob_param->frequency;
9983*4882a593Smuzhiyun 	if ((freq < 70000) || (freq > 130000))
9984*4882a593Smuzhiyun 		return -EIO;
9985*4882a593Smuzhiyun 	freq = (freq - 50000) / 50;
9986*4882a593Smuzhiyun 
9987*4882a593Smuzhiyun 	{
9988*4882a593Smuzhiyun 		u16 index = 0;
9989*4882a593Smuzhiyun 		u16 remainder = 0;
9990*4882a593Smuzhiyun 		u16 *trk_filtercfg = ext_attr->oob_trk_filter_cfg;
9991*4882a593Smuzhiyun 
9992*4882a593Smuzhiyun 		index = (u16) ((freq - 400) / 200);
9993*4882a593Smuzhiyun 		remainder = (u16) ((freq - 400) % 200);
9994*4882a593Smuzhiyun 		trk_filter_value =
9995*4882a593Smuzhiyun 		    trk_filtercfg[index] - (trk_filtercfg[index] -
9996*4882a593Smuzhiyun 					   trk_filtercfg[index +
9997*4882a593Smuzhiyun 							1]) / 10 * remainder /
9998*4882a593Smuzhiyun 		    20;
9999*4882a593Smuzhiyun 	}
10000*4882a593Smuzhiyun 
10001*4882a593Smuzhiyun    /********/
10002*4882a593Smuzhiyun 	/* Stop  */
10003*4882a593Smuzhiyun    /********/
10004*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_COMM_EXEC__A, ORX_COMM_EXEC_STOP, 0);
10005*4882a593Smuzhiyun 	if (rc != 0) {
10006*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10007*4882a593Smuzhiyun 		goto rw_error;
10008*4882a593Smuzhiyun 	}
10009*4882a593Smuzhiyun 	scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
10010*4882a593Smuzhiyun 	    | SCU_RAM_COMMAND_CMD_DEMOD_STOP;
10011*4882a593Smuzhiyun 	scu_cmd.parameter_len = 0;
10012*4882a593Smuzhiyun 	scu_cmd.result_len = 1;
10013*4882a593Smuzhiyun 	scu_cmd.result = cmd_result;
10014*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &scu_cmd);
10015*4882a593Smuzhiyun 	if (rc != 0) {
10016*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10017*4882a593Smuzhiyun 		goto rw_error;
10018*4882a593Smuzhiyun 	}
10019*4882a593Smuzhiyun    /********/
10020*4882a593Smuzhiyun 	/* Reset */
10021*4882a593Smuzhiyun    /********/
10022*4882a593Smuzhiyun 	scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
10023*4882a593Smuzhiyun 	    | SCU_RAM_COMMAND_CMD_DEMOD_RESET;
10024*4882a593Smuzhiyun 	scu_cmd.parameter_len = 0;
10025*4882a593Smuzhiyun 	scu_cmd.result_len = 1;
10026*4882a593Smuzhiyun 	scu_cmd.result = cmd_result;
10027*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &scu_cmd);
10028*4882a593Smuzhiyun 	if (rc != 0) {
10029*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10030*4882a593Smuzhiyun 		goto rw_error;
10031*4882a593Smuzhiyun 	}
10032*4882a593Smuzhiyun    /**********/
10033*4882a593Smuzhiyun 	/* SET_ENV */
10034*4882a593Smuzhiyun    /**********/
10035*4882a593Smuzhiyun 	/* set frequency, spectrum inversion and data rate */
10036*4882a593Smuzhiyun 	scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
10037*4882a593Smuzhiyun 	    | SCU_RAM_COMMAND_CMD_DEMOD_SET_ENV;
10038*4882a593Smuzhiyun 	scu_cmd.parameter_len = 3;
10039*4882a593Smuzhiyun 	/* 1-data rate;2-frequency */
10040*4882a593Smuzhiyun 	switch (oob_param->standard) {
10041*4882a593Smuzhiyun 	case DRX_OOB_MODE_A:
10042*4882a593Smuzhiyun 		if (
10043*4882a593Smuzhiyun 			   /* signal is transmitted inverted */
10044*4882a593Smuzhiyun 			   ((oob_param->spectrum_inverted == true) &&
10045*4882a593Smuzhiyun 			    /* and tuner is not mirroring the signal */
10046*4882a593Smuzhiyun 			    (!mirror_freq_spect_oob)) |
10047*4882a593Smuzhiyun 			   /* or */
10048*4882a593Smuzhiyun 			   /* signal is transmitted noninverted */
10049*4882a593Smuzhiyun 			   ((oob_param->spectrum_inverted == false) &&
10050*4882a593Smuzhiyun 			    /* and tuner is mirroring the signal */
10051*4882a593Smuzhiyun 			    (mirror_freq_spect_oob))
10052*4882a593Smuzhiyun 		    )
10053*4882a593Smuzhiyun 			set_param_parameters[0] =
10054*4882a593Smuzhiyun 			    SCU_RAM_ORX_RF_RX_DATA_RATE_2048KBPS_INVSPEC;
10055*4882a593Smuzhiyun 		else
10056*4882a593Smuzhiyun 			set_param_parameters[0] =
10057*4882a593Smuzhiyun 			    SCU_RAM_ORX_RF_RX_DATA_RATE_2048KBPS_REGSPEC;
10058*4882a593Smuzhiyun 		break;
10059*4882a593Smuzhiyun 	case DRX_OOB_MODE_B_GRADE_A:
10060*4882a593Smuzhiyun 		if (
10061*4882a593Smuzhiyun 			   /* signal is transmitted inverted */
10062*4882a593Smuzhiyun 			   ((oob_param->spectrum_inverted == true) &&
10063*4882a593Smuzhiyun 			    /* and tuner is not mirroring the signal */
10064*4882a593Smuzhiyun 			    (!mirror_freq_spect_oob)) |
10065*4882a593Smuzhiyun 			   /* or */
10066*4882a593Smuzhiyun 			   /* signal is transmitted noninverted */
10067*4882a593Smuzhiyun 			   ((oob_param->spectrum_inverted == false) &&
10068*4882a593Smuzhiyun 			    /* and tuner is mirroring the signal */
10069*4882a593Smuzhiyun 			    (mirror_freq_spect_oob))
10070*4882a593Smuzhiyun 		    )
10071*4882a593Smuzhiyun 			set_param_parameters[0] =
10072*4882a593Smuzhiyun 			    SCU_RAM_ORX_RF_RX_DATA_RATE_1544KBPS_INVSPEC;
10073*4882a593Smuzhiyun 		else
10074*4882a593Smuzhiyun 			set_param_parameters[0] =
10075*4882a593Smuzhiyun 			    SCU_RAM_ORX_RF_RX_DATA_RATE_1544KBPS_REGSPEC;
10076*4882a593Smuzhiyun 		break;
10077*4882a593Smuzhiyun 	case DRX_OOB_MODE_B_GRADE_B:
10078*4882a593Smuzhiyun 	default:
10079*4882a593Smuzhiyun 		if (
10080*4882a593Smuzhiyun 			   /* signal is transmitted inverted */
10081*4882a593Smuzhiyun 			   ((oob_param->spectrum_inverted == true) &&
10082*4882a593Smuzhiyun 			    /* and tuner is not mirroring the signal */
10083*4882a593Smuzhiyun 			    (!mirror_freq_spect_oob)) |
10084*4882a593Smuzhiyun 			   /* or */
10085*4882a593Smuzhiyun 			   /* signal is transmitted noninverted */
10086*4882a593Smuzhiyun 			   ((oob_param->spectrum_inverted == false) &&
10087*4882a593Smuzhiyun 			    /* and tuner is mirroring the signal */
10088*4882a593Smuzhiyun 			    (mirror_freq_spect_oob))
10089*4882a593Smuzhiyun 		    )
10090*4882a593Smuzhiyun 			set_param_parameters[0] =
10091*4882a593Smuzhiyun 			    SCU_RAM_ORX_RF_RX_DATA_RATE_3088KBPS_INVSPEC;
10092*4882a593Smuzhiyun 		else
10093*4882a593Smuzhiyun 			set_param_parameters[0] =
10094*4882a593Smuzhiyun 			    SCU_RAM_ORX_RF_RX_DATA_RATE_3088KBPS_REGSPEC;
10095*4882a593Smuzhiyun 		break;
10096*4882a593Smuzhiyun 	}
10097*4882a593Smuzhiyun 	set_param_parameters[1] = (u16) (freq & 0xFFFF);
10098*4882a593Smuzhiyun 	set_param_parameters[2] = trk_filter_value;
10099*4882a593Smuzhiyun 	scu_cmd.parameter = set_param_parameters;
10100*4882a593Smuzhiyun 	scu_cmd.result_len = 1;
10101*4882a593Smuzhiyun 	scu_cmd.result = cmd_result;
10102*4882a593Smuzhiyun 	mode_index = mode_val[(set_param_parameters[0] & 0xC0) >> 6];
10103*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &scu_cmd);
10104*4882a593Smuzhiyun 	if (rc != 0) {
10105*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10106*4882a593Smuzhiyun 		goto rw_error;
10107*4882a593Smuzhiyun 	}
10108*4882a593Smuzhiyun 
10109*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0xFABA, 0);
10110*4882a593Smuzhiyun 	if (rc != 0) {
10111*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10112*4882a593Smuzhiyun 		goto rw_error;
10113*4882a593Smuzhiyun 	}	/*  Write magic word to enable pdr reg write  */
10114*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_OOB_CRX_CFG__A, OOB_CRX_DRIVE_STRENGTH << SIO_PDR_OOB_CRX_CFG_DRIVE__B | 0x03 << SIO_PDR_OOB_CRX_CFG_MODE__B, 0);
10115*4882a593Smuzhiyun 	if (rc != 0) {
10116*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10117*4882a593Smuzhiyun 		goto rw_error;
10118*4882a593Smuzhiyun 	}
10119*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_OOB_DRX_CFG__A, OOB_DRX_DRIVE_STRENGTH << SIO_PDR_OOB_DRX_CFG_DRIVE__B | 0x03 << SIO_PDR_OOB_DRX_CFG_MODE__B, 0);
10120*4882a593Smuzhiyun 	if (rc != 0) {
10121*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10122*4882a593Smuzhiyun 		goto rw_error;
10123*4882a593Smuzhiyun 	}
10124*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
10125*4882a593Smuzhiyun 	if (rc != 0) {
10126*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10127*4882a593Smuzhiyun 		goto rw_error;
10128*4882a593Smuzhiyun 	}	/*  Write magic word to disable pdr reg write */
10129*4882a593Smuzhiyun 
10130*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_TOP_COMM_KEY__A, 0, 0);
10131*4882a593Smuzhiyun 	if (rc != 0) {
10132*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10133*4882a593Smuzhiyun 		goto rw_error;
10134*4882a593Smuzhiyun 	}
10135*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_AAG_LEN_W__A, 16000, 0);
10136*4882a593Smuzhiyun 	if (rc != 0) {
10137*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10138*4882a593Smuzhiyun 		goto rw_error;
10139*4882a593Smuzhiyun 	}
10140*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_AAG_THR_W__A, 40, 0);
10141*4882a593Smuzhiyun 	if (rc != 0) {
10142*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10143*4882a593Smuzhiyun 		goto rw_error;
10144*4882a593Smuzhiyun 	}
10145*4882a593Smuzhiyun 
10146*4882a593Smuzhiyun 	/* ddc */
10147*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_DDC_OFO_SET_W__A, ORX_DDC_OFO_SET_W__PRE, 0);
10148*4882a593Smuzhiyun 	if (rc != 0) {
10149*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10150*4882a593Smuzhiyun 		goto rw_error;
10151*4882a593Smuzhiyun 	}
10152*4882a593Smuzhiyun 
10153*4882a593Smuzhiyun 	/* nsu */
10154*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_NSU_AOX_LOPOW_W__A, ext_attr->oob_lo_pow, 0);
10155*4882a593Smuzhiyun 	if (rc != 0) {
10156*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10157*4882a593Smuzhiyun 		goto rw_error;
10158*4882a593Smuzhiyun 	}
10159*4882a593Smuzhiyun 
10160*4882a593Smuzhiyun 	/* initialization for target mode */
10161*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TARGET_MODE__A, SCU_RAM_ORX_TARGET_MODE_2048KBPS_SQRT, 0);
10162*4882a593Smuzhiyun 	if (rc != 0) {
10163*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10164*4882a593Smuzhiyun 		goto rw_error;
10165*4882a593Smuzhiyun 	}
10166*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FREQ_GAIN_CORR__A, SCU_RAM_ORX_FREQ_GAIN_CORR_2048KBPS, 0);
10167*4882a593Smuzhiyun 	if (rc != 0) {
10168*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10169*4882a593Smuzhiyun 		goto rw_error;
10170*4882a593Smuzhiyun 	}
10171*4882a593Smuzhiyun 
10172*4882a593Smuzhiyun 	/* Reset bits for timing and freq. recovery */
10173*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_CPH__A, 0x0001, 0);
10174*4882a593Smuzhiyun 	if (rc != 0) {
10175*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10176*4882a593Smuzhiyun 		goto rw_error;
10177*4882a593Smuzhiyun 	}
10178*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_CTI__A, 0x0002, 0);
10179*4882a593Smuzhiyun 	if (rc != 0) {
10180*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10181*4882a593Smuzhiyun 		goto rw_error;
10182*4882a593Smuzhiyun 	}
10183*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_KRN__A, 0x0004, 0);
10184*4882a593Smuzhiyun 	if (rc != 0) {
10185*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10186*4882a593Smuzhiyun 		goto rw_error;
10187*4882a593Smuzhiyun 	}
10188*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_KRP__A, 0x0008, 0);
10189*4882a593Smuzhiyun 	if (rc != 0) {
10190*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10191*4882a593Smuzhiyun 		goto rw_error;
10192*4882a593Smuzhiyun 	}
10193*4882a593Smuzhiyun 
10194*4882a593Smuzhiyun 	/* AGN_LOCK = {2048>>3, -2048, 8, -8, 0, 1}; */
10195*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_LOCK_TH__A, 2048 >> 3, 0);
10196*4882a593Smuzhiyun 	if (rc != 0) {
10197*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10198*4882a593Smuzhiyun 		goto rw_error;
10199*4882a593Smuzhiyun 	}
10200*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_LOCK_TOTH__A, (u16)(-2048), 0);
10201*4882a593Smuzhiyun 	if (rc != 0) {
10202*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10203*4882a593Smuzhiyun 		goto rw_error;
10204*4882a593Smuzhiyun 	}
10205*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_ONLOCK_TTH__A, 8, 0);
10206*4882a593Smuzhiyun 	if (rc != 0) {
10207*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10208*4882a593Smuzhiyun 		goto rw_error;
10209*4882a593Smuzhiyun 	}
10210*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_UNLOCK_TTH__A, (u16)(-8), 0);
10211*4882a593Smuzhiyun 	if (rc != 0) {
10212*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10213*4882a593Smuzhiyun 		goto rw_error;
10214*4882a593Smuzhiyun 	}
10215*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_LOCK_MASK__A, 1, 0);
10216*4882a593Smuzhiyun 	if (rc != 0) {
10217*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10218*4882a593Smuzhiyun 		goto rw_error;
10219*4882a593Smuzhiyun 	}
10220*4882a593Smuzhiyun 
10221*4882a593Smuzhiyun 	/* DGN_LOCK = {10, -2048, 8, -8, 0, 1<<1}; */
10222*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_LOCK_TH__A, 10, 0);
10223*4882a593Smuzhiyun 	if (rc != 0) {
10224*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10225*4882a593Smuzhiyun 		goto rw_error;
10226*4882a593Smuzhiyun 	}
10227*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_LOCK_TOTH__A, (u16)(-2048), 0);
10228*4882a593Smuzhiyun 	if (rc != 0) {
10229*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10230*4882a593Smuzhiyun 		goto rw_error;
10231*4882a593Smuzhiyun 	}
10232*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_ONLOCK_TTH__A, 8, 0);
10233*4882a593Smuzhiyun 	if (rc != 0) {
10234*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10235*4882a593Smuzhiyun 		goto rw_error;
10236*4882a593Smuzhiyun 	}
10237*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_UNLOCK_TTH__A, (u16)(-8), 0);
10238*4882a593Smuzhiyun 	if (rc != 0) {
10239*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10240*4882a593Smuzhiyun 		goto rw_error;
10241*4882a593Smuzhiyun 	}
10242*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_LOCK_MASK__A, 1 << 1, 0);
10243*4882a593Smuzhiyun 	if (rc != 0) {
10244*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10245*4882a593Smuzhiyun 		goto rw_error;
10246*4882a593Smuzhiyun 	}
10247*4882a593Smuzhiyun 
10248*4882a593Smuzhiyun 	/* FRQ_LOCK = {15,-2048, 8, -8, 0, 1<<2}; */
10249*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_LOCK_TH__A, 17, 0);
10250*4882a593Smuzhiyun 	if (rc != 0) {
10251*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10252*4882a593Smuzhiyun 		goto rw_error;
10253*4882a593Smuzhiyun 	}
10254*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_LOCK_TOTH__A, (u16)(-2048), 0);
10255*4882a593Smuzhiyun 	if (rc != 0) {
10256*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10257*4882a593Smuzhiyun 		goto rw_error;
10258*4882a593Smuzhiyun 	}
10259*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_ONLOCK_TTH__A, 8, 0);
10260*4882a593Smuzhiyun 	if (rc != 0) {
10261*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10262*4882a593Smuzhiyun 		goto rw_error;
10263*4882a593Smuzhiyun 	}
10264*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_UNLOCK_TTH__A, (u16)(-8), 0);
10265*4882a593Smuzhiyun 	if (rc != 0) {
10266*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10267*4882a593Smuzhiyun 		goto rw_error;
10268*4882a593Smuzhiyun 	}
10269*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_LOCK_MASK__A, 1 << 2, 0);
10270*4882a593Smuzhiyun 	if (rc != 0) {
10271*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10272*4882a593Smuzhiyun 		goto rw_error;
10273*4882a593Smuzhiyun 	}
10274*4882a593Smuzhiyun 
10275*4882a593Smuzhiyun 	/* PHA_LOCK = {5000, -2048, 8, -8, 0, 1<<3}; */
10276*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_LOCK_TH__A, 3000, 0);
10277*4882a593Smuzhiyun 	if (rc != 0) {
10278*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10279*4882a593Smuzhiyun 		goto rw_error;
10280*4882a593Smuzhiyun 	}
10281*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_LOCK_TOTH__A, (u16)(-2048), 0);
10282*4882a593Smuzhiyun 	if (rc != 0) {
10283*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10284*4882a593Smuzhiyun 		goto rw_error;
10285*4882a593Smuzhiyun 	}
10286*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_ONLOCK_TTH__A, 8, 0);
10287*4882a593Smuzhiyun 	if (rc != 0) {
10288*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10289*4882a593Smuzhiyun 		goto rw_error;
10290*4882a593Smuzhiyun 	}
10291*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_UNLOCK_TTH__A, (u16)(-8), 0);
10292*4882a593Smuzhiyun 	if (rc != 0) {
10293*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10294*4882a593Smuzhiyun 		goto rw_error;
10295*4882a593Smuzhiyun 	}
10296*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_LOCK_MASK__A, 1 << 3, 0);
10297*4882a593Smuzhiyun 	if (rc != 0) {
10298*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10299*4882a593Smuzhiyun 		goto rw_error;
10300*4882a593Smuzhiyun 	}
10301*4882a593Smuzhiyun 
10302*4882a593Smuzhiyun 	/* TIM_LOCK = {300,      -2048, 8, -8, 0, 1<<4}; */
10303*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_LOCK_TH__A, 400, 0);
10304*4882a593Smuzhiyun 	if (rc != 0) {
10305*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10306*4882a593Smuzhiyun 		goto rw_error;
10307*4882a593Smuzhiyun 	}
10308*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_LOCK_TOTH__A, (u16)(-2048), 0);
10309*4882a593Smuzhiyun 	if (rc != 0) {
10310*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10311*4882a593Smuzhiyun 		goto rw_error;
10312*4882a593Smuzhiyun 	}
10313*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_ONLOCK_TTH__A, 8, 0);
10314*4882a593Smuzhiyun 	if (rc != 0) {
10315*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10316*4882a593Smuzhiyun 		goto rw_error;
10317*4882a593Smuzhiyun 	}
10318*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_UNLOCK_TTH__A, (u16)(-8), 0);
10319*4882a593Smuzhiyun 	if (rc != 0) {
10320*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10321*4882a593Smuzhiyun 		goto rw_error;
10322*4882a593Smuzhiyun 	}
10323*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_LOCK_MASK__A, 1 << 4, 0);
10324*4882a593Smuzhiyun 	if (rc != 0) {
10325*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10326*4882a593Smuzhiyun 		goto rw_error;
10327*4882a593Smuzhiyun 	}
10328*4882a593Smuzhiyun 
10329*4882a593Smuzhiyun 	/* EQU_LOCK = {20,      -2048, 8, -8, 0, 1<<5}; */
10330*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_LOCK_TH__A, 20, 0);
10331*4882a593Smuzhiyun 	if (rc != 0) {
10332*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10333*4882a593Smuzhiyun 		goto rw_error;
10334*4882a593Smuzhiyun 	}
10335*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_LOCK_TOTH__A, (u16)(-2048), 0);
10336*4882a593Smuzhiyun 	if (rc != 0) {
10337*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10338*4882a593Smuzhiyun 		goto rw_error;
10339*4882a593Smuzhiyun 	}
10340*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_ONLOCK_TTH__A, 4, 0);
10341*4882a593Smuzhiyun 	if (rc != 0) {
10342*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10343*4882a593Smuzhiyun 		goto rw_error;
10344*4882a593Smuzhiyun 	}
10345*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_UNLOCK_TTH__A, (u16)(-4), 0);
10346*4882a593Smuzhiyun 	if (rc != 0) {
10347*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10348*4882a593Smuzhiyun 		goto rw_error;
10349*4882a593Smuzhiyun 	}
10350*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_LOCK_MASK__A, 1 << 5, 0);
10351*4882a593Smuzhiyun 	if (rc != 0) {
10352*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10353*4882a593Smuzhiyun 		goto rw_error;
10354*4882a593Smuzhiyun 	}
10355*4882a593Smuzhiyun 
10356*4882a593Smuzhiyun 	/* PRE-Filter coefficients (PFI) */
10357*4882a593Smuzhiyun 	rc = drxdap_fasi_write_block(dev_addr, ORX_FWP_PFI_A_W__A, sizeof(pfi_coeffs[mode_index]), ((u8 *)pfi_coeffs[mode_index]), 0);
10358*4882a593Smuzhiyun 	if (rc != 0) {
10359*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10360*4882a593Smuzhiyun 		goto rw_error;
10361*4882a593Smuzhiyun 	}
10362*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_TOP_MDE_W__A, mode_index, 0);
10363*4882a593Smuzhiyun 	if (rc != 0) {
10364*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10365*4882a593Smuzhiyun 		goto rw_error;
10366*4882a593Smuzhiyun 	}
10367*4882a593Smuzhiyun 
10368*4882a593Smuzhiyun 	/* NYQUIST-Filter coefficients (NYQ) */
10369*4882a593Smuzhiyun 	for (i = 0; i < (NYQFILTERLEN + 1) / 2; i++) {
10370*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_NYQ_ADR_W__A, i, 0);
10371*4882a593Smuzhiyun 		if (rc != 0) {
10372*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10373*4882a593Smuzhiyun 			goto rw_error;
10374*4882a593Smuzhiyun 		}
10375*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_NYQ_COF_RW__A, nyquist_coeffs[mode_index][i], 0);
10376*4882a593Smuzhiyun 		if (rc != 0) {
10377*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10378*4882a593Smuzhiyun 			goto rw_error;
10379*4882a593Smuzhiyun 		}
10380*4882a593Smuzhiyun 	}
10381*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_NYQ_ADR_W__A, 31, 0);
10382*4882a593Smuzhiyun 	if (rc != 0) {
10383*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10384*4882a593Smuzhiyun 		goto rw_error;
10385*4882a593Smuzhiyun 	}
10386*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_COMM_EXEC__A, ORX_COMM_EXEC_ACTIVE, 0);
10387*4882a593Smuzhiyun 	if (rc != 0) {
10388*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10389*4882a593Smuzhiyun 		goto rw_error;
10390*4882a593Smuzhiyun 	}
10391*4882a593Smuzhiyun 	/********/
10392*4882a593Smuzhiyun 	/* Start */
10393*4882a593Smuzhiyun 	/********/
10394*4882a593Smuzhiyun 	scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
10395*4882a593Smuzhiyun 	    | SCU_RAM_COMMAND_CMD_DEMOD_START;
10396*4882a593Smuzhiyun 	scu_cmd.parameter_len = 0;
10397*4882a593Smuzhiyun 	scu_cmd.result_len = 1;
10398*4882a593Smuzhiyun 	scu_cmd.result = cmd_result;
10399*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &scu_cmd);
10400*4882a593Smuzhiyun 	if (rc != 0) {
10401*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10402*4882a593Smuzhiyun 		goto rw_error;
10403*4882a593Smuzhiyun 	}
10404*4882a593Smuzhiyun 
10405*4882a593Smuzhiyun 	rc = set_orx_nsu_aox(demod, true);
10406*4882a593Smuzhiyun 	if (rc != 0) {
10407*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10408*4882a593Smuzhiyun 		goto rw_error;
10409*4882a593Smuzhiyun 	}
10410*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ORX_NSU_AOX_STHR_W__A, ext_attr->oob_pre_saw, 0);
10411*4882a593Smuzhiyun 	if (rc != 0) {
10412*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10413*4882a593Smuzhiyun 		goto rw_error;
10414*4882a593Smuzhiyun 	}
10415*4882a593Smuzhiyun 
10416*4882a593Smuzhiyun 	ext_attr->oob_power_on = true;
10417*4882a593Smuzhiyun 
10418*4882a593Smuzhiyun 	return 0;
10419*4882a593Smuzhiyun rw_error:
10420*4882a593Smuzhiyun 	return rc;
10421*4882a593Smuzhiyun }
10422*4882a593Smuzhiyun 
10423*4882a593Smuzhiyun /*============================================================================*/
10424*4882a593Smuzhiyun /*==                     END OOB DATAPATH FUNCTIONS                         ==*/
10425*4882a593Smuzhiyun /*============================================================================*/
10426*4882a593Smuzhiyun 
10427*4882a593Smuzhiyun /*=============================================================================
10428*4882a593Smuzhiyun   ===== MC command related functions ==========================================
10429*4882a593Smuzhiyun   ===========================================================================*/
10430*4882a593Smuzhiyun 
10431*4882a593Smuzhiyun /*=============================================================================
10432*4882a593Smuzhiyun   ===== ctrl_set_channel() ==========================================================
10433*4882a593Smuzhiyun   ===========================================================================*/
10434*4882a593Smuzhiyun /*
10435*4882a593Smuzhiyun * \fn int ctrl_set_channel()
10436*4882a593Smuzhiyun * \brief Select a new transmission channel.
10437*4882a593Smuzhiyun * \param demod instance of demod.
10438*4882a593Smuzhiyun * \param channel Pointer to channel data.
10439*4882a593Smuzhiyun * \return int.
10440*4882a593Smuzhiyun *
10441*4882a593Smuzhiyun * In case the tuner module is not used and in case of NTSC/FM the pogrammer
10442*4882a593Smuzhiyun * must tune the tuner to the centre frequency of the NTSC/FM channel.
10443*4882a593Smuzhiyun *
10444*4882a593Smuzhiyun */
10445*4882a593Smuzhiyun static int
ctrl_set_channel(struct drx_demod_instance * demod,struct drx_channel * channel)10446*4882a593Smuzhiyun ctrl_set_channel(struct drx_demod_instance *demod, struct drx_channel *channel)
10447*4882a593Smuzhiyun {
10448*4882a593Smuzhiyun 	int rc;
10449*4882a593Smuzhiyun 	s32 tuner_freq_offset = 0;
10450*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
10451*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
10452*4882a593Smuzhiyun 	enum drx_standard standard = DRX_STANDARD_UNKNOWN;
10453*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10454*4882a593Smuzhiyun 	u32 min_symbol_rate = 0;
10455*4882a593Smuzhiyun 	u32 max_symbol_rate = 0;
10456*4882a593Smuzhiyun 	int bandwidth_temp = 0;
10457*4882a593Smuzhiyun 	int bandwidth = 0;
10458*4882a593Smuzhiyun #endif
10459*4882a593Smuzhiyun    /*== check arguments ======================================================*/
10460*4882a593Smuzhiyun 	if ((demod == NULL) || (channel == NULL))
10461*4882a593Smuzhiyun 		return -EINVAL;
10462*4882a593Smuzhiyun 
10463*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
10464*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
10465*4882a593Smuzhiyun 	standard = ext_attr->standard;
10466*4882a593Smuzhiyun 
10467*4882a593Smuzhiyun 	/* check valid standards */
10468*4882a593Smuzhiyun 	switch (standard) {
10469*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
10470*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10471*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
10472*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
10473*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
10474*4882a593Smuzhiyun #endif /* DRXJ_VSB_ONLY */
10475*4882a593Smuzhiyun 		break;
10476*4882a593Smuzhiyun 	case DRX_STANDARD_UNKNOWN:
10477*4882a593Smuzhiyun 	default:
10478*4882a593Smuzhiyun 		return -EINVAL;
10479*4882a593Smuzhiyun 	}
10480*4882a593Smuzhiyun 
10481*4882a593Smuzhiyun 	/* check bandwidth QAM annex B, NTSC and 8VSB */
10482*4882a593Smuzhiyun 	if ((standard == DRX_STANDARD_ITU_B) ||
10483*4882a593Smuzhiyun 	    (standard == DRX_STANDARD_8VSB) ||
10484*4882a593Smuzhiyun 	    (standard == DRX_STANDARD_NTSC)) {
10485*4882a593Smuzhiyun 		switch (channel->bandwidth) {
10486*4882a593Smuzhiyun 		case DRX_BANDWIDTH_6MHZ:
10487*4882a593Smuzhiyun 		case DRX_BANDWIDTH_UNKNOWN:
10488*4882a593Smuzhiyun 			channel->bandwidth = DRX_BANDWIDTH_6MHZ;
10489*4882a593Smuzhiyun 			break;
10490*4882a593Smuzhiyun 		case DRX_BANDWIDTH_8MHZ:
10491*4882a593Smuzhiyun 		case DRX_BANDWIDTH_7MHZ:
10492*4882a593Smuzhiyun 		default:
10493*4882a593Smuzhiyun 			return -EINVAL;
10494*4882a593Smuzhiyun 		}
10495*4882a593Smuzhiyun 	}
10496*4882a593Smuzhiyun 
10497*4882a593Smuzhiyun 	/* For QAM annex A and annex C:
10498*4882a593Smuzhiyun 	   -check symbolrate and constellation
10499*4882a593Smuzhiyun 	   -derive bandwidth from symbolrate (input bandwidth is ignored)
10500*4882a593Smuzhiyun 	 */
10501*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10502*4882a593Smuzhiyun 	if ((standard == DRX_STANDARD_ITU_A) ||
10503*4882a593Smuzhiyun 	    (standard == DRX_STANDARD_ITU_C)) {
10504*4882a593Smuzhiyun 		struct drxuio_cfg uio_cfg = { DRX_UIO1, DRX_UIO_MODE_FIRMWARE_SAW };
10505*4882a593Smuzhiyun 		int bw_rolloff_factor = 0;
10506*4882a593Smuzhiyun 
10507*4882a593Smuzhiyun 		bw_rolloff_factor = (standard == DRX_STANDARD_ITU_A) ? 115 : 113;
10508*4882a593Smuzhiyun 		min_symbol_rate = DRXJ_QAM_SYMBOLRATE_MIN;
10509*4882a593Smuzhiyun 		max_symbol_rate = DRXJ_QAM_SYMBOLRATE_MAX;
10510*4882a593Smuzhiyun 		/* config SMA_TX pin to SAW switch mode */
10511*4882a593Smuzhiyun 		rc = ctrl_set_uio_cfg(demod, &uio_cfg);
10512*4882a593Smuzhiyun 		if (rc != 0) {
10513*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10514*4882a593Smuzhiyun 			goto rw_error;
10515*4882a593Smuzhiyun 		}
10516*4882a593Smuzhiyun 
10517*4882a593Smuzhiyun 		if (channel->symbolrate < min_symbol_rate ||
10518*4882a593Smuzhiyun 		    channel->symbolrate > max_symbol_rate) {
10519*4882a593Smuzhiyun 			return -EINVAL;
10520*4882a593Smuzhiyun 		}
10521*4882a593Smuzhiyun 
10522*4882a593Smuzhiyun 		switch (channel->constellation) {
10523*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM16:
10524*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM32:
10525*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM64:
10526*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM128:
10527*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM256:
10528*4882a593Smuzhiyun 			bandwidth_temp = channel->symbolrate * bw_rolloff_factor;
10529*4882a593Smuzhiyun 			bandwidth = bandwidth_temp / 100;
10530*4882a593Smuzhiyun 
10531*4882a593Smuzhiyun 			if ((bandwidth_temp % 100) >= 50)
10532*4882a593Smuzhiyun 				bandwidth++;
10533*4882a593Smuzhiyun 
10534*4882a593Smuzhiyun 			if (bandwidth <= 6100000) {
10535*4882a593Smuzhiyun 				channel->bandwidth = DRX_BANDWIDTH_6MHZ;
10536*4882a593Smuzhiyun 			} else if ((bandwidth > 6100000)
10537*4882a593Smuzhiyun 				   && (bandwidth <= 7100000)) {
10538*4882a593Smuzhiyun 				channel->bandwidth = DRX_BANDWIDTH_7MHZ;
10539*4882a593Smuzhiyun 			} else if (bandwidth > 7100000) {
10540*4882a593Smuzhiyun 				channel->bandwidth = DRX_BANDWIDTH_8MHZ;
10541*4882a593Smuzhiyun 			}
10542*4882a593Smuzhiyun 			break;
10543*4882a593Smuzhiyun 		default:
10544*4882a593Smuzhiyun 			return -EINVAL;
10545*4882a593Smuzhiyun 		}
10546*4882a593Smuzhiyun 	}
10547*4882a593Smuzhiyun 
10548*4882a593Smuzhiyun 	/* For QAM annex B:
10549*4882a593Smuzhiyun 	   -check constellation
10550*4882a593Smuzhiyun 	 */
10551*4882a593Smuzhiyun 	if (standard == DRX_STANDARD_ITU_B) {
10552*4882a593Smuzhiyun 		switch (channel->constellation) {
10553*4882a593Smuzhiyun 		case DRX_CONSTELLATION_AUTO:
10554*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM256:
10555*4882a593Smuzhiyun 		case DRX_CONSTELLATION_QAM64:
10556*4882a593Smuzhiyun 			break;
10557*4882a593Smuzhiyun 		default:
10558*4882a593Smuzhiyun 			return -EINVAL;
10559*4882a593Smuzhiyun 		}
10560*4882a593Smuzhiyun 
10561*4882a593Smuzhiyun 		switch (channel->interleavemode) {
10562*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J1:
10563*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J1_V2:
10564*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J2:
10565*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I64_J2:
10566*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J3:
10567*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I32_J4:
10568*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J4:
10569*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I16_J8:
10570*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J5:
10571*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I8_J16:
10572*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J6:
10573*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J7:
10574*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I128_J8:
10575*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I12_J17:
10576*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_I5_J4:
10577*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_B52_M240:
10578*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_B52_M720:
10579*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_UNKNOWN:
10580*4882a593Smuzhiyun 		case DRX_INTERLEAVEMODE_AUTO:
10581*4882a593Smuzhiyun 			break;
10582*4882a593Smuzhiyun 		default:
10583*4882a593Smuzhiyun 			return -EINVAL;
10584*4882a593Smuzhiyun 		}
10585*4882a593Smuzhiyun 	}
10586*4882a593Smuzhiyun 
10587*4882a593Smuzhiyun 	if ((ext_attr->uio_sma_tx_mode) == DRX_UIO_MODE_FIRMWARE_SAW) {
10588*4882a593Smuzhiyun 		/* SAW SW, user UIO is used for switchable SAW */
10589*4882a593Smuzhiyun 		struct drxuio_data uio1 = { DRX_UIO1, false };
10590*4882a593Smuzhiyun 
10591*4882a593Smuzhiyun 		switch (channel->bandwidth) {
10592*4882a593Smuzhiyun 		case DRX_BANDWIDTH_8MHZ:
10593*4882a593Smuzhiyun 			uio1.value = true;
10594*4882a593Smuzhiyun 			break;
10595*4882a593Smuzhiyun 		case DRX_BANDWIDTH_7MHZ:
10596*4882a593Smuzhiyun 			uio1.value = false;
10597*4882a593Smuzhiyun 			break;
10598*4882a593Smuzhiyun 		case DRX_BANDWIDTH_6MHZ:
10599*4882a593Smuzhiyun 			uio1.value = false;
10600*4882a593Smuzhiyun 			break;
10601*4882a593Smuzhiyun 		case DRX_BANDWIDTH_UNKNOWN:
10602*4882a593Smuzhiyun 		default:
10603*4882a593Smuzhiyun 			return -EINVAL;
10604*4882a593Smuzhiyun 		}
10605*4882a593Smuzhiyun 
10606*4882a593Smuzhiyun 		rc = ctrl_uio_write(demod, &uio1);
10607*4882a593Smuzhiyun 		if (rc != 0) {
10608*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10609*4882a593Smuzhiyun 			goto rw_error;
10610*4882a593Smuzhiyun 		}
10611*4882a593Smuzhiyun 	}
10612*4882a593Smuzhiyun #endif /* DRXJ_VSB_ONLY */
10613*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_ACTIVE, 0);
10614*4882a593Smuzhiyun 	if (rc != 0) {
10615*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10616*4882a593Smuzhiyun 		goto rw_error;
10617*4882a593Smuzhiyun 	}
10618*4882a593Smuzhiyun 
10619*4882a593Smuzhiyun 	tuner_freq_offset = 0;
10620*4882a593Smuzhiyun 
10621*4882a593Smuzhiyun    /*== Setup demod for specific standard ====================================*/
10622*4882a593Smuzhiyun 	switch (standard) {
10623*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
10624*4882a593Smuzhiyun 		if (channel->mirror == DRX_MIRROR_AUTO)
10625*4882a593Smuzhiyun 			ext_attr->mirror = DRX_MIRROR_NO;
10626*4882a593Smuzhiyun 		else
10627*4882a593Smuzhiyun 			ext_attr->mirror = channel->mirror;
10628*4882a593Smuzhiyun 		rc = set_vsb(demod);
10629*4882a593Smuzhiyun 		if (rc != 0) {
10630*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10631*4882a593Smuzhiyun 			goto rw_error;
10632*4882a593Smuzhiyun 		}
10633*4882a593Smuzhiyun 		rc = set_frequency(demod, channel, tuner_freq_offset);
10634*4882a593Smuzhiyun 		if (rc != 0) {
10635*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10636*4882a593Smuzhiyun 			goto rw_error;
10637*4882a593Smuzhiyun 		}
10638*4882a593Smuzhiyun 		break;
10639*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10640*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
10641*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
10642*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
10643*4882a593Smuzhiyun 		rc = set_qam_channel(demod, channel, tuner_freq_offset);
10644*4882a593Smuzhiyun 		if (rc != 0) {
10645*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10646*4882a593Smuzhiyun 			goto rw_error;
10647*4882a593Smuzhiyun 		}
10648*4882a593Smuzhiyun 		break;
10649*4882a593Smuzhiyun #endif
10650*4882a593Smuzhiyun 	case DRX_STANDARD_UNKNOWN:
10651*4882a593Smuzhiyun 	default:
10652*4882a593Smuzhiyun 		return -EIO;
10653*4882a593Smuzhiyun 	}
10654*4882a593Smuzhiyun 
10655*4882a593Smuzhiyun 	/* flag the packet error counter reset */
10656*4882a593Smuzhiyun 	ext_attr->reset_pkt_err_acc = true;
10657*4882a593Smuzhiyun 
10658*4882a593Smuzhiyun 	return 0;
10659*4882a593Smuzhiyun rw_error:
10660*4882a593Smuzhiyun 	return rc;
10661*4882a593Smuzhiyun }
10662*4882a593Smuzhiyun 
10663*4882a593Smuzhiyun /*=============================================================================
10664*4882a593Smuzhiyun   ===== SigQuality() ==========================================================
10665*4882a593Smuzhiyun   ===========================================================================*/
10666*4882a593Smuzhiyun 
10667*4882a593Smuzhiyun /*
10668*4882a593Smuzhiyun * \fn int ctrl_sig_quality()
10669*4882a593Smuzhiyun * \brief Retrieve signal quality form device.
10670*4882a593Smuzhiyun * \param devmod Pointer to demodulator instance.
10671*4882a593Smuzhiyun * \param sig_quality Pointer to signal quality data.
10672*4882a593Smuzhiyun * \return int.
10673*4882a593Smuzhiyun * \retval 0 sig_quality contains valid data.
10674*4882a593Smuzhiyun * \retval -EINVAL sig_quality is NULL.
10675*4882a593Smuzhiyun * \retval -EIO Erroneous data, sig_quality contains invalid data.
10676*4882a593Smuzhiyun 
10677*4882a593Smuzhiyun */
10678*4882a593Smuzhiyun static int
ctrl_sig_quality(struct drx_demod_instance * demod,enum drx_lock_status lock_status)10679*4882a593Smuzhiyun ctrl_sig_quality(struct drx_demod_instance *demod,
10680*4882a593Smuzhiyun 		 enum drx_lock_status lock_status)
10681*4882a593Smuzhiyun {
10682*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
10683*4882a593Smuzhiyun 	struct drxj_data *ext_attr = demod->my_ext_attr;
10684*4882a593Smuzhiyun 	struct drx39xxj_state *state = dev_addr->user_data;
10685*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
10686*4882a593Smuzhiyun 	enum drx_standard standard = ext_attr->standard;
10687*4882a593Smuzhiyun 	int rc;
10688*4882a593Smuzhiyun 	u32 ber, cnt, err, pkt;
10689*4882a593Smuzhiyun 	u16 mer, strength = 0;
10690*4882a593Smuzhiyun 
10691*4882a593Smuzhiyun 	rc = get_sig_strength(demod, &strength);
10692*4882a593Smuzhiyun 	if (rc < 0) {
10693*4882a593Smuzhiyun 		pr_err("error getting signal strength %d\n", rc);
10694*4882a593Smuzhiyun 		p->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10695*4882a593Smuzhiyun 	} else {
10696*4882a593Smuzhiyun 		p->strength.stat[0].scale = FE_SCALE_RELATIVE;
10697*4882a593Smuzhiyun 		p->strength.stat[0].uvalue = 65535UL *  strength/ 100;
10698*4882a593Smuzhiyun 	}
10699*4882a593Smuzhiyun 
10700*4882a593Smuzhiyun 	switch (standard) {
10701*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
10702*4882a593Smuzhiyun #ifdef DRXJ_SIGNAL_ACCUM_ERR
10703*4882a593Smuzhiyun 		rc = get_acc_pkt_err(demod, &pkt);
10704*4882a593Smuzhiyun 		if (rc != 0) {
10705*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10706*4882a593Smuzhiyun 			goto rw_error;
10707*4882a593Smuzhiyun 		}
10708*4882a593Smuzhiyun #endif
10709*4882a593Smuzhiyun 		if (lock_status != DRXJ_DEMOD_LOCK && lock_status != DRX_LOCKED) {
10710*4882a593Smuzhiyun 			p->pre_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10711*4882a593Smuzhiyun 			p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10712*4882a593Smuzhiyun 			p->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10713*4882a593Smuzhiyun 			p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10714*4882a593Smuzhiyun 			p->block_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10715*4882a593Smuzhiyun 			p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10716*4882a593Smuzhiyun 			p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10717*4882a593Smuzhiyun 		} else {
10718*4882a593Smuzhiyun 			rc = get_vsb_post_rs_pck_err(dev_addr, &err, &pkt);
10719*4882a593Smuzhiyun 			if (rc != 0) {
10720*4882a593Smuzhiyun 				pr_err("error %d getting UCB\n", rc);
10721*4882a593Smuzhiyun 				p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10722*4882a593Smuzhiyun 			} else {
10723*4882a593Smuzhiyun 				p->block_error.stat[0].scale = FE_SCALE_COUNTER;
10724*4882a593Smuzhiyun 				p->block_error.stat[0].uvalue += err;
10725*4882a593Smuzhiyun 				p->block_count.stat[0].scale = FE_SCALE_COUNTER;
10726*4882a593Smuzhiyun 				p->block_count.stat[0].uvalue += pkt;
10727*4882a593Smuzhiyun 			}
10728*4882a593Smuzhiyun 
10729*4882a593Smuzhiyun 			/* PostViterbi is compute in steps of 10^(-6) */
10730*4882a593Smuzhiyun 			rc = get_vs_bpre_viterbi_ber(dev_addr, &ber, &cnt);
10731*4882a593Smuzhiyun 			if (rc != 0) {
10732*4882a593Smuzhiyun 				pr_err("error %d getting pre-ber\n", rc);
10733*4882a593Smuzhiyun 				p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10734*4882a593Smuzhiyun 			} else {
10735*4882a593Smuzhiyun 				p->pre_bit_error.stat[0].scale = FE_SCALE_COUNTER;
10736*4882a593Smuzhiyun 				p->pre_bit_error.stat[0].uvalue += ber;
10737*4882a593Smuzhiyun 				p->pre_bit_count.stat[0].scale = FE_SCALE_COUNTER;
10738*4882a593Smuzhiyun 				p->pre_bit_count.stat[0].uvalue += cnt;
10739*4882a593Smuzhiyun 			}
10740*4882a593Smuzhiyun 
10741*4882a593Smuzhiyun 			rc = get_vs_bpost_viterbi_ber(dev_addr, &ber, &cnt);
10742*4882a593Smuzhiyun 			if (rc != 0) {
10743*4882a593Smuzhiyun 				pr_err("error %d getting post-ber\n", rc);
10744*4882a593Smuzhiyun 				p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10745*4882a593Smuzhiyun 			} else {
10746*4882a593Smuzhiyun 				p->post_bit_error.stat[0].scale = FE_SCALE_COUNTER;
10747*4882a593Smuzhiyun 				p->post_bit_error.stat[0].uvalue += ber;
10748*4882a593Smuzhiyun 				p->post_bit_count.stat[0].scale = FE_SCALE_COUNTER;
10749*4882a593Smuzhiyun 				p->post_bit_count.stat[0].uvalue += cnt;
10750*4882a593Smuzhiyun 			}
10751*4882a593Smuzhiyun 			rc = get_vsbmer(dev_addr, &mer);
10752*4882a593Smuzhiyun 			if (rc != 0) {
10753*4882a593Smuzhiyun 				pr_err("error %d getting MER\n", rc);
10754*4882a593Smuzhiyun 				p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
10755*4882a593Smuzhiyun 			} else {
10756*4882a593Smuzhiyun 				p->cnr.stat[0].svalue = mer * 100;
10757*4882a593Smuzhiyun 				p->cnr.stat[0].scale = FE_SCALE_DECIBEL;
10758*4882a593Smuzhiyun 			}
10759*4882a593Smuzhiyun 		}
10760*4882a593Smuzhiyun 		break;
10761*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10762*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
10763*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
10764*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
10765*4882a593Smuzhiyun 		rc = ctrl_get_qam_sig_quality(demod);
10766*4882a593Smuzhiyun 		if (rc != 0) {
10767*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10768*4882a593Smuzhiyun 			goto rw_error;
10769*4882a593Smuzhiyun 		}
10770*4882a593Smuzhiyun 		break;
10771*4882a593Smuzhiyun #endif
10772*4882a593Smuzhiyun 	default:
10773*4882a593Smuzhiyun 		return -EIO;
10774*4882a593Smuzhiyun 	}
10775*4882a593Smuzhiyun 
10776*4882a593Smuzhiyun 	return 0;
10777*4882a593Smuzhiyun rw_error:
10778*4882a593Smuzhiyun 	return rc;
10779*4882a593Smuzhiyun }
10780*4882a593Smuzhiyun 
10781*4882a593Smuzhiyun /*============================================================================*/
10782*4882a593Smuzhiyun 
10783*4882a593Smuzhiyun /*
10784*4882a593Smuzhiyun * \fn int ctrl_lock_status()
10785*4882a593Smuzhiyun * \brief Retrieve lock status .
10786*4882a593Smuzhiyun * \param dev_addr Pointer to demodulator device address.
10787*4882a593Smuzhiyun * \param lock_stat Pointer to lock status structure.
10788*4882a593Smuzhiyun * \return int.
10789*4882a593Smuzhiyun *
10790*4882a593Smuzhiyun */
10791*4882a593Smuzhiyun static int
ctrl_lock_status(struct drx_demod_instance * demod,enum drx_lock_status * lock_stat)10792*4882a593Smuzhiyun ctrl_lock_status(struct drx_demod_instance *demod, enum drx_lock_status *lock_stat)
10793*4882a593Smuzhiyun {
10794*4882a593Smuzhiyun 	enum drx_standard standard = DRX_STANDARD_UNKNOWN;
10795*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
10796*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
10797*4882a593Smuzhiyun 	struct drxjscu_cmd cmd_scu = { /* command      */ 0,
10798*4882a593Smuzhiyun 		/* parameter_len */ 0,
10799*4882a593Smuzhiyun 		/* result_len    */ 0,
10800*4882a593Smuzhiyun 		/* *parameter   */ NULL,
10801*4882a593Smuzhiyun 		/* *result      */ NULL
10802*4882a593Smuzhiyun 	};
10803*4882a593Smuzhiyun 	int rc;
10804*4882a593Smuzhiyun 	u16 cmd_result[2] = { 0, 0 };
10805*4882a593Smuzhiyun 	u16 demod_lock = SCU_RAM_PARAM_1_RES_DEMOD_GET_LOCK_DEMOD_LOCKED;
10806*4882a593Smuzhiyun 
10807*4882a593Smuzhiyun 	/* check arguments */
10808*4882a593Smuzhiyun 	if ((demod == NULL) || (lock_stat == NULL))
10809*4882a593Smuzhiyun 		return -EINVAL;
10810*4882a593Smuzhiyun 
10811*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
10812*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
10813*4882a593Smuzhiyun 	standard = ext_attr->standard;
10814*4882a593Smuzhiyun 
10815*4882a593Smuzhiyun 	*lock_stat = DRX_NOT_LOCKED;
10816*4882a593Smuzhiyun 
10817*4882a593Smuzhiyun 	/* define the SCU command code */
10818*4882a593Smuzhiyun 	switch (standard) {
10819*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
10820*4882a593Smuzhiyun 		cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB |
10821*4882a593Smuzhiyun 		    SCU_RAM_COMMAND_CMD_DEMOD_GET_LOCK;
10822*4882a593Smuzhiyun 		demod_lock |= 0x6;
10823*4882a593Smuzhiyun 		break;
10824*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10825*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
10826*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
10827*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
10828*4882a593Smuzhiyun 		cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
10829*4882a593Smuzhiyun 		    SCU_RAM_COMMAND_CMD_DEMOD_GET_LOCK;
10830*4882a593Smuzhiyun 		break;
10831*4882a593Smuzhiyun #endif
10832*4882a593Smuzhiyun 	case DRX_STANDARD_UNKNOWN:
10833*4882a593Smuzhiyun 	default:
10834*4882a593Smuzhiyun 		return -EIO;
10835*4882a593Smuzhiyun 	}
10836*4882a593Smuzhiyun 
10837*4882a593Smuzhiyun 	/* define the SCU command parameters and execute the command */
10838*4882a593Smuzhiyun 	cmd_scu.parameter_len = 0;
10839*4882a593Smuzhiyun 	cmd_scu.result_len = 2;
10840*4882a593Smuzhiyun 	cmd_scu.parameter = NULL;
10841*4882a593Smuzhiyun 	cmd_scu.result = cmd_result;
10842*4882a593Smuzhiyun 	rc = scu_command(dev_addr, &cmd_scu);
10843*4882a593Smuzhiyun 	if (rc != 0) {
10844*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
10845*4882a593Smuzhiyun 		goto rw_error;
10846*4882a593Smuzhiyun 	}
10847*4882a593Smuzhiyun 
10848*4882a593Smuzhiyun 	/* set the lock status */
10849*4882a593Smuzhiyun 	if (cmd_scu.result[1] < demod_lock) {
10850*4882a593Smuzhiyun 		/* 0x0000 NOT LOCKED */
10851*4882a593Smuzhiyun 		*lock_stat = DRX_NOT_LOCKED;
10852*4882a593Smuzhiyun 	} else if (cmd_scu.result[1] < SCU_RAM_PARAM_1_RES_DEMOD_GET_LOCK_LOCKED) {
10853*4882a593Smuzhiyun 		*lock_stat = DRXJ_DEMOD_LOCK;
10854*4882a593Smuzhiyun 	} else if (cmd_scu.result[1] <
10855*4882a593Smuzhiyun 		   SCU_RAM_PARAM_1_RES_DEMOD_GET_LOCK_NEVER_LOCK) {
10856*4882a593Smuzhiyun 		/* 0x8000 DEMOD + FEC LOCKED (system lock) */
10857*4882a593Smuzhiyun 		*lock_stat = DRX_LOCKED;
10858*4882a593Smuzhiyun 	} else {
10859*4882a593Smuzhiyun 		/* 0xC000 NEVER LOCKED */
10860*4882a593Smuzhiyun 		/* (system will never be able to lock to the signal) */
10861*4882a593Smuzhiyun 		*lock_stat = DRX_NEVER_LOCK;
10862*4882a593Smuzhiyun 	}
10863*4882a593Smuzhiyun 
10864*4882a593Smuzhiyun 	return 0;
10865*4882a593Smuzhiyun rw_error:
10866*4882a593Smuzhiyun 	return rc;
10867*4882a593Smuzhiyun }
10868*4882a593Smuzhiyun 
10869*4882a593Smuzhiyun /*============================================================================*/
10870*4882a593Smuzhiyun 
10871*4882a593Smuzhiyun /*
10872*4882a593Smuzhiyun * \fn int ctrl_set_standard()
10873*4882a593Smuzhiyun * \brief Set modulation standard to be used.
10874*4882a593Smuzhiyun * \param standard Modulation standard.
10875*4882a593Smuzhiyun * \return int.
10876*4882a593Smuzhiyun *
10877*4882a593Smuzhiyun * Setup stuff for the desired demodulation standard.
10878*4882a593Smuzhiyun * Disable and power down the previous selected demodulation standard
10879*4882a593Smuzhiyun *
10880*4882a593Smuzhiyun */
10881*4882a593Smuzhiyun static int
ctrl_set_standard(struct drx_demod_instance * demod,enum drx_standard * standard)10882*4882a593Smuzhiyun ctrl_set_standard(struct drx_demod_instance *demod, enum drx_standard *standard)
10883*4882a593Smuzhiyun {
10884*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
10885*4882a593Smuzhiyun 	int rc;
10886*4882a593Smuzhiyun 	enum drx_standard prev_standard;
10887*4882a593Smuzhiyun 
10888*4882a593Smuzhiyun 	/* check arguments */
10889*4882a593Smuzhiyun 	if ((standard == NULL) || (demod == NULL))
10890*4882a593Smuzhiyun 		return -EINVAL;
10891*4882a593Smuzhiyun 
10892*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
10893*4882a593Smuzhiyun 	prev_standard = ext_attr->standard;
10894*4882a593Smuzhiyun 
10895*4882a593Smuzhiyun 	/*
10896*4882a593Smuzhiyun 	   Stop and power down previous standard
10897*4882a593Smuzhiyun 	 */
10898*4882a593Smuzhiyun 	switch (prev_standard) {
10899*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10900*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
10901*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
10902*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
10903*4882a593Smuzhiyun 		rc = power_down_qam(demod, false);
10904*4882a593Smuzhiyun 		if (rc != 0) {
10905*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10906*4882a593Smuzhiyun 			goto rw_error;
10907*4882a593Smuzhiyun 		}
10908*4882a593Smuzhiyun 		break;
10909*4882a593Smuzhiyun #endif
10910*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
10911*4882a593Smuzhiyun 		rc = power_down_vsb(demod, false);
10912*4882a593Smuzhiyun 		if (rc != 0) {
10913*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10914*4882a593Smuzhiyun 			goto rw_error;
10915*4882a593Smuzhiyun 		}
10916*4882a593Smuzhiyun 		break;
10917*4882a593Smuzhiyun 	case DRX_STANDARD_UNKNOWN:
10918*4882a593Smuzhiyun 		/* Do nothing */
10919*4882a593Smuzhiyun 		break;
10920*4882a593Smuzhiyun 	case DRX_STANDARD_AUTO:
10921*4882a593Smuzhiyun 	default:
10922*4882a593Smuzhiyun 		return -EINVAL;
10923*4882a593Smuzhiyun 	}
10924*4882a593Smuzhiyun 
10925*4882a593Smuzhiyun 	/*
10926*4882a593Smuzhiyun 	   Initialize channel independent registers
10927*4882a593Smuzhiyun 	   Power up new standard
10928*4882a593Smuzhiyun 	 */
10929*4882a593Smuzhiyun 	ext_attr->standard = *standard;
10930*4882a593Smuzhiyun 
10931*4882a593Smuzhiyun 	switch (*standard) {
10932*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10933*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
10934*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
10935*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
10936*4882a593Smuzhiyun 		do {
10937*4882a593Smuzhiyun 			u16 dummy;
10938*4882a593Smuzhiyun 			rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SCU_RAM_VERSION_HI__A, &dummy, 0);
10939*4882a593Smuzhiyun 			if (rc != 0) {
10940*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
10941*4882a593Smuzhiyun 				goto rw_error;
10942*4882a593Smuzhiyun 			}
10943*4882a593Smuzhiyun 		} while (0);
10944*4882a593Smuzhiyun 		break;
10945*4882a593Smuzhiyun #endif
10946*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
10947*4882a593Smuzhiyun 		rc = set_vsb_leak_n_gain(demod);
10948*4882a593Smuzhiyun 		if (rc != 0) {
10949*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
10950*4882a593Smuzhiyun 			goto rw_error;
10951*4882a593Smuzhiyun 		}
10952*4882a593Smuzhiyun 		break;
10953*4882a593Smuzhiyun 	default:
10954*4882a593Smuzhiyun 		ext_attr->standard = DRX_STANDARD_UNKNOWN;
10955*4882a593Smuzhiyun 		return -EINVAL;
10956*4882a593Smuzhiyun 		break;
10957*4882a593Smuzhiyun 	}
10958*4882a593Smuzhiyun 
10959*4882a593Smuzhiyun 	return 0;
10960*4882a593Smuzhiyun rw_error:
10961*4882a593Smuzhiyun 	/* Don't know what the standard is now ... try again */
10962*4882a593Smuzhiyun 	ext_attr->standard = DRX_STANDARD_UNKNOWN;
10963*4882a593Smuzhiyun 	return rc;
10964*4882a593Smuzhiyun }
10965*4882a593Smuzhiyun 
10966*4882a593Smuzhiyun /*============================================================================*/
10967*4882a593Smuzhiyun 
drxj_reset_mode(struct drxj_data * ext_attr)10968*4882a593Smuzhiyun static void drxj_reset_mode(struct drxj_data *ext_attr)
10969*4882a593Smuzhiyun {
10970*4882a593Smuzhiyun 	/* Initialize default AFE configuration for QAM */
10971*4882a593Smuzhiyun 	if (ext_attr->has_lna) {
10972*4882a593Smuzhiyun 		/* IF AGC off, PGA active */
10973*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10974*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.standard = DRX_STANDARD_ITU_B;
10975*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_OFF;
10976*4882a593Smuzhiyun 		ext_attr->qam_pga_cfg = 140 + (11 * 13);
10977*4882a593Smuzhiyun #endif
10978*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.standard = DRX_STANDARD_8VSB;
10979*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_OFF;
10980*4882a593Smuzhiyun 		ext_attr->vsb_pga_cfg = 140 + (11 * 13);
10981*4882a593Smuzhiyun 	} else {
10982*4882a593Smuzhiyun 		/* IF AGC on, PGA not active */
10983*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
10984*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.standard = DRX_STANDARD_ITU_B;
10985*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
10986*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.min_output_level = 0;
10987*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.max_output_level = 0x7FFF;
10988*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.speed = 3;
10989*4882a593Smuzhiyun 		ext_attr->qam_if_agc_cfg.top = 1297;
10990*4882a593Smuzhiyun 		ext_attr->qam_pga_cfg = 140;
10991*4882a593Smuzhiyun #endif
10992*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.standard = DRX_STANDARD_8VSB;
10993*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
10994*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.min_output_level = 0;
10995*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.max_output_level = 0x7FFF;
10996*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.speed = 3;
10997*4882a593Smuzhiyun 		ext_attr->vsb_if_agc_cfg.top = 1024;
10998*4882a593Smuzhiyun 		ext_attr->vsb_pga_cfg = 140;
10999*4882a593Smuzhiyun 	}
11000*4882a593Smuzhiyun 	/* TODO: remove min_output_level and max_output_level for both QAM and VSB after */
11001*4882a593Smuzhiyun 	/* mc has not used them */
11002*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
11003*4882a593Smuzhiyun 	ext_attr->qam_rf_agc_cfg.standard = DRX_STANDARD_ITU_B;
11004*4882a593Smuzhiyun 	ext_attr->qam_rf_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
11005*4882a593Smuzhiyun 	ext_attr->qam_rf_agc_cfg.min_output_level = 0;
11006*4882a593Smuzhiyun 	ext_attr->qam_rf_agc_cfg.max_output_level = 0x7FFF;
11007*4882a593Smuzhiyun 	ext_attr->qam_rf_agc_cfg.speed = 3;
11008*4882a593Smuzhiyun 	ext_attr->qam_rf_agc_cfg.top = 9500;
11009*4882a593Smuzhiyun 	ext_attr->qam_rf_agc_cfg.cut_off_current = 4000;
11010*4882a593Smuzhiyun 	ext_attr->qam_pre_saw_cfg.standard = DRX_STANDARD_ITU_B;
11011*4882a593Smuzhiyun 	ext_attr->qam_pre_saw_cfg.reference = 0x07;
11012*4882a593Smuzhiyun 	ext_attr->qam_pre_saw_cfg.use_pre_saw = true;
11013*4882a593Smuzhiyun #endif
11014*4882a593Smuzhiyun 	/* Initialize default AFE configuration for VSB */
11015*4882a593Smuzhiyun 	ext_attr->vsb_rf_agc_cfg.standard = DRX_STANDARD_8VSB;
11016*4882a593Smuzhiyun 	ext_attr->vsb_rf_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
11017*4882a593Smuzhiyun 	ext_attr->vsb_rf_agc_cfg.min_output_level = 0;
11018*4882a593Smuzhiyun 	ext_attr->vsb_rf_agc_cfg.max_output_level = 0x7FFF;
11019*4882a593Smuzhiyun 	ext_attr->vsb_rf_agc_cfg.speed = 3;
11020*4882a593Smuzhiyun 	ext_attr->vsb_rf_agc_cfg.top = 9500;
11021*4882a593Smuzhiyun 	ext_attr->vsb_rf_agc_cfg.cut_off_current = 4000;
11022*4882a593Smuzhiyun 	ext_attr->vsb_pre_saw_cfg.standard = DRX_STANDARD_8VSB;
11023*4882a593Smuzhiyun 	ext_attr->vsb_pre_saw_cfg.reference = 0x07;
11024*4882a593Smuzhiyun 	ext_attr->vsb_pre_saw_cfg.use_pre_saw = true;
11025*4882a593Smuzhiyun }
11026*4882a593Smuzhiyun 
11027*4882a593Smuzhiyun /*
11028*4882a593Smuzhiyun * \fn int ctrl_power_mode()
11029*4882a593Smuzhiyun * \brief Set the power mode of the device to the specified power mode
11030*4882a593Smuzhiyun * \param demod Pointer to demodulator instance.
11031*4882a593Smuzhiyun * \param mode  Pointer to new power mode.
11032*4882a593Smuzhiyun * \return int.
11033*4882a593Smuzhiyun * \retval 0          Success
11034*4882a593Smuzhiyun * \retval -EIO       I2C error or other failure
11035*4882a593Smuzhiyun * \retval -EINVAL Invalid mode argument.
11036*4882a593Smuzhiyun *
11037*4882a593Smuzhiyun *
11038*4882a593Smuzhiyun */
11039*4882a593Smuzhiyun static int
ctrl_power_mode(struct drx_demod_instance * demod,enum drx_power_mode * mode)11040*4882a593Smuzhiyun ctrl_power_mode(struct drx_demod_instance *demod, enum drx_power_mode *mode)
11041*4882a593Smuzhiyun {
11042*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = (struct drx_common_attr *) NULL;
11043*4882a593Smuzhiyun 	struct drxj_data *ext_attr = (struct drxj_data *) NULL;
11044*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)NULL;
11045*4882a593Smuzhiyun 	int rc;
11046*4882a593Smuzhiyun 	u16 sio_cc_pwd_mode = 0;
11047*4882a593Smuzhiyun 
11048*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
11049*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
11050*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
11051*4882a593Smuzhiyun 
11052*4882a593Smuzhiyun 	/* Check arguments */
11053*4882a593Smuzhiyun 	if (mode == NULL)
11054*4882a593Smuzhiyun 		return -EINVAL;
11055*4882a593Smuzhiyun 
11056*4882a593Smuzhiyun 	/* If already in requested power mode, do nothing */
11057*4882a593Smuzhiyun 	if (common_attr->current_power_mode == *mode)
11058*4882a593Smuzhiyun 		return 0;
11059*4882a593Smuzhiyun 
11060*4882a593Smuzhiyun 	switch (*mode) {
11061*4882a593Smuzhiyun 	case DRX_POWER_UP:
11062*4882a593Smuzhiyun 	case DRXJ_POWER_DOWN_MAIN_PATH:
11063*4882a593Smuzhiyun 		sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_NONE;
11064*4882a593Smuzhiyun 		break;
11065*4882a593Smuzhiyun 	case DRXJ_POWER_DOWN_CORE:
11066*4882a593Smuzhiyun 		sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_CLOCK;
11067*4882a593Smuzhiyun 		break;
11068*4882a593Smuzhiyun 	case DRXJ_POWER_DOWN_PLL:
11069*4882a593Smuzhiyun 		sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_PLL;
11070*4882a593Smuzhiyun 		break;
11071*4882a593Smuzhiyun 	case DRX_POWER_DOWN:
11072*4882a593Smuzhiyun 		sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_OSC;
11073*4882a593Smuzhiyun 		break;
11074*4882a593Smuzhiyun 	default:
11075*4882a593Smuzhiyun 		/* Unknow sleep mode */
11076*4882a593Smuzhiyun 		return -EINVAL;
11077*4882a593Smuzhiyun 		break;
11078*4882a593Smuzhiyun 	}
11079*4882a593Smuzhiyun 
11080*4882a593Smuzhiyun 	/* Check if device needs to be powered up */
11081*4882a593Smuzhiyun 	if ((common_attr->current_power_mode != DRX_POWER_UP)) {
11082*4882a593Smuzhiyun 		rc = power_up_device(demod);
11083*4882a593Smuzhiyun 		if (rc != 0) {
11084*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
11085*4882a593Smuzhiyun 			goto rw_error;
11086*4882a593Smuzhiyun 		}
11087*4882a593Smuzhiyun 	}
11088*4882a593Smuzhiyun 
11089*4882a593Smuzhiyun 	if (*mode == DRX_POWER_UP) {
11090*4882a593Smuzhiyun 		/* Restore analog & pin configuration */
11091*4882a593Smuzhiyun 
11092*4882a593Smuzhiyun 		/* Initialize default AFE configuration for VSB */
11093*4882a593Smuzhiyun 		drxj_reset_mode(ext_attr);
11094*4882a593Smuzhiyun 	} else {
11095*4882a593Smuzhiyun 		/* Power down to requested mode */
11096*4882a593Smuzhiyun 		/* Backup some register settings */
11097*4882a593Smuzhiyun 		/* Set pins with possible pull-ups connected to them in input mode */
11098*4882a593Smuzhiyun 		/* Analog power down */
11099*4882a593Smuzhiyun 		/* ADC power down */
11100*4882a593Smuzhiyun 		/* Power down device */
11101*4882a593Smuzhiyun 		/* stop all comm_exec */
11102*4882a593Smuzhiyun 		/*
11103*4882a593Smuzhiyun 		   Stop and power down previous standard
11104*4882a593Smuzhiyun 		 */
11105*4882a593Smuzhiyun 
11106*4882a593Smuzhiyun 		switch (ext_attr->standard) {
11107*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_A:
11108*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_B:
11109*4882a593Smuzhiyun 		case DRX_STANDARD_ITU_C:
11110*4882a593Smuzhiyun 			rc = power_down_qam(demod, true);
11111*4882a593Smuzhiyun 			if (rc != 0) {
11112*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
11113*4882a593Smuzhiyun 				goto rw_error;
11114*4882a593Smuzhiyun 			}
11115*4882a593Smuzhiyun 			break;
11116*4882a593Smuzhiyun 		case DRX_STANDARD_8VSB:
11117*4882a593Smuzhiyun 			rc = power_down_vsb(demod, true);
11118*4882a593Smuzhiyun 			if (rc != 0) {
11119*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
11120*4882a593Smuzhiyun 				goto rw_error;
11121*4882a593Smuzhiyun 			}
11122*4882a593Smuzhiyun 			break;
11123*4882a593Smuzhiyun 		case DRX_STANDARD_PAL_SECAM_BG:
11124*4882a593Smuzhiyun 		case DRX_STANDARD_PAL_SECAM_DK:
11125*4882a593Smuzhiyun 		case DRX_STANDARD_PAL_SECAM_I:
11126*4882a593Smuzhiyun 		case DRX_STANDARD_PAL_SECAM_L:
11127*4882a593Smuzhiyun 		case DRX_STANDARD_PAL_SECAM_LP:
11128*4882a593Smuzhiyun 		case DRX_STANDARD_NTSC:
11129*4882a593Smuzhiyun 		case DRX_STANDARD_FM:
11130*4882a593Smuzhiyun 			rc = power_down_atv(demod, ext_attr->standard, true);
11131*4882a593Smuzhiyun 			if (rc != 0) {
11132*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
11133*4882a593Smuzhiyun 				goto rw_error;
11134*4882a593Smuzhiyun 			}
11135*4882a593Smuzhiyun 			break;
11136*4882a593Smuzhiyun 		case DRX_STANDARD_UNKNOWN:
11137*4882a593Smuzhiyun 			/* Do nothing */
11138*4882a593Smuzhiyun 			break;
11139*4882a593Smuzhiyun 		case DRX_STANDARD_AUTO:
11140*4882a593Smuzhiyun 		default:
11141*4882a593Smuzhiyun 			return -EIO;
11142*4882a593Smuzhiyun 		}
11143*4882a593Smuzhiyun 		ext_attr->standard = DRX_STANDARD_UNKNOWN;
11144*4882a593Smuzhiyun 	}
11145*4882a593Smuzhiyun 
11146*4882a593Smuzhiyun 	if (*mode != DRXJ_POWER_DOWN_MAIN_PATH) {
11147*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_CC_PWD_MODE__A, sio_cc_pwd_mode, 0);
11148*4882a593Smuzhiyun 		if (rc != 0) {
11149*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
11150*4882a593Smuzhiyun 			goto rw_error;
11151*4882a593Smuzhiyun 		}
11152*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, SIO_CC_UPDATE__A, SIO_CC_UPDATE_KEY, 0);
11153*4882a593Smuzhiyun 		if (rc != 0) {
11154*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
11155*4882a593Smuzhiyun 			goto rw_error;
11156*4882a593Smuzhiyun 		}
11157*4882a593Smuzhiyun 
11158*4882a593Smuzhiyun 		if ((*mode != DRX_POWER_UP)) {
11159*4882a593Smuzhiyun 			/* Initialize HI, wakeup key especially before put IC to sleep */
11160*4882a593Smuzhiyun 			rc = init_hi(demod);
11161*4882a593Smuzhiyun 			if (rc != 0) {
11162*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
11163*4882a593Smuzhiyun 				goto rw_error;
11164*4882a593Smuzhiyun 			}
11165*4882a593Smuzhiyun 
11166*4882a593Smuzhiyun 			ext_attr->hi_cfg_ctrl |= SIO_HI_RA_RAM_PAR_5_CFG_SLEEP_ZZZ;
11167*4882a593Smuzhiyun 			rc = hi_cfg_command(demod);
11168*4882a593Smuzhiyun 			if (rc != 0) {
11169*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
11170*4882a593Smuzhiyun 				goto rw_error;
11171*4882a593Smuzhiyun 			}
11172*4882a593Smuzhiyun 		}
11173*4882a593Smuzhiyun 	}
11174*4882a593Smuzhiyun 
11175*4882a593Smuzhiyun 	common_attr->current_power_mode = *mode;
11176*4882a593Smuzhiyun 
11177*4882a593Smuzhiyun 	return 0;
11178*4882a593Smuzhiyun rw_error:
11179*4882a593Smuzhiyun 	return rc;
11180*4882a593Smuzhiyun }
11181*4882a593Smuzhiyun 
11182*4882a593Smuzhiyun /*============================================================================*/
11183*4882a593Smuzhiyun /*== CTRL Set/Get Config related functions ===================================*/
11184*4882a593Smuzhiyun /*============================================================================*/
11185*4882a593Smuzhiyun 
11186*4882a593Smuzhiyun /*
11187*4882a593Smuzhiyun * \fn int ctrl_set_cfg_pre_saw()
11188*4882a593Smuzhiyun * \brief Set Pre-saw reference.
11189*4882a593Smuzhiyun * \param demod demod instance
11190*4882a593Smuzhiyun * \param u16 *
11191*4882a593Smuzhiyun * \return int.
11192*4882a593Smuzhiyun *
11193*4882a593Smuzhiyun * Check arguments
11194*4882a593Smuzhiyun * Dispatch handling to standard specific function.
11195*4882a593Smuzhiyun *
11196*4882a593Smuzhiyun */
11197*4882a593Smuzhiyun static int
ctrl_set_cfg_pre_saw(struct drx_demod_instance * demod,struct drxj_cfg_pre_saw * pre_saw)11198*4882a593Smuzhiyun ctrl_set_cfg_pre_saw(struct drx_demod_instance *demod, struct drxj_cfg_pre_saw *pre_saw)
11199*4882a593Smuzhiyun {
11200*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
11201*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
11202*4882a593Smuzhiyun 	int rc;
11203*4882a593Smuzhiyun 
11204*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
11205*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
11206*4882a593Smuzhiyun 
11207*4882a593Smuzhiyun 	/* check arguments */
11208*4882a593Smuzhiyun 	if ((pre_saw == NULL) || (pre_saw->reference > IQM_AF_PDREF__M)
11209*4882a593Smuzhiyun 	    ) {
11210*4882a593Smuzhiyun 		return -EINVAL;
11211*4882a593Smuzhiyun 	}
11212*4882a593Smuzhiyun 
11213*4882a593Smuzhiyun 	/* Only if standard is currently active */
11214*4882a593Smuzhiyun 	if ((ext_attr->standard == pre_saw->standard) ||
11215*4882a593Smuzhiyun 	    (DRXJ_ISQAMSTD(ext_attr->standard) &&
11216*4882a593Smuzhiyun 	     DRXJ_ISQAMSTD(pre_saw->standard)) ||
11217*4882a593Smuzhiyun 	    (DRXJ_ISATVSTD(ext_attr->standard) &&
11218*4882a593Smuzhiyun 	     DRXJ_ISATVSTD(pre_saw->standard))) {
11219*4882a593Smuzhiyun 		rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PDREF__A, pre_saw->reference, 0);
11220*4882a593Smuzhiyun 		if (rc != 0) {
11221*4882a593Smuzhiyun 			pr_err("error %d\n", rc);
11222*4882a593Smuzhiyun 			goto rw_error;
11223*4882a593Smuzhiyun 		}
11224*4882a593Smuzhiyun 	}
11225*4882a593Smuzhiyun 
11226*4882a593Smuzhiyun 	/* Store pre-saw settings */
11227*4882a593Smuzhiyun 	switch (pre_saw->standard) {
11228*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
11229*4882a593Smuzhiyun 		ext_attr->vsb_pre_saw_cfg = *pre_saw;
11230*4882a593Smuzhiyun 		break;
11231*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
11232*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
11233*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
11234*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
11235*4882a593Smuzhiyun 		ext_attr->qam_pre_saw_cfg = *pre_saw;
11236*4882a593Smuzhiyun 		break;
11237*4882a593Smuzhiyun #endif
11238*4882a593Smuzhiyun 	default:
11239*4882a593Smuzhiyun 		return -EINVAL;
11240*4882a593Smuzhiyun 	}
11241*4882a593Smuzhiyun 
11242*4882a593Smuzhiyun 	return 0;
11243*4882a593Smuzhiyun rw_error:
11244*4882a593Smuzhiyun 	return rc;
11245*4882a593Smuzhiyun }
11246*4882a593Smuzhiyun 
11247*4882a593Smuzhiyun /*============================================================================*/
11248*4882a593Smuzhiyun 
11249*4882a593Smuzhiyun /*
11250*4882a593Smuzhiyun * \fn int ctrl_set_cfg_afe_gain()
11251*4882a593Smuzhiyun * \brief Set AFE Gain.
11252*4882a593Smuzhiyun * \param demod demod instance
11253*4882a593Smuzhiyun * \param u16 *
11254*4882a593Smuzhiyun * \return int.
11255*4882a593Smuzhiyun *
11256*4882a593Smuzhiyun * Check arguments
11257*4882a593Smuzhiyun * Dispatch handling to standard specific function.
11258*4882a593Smuzhiyun *
11259*4882a593Smuzhiyun */
11260*4882a593Smuzhiyun static int
ctrl_set_cfg_afe_gain(struct drx_demod_instance * demod,struct drxj_cfg_afe_gain * afe_gain)11261*4882a593Smuzhiyun ctrl_set_cfg_afe_gain(struct drx_demod_instance *demod, struct drxj_cfg_afe_gain *afe_gain)
11262*4882a593Smuzhiyun {
11263*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
11264*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
11265*4882a593Smuzhiyun 	int rc;
11266*4882a593Smuzhiyun 	u8 gain = 0;
11267*4882a593Smuzhiyun 
11268*4882a593Smuzhiyun 	/* check arguments */
11269*4882a593Smuzhiyun 	if (afe_gain == NULL)
11270*4882a593Smuzhiyun 		return -EINVAL;
11271*4882a593Smuzhiyun 
11272*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
11273*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
11274*4882a593Smuzhiyun 
11275*4882a593Smuzhiyun 	switch (afe_gain->standard) {
11276*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:	fallthrough;
11277*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
11278*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
11279*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
11280*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
11281*4882a593Smuzhiyun #endif
11282*4882a593Smuzhiyun 		/* Do nothing */
11283*4882a593Smuzhiyun 		break;
11284*4882a593Smuzhiyun 	default:
11285*4882a593Smuzhiyun 		return -EINVAL;
11286*4882a593Smuzhiyun 	}
11287*4882a593Smuzhiyun 
11288*4882a593Smuzhiyun 	/* TODO PGA gain is also written by microcode (at least by QAM and VSB)
11289*4882a593Smuzhiyun 	   So I (PJ) think interface requires choice between auto, user mode */
11290*4882a593Smuzhiyun 
11291*4882a593Smuzhiyun 	if (afe_gain->gain >= 329)
11292*4882a593Smuzhiyun 		gain = 15;
11293*4882a593Smuzhiyun 	else if (afe_gain->gain <= 147)
11294*4882a593Smuzhiyun 		gain = 0;
11295*4882a593Smuzhiyun 	else
11296*4882a593Smuzhiyun 		gain = (afe_gain->gain - 140 + 6) / 13;
11297*4882a593Smuzhiyun 
11298*4882a593Smuzhiyun 	/* Only if standard is currently active */
11299*4882a593Smuzhiyun 	if (ext_attr->standard == afe_gain->standard) {
11300*4882a593Smuzhiyun 			rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PGA_GAIN__A, gain, 0);
11301*4882a593Smuzhiyun 			if (rc != 0) {
11302*4882a593Smuzhiyun 				pr_err("error %d\n", rc);
11303*4882a593Smuzhiyun 				goto rw_error;
11304*4882a593Smuzhiyun 			}
11305*4882a593Smuzhiyun 		}
11306*4882a593Smuzhiyun 
11307*4882a593Smuzhiyun 	/* Store AFE Gain settings */
11308*4882a593Smuzhiyun 	switch (afe_gain->standard) {
11309*4882a593Smuzhiyun 	case DRX_STANDARD_8VSB:
11310*4882a593Smuzhiyun 		ext_attr->vsb_pga_cfg = gain * 13 + 140;
11311*4882a593Smuzhiyun 		break;
11312*4882a593Smuzhiyun #ifndef DRXJ_VSB_ONLY
11313*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_A:
11314*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_B:
11315*4882a593Smuzhiyun 	case DRX_STANDARD_ITU_C:
11316*4882a593Smuzhiyun 		ext_attr->qam_pga_cfg = gain * 13 + 140;
11317*4882a593Smuzhiyun 		break;
11318*4882a593Smuzhiyun #endif
11319*4882a593Smuzhiyun 	default:
11320*4882a593Smuzhiyun 		return -EIO;
11321*4882a593Smuzhiyun 	}
11322*4882a593Smuzhiyun 
11323*4882a593Smuzhiyun 	return 0;
11324*4882a593Smuzhiyun rw_error:
11325*4882a593Smuzhiyun 	return rc;
11326*4882a593Smuzhiyun }
11327*4882a593Smuzhiyun 
11328*4882a593Smuzhiyun /*============================================================================*/
11329*4882a593Smuzhiyun 
11330*4882a593Smuzhiyun 
11331*4882a593Smuzhiyun /*=============================================================================
11332*4882a593Smuzhiyun ===== EXPORTED FUNCTIONS ====================================================*/
11333*4882a593Smuzhiyun 
11334*4882a593Smuzhiyun static int drx_ctrl_u_code(struct drx_demod_instance *demod,
11335*4882a593Smuzhiyun 		       struct drxu_code_info *mc_info,
11336*4882a593Smuzhiyun 		       enum drxu_code_action action);
11337*4882a593Smuzhiyun static int drxj_set_lna_state(struct drx_demod_instance *demod, bool state);
11338*4882a593Smuzhiyun 
11339*4882a593Smuzhiyun /*
11340*4882a593Smuzhiyun * \fn drxj_open()
11341*4882a593Smuzhiyun * \brief Open the demod instance, configure device, configure drxdriver
11342*4882a593Smuzhiyun * \return Status_t Return status.
11343*4882a593Smuzhiyun *
11344*4882a593Smuzhiyun * drxj_open() can be called with a NULL ucode image => no ucode upload.
11345*4882a593Smuzhiyun * This means that drxj_open() must NOT contain SCU commands or, in general,
11346*4882a593Smuzhiyun * rely on SCU or AUD ucode to be present.
11347*4882a593Smuzhiyun *
11348*4882a593Smuzhiyun */
11349*4882a593Smuzhiyun 
drxj_open(struct drx_demod_instance * demod)11350*4882a593Smuzhiyun static int drxj_open(struct drx_demod_instance *demod)
11351*4882a593Smuzhiyun {
11352*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = NULL;
11353*4882a593Smuzhiyun 	struct drxj_data *ext_attr = NULL;
11354*4882a593Smuzhiyun 	struct drx_common_attr *common_attr = NULL;
11355*4882a593Smuzhiyun 	u32 driver_version = 0;
11356*4882a593Smuzhiyun 	struct drxu_code_info ucode_info;
11357*4882a593Smuzhiyun 	struct drx_cfg_mpeg_output cfg_mpeg_output;
11358*4882a593Smuzhiyun 	int rc;
11359*4882a593Smuzhiyun 	enum drx_power_mode power_mode = DRX_POWER_UP;
11360*4882a593Smuzhiyun 
11361*4882a593Smuzhiyun 	if ((demod == NULL) ||
11362*4882a593Smuzhiyun 	    (demod->my_common_attr == NULL) ||
11363*4882a593Smuzhiyun 	    (demod->my_ext_attr == NULL) ||
11364*4882a593Smuzhiyun 	    (demod->my_i2c_dev_addr == NULL) ||
11365*4882a593Smuzhiyun 	    (demod->my_common_attr->is_opened)) {
11366*4882a593Smuzhiyun 		return -EINVAL;
11367*4882a593Smuzhiyun 	}
11368*4882a593Smuzhiyun 
11369*4882a593Smuzhiyun 	/* Check arguments */
11370*4882a593Smuzhiyun 	if (demod->my_ext_attr == NULL)
11371*4882a593Smuzhiyun 		return -EINVAL;
11372*4882a593Smuzhiyun 
11373*4882a593Smuzhiyun 	dev_addr = demod->my_i2c_dev_addr;
11374*4882a593Smuzhiyun 	ext_attr = (struct drxj_data *) demod->my_ext_attr;
11375*4882a593Smuzhiyun 	common_attr = (struct drx_common_attr *) demod->my_common_attr;
11376*4882a593Smuzhiyun 
11377*4882a593Smuzhiyun 	rc = ctrl_power_mode(demod, &power_mode);
11378*4882a593Smuzhiyun 	if (rc != 0) {
11379*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11380*4882a593Smuzhiyun 		goto rw_error;
11381*4882a593Smuzhiyun 	}
11382*4882a593Smuzhiyun 	if (power_mode != DRX_POWER_UP) {
11383*4882a593Smuzhiyun 		rc = -EINVAL;
11384*4882a593Smuzhiyun 		pr_err("failed to powerup device\n");
11385*4882a593Smuzhiyun 		goto rw_error;
11386*4882a593Smuzhiyun 	}
11387*4882a593Smuzhiyun 
11388*4882a593Smuzhiyun 	/* has to be in front of setIqmAf and setOrxNsuAox */
11389*4882a593Smuzhiyun 	rc = get_device_capabilities(demod);
11390*4882a593Smuzhiyun 	if (rc != 0) {
11391*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11392*4882a593Smuzhiyun 		goto rw_error;
11393*4882a593Smuzhiyun 	}
11394*4882a593Smuzhiyun 
11395*4882a593Smuzhiyun 	/*
11396*4882a593Smuzhiyun 	 * Soft reset of sys- and osc-clockdomain
11397*4882a593Smuzhiyun 	 *
11398*4882a593Smuzhiyun 	 * HACK: On windows, it writes a 0x07 here, instead of just 0x03.
11399*4882a593Smuzhiyun 	 * As we didn't load the firmware here yet, we should do the same.
11400*4882a593Smuzhiyun 	 * Btw, this is coherent with DRX-K, where we send reset codes
11401*4882a593Smuzhiyun 	 * for modulation (OFTM, in DRX-k), SYS and OSC clock domains.
11402*4882a593Smuzhiyun 	 */
11403*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_CC_SOFT_RST__A, (0x04 | SIO_CC_SOFT_RST_SYS__M | SIO_CC_SOFT_RST_OSC__M), 0);
11404*4882a593Smuzhiyun 	if (rc != 0) {
11405*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11406*4882a593Smuzhiyun 		goto rw_error;
11407*4882a593Smuzhiyun 	}
11408*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SIO_CC_UPDATE__A, SIO_CC_UPDATE_KEY, 0);
11409*4882a593Smuzhiyun 	if (rc != 0) {
11410*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11411*4882a593Smuzhiyun 		goto rw_error;
11412*4882a593Smuzhiyun 	}
11413*4882a593Smuzhiyun 	msleep(1);
11414*4882a593Smuzhiyun 
11415*4882a593Smuzhiyun 	/* TODO first make sure that everything keeps working before enabling this */
11416*4882a593Smuzhiyun 	/* PowerDownAnalogBlocks() */
11417*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, ATV_TOP_STDBY__A, (~ATV_TOP_STDBY_CVBS_STDBY_A2_ACTIVE) | ATV_TOP_STDBY_SIF_STDBY_STANDBY, 0);
11418*4882a593Smuzhiyun 	if (rc != 0) {
11419*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11420*4882a593Smuzhiyun 		goto rw_error;
11421*4882a593Smuzhiyun 	}
11422*4882a593Smuzhiyun 
11423*4882a593Smuzhiyun 	rc = set_iqm_af(demod, false);
11424*4882a593Smuzhiyun 	if (rc != 0) {
11425*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11426*4882a593Smuzhiyun 		goto rw_error;
11427*4882a593Smuzhiyun 	}
11428*4882a593Smuzhiyun 	rc = set_orx_nsu_aox(demod, false);
11429*4882a593Smuzhiyun 	if (rc != 0) {
11430*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11431*4882a593Smuzhiyun 		goto rw_error;
11432*4882a593Smuzhiyun 	}
11433*4882a593Smuzhiyun 
11434*4882a593Smuzhiyun 	rc = init_hi(demod);
11435*4882a593Smuzhiyun 	if (rc != 0) {
11436*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11437*4882a593Smuzhiyun 		goto rw_error;
11438*4882a593Smuzhiyun 	}
11439*4882a593Smuzhiyun 
11440*4882a593Smuzhiyun 	/* disable mpegoutput pins */
11441*4882a593Smuzhiyun 	memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
11442*4882a593Smuzhiyun 	cfg_mpeg_output.enable_mpeg_output = false;
11443*4882a593Smuzhiyun 
11444*4882a593Smuzhiyun 	rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
11445*4882a593Smuzhiyun 	if (rc != 0) {
11446*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11447*4882a593Smuzhiyun 		goto rw_error;
11448*4882a593Smuzhiyun 	}
11449*4882a593Smuzhiyun 	/* Stop AUD Inform SetAudio it will need to do all setting */
11450*4882a593Smuzhiyun 	rc = power_down_aud(demod);
11451*4882a593Smuzhiyun 	if (rc != 0) {
11452*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11453*4882a593Smuzhiyun 		goto rw_error;
11454*4882a593Smuzhiyun 	}
11455*4882a593Smuzhiyun 	/* Stop SCU */
11456*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_STOP, 0);
11457*4882a593Smuzhiyun 	if (rc != 0) {
11458*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11459*4882a593Smuzhiyun 		goto rw_error;
11460*4882a593Smuzhiyun 	}
11461*4882a593Smuzhiyun 
11462*4882a593Smuzhiyun 	/* Upload microcode */
11463*4882a593Smuzhiyun 	if (common_attr->microcode_file != NULL) {
11464*4882a593Smuzhiyun 		/* Dirty trick to use common ucode upload & verify,
11465*4882a593Smuzhiyun 		   pretend device is already open */
11466*4882a593Smuzhiyun 		common_attr->is_opened = true;
11467*4882a593Smuzhiyun 		ucode_info.mc_file = common_attr->microcode_file;
11468*4882a593Smuzhiyun 
11469*4882a593Smuzhiyun 		if (DRX_ISPOWERDOWNMODE(demod->my_common_attr->current_power_mode)) {
11470*4882a593Smuzhiyun 			pr_err("Should powerup before loading the firmware.");
11471*4882a593Smuzhiyun 			return -EINVAL;
11472*4882a593Smuzhiyun 		}
11473*4882a593Smuzhiyun 
11474*4882a593Smuzhiyun 		rc = drx_ctrl_u_code(demod, &ucode_info, UCODE_UPLOAD);
11475*4882a593Smuzhiyun 		if (rc != 0) {
11476*4882a593Smuzhiyun 			pr_err("error %d while uploading the firmware\n", rc);
11477*4882a593Smuzhiyun 			goto rw_error;
11478*4882a593Smuzhiyun 		}
11479*4882a593Smuzhiyun 		if (common_attr->verify_microcode == true) {
11480*4882a593Smuzhiyun 			rc = drx_ctrl_u_code(demod, &ucode_info, UCODE_VERIFY);
11481*4882a593Smuzhiyun 			if (rc != 0) {
11482*4882a593Smuzhiyun 				pr_err("error %d while verifying the firmware\n",
11483*4882a593Smuzhiyun 				       rc);
11484*4882a593Smuzhiyun 				goto rw_error;
11485*4882a593Smuzhiyun 			}
11486*4882a593Smuzhiyun 		}
11487*4882a593Smuzhiyun 		common_attr->is_opened = false;
11488*4882a593Smuzhiyun 	}
11489*4882a593Smuzhiyun 
11490*4882a593Smuzhiyun 	/* Run SCU for a little while to initialize microcode version numbers */
11491*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_ACTIVE, 0);
11492*4882a593Smuzhiyun 	if (rc != 0) {
11493*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11494*4882a593Smuzhiyun 		goto rw_error;
11495*4882a593Smuzhiyun 	}
11496*4882a593Smuzhiyun 
11497*4882a593Smuzhiyun 	/* Initialize scan timeout */
11498*4882a593Smuzhiyun 	common_attr->scan_demod_lock_timeout = DRXJ_SCAN_TIMEOUT;
11499*4882a593Smuzhiyun 	common_attr->scan_desired_lock = DRX_LOCKED;
11500*4882a593Smuzhiyun 
11501*4882a593Smuzhiyun 	drxj_reset_mode(ext_attr);
11502*4882a593Smuzhiyun 	ext_attr->standard = DRX_STANDARD_UNKNOWN;
11503*4882a593Smuzhiyun 
11504*4882a593Smuzhiyun 	rc = smart_ant_init(demod);
11505*4882a593Smuzhiyun 	if (rc != 0) {
11506*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11507*4882a593Smuzhiyun 		goto rw_error;
11508*4882a593Smuzhiyun 	}
11509*4882a593Smuzhiyun 
11510*4882a593Smuzhiyun 	/* Stamp driver version number in SCU data RAM in BCD code
11511*4882a593Smuzhiyun 	   Done to enable field application engineers to retrieve drxdriver version
11512*4882a593Smuzhiyun 	   via I2C from SCU RAM
11513*4882a593Smuzhiyun 	 */
11514*4882a593Smuzhiyun 	driver_version = (VERSION_MAJOR / 100) % 10;
11515*4882a593Smuzhiyun 	driver_version <<= 4;
11516*4882a593Smuzhiyun 	driver_version += (VERSION_MAJOR / 10) % 10;
11517*4882a593Smuzhiyun 	driver_version <<= 4;
11518*4882a593Smuzhiyun 	driver_version += (VERSION_MAJOR % 10);
11519*4882a593Smuzhiyun 	driver_version <<= 4;
11520*4882a593Smuzhiyun 	driver_version += (VERSION_MINOR % 10);
11521*4882a593Smuzhiyun 	driver_version <<= 4;
11522*4882a593Smuzhiyun 	driver_version += (VERSION_PATCH / 1000) % 10;
11523*4882a593Smuzhiyun 	driver_version <<= 4;
11524*4882a593Smuzhiyun 	driver_version += (VERSION_PATCH / 100) % 10;
11525*4882a593Smuzhiyun 	driver_version <<= 4;
11526*4882a593Smuzhiyun 	driver_version += (VERSION_PATCH / 10) % 10;
11527*4882a593Smuzhiyun 	driver_version <<= 4;
11528*4882a593Smuzhiyun 	driver_version += (VERSION_PATCH % 10);
11529*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_DRIVER_VER_HI__A, (u16)(driver_version >> 16), 0);
11530*4882a593Smuzhiyun 	if (rc != 0) {
11531*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11532*4882a593Smuzhiyun 		goto rw_error;
11533*4882a593Smuzhiyun 	}
11534*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_DRIVER_VER_LO__A, (u16)(driver_version & 0xFFFF), 0);
11535*4882a593Smuzhiyun 	if (rc != 0) {
11536*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11537*4882a593Smuzhiyun 		goto rw_error;
11538*4882a593Smuzhiyun 	}
11539*4882a593Smuzhiyun 
11540*4882a593Smuzhiyun 	rc = ctrl_set_oob(demod, NULL);
11541*4882a593Smuzhiyun 	if (rc != 0) {
11542*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11543*4882a593Smuzhiyun 		goto rw_error;
11544*4882a593Smuzhiyun 	}
11545*4882a593Smuzhiyun 
11546*4882a593Smuzhiyun 	/* refresh the audio data structure with default */
11547*4882a593Smuzhiyun 	ext_attr->aud_data = drxj_default_aud_data_g;
11548*4882a593Smuzhiyun 
11549*4882a593Smuzhiyun 	demod->my_common_attr->is_opened = true;
11550*4882a593Smuzhiyun 	drxj_set_lna_state(demod, false);
11551*4882a593Smuzhiyun 	return 0;
11552*4882a593Smuzhiyun rw_error:
11553*4882a593Smuzhiyun 	common_attr->is_opened = false;
11554*4882a593Smuzhiyun 	return rc;
11555*4882a593Smuzhiyun }
11556*4882a593Smuzhiyun 
11557*4882a593Smuzhiyun /*============================================================================*/
11558*4882a593Smuzhiyun /*
11559*4882a593Smuzhiyun * \fn drxj_close()
11560*4882a593Smuzhiyun * \brief Close the demod instance, power down the device
11561*4882a593Smuzhiyun * \return Status_t Return status.
11562*4882a593Smuzhiyun *
11563*4882a593Smuzhiyun */
drxj_close(struct drx_demod_instance * demod)11564*4882a593Smuzhiyun static int drxj_close(struct drx_demod_instance *demod)
11565*4882a593Smuzhiyun {
11566*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
11567*4882a593Smuzhiyun 	int rc;
11568*4882a593Smuzhiyun 	enum drx_power_mode power_mode = DRX_POWER_UP;
11569*4882a593Smuzhiyun 
11570*4882a593Smuzhiyun 	if ((demod->my_common_attr == NULL) ||
11571*4882a593Smuzhiyun 	    (demod->my_ext_attr == NULL) ||
11572*4882a593Smuzhiyun 	    (demod->my_i2c_dev_addr == NULL) ||
11573*4882a593Smuzhiyun 	    (!demod->my_common_attr->is_opened)) {
11574*4882a593Smuzhiyun 		return -EINVAL;
11575*4882a593Smuzhiyun 	}
11576*4882a593Smuzhiyun 
11577*4882a593Smuzhiyun 	/* power up */
11578*4882a593Smuzhiyun 	rc = ctrl_power_mode(demod, &power_mode);
11579*4882a593Smuzhiyun 	if (rc != 0) {
11580*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11581*4882a593Smuzhiyun 		goto rw_error;
11582*4882a593Smuzhiyun 	}
11583*4882a593Smuzhiyun 
11584*4882a593Smuzhiyun 	rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_ACTIVE, 0);
11585*4882a593Smuzhiyun 	if (rc != 0) {
11586*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11587*4882a593Smuzhiyun 		goto rw_error;
11588*4882a593Smuzhiyun 	}
11589*4882a593Smuzhiyun 	power_mode = DRX_POWER_DOWN;
11590*4882a593Smuzhiyun 	rc = ctrl_power_mode(demod, &power_mode);
11591*4882a593Smuzhiyun 	if (rc != 0) {
11592*4882a593Smuzhiyun 		pr_err("error %d\n", rc);
11593*4882a593Smuzhiyun 		goto rw_error;
11594*4882a593Smuzhiyun 	}
11595*4882a593Smuzhiyun 
11596*4882a593Smuzhiyun 	DRX_ATTR_ISOPENED(demod) = false;
11597*4882a593Smuzhiyun 
11598*4882a593Smuzhiyun 	return 0;
11599*4882a593Smuzhiyun rw_error:
11600*4882a593Smuzhiyun 	DRX_ATTR_ISOPENED(demod) = false;
11601*4882a593Smuzhiyun 
11602*4882a593Smuzhiyun 	return rc;
11603*4882a593Smuzhiyun }
11604*4882a593Smuzhiyun 
11605*4882a593Smuzhiyun /*
11606*4882a593Smuzhiyun  * Microcode related functions
11607*4882a593Smuzhiyun  */
11608*4882a593Smuzhiyun 
11609*4882a593Smuzhiyun /*
11610*4882a593Smuzhiyun  * drx_u_code_compute_crc	- Compute CRC of block of microcode data.
11611*4882a593Smuzhiyun  * @block_data: Pointer to microcode data.
11612*4882a593Smuzhiyun  * @nr_words:   Size of microcode block (number of 16 bits words).
11613*4882a593Smuzhiyun  *
11614*4882a593Smuzhiyun  * returns The computed CRC residue.
11615*4882a593Smuzhiyun  */
drx_u_code_compute_crc(u8 * block_data,u16 nr_words)11616*4882a593Smuzhiyun static u16 drx_u_code_compute_crc(u8 *block_data, u16 nr_words)
11617*4882a593Smuzhiyun {
11618*4882a593Smuzhiyun 	u16 i = 0;
11619*4882a593Smuzhiyun 	u16 j = 0;
11620*4882a593Smuzhiyun 	u32 crc_word = 0;
11621*4882a593Smuzhiyun 	u32 carry = 0;
11622*4882a593Smuzhiyun 
11623*4882a593Smuzhiyun 	while (i < nr_words) {
11624*4882a593Smuzhiyun 		crc_word |= (u32)be16_to_cpu(*(__be16 *)(block_data));
11625*4882a593Smuzhiyun 		for (j = 0; j < 16; j++) {
11626*4882a593Smuzhiyun 			crc_word <<= 1;
11627*4882a593Smuzhiyun 			if (carry != 0)
11628*4882a593Smuzhiyun 				crc_word ^= 0x80050000UL;
11629*4882a593Smuzhiyun 			carry = crc_word & 0x80000000UL;
11630*4882a593Smuzhiyun 		}
11631*4882a593Smuzhiyun 		i++;
11632*4882a593Smuzhiyun 		block_data += (sizeof(u16));
11633*4882a593Smuzhiyun 	}
11634*4882a593Smuzhiyun 	return (u16)(crc_word >> 16);
11635*4882a593Smuzhiyun }
11636*4882a593Smuzhiyun 
11637*4882a593Smuzhiyun /*
11638*4882a593Smuzhiyun  * drx_check_firmware - checks if the loaded firmware is valid
11639*4882a593Smuzhiyun  *
11640*4882a593Smuzhiyun  * @demod:	demod structure
11641*4882a593Smuzhiyun  * @mc_data:	pointer to the start of the firmware
11642*4882a593Smuzhiyun  * @size:	firmware size
11643*4882a593Smuzhiyun  */
drx_check_firmware(struct drx_demod_instance * demod,u8 * mc_data,unsigned size)11644*4882a593Smuzhiyun static int drx_check_firmware(struct drx_demod_instance *demod, u8 *mc_data,
11645*4882a593Smuzhiyun 			  unsigned size)
11646*4882a593Smuzhiyun {
11647*4882a593Smuzhiyun 	struct drxu_code_block_hdr block_hdr;
11648*4882a593Smuzhiyun 	int i;
11649*4882a593Smuzhiyun 	unsigned count = 2 * sizeof(u16);
11650*4882a593Smuzhiyun 	u32 mc_dev_type, mc_version, mc_base_version;
11651*4882a593Smuzhiyun 	u16 mc_nr_of_blks = be16_to_cpu(*(__be16 *)(mc_data + sizeof(u16)));
11652*4882a593Smuzhiyun 
11653*4882a593Smuzhiyun 	/*
11654*4882a593Smuzhiyun 	 * Scan microcode blocks first for version info
11655*4882a593Smuzhiyun 	 * and firmware check
11656*4882a593Smuzhiyun 	 */
11657*4882a593Smuzhiyun 
11658*4882a593Smuzhiyun 	/* Clear version block */
11659*4882a593Smuzhiyun 	DRX_ATTR_MCRECORD(demod).aux_type = 0;
11660*4882a593Smuzhiyun 	DRX_ATTR_MCRECORD(demod).mc_dev_type = 0;
11661*4882a593Smuzhiyun 	DRX_ATTR_MCRECORD(demod).mc_version = 0;
11662*4882a593Smuzhiyun 	DRX_ATTR_MCRECORD(demod).mc_base_version = 0;
11663*4882a593Smuzhiyun 
11664*4882a593Smuzhiyun 	for (i = 0; i < mc_nr_of_blks; i++) {
11665*4882a593Smuzhiyun 		if (count + 3 * sizeof(u16) + sizeof(u32) > size)
11666*4882a593Smuzhiyun 			goto eof;
11667*4882a593Smuzhiyun 
11668*4882a593Smuzhiyun 		/* Process block header */
11669*4882a593Smuzhiyun 		block_hdr.addr = be32_to_cpu(*(__be32 *)(mc_data + count));
11670*4882a593Smuzhiyun 		count += sizeof(u32);
11671*4882a593Smuzhiyun 		block_hdr.size = be16_to_cpu(*(__be16 *)(mc_data + count));
11672*4882a593Smuzhiyun 		count += sizeof(u16);
11673*4882a593Smuzhiyun 		block_hdr.flags = be16_to_cpu(*(__be16 *)(mc_data + count));
11674*4882a593Smuzhiyun 		count += sizeof(u16);
11675*4882a593Smuzhiyun 		block_hdr.CRC = be16_to_cpu(*(__be16 *)(mc_data + count));
11676*4882a593Smuzhiyun 		count += sizeof(u16);
11677*4882a593Smuzhiyun 
11678*4882a593Smuzhiyun 		pr_debug("%u: addr %u, size %u, flags 0x%04x, CRC 0x%04x\n",
11679*4882a593Smuzhiyun 			count, block_hdr.addr, block_hdr.size, block_hdr.flags,
11680*4882a593Smuzhiyun 			block_hdr.CRC);
11681*4882a593Smuzhiyun 
11682*4882a593Smuzhiyun 		if (block_hdr.flags & 0x8) {
11683*4882a593Smuzhiyun 			u8 *auxblk = ((void *)mc_data) + block_hdr.addr;
11684*4882a593Smuzhiyun 			u16 auxtype;
11685*4882a593Smuzhiyun 
11686*4882a593Smuzhiyun 			if (block_hdr.addr + sizeof(u16) > size)
11687*4882a593Smuzhiyun 				goto eof;
11688*4882a593Smuzhiyun 
11689*4882a593Smuzhiyun 			auxtype = be16_to_cpu(*(__be16 *)(auxblk));
11690*4882a593Smuzhiyun 
11691*4882a593Smuzhiyun 			/* Aux block. Check type */
11692*4882a593Smuzhiyun 			if (DRX_ISMCVERTYPE(auxtype)) {
11693*4882a593Smuzhiyun 				if (block_hdr.addr + 2 * sizeof(u16) + 2 * sizeof (u32) > size)
11694*4882a593Smuzhiyun 					goto eof;
11695*4882a593Smuzhiyun 
11696*4882a593Smuzhiyun 				auxblk += sizeof(u16);
11697*4882a593Smuzhiyun 				mc_dev_type = be32_to_cpu(*(__be32 *)(auxblk));
11698*4882a593Smuzhiyun 				auxblk += sizeof(u32);
11699*4882a593Smuzhiyun 				mc_version = be32_to_cpu(*(__be32 *)(auxblk));
11700*4882a593Smuzhiyun 				auxblk += sizeof(u32);
11701*4882a593Smuzhiyun 				mc_base_version = be32_to_cpu(*(__be32 *)(auxblk));
11702*4882a593Smuzhiyun 
11703*4882a593Smuzhiyun 				DRX_ATTR_MCRECORD(demod).aux_type = auxtype;
11704*4882a593Smuzhiyun 				DRX_ATTR_MCRECORD(demod).mc_dev_type = mc_dev_type;
11705*4882a593Smuzhiyun 				DRX_ATTR_MCRECORD(demod).mc_version = mc_version;
11706*4882a593Smuzhiyun 				DRX_ATTR_MCRECORD(demod).mc_base_version = mc_base_version;
11707*4882a593Smuzhiyun 
11708*4882a593Smuzhiyun 				pr_info("Firmware dev %x, ver %x, base ver %x\n",
11709*4882a593Smuzhiyun 					mc_dev_type, mc_version, mc_base_version);
11710*4882a593Smuzhiyun 
11711*4882a593Smuzhiyun 			}
11712*4882a593Smuzhiyun 		} else if (count + block_hdr.size * sizeof(u16) > size)
11713*4882a593Smuzhiyun 			goto eof;
11714*4882a593Smuzhiyun 
11715*4882a593Smuzhiyun 		count += block_hdr.size * sizeof(u16);
11716*4882a593Smuzhiyun 	}
11717*4882a593Smuzhiyun 	return 0;
11718*4882a593Smuzhiyun eof:
11719*4882a593Smuzhiyun 	pr_err("Firmware is truncated at pos %u/%u\n", count, size);
11720*4882a593Smuzhiyun 	return -EINVAL;
11721*4882a593Smuzhiyun }
11722*4882a593Smuzhiyun 
11723*4882a593Smuzhiyun /*
11724*4882a593Smuzhiyun  * drx_ctrl_u_code - Handle microcode upload or verify.
11725*4882a593Smuzhiyun  * @dev_addr: Address of device.
11726*4882a593Smuzhiyun  * @mc_info:  Pointer to information about microcode data.
11727*4882a593Smuzhiyun  * @action:  Either UCODE_UPLOAD or UCODE_VERIFY
11728*4882a593Smuzhiyun  *
11729*4882a593Smuzhiyun  * This function returns:
11730*4882a593Smuzhiyun  *	0:
11731*4882a593Smuzhiyun  *		- In case of UCODE_UPLOAD: code is successfully uploaded.
11732*4882a593Smuzhiyun  *               - In case of UCODE_VERIFY: image on device is equal to
11733*4882a593Smuzhiyun  *		  image provided to this control function.
11734*4882a593Smuzhiyun  *	-EIO:
11735*4882a593Smuzhiyun  *		- In case of UCODE_UPLOAD: I2C error.
11736*4882a593Smuzhiyun  *		- In case of UCODE_VERIFY: I2C error or image on device
11737*4882a593Smuzhiyun  *		  is not equal to image provided to this control function.
11738*4882a593Smuzhiyun  *	-EINVAL:
11739*4882a593Smuzhiyun  *		- Invalid arguments.
11740*4882a593Smuzhiyun  *		- Provided image is corrupt
11741*4882a593Smuzhiyun  */
drx_ctrl_u_code(struct drx_demod_instance * demod,struct drxu_code_info * mc_info,enum drxu_code_action action)11742*4882a593Smuzhiyun static int drx_ctrl_u_code(struct drx_demod_instance *demod,
11743*4882a593Smuzhiyun 		       struct drxu_code_info *mc_info,
11744*4882a593Smuzhiyun 		       enum drxu_code_action action)
11745*4882a593Smuzhiyun {
11746*4882a593Smuzhiyun 	struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
11747*4882a593Smuzhiyun 	int rc;
11748*4882a593Smuzhiyun 	u16 i = 0;
11749*4882a593Smuzhiyun 	u16 mc_nr_of_blks = 0;
11750*4882a593Smuzhiyun 	u16 mc_magic_word = 0;
11751*4882a593Smuzhiyun 	const u8 *mc_data_init = NULL;
11752*4882a593Smuzhiyun 	u8 *mc_data = NULL;
11753*4882a593Smuzhiyun 	unsigned size;
11754*4882a593Smuzhiyun 	char *mc_file;
11755*4882a593Smuzhiyun 
11756*4882a593Smuzhiyun 	/* Check arguments */
11757*4882a593Smuzhiyun 	if (!mc_info || !mc_info->mc_file)
11758*4882a593Smuzhiyun 		return -EINVAL;
11759*4882a593Smuzhiyun 
11760*4882a593Smuzhiyun 	mc_file = mc_info->mc_file;
11761*4882a593Smuzhiyun 
11762*4882a593Smuzhiyun 	if (!demod->firmware) {
11763*4882a593Smuzhiyun 		const struct firmware *fw = NULL;
11764*4882a593Smuzhiyun 
11765*4882a593Smuzhiyun 		rc = request_firmware(&fw, mc_file, demod->i2c->dev.parent);
11766*4882a593Smuzhiyun 		if (rc < 0) {
11767*4882a593Smuzhiyun 			pr_err("Couldn't read firmware %s\n", mc_file);
11768*4882a593Smuzhiyun 			return rc;
11769*4882a593Smuzhiyun 		}
11770*4882a593Smuzhiyun 		demod->firmware = fw;
11771*4882a593Smuzhiyun 
11772*4882a593Smuzhiyun 		if (demod->firmware->size < 2 * sizeof(u16)) {
11773*4882a593Smuzhiyun 			rc = -EINVAL;
11774*4882a593Smuzhiyun 			pr_err("Firmware is too short!\n");
11775*4882a593Smuzhiyun 			goto release;
11776*4882a593Smuzhiyun 		}
11777*4882a593Smuzhiyun 
11778*4882a593Smuzhiyun 		pr_info("Firmware %s, size %zu\n",
11779*4882a593Smuzhiyun 			mc_file, demod->firmware->size);
11780*4882a593Smuzhiyun 	}
11781*4882a593Smuzhiyun 
11782*4882a593Smuzhiyun 	mc_data_init = demod->firmware->data;
11783*4882a593Smuzhiyun 	size = demod->firmware->size;
11784*4882a593Smuzhiyun 
11785*4882a593Smuzhiyun 	mc_data = (void *)mc_data_init;
11786*4882a593Smuzhiyun 	/* Check data */
11787*4882a593Smuzhiyun 	mc_magic_word = be16_to_cpu(*(__be16 *)(mc_data));
11788*4882a593Smuzhiyun 	mc_data += sizeof(u16);
11789*4882a593Smuzhiyun 	mc_nr_of_blks = be16_to_cpu(*(__be16 *)(mc_data));
11790*4882a593Smuzhiyun 	mc_data += sizeof(u16);
11791*4882a593Smuzhiyun 
11792*4882a593Smuzhiyun 	if ((mc_magic_word != DRX_UCODE_MAGIC_WORD) || (mc_nr_of_blks == 0)) {
11793*4882a593Smuzhiyun 		rc = -EINVAL;
11794*4882a593Smuzhiyun 		pr_err("Firmware magic word doesn't match\n");
11795*4882a593Smuzhiyun 		goto release;
11796*4882a593Smuzhiyun 	}
11797*4882a593Smuzhiyun 
11798*4882a593Smuzhiyun 	if (action == UCODE_UPLOAD) {
11799*4882a593Smuzhiyun 		rc = drx_check_firmware(demod, (u8 *)mc_data_init, size);
11800*4882a593Smuzhiyun 		if (rc)
11801*4882a593Smuzhiyun 			goto release;
11802*4882a593Smuzhiyun 		pr_info("Uploading firmware %s\n", mc_file);
11803*4882a593Smuzhiyun 	} else {
11804*4882a593Smuzhiyun 		pr_info("Verifying if firmware upload was ok.\n");
11805*4882a593Smuzhiyun 	}
11806*4882a593Smuzhiyun 
11807*4882a593Smuzhiyun 	/* Process microcode blocks */
11808*4882a593Smuzhiyun 	for (i = 0; i < mc_nr_of_blks; i++) {
11809*4882a593Smuzhiyun 		struct drxu_code_block_hdr block_hdr;
11810*4882a593Smuzhiyun 		u16 mc_block_nr_bytes = 0;
11811*4882a593Smuzhiyun 
11812*4882a593Smuzhiyun 		/* Process block header */
11813*4882a593Smuzhiyun 		block_hdr.addr = be32_to_cpu(*(__be32 *)(mc_data));
11814*4882a593Smuzhiyun 		mc_data += sizeof(u32);
11815*4882a593Smuzhiyun 		block_hdr.size = be16_to_cpu(*(__be16 *)(mc_data));
11816*4882a593Smuzhiyun 		mc_data += sizeof(u16);
11817*4882a593Smuzhiyun 		block_hdr.flags = be16_to_cpu(*(__be16 *)(mc_data));
11818*4882a593Smuzhiyun 		mc_data += sizeof(u16);
11819*4882a593Smuzhiyun 		block_hdr.CRC = be16_to_cpu(*(__be16 *)(mc_data));
11820*4882a593Smuzhiyun 		mc_data += sizeof(u16);
11821*4882a593Smuzhiyun 
11822*4882a593Smuzhiyun 		pr_debug("%zd: addr %u, size %u, flags 0x%04x, CRC 0x%04x\n",
11823*4882a593Smuzhiyun 			(mc_data - mc_data_init), block_hdr.addr,
11824*4882a593Smuzhiyun 			 block_hdr.size, block_hdr.flags, block_hdr.CRC);
11825*4882a593Smuzhiyun 
11826*4882a593Smuzhiyun 		/* Check block header on:
11827*4882a593Smuzhiyun 		   - data larger than 64Kb
11828*4882a593Smuzhiyun 		   - if CRC enabled check CRC
11829*4882a593Smuzhiyun 		 */
11830*4882a593Smuzhiyun 		if ((block_hdr.size > 0x7FFF) ||
11831*4882a593Smuzhiyun 		    (((block_hdr.flags & DRX_UCODE_CRC_FLAG) != 0) &&
11832*4882a593Smuzhiyun 		     (block_hdr.CRC != drx_u_code_compute_crc(mc_data, block_hdr.size)))
11833*4882a593Smuzhiyun 		    ) {
11834*4882a593Smuzhiyun 			/* Wrong data ! */
11835*4882a593Smuzhiyun 			rc = -EINVAL;
11836*4882a593Smuzhiyun 			pr_err("firmware CRC is wrong\n");
11837*4882a593Smuzhiyun 			goto release;
11838*4882a593Smuzhiyun 		}
11839*4882a593Smuzhiyun 
11840*4882a593Smuzhiyun 		if (!block_hdr.size)
11841*4882a593Smuzhiyun 			continue;
11842*4882a593Smuzhiyun 
11843*4882a593Smuzhiyun 		mc_block_nr_bytes = block_hdr.size * ((u16) sizeof(u16));
11844*4882a593Smuzhiyun 
11845*4882a593Smuzhiyun 		/* Perform the desired action */
11846*4882a593Smuzhiyun 		switch (action) {
11847*4882a593Smuzhiyun 		case UCODE_UPLOAD:	/* Upload microcode */
11848*4882a593Smuzhiyun 			if (drxdap_fasi_write_block(dev_addr,
11849*4882a593Smuzhiyun 							block_hdr.addr,
11850*4882a593Smuzhiyun 							mc_block_nr_bytes,
11851*4882a593Smuzhiyun 							mc_data, 0x0000)) {
11852*4882a593Smuzhiyun 				rc = -EIO;
11853*4882a593Smuzhiyun 				pr_err("error writing firmware at pos %zd\n",
11854*4882a593Smuzhiyun 				       mc_data - mc_data_init);
11855*4882a593Smuzhiyun 				goto release;
11856*4882a593Smuzhiyun 			}
11857*4882a593Smuzhiyun 			break;
11858*4882a593Smuzhiyun 		case UCODE_VERIFY: {	/* Verify uploaded microcode */
11859*4882a593Smuzhiyun 			int result = 0;
11860*4882a593Smuzhiyun 			u8 mc_data_buffer[DRX_UCODE_MAX_BUF_SIZE];
11861*4882a593Smuzhiyun 			u32 bytes_to_comp = 0;
11862*4882a593Smuzhiyun 			u32 bytes_left = mc_block_nr_bytes;
11863*4882a593Smuzhiyun 			u32 curr_addr = block_hdr.addr;
11864*4882a593Smuzhiyun 			u8 *curr_ptr = mc_data;
11865*4882a593Smuzhiyun 
11866*4882a593Smuzhiyun 			while (bytes_left != 0) {
11867*4882a593Smuzhiyun 				if (bytes_left > DRX_UCODE_MAX_BUF_SIZE)
11868*4882a593Smuzhiyun 					bytes_to_comp = DRX_UCODE_MAX_BUF_SIZE;
11869*4882a593Smuzhiyun 				else
11870*4882a593Smuzhiyun 					bytes_to_comp = bytes_left;
11871*4882a593Smuzhiyun 
11872*4882a593Smuzhiyun 				if (drxdap_fasi_read_block(dev_addr,
11873*4882a593Smuzhiyun 						    curr_addr,
11874*4882a593Smuzhiyun 						    (u16)bytes_to_comp,
11875*4882a593Smuzhiyun 						    (u8 *)mc_data_buffer,
11876*4882a593Smuzhiyun 						    0x0000)) {
11877*4882a593Smuzhiyun 					pr_err("error reading firmware at pos %zd\n",
11878*4882a593Smuzhiyun 					       mc_data - mc_data_init);
11879*4882a593Smuzhiyun 					return -EIO;
11880*4882a593Smuzhiyun 				}
11881*4882a593Smuzhiyun 
11882*4882a593Smuzhiyun 				result = memcmp(curr_ptr, mc_data_buffer,
11883*4882a593Smuzhiyun 						bytes_to_comp);
11884*4882a593Smuzhiyun 
11885*4882a593Smuzhiyun 				if (result) {
11886*4882a593Smuzhiyun 					pr_err("error verifying firmware at pos %zd\n",
11887*4882a593Smuzhiyun 					       mc_data - mc_data_init);
11888*4882a593Smuzhiyun 					return -EIO;
11889*4882a593Smuzhiyun 				}
11890*4882a593Smuzhiyun 
11891*4882a593Smuzhiyun 				curr_addr += ((dr_xaddr_t)(bytes_to_comp / 2));
11892*4882a593Smuzhiyun 				curr_ptr =&(curr_ptr[bytes_to_comp]);
11893*4882a593Smuzhiyun 				bytes_left -=((u32) bytes_to_comp);
11894*4882a593Smuzhiyun 			}
11895*4882a593Smuzhiyun 			break;
11896*4882a593Smuzhiyun 		}
11897*4882a593Smuzhiyun 		default:
11898*4882a593Smuzhiyun 			return -EINVAL;
11899*4882a593Smuzhiyun 			break;
11900*4882a593Smuzhiyun 
11901*4882a593Smuzhiyun 		}
11902*4882a593Smuzhiyun 		mc_data += mc_block_nr_bytes;
11903*4882a593Smuzhiyun 	}
11904*4882a593Smuzhiyun 
11905*4882a593Smuzhiyun 	return 0;
11906*4882a593Smuzhiyun 
11907*4882a593Smuzhiyun release:
11908*4882a593Smuzhiyun 	release_firmware(demod->firmware);
11909*4882a593Smuzhiyun 	demod->firmware = NULL;
11910*4882a593Smuzhiyun 
11911*4882a593Smuzhiyun 	return rc;
11912*4882a593Smuzhiyun }
11913*4882a593Smuzhiyun 
11914*4882a593Smuzhiyun /* caller is expected to check if lna is supported before enabling */
drxj_set_lna_state(struct drx_demod_instance * demod,bool state)11915*4882a593Smuzhiyun static int drxj_set_lna_state(struct drx_demod_instance *demod, bool state)
11916*4882a593Smuzhiyun {
11917*4882a593Smuzhiyun 	struct drxuio_cfg uio_cfg;
11918*4882a593Smuzhiyun 	struct drxuio_data uio_data;
11919*4882a593Smuzhiyun 	int result;
11920*4882a593Smuzhiyun 
11921*4882a593Smuzhiyun 	uio_cfg.uio = DRX_UIO1;
11922*4882a593Smuzhiyun 	uio_cfg.mode = DRX_UIO_MODE_READWRITE;
11923*4882a593Smuzhiyun 	/* Configure user-I/O #3: enable read/write */
11924*4882a593Smuzhiyun 	result = ctrl_set_uio_cfg(demod, &uio_cfg);
11925*4882a593Smuzhiyun 	if (result) {
11926*4882a593Smuzhiyun 		pr_err("Failed to setup LNA GPIO!\n");
11927*4882a593Smuzhiyun 		return result;
11928*4882a593Smuzhiyun 	}
11929*4882a593Smuzhiyun 
11930*4882a593Smuzhiyun 	uio_data.uio = DRX_UIO1;
11931*4882a593Smuzhiyun 	uio_data.value = state;
11932*4882a593Smuzhiyun 	result = ctrl_uio_write(demod, &uio_data);
11933*4882a593Smuzhiyun 	if (result != 0) {
11934*4882a593Smuzhiyun 		pr_err("Failed to %sable LNA!\n",
11935*4882a593Smuzhiyun 		       state ? "en" : "dis");
11936*4882a593Smuzhiyun 		return result;
11937*4882a593Smuzhiyun 	}
11938*4882a593Smuzhiyun 	return 0;
11939*4882a593Smuzhiyun }
11940*4882a593Smuzhiyun 
11941*4882a593Smuzhiyun /*
11942*4882a593Smuzhiyun  * The Linux DVB Driver for Micronas DRX39xx family (drx3933j)
11943*4882a593Smuzhiyun  *
11944*4882a593Smuzhiyun  * Written by Devin Heitmueller <devin.heitmueller@kernellabs.com>
11945*4882a593Smuzhiyun  */
11946*4882a593Smuzhiyun 
drx39xxj_set_powerstate(struct dvb_frontend * fe,int enable)11947*4882a593Smuzhiyun static int drx39xxj_set_powerstate(struct dvb_frontend *fe, int enable)
11948*4882a593Smuzhiyun {
11949*4882a593Smuzhiyun 	struct drx39xxj_state *state = fe->demodulator_priv;
11950*4882a593Smuzhiyun 	struct drx_demod_instance *demod = state->demod;
11951*4882a593Smuzhiyun 	int result;
11952*4882a593Smuzhiyun 	enum drx_power_mode power_mode;
11953*4882a593Smuzhiyun 
11954*4882a593Smuzhiyun 	if (enable)
11955*4882a593Smuzhiyun 		power_mode = DRX_POWER_UP;
11956*4882a593Smuzhiyun 	else
11957*4882a593Smuzhiyun 		power_mode = DRX_POWER_DOWN;
11958*4882a593Smuzhiyun 
11959*4882a593Smuzhiyun 	result = ctrl_power_mode(demod, &power_mode);
11960*4882a593Smuzhiyun 	if (result != 0) {
11961*4882a593Smuzhiyun 		pr_err("Power state change failed\n");
11962*4882a593Smuzhiyun 		return 0;
11963*4882a593Smuzhiyun 	}
11964*4882a593Smuzhiyun 
11965*4882a593Smuzhiyun 	return 0;
11966*4882a593Smuzhiyun }
11967*4882a593Smuzhiyun 
drx39xxj_read_status(struct dvb_frontend * fe,enum fe_status * status)11968*4882a593Smuzhiyun static int drx39xxj_read_status(struct dvb_frontend *fe, enum fe_status *status)
11969*4882a593Smuzhiyun {
11970*4882a593Smuzhiyun 	struct drx39xxj_state *state = fe->demodulator_priv;
11971*4882a593Smuzhiyun 	struct drx_demod_instance *demod = state->demod;
11972*4882a593Smuzhiyun 	int result;
11973*4882a593Smuzhiyun 	enum drx_lock_status lock_status;
11974*4882a593Smuzhiyun 
11975*4882a593Smuzhiyun 	*status = 0;
11976*4882a593Smuzhiyun 
11977*4882a593Smuzhiyun 	result = ctrl_lock_status(demod, &lock_status);
11978*4882a593Smuzhiyun 	if (result != 0) {
11979*4882a593Smuzhiyun 		pr_err("drx39xxj: could not get lock status!\n");
11980*4882a593Smuzhiyun 		*status = 0;
11981*4882a593Smuzhiyun 	}
11982*4882a593Smuzhiyun 
11983*4882a593Smuzhiyun 	switch (lock_status) {
11984*4882a593Smuzhiyun 	case DRX_NEVER_LOCK:
11985*4882a593Smuzhiyun 		*status = 0;
11986*4882a593Smuzhiyun 		pr_err("drx says NEVER_LOCK\n");
11987*4882a593Smuzhiyun 		break;
11988*4882a593Smuzhiyun 	case DRX_NOT_LOCKED:
11989*4882a593Smuzhiyun 		*status = 0;
11990*4882a593Smuzhiyun 		break;
11991*4882a593Smuzhiyun 	case DRX_LOCK_STATE_1:
11992*4882a593Smuzhiyun 	case DRX_LOCK_STATE_2:
11993*4882a593Smuzhiyun 	case DRX_LOCK_STATE_3:
11994*4882a593Smuzhiyun 	case DRX_LOCK_STATE_4:
11995*4882a593Smuzhiyun 	case DRX_LOCK_STATE_5:
11996*4882a593Smuzhiyun 	case DRX_LOCK_STATE_6:
11997*4882a593Smuzhiyun 	case DRX_LOCK_STATE_7:
11998*4882a593Smuzhiyun 	case DRX_LOCK_STATE_8:
11999*4882a593Smuzhiyun 	case DRX_LOCK_STATE_9:
12000*4882a593Smuzhiyun 		*status = FE_HAS_SIGNAL
12001*4882a593Smuzhiyun 		    | FE_HAS_CARRIER | FE_HAS_VITERBI | FE_HAS_SYNC;
12002*4882a593Smuzhiyun 		break;
12003*4882a593Smuzhiyun 	case DRX_LOCKED:
12004*4882a593Smuzhiyun 		*status = FE_HAS_SIGNAL
12005*4882a593Smuzhiyun 		    | FE_HAS_CARRIER
12006*4882a593Smuzhiyun 		    | FE_HAS_VITERBI | FE_HAS_SYNC | FE_HAS_LOCK;
12007*4882a593Smuzhiyun 		break;
12008*4882a593Smuzhiyun 	default:
12009*4882a593Smuzhiyun 		pr_err("Lock state unknown %d\n", lock_status);
12010*4882a593Smuzhiyun 	}
12011*4882a593Smuzhiyun 	ctrl_sig_quality(demod, lock_status);
12012*4882a593Smuzhiyun 
12013*4882a593Smuzhiyun 	return 0;
12014*4882a593Smuzhiyun }
12015*4882a593Smuzhiyun 
drx39xxj_read_ber(struct dvb_frontend * fe,u32 * ber)12016*4882a593Smuzhiyun static int drx39xxj_read_ber(struct dvb_frontend *fe, u32 *ber)
12017*4882a593Smuzhiyun {
12018*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &fe->dtv_property_cache;
12019*4882a593Smuzhiyun 
12020*4882a593Smuzhiyun 	if (p->pre_bit_error.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
12021*4882a593Smuzhiyun 		*ber = 0;
12022*4882a593Smuzhiyun 		return 0;
12023*4882a593Smuzhiyun 	}
12024*4882a593Smuzhiyun 
12025*4882a593Smuzhiyun 	if (!p->pre_bit_count.stat[0].uvalue) {
12026*4882a593Smuzhiyun 		if (!p->pre_bit_error.stat[0].uvalue)
12027*4882a593Smuzhiyun 			*ber = 0;
12028*4882a593Smuzhiyun 		else
12029*4882a593Smuzhiyun 			*ber = 1000000;
12030*4882a593Smuzhiyun 	} else {
12031*4882a593Smuzhiyun 		*ber = frac_times1e6(p->pre_bit_error.stat[0].uvalue,
12032*4882a593Smuzhiyun 				     p->pre_bit_count.stat[0].uvalue);
12033*4882a593Smuzhiyun 	}
12034*4882a593Smuzhiyun 	return 0;
12035*4882a593Smuzhiyun }
12036*4882a593Smuzhiyun 
drx39xxj_read_signal_strength(struct dvb_frontend * fe,u16 * strength)12037*4882a593Smuzhiyun static int drx39xxj_read_signal_strength(struct dvb_frontend *fe,
12038*4882a593Smuzhiyun 					 u16 *strength)
12039*4882a593Smuzhiyun {
12040*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &fe->dtv_property_cache;
12041*4882a593Smuzhiyun 
12042*4882a593Smuzhiyun 	if (p->strength.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
12043*4882a593Smuzhiyun 		*strength = 0;
12044*4882a593Smuzhiyun 		return 0;
12045*4882a593Smuzhiyun 	}
12046*4882a593Smuzhiyun 
12047*4882a593Smuzhiyun 	*strength = p->strength.stat[0].uvalue;
12048*4882a593Smuzhiyun 	return 0;
12049*4882a593Smuzhiyun }
12050*4882a593Smuzhiyun 
drx39xxj_read_snr(struct dvb_frontend * fe,u16 * snr)12051*4882a593Smuzhiyun static int drx39xxj_read_snr(struct dvb_frontend *fe, u16 *snr)
12052*4882a593Smuzhiyun {
12053*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &fe->dtv_property_cache;
12054*4882a593Smuzhiyun 	u64 tmp64;
12055*4882a593Smuzhiyun 
12056*4882a593Smuzhiyun 	if (p->cnr.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
12057*4882a593Smuzhiyun 		*snr = 0;
12058*4882a593Smuzhiyun 		return 0;
12059*4882a593Smuzhiyun 	}
12060*4882a593Smuzhiyun 
12061*4882a593Smuzhiyun 	tmp64 = p->cnr.stat[0].svalue;
12062*4882a593Smuzhiyun 	do_div(tmp64, 10);
12063*4882a593Smuzhiyun 	*snr = tmp64;
12064*4882a593Smuzhiyun 	return 0;
12065*4882a593Smuzhiyun }
12066*4882a593Smuzhiyun 
drx39xxj_read_ucblocks(struct dvb_frontend * fe,u32 * ucb)12067*4882a593Smuzhiyun static int drx39xxj_read_ucblocks(struct dvb_frontend *fe, u32 *ucb)
12068*4882a593Smuzhiyun {
12069*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &fe->dtv_property_cache;
12070*4882a593Smuzhiyun 
12071*4882a593Smuzhiyun 	if (p->block_error.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
12072*4882a593Smuzhiyun 		*ucb = 0;
12073*4882a593Smuzhiyun 		return 0;
12074*4882a593Smuzhiyun 	}
12075*4882a593Smuzhiyun 
12076*4882a593Smuzhiyun 	*ucb = p->block_error.stat[0].uvalue;
12077*4882a593Smuzhiyun 	return 0;
12078*4882a593Smuzhiyun }
12079*4882a593Smuzhiyun 
drx39xxj_set_frontend(struct dvb_frontend * fe)12080*4882a593Smuzhiyun static int drx39xxj_set_frontend(struct dvb_frontend *fe)
12081*4882a593Smuzhiyun {
12082*4882a593Smuzhiyun #ifdef DJH_DEBUG
12083*4882a593Smuzhiyun 	int i;
12084*4882a593Smuzhiyun #endif
12085*4882a593Smuzhiyun 	struct dtv_frontend_properties *p = &fe->dtv_property_cache;
12086*4882a593Smuzhiyun 	struct drx39xxj_state *state = fe->demodulator_priv;
12087*4882a593Smuzhiyun 	struct drx_demod_instance *demod = state->demod;
12088*4882a593Smuzhiyun 	enum drx_standard standard = DRX_STANDARD_8VSB;
12089*4882a593Smuzhiyun 	struct drx_channel channel;
12090*4882a593Smuzhiyun 	int result;
12091*4882a593Smuzhiyun 	static const struct drx_channel def_channel = {
12092*4882a593Smuzhiyun 		/* frequency      */ 0,
12093*4882a593Smuzhiyun 		/* bandwidth      */ DRX_BANDWIDTH_6MHZ,
12094*4882a593Smuzhiyun 		/* mirror         */ DRX_MIRROR_NO,
12095*4882a593Smuzhiyun 		/* constellation  */ DRX_CONSTELLATION_AUTO,
12096*4882a593Smuzhiyun 		/* hierarchy      */ DRX_HIERARCHY_UNKNOWN,
12097*4882a593Smuzhiyun 		/* priority       */ DRX_PRIORITY_UNKNOWN,
12098*4882a593Smuzhiyun 		/* coderate       */ DRX_CODERATE_UNKNOWN,
12099*4882a593Smuzhiyun 		/* guard          */ DRX_GUARD_UNKNOWN,
12100*4882a593Smuzhiyun 		/* fftmode        */ DRX_FFTMODE_UNKNOWN,
12101*4882a593Smuzhiyun 		/* classification */ DRX_CLASSIFICATION_AUTO,
12102*4882a593Smuzhiyun 		/* symbolrate     */ 5057000,
12103*4882a593Smuzhiyun 		/* interleavemode */ DRX_INTERLEAVEMODE_UNKNOWN,
12104*4882a593Smuzhiyun 		/* ldpc           */ DRX_LDPC_UNKNOWN,
12105*4882a593Smuzhiyun 		/* carrier        */ DRX_CARRIER_UNKNOWN,
12106*4882a593Smuzhiyun 		/* frame mode     */ DRX_FRAMEMODE_UNKNOWN
12107*4882a593Smuzhiyun 	};
12108*4882a593Smuzhiyun 	u32 constellation = DRX_CONSTELLATION_AUTO;
12109*4882a593Smuzhiyun 
12110*4882a593Smuzhiyun 	/* Bring the demod out of sleep */
12111*4882a593Smuzhiyun 	drx39xxj_set_powerstate(fe, 1);
12112*4882a593Smuzhiyun 
12113*4882a593Smuzhiyun 	if (fe->ops.tuner_ops.set_params) {
12114*4882a593Smuzhiyun 		u32 int_freq;
12115*4882a593Smuzhiyun 
12116*4882a593Smuzhiyun 		if (fe->ops.i2c_gate_ctrl)
12117*4882a593Smuzhiyun 			fe->ops.i2c_gate_ctrl(fe, 1);
12118*4882a593Smuzhiyun 
12119*4882a593Smuzhiyun 		/* Set tuner to desired frequency and standard */
12120*4882a593Smuzhiyun 		fe->ops.tuner_ops.set_params(fe);
12121*4882a593Smuzhiyun 
12122*4882a593Smuzhiyun 		/* Use the tuner's IF */
12123*4882a593Smuzhiyun 		if (fe->ops.tuner_ops.get_if_frequency) {
12124*4882a593Smuzhiyun 			fe->ops.tuner_ops.get_if_frequency(fe, &int_freq);
12125*4882a593Smuzhiyun 			demod->my_common_attr->intermediate_freq = int_freq / 1000;
12126*4882a593Smuzhiyun 		}
12127*4882a593Smuzhiyun 
12128*4882a593Smuzhiyun 		if (fe->ops.i2c_gate_ctrl)
12129*4882a593Smuzhiyun 			fe->ops.i2c_gate_ctrl(fe, 0);
12130*4882a593Smuzhiyun 	}
12131*4882a593Smuzhiyun 
12132*4882a593Smuzhiyun 	switch (p->delivery_system) {
12133*4882a593Smuzhiyun 	case SYS_ATSC:
12134*4882a593Smuzhiyun 		standard = DRX_STANDARD_8VSB;
12135*4882a593Smuzhiyun 		break;
12136*4882a593Smuzhiyun 	case SYS_DVBC_ANNEX_B:
12137*4882a593Smuzhiyun 		standard = DRX_STANDARD_ITU_B;
12138*4882a593Smuzhiyun 
12139*4882a593Smuzhiyun 		switch (p->modulation) {
12140*4882a593Smuzhiyun 		case QAM_64:
12141*4882a593Smuzhiyun 			constellation = DRX_CONSTELLATION_QAM64;
12142*4882a593Smuzhiyun 			break;
12143*4882a593Smuzhiyun 		case QAM_256:
12144*4882a593Smuzhiyun 			constellation = DRX_CONSTELLATION_QAM256;
12145*4882a593Smuzhiyun 			break;
12146*4882a593Smuzhiyun 		default:
12147*4882a593Smuzhiyun 			constellation = DRX_CONSTELLATION_AUTO;
12148*4882a593Smuzhiyun 			break;
12149*4882a593Smuzhiyun 		}
12150*4882a593Smuzhiyun 		break;
12151*4882a593Smuzhiyun 	default:
12152*4882a593Smuzhiyun 		return -EINVAL;
12153*4882a593Smuzhiyun 	}
12154*4882a593Smuzhiyun 	/* Set the standard (will be powered up if necessary */
12155*4882a593Smuzhiyun 	result = ctrl_set_standard(demod, &standard);
12156*4882a593Smuzhiyun 	if (result != 0) {
12157*4882a593Smuzhiyun 		pr_err("Failed to set standard! result=%02x\n",
12158*4882a593Smuzhiyun 			result);
12159*4882a593Smuzhiyun 		return -EINVAL;
12160*4882a593Smuzhiyun 	}
12161*4882a593Smuzhiyun 
12162*4882a593Smuzhiyun 	/* set channel parameters */
12163*4882a593Smuzhiyun 	channel = def_channel;
12164*4882a593Smuzhiyun 	channel.frequency = p->frequency / 1000;
12165*4882a593Smuzhiyun 	channel.bandwidth = DRX_BANDWIDTH_6MHZ;
12166*4882a593Smuzhiyun 	channel.constellation = constellation;
12167*4882a593Smuzhiyun 
12168*4882a593Smuzhiyun 	/* program channel */
12169*4882a593Smuzhiyun 	result = ctrl_set_channel(demod, &channel);
12170*4882a593Smuzhiyun 	if (result != 0) {
12171*4882a593Smuzhiyun 		pr_err("Failed to set channel!\n");
12172*4882a593Smuzhiyun 		return -EINVAL;
12173*4882a593Smuzhiyun 	}
12174*4882a593Smuzhiyun 	/* Just for giggles, let's shut off the LNA again.... */
12175*4882a593Smuzhiyun 	drxj_set_lna_state(demod, false);
12176*4882a593Smuzhiyun 
12177*4882a593Smuzhiyun 	/* After set_frontend, except for strength, stats aren't available */
12178*4882a593Smuzhiyun 	p->strength.stat[0].scale = FE_SCALE_RELATIVE;
12179*4882a593Smuzhiyun 
12180*4882a593Smuzhiyun 	return 0;
12181*4882a593Smuzhiyun }
12182*4882a593Smuzhiyun 
drx39xxj_sleep(struct dvb_frontend * fe)12183*4882a593Smuzhiyun static int drx39xxj_sleep(struct dvb_frontend *fe)
12184*4882a593Smuzhiyun {
12185*4882a593Smuzhiyun 	/* power-down the demodulator */
12186*4882a593Smuzhiyun 	return drx39xxj_set_powerstate(fe, 0);
12187*4882a593Smuzhiyun }
12188*4882a593Smuzhiyun 
drx39xxj_i2c_gate_ctrl(struct dvb_frontend * fe,int enable)12189*4882a593Smuzhiyun static int drx39xxj_i2c_gate_ctrl(struct dvb_frontend *fe, int enable)
12190*4882a593Smuzhiyun {
12191*4882a593Smuzhiyun 	struct drx39xxj_state *state = fe->demodulator_priv;
12192*4882a593Smuzhiyun 	struct drx_demod_instance *demod = state->demod;
12193*4882a593Smuzhiyun 	bool i2c_gate_state;
12194*4882a593Smuzhiyun 	int result;
12195*4882a593Smuzhiyun 
12196*4882a593Smuzhiyun #ifdef DJH_DEBUG
12197*4882a593Smuzhiyun 	pr_debug("i2c gate call: enable=%d state=%d\n", enable,
12198*4882a593Smuzhiyun 	       state->i2c_gate_open);
12199*4882a593Smuzhiyun #endif
12200*4882a593Smuzhiyun 
12201*4882a593Smuzhiyun 	if (enable)
12202*4882a593Smuzhiyun 		i2c_gate_state = true;
12203*4882a593Smuzhiyun 	else
12204*4882a593Smuzhiyun 		i2c_gate_state = false;
12205*4882a593Smuzhiyun 
12206*4882a593Smuzhiyun 	if (state->i2c_gate_open == enable) {
12207*4882a593Smuzhiyun 		/* We're already in the desired state */
12208*4882a593Smuzhiyun 		return 0;
12209*4882a593Smuzhiyun 	}
12210*4882a593Smuzhiyun 
12211*4882a593Smuzhiyun 	result = ctrl_i2c_bridge(demod, &i2c_gate_state);
12212*4882a593Smuzhiyun 	if (result != 0) {
12213*4882a593Smuzhiyun 		pr_err("drx39xxj: could not open i2c gate [%d]\n",
12214*4882a593Smuzhiyun 		       result);
12215*4882a593Smuzhiyun 		dump_stack();
12216*4882a593Smuzhiyun 	} else {
12217*4882a593Smuzhiyun 		state->i2c_gate_open = enable;
12218*4882a593Smuzhiyun 	}
12219*4882a593Smuzhiyun 	return 0;
12220*4882a593Smuzhiyun }
12221*4882a593Smuzhiyun 
drx39xxj_init(struct dvb_frontend * fe)12222*4882a593Smuzhiyun static int drx39xxj_init(struct dvb_frontend *fe)
12223*4882a593Smuzhiyun {
12224*4882a593Smuzhiyun 	struct drx39xxj_state *state = fe->demodulator_priv;
12225*4882a593Smuzhiyun 	struct drx_demod_instance *demod = state->demod;
12226*4882a593Smuzhiyun 	int rc = 0;
12227*4882a593Smuzhiyun 
12228*4882a593Smuzhiyun 	if (fe->exit == DVB_FE_DEVICE_RESUME) {
12229*4882a593Smuzhiyun 		/* so drxj_open() does what it needs to do */
12230*4882a593Smuzhiyun 		demod->my_common_attr->is_opened = false;
12231*4882a593Smuzhiyun 		rc = drxj_open(demod);
12232*4882a593Smuzhiyun 		if (rc != 0)
12233*4882a593Smuzhiyun 			pr_err("drx39xxj_init(): DRX open failed rc=%d!\n", rc);
12234*4882a593Smuzhiyun 	} else
12235*4882a593Smuzhiyun 		drx39xxj_set_powerstate(fe, 1);
12236*4882a593Smuzhiyun 
12237*4882a593Smuzhiyun 	return rc;
12238*4882a593Smuzhiyun }
12239*4882a593Smuzhiyun 
drx39xxj_set_lna(struct dvb_frontend * fe)12240*4882a593Smuzhiyun static int drx39xxj_set_lna(struct dvb_frontend *fe)
12241*4882a593Smuzhiyun {
12242*4882a593Smuzhiyun 	struct dtv_frontend_properties *c = &fe->dtv_property_cache;
12243*4882a593Smuzhiyun 	struct drx39xxj_state *state = fe->demodulator_priv;
12244*4882a593Smuzhiyun 	struct drx_demod_instance *demod = state->demod;
12245*4882a593Smuzhiyun 	struct drxj_data *ext_attr = demod->my_ext_attr;
12246*4882a593Smuzhiyun 
12247*4882a593Smuzhiyun 	if (c->lna) {
12248*4882a593Smuzhiyun 		if (!ext_attr->has_lna) {
12249*4882a593Smuzhiyun 			pr_err("LNA is not supported on this device!\n");
12250*4882a593Smuzhiyun 			return -EINVAL;
12251*4882a593Smuzhiyun 
12252*4882a593Smuzhiyun 		}
12253*4882a593Smuzhiyun 	}
12254*4882a593Smuzhiyun 
12255*4882a593Smuzhiyun 	return drxj_set_lna_state(demod, c->lna);
12256*4882a593Smuzhiyun }
12257*4882a593Smuzhiyun 
drx39xxj_get_tune_settings(struct dvb_frontend * fe,struct dvb_frontend_tune_settings * tune)12258*4882a593Smuzhiyun static int drx39xxj_get_tune_settings(struct dvb_frontend *fe,
12259*4882a593Smuzhiyun 				      struct dvb_frontend_tune_settings *tune)
12260*4882a593Smuzhiyun {
12261*4882a593Smuzhiyun 	tune->min_delay_ms = 1000;
12262*4882a593Smuzhiyun 	return 0;
12263*4882a593Smuzhiyun }
12264*4882a593Smuzhiyun 
drx39xxj_release(struct dvb_frontend * fe)12265*4882a593Smuzhiyun static void drx39xxj_release(struct dvb_frontend *fe)
12266*4882a593Smuzhiyun {
12267*4882a593Smuzhiyun 	struct drx39xxj_state *state = fe->demodulator_priv;
12268*4882a593Smuzhiyun 	struct drx_demod_instance *demod = state->demod;
12269*4882a593Smuzhiyun 
12270*4882a593Smuzhiyun 	/* if device is removed don't access it */
12271*4882a593Smuzhiyun 	if (fe->exit != DVB_FE_DEVICE_REMOVED)
12272*4882a593Smuzhiyun 		drxj_close(demod);
12273*4882a593Smuzhiyun 
12274*4882a593Smuzhiyun 	kfree(demod->my_ext_attr);
12275*4882a593Smuzhiyun 	kfree(demod->my_common_attr);
12276*4882a593Smuzhiyun 	kfree(demod->my_i2c_dev_addr);
12277*4882a593Smuzhiyun 	release_firmware(demod->firmware);
12278*4882a593Smuzhiyun 	kfree(demod);
12279*4882a593Smuzhiyun 	kfree(state);
12280*4882a593Smuzhiyun }
12281*4882a593Smuzhiyun 
12282*4882a593Smuzhiyun static const struct dvb_frontend_ops drx39xxj_ops;
12283*4882a593Smuzhiyun 
drx39xxj_attach(struct i2c_adapter * i2c)12284*4882a593Smuzhiyun struct dvb_frontend *drx39xxj_attach(struct i2c_adapter *i2c)
12285*4882a593Smuzhiyun {
12286*4882a593Smuzhiyun 	struct drx39xxj_state *state = NULL;
12287*4882a593Smuzhiyun 	struct i2c_device_addr *demod_addr = NULL;
12288*4882a593Smuzhiyun 	struct drx_common_attr *demod_comm_attr = NULL;
12289*4882a593Smuzhiyun 	struct drxj_data *demod_ext_attr = NULL;
12290*4882a593Smuzhiyun 	struct drx_demod_instance *demod = NULL;
12291*4882a593Smuzhiyun 	struct dtv_frontend_properties *p;
12292*4882a593Smuzhiyun 	int result;
12293*4882a593Smuzhiyun 
12294*4882a593Smuzhiyun 	/* allocate memory for the internal state */
12295*4882a593Smuzhiyun 	state = kzalloc(sizeof(struct drx39xxj_state), GFP_KERNEL);
12296*4882a593Smuzhiyun 	if (state == NULL)
12297*4882a593Smuzhiyun 		goto error;
12298*4882a593Smuzhiyun 
12299*4882a593Smuzhiyun 	demod = kmemdup(&drxj_default_demod_g,
12300*4882a593Smuzhiyun 			sizeof(struct drx_demod_instance), GFP_KERNEL);
12301*4882a593Smuzhiyun 	if (demod == NULL)
12302*4882a593Smuzhiyun 		goto error;
12303*4882a593Smuzhiyun 
12304*4882a593Smuzhiyun 	demod_addr = kmemdup(&drxj_default_addr_g,
12305*4882a593Smuzhiyun 			     sizeof(struct i2c_device_addr), GFP_KERNEL);
12306*4882a593Smuzhiyun 	if (demod_addr == NULL)
12307*4882a593Smuzhiyun 		goto error;
12308*4882a593Smuzhiyun 
12309*4882a593Smuzhiyun 	demod_comm_attr = kmemdup(&drxj_default_comm_attr_g,
12310*4882a593Smuzhiyun 				  sizeof(struct drx_common_attr), GFP_KERNEL);
12311*4882a593Smuzhiyun 	if (demod_comm_attr == NULL)
12312*4882a593Smuzhiyun 		goto error;
12313*4882a593Smuzhiyun 
12314*4882a593Smuzhiyun 	demod_ext_attr = kmemdup(&drxj_data_g, sizeof(struct drxj_data),
12315*4882a593Smuzhiyun 				 GFP_KERNEL);
12316*4882a593Smuzhiyun 	if (demod_ext_attr == NULL)
12317*4882a593Smuzhiyun 		goto error;
12318*4882a593Smuzhiyun 
12319*4882a593Smuzhiyun 	/* setup the state */
12320*4882a593Smuzhiyun 	state->i2c = i2c;
12321*4882a593Smuzhiyun 	state->demod = demod;
12322*4882a593Smuzhiyun 
12323*4882a593Smuzhiyun 	/* setup the demod data */
12324*4882a593Smuzhiyun 	demod->my_i2c_dev_addr = demod_addr;
12325*4882a593Smuzhiyun 	demod->my_common_attr = demod_comm_attr;
12326*4882a593Smuzhiyun 	demod->my_i2c_dev_addr->user_data = state;
12327*4882a593Smuzhiyun 	demod->my_common_attr->microcode_file = DRX39XX_MAIN_FIRMWARE;
12328*4882a593Smuzhiyun 	demod->my_common_attr->verify_microcode = true;
12329*4882a593Smuzhiyun 	demod->my_common_attr->intermediate_freq = 5000;
12330*4882a593Smuzhiyun 	demod->my_common_attr->current_power_mode = DRX_POWER_DOWN;
12331*4882a593Smuzhiyun 	demod->my_ext_attr = demod_ext_attr;
12332*4882a593Smuzhiyun 	((struct drxj_data *)demod_ext_attr)->uio_sma_tx_mode = DRX_UIO_MODE_READWRITE;
12333*4882a593Smuzhiyun 	demod->i2c = i2c;
12334*4882a593Smuzhiyun 
12335*4882a593Smuzhiyun 	result = drxj_open(demod);
12336*4882a593Smuzhiyun 	if (result != 0) {
12337*4882a593Smuzhiyun 		pr_err("DRX open failed!  Aborting\n");
12338*4882a593Smuzhiyun 		goto error;
12339*4882a593Smuzhiyun 	}
12340*4882a593Smuzhiyun 
12341*4882a593Smuzhiyun 	/* create dvb_frontend */
12342*4882a593Smuzhiyun 	memcpy(&state->frontend.ops, &drx39xxj_ops,
12343*4882a593Smuzhiyun 	       sizeof(struct dvb_frontend_ops));
12344*4882a593Smuzhiyun 
12345*4882a593Smuzhiyun 	state->frontend.demodulator_priv = state;
12346*4882a593Smuzhiyun 
12347*4882a593Smuzhiyun 	/* Initialize stats - needed for DVBv5 stats to work */
12348*4882a593Smuzhiyun 	p = &state->frontend.dtv_property_cache;
12349*4882a593Smuzhiyun 	p->strength.len = 1;
12350*4882a593Smuzhiyun 	p->pre_bit_count.len = 1;
12351*4882a593Smuzhiyun 	p->pre_bit_error.len = 1;
12352*4882a593Smuzhiyun 	p->post_bit_count.len = 1;
12353*4882a593Smuzhiyun 	p->post_bit_error.len = 1;
12354*4882a593Smuzhiyun 	p->block_count.len = 1;
12355*4882a593Smuzhiyun 	p->block_error.len = 1;
12356*4882a593Smuzhiyun 	p->cnr.len = 1;
12357*4882a593Smuzhiyun 
12358*4882a593Smuzhiyun 	p->strength.stat[0].scale = FE_SCALE_RELATIVE;
12359*4882a593Smuzhiyun 	p->pre_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12360*4882a593Smuzhiyun 	p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12361*4882a593Smuzhiyun 	p->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12362*4882a593Smuzhiyun 	p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12363*4882a593Smuzhiyun 	p->block_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12364*4882a593Smuzhiyun 	p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12365*4882a593Smuzhiyun 	p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12366*4882a593Smuzhiyun 
12367*4882a593Smuzhiyun 	return &state->frontend;
12368*4882a593Smuzhiyun 
12369*4882a593Smuzhiyun error:
12370*4882a593Smuzhiyun 	kfree(demod_ext_attr);
12371*4882a593Smuzhiyun 	kfree(demod_comm_attr);
12372*4882a593Smuzhiyun 	kfree(demod_addr);
12373*4882a593Smuzhiyun 	kfree(demod);
12374*4882a593Smuzhiyun 	kfree(state);
12375*4882a593Smuzhiyun 
12376*4882a593Smuzhiyun 	return NULL;
12377*4882a593Smuzhiyun }
12378*4882a593Smuzhiyun EXPORT_SYMBOL(drx39xxj_attach);
12379*4882a593Smuzhiyun 
12380*4882a593Smuzhiyun static const struct dvb_frontend_ops drx39xxj_ops = {
12381*4882a593Smuzhiyun 	.delsys = { SYS_ATSC, SYS_DVBC_ANNEX_B },
12382*4882a593Smuzhiyun 	.info = {
12383*4882a593Smuzhiyun 		 .name = "Micronas DRX39xxj family Frontend",
12384*4882a593Smuzhiyun 		 .frequency_min_hz =  51 * MHz,
12385*4882a593Smuzhiyun 		 .frequency_max_hz = 858 * MHz,
12386*4882a593Smuzhiyun 		 .frequency_stepsize_hz = 62500,
12387*4882a593Smuzhiyun 		 .caps = FE_CAN_QAM_64 | FE_CAN_QAM_256 | FE_CAN_8VSB
12388*4882a593Smuzhiyun 	},
12389*4882a593Smuzhiyun 
12390*4882a593Smuzhiyun 	.init = drx39xxj_init,
12391*4882a593Smuzhiyun 	.i2c_gate_ctrl = drx39xxj_i2c_gate_ctrl,
12392*4882a593Smuzhiyun 	.sleep = drx39xxj_sleep,
12393*4882a593Smuzhiyun 	.set_frontend = drx39xxj_set_frontend,
12394*4882a593Smuzhiyun 	.get_tune_settings = drx39xxj_get_tune_settings,
12395*4882a593Smuzhiyun 	.read_status = drx39xxj_read_status,
12396*4882a593Smuzhiyun 	.read_ber = drx39xxj_read_ber,
12397*4882a593Smuzhiyun 	.read_signal_strength = drx39xxj_read_signal_strength,
12398*4882a593Smuzhiyun 	.read_snr = drx39xxj_read_snr,
12399*4882a593Smuzhiyun 	.read_ucblocks = drx39xxj_read_ucblocks,
12400*4882a593Smuzhiyun 	.release = drx39xxj_release,
12401*4882a593Smuzhiyun 	.set_lna = drx39xxj_set_lna,
12402*4882a593Smuzhiyun };
12403*4882a593Smuzhiyun 
12404*4882a593Smuzhiyun MODULE_DESCRIPTION("Micronas DRX39xxj Frontend");
12405*4882a593Smuzhiyun MODULE_AUTHOR("Devin Heitmueller");
12406*4882a593Smuzhiyun MODULE_LICENSE("GPL");
12407*4882a593Smuzhiyun MODULE_FIRMWARE(DRX39XX_MAIN_FIRMWARE);
12408