1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Copyright (C) 2008-2009 Texas Instruments Inc
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Image Sensor Interface (ISIF) driver
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * This driver is for configuring the ISIF IP available on DM365 or any other
8*4882a593Smuzhiyun * TI SoCs. This is used for capturing yuv or bayer video or image data
9*4882a593Smuzhiyun * from a decoder or sensor. This IP is similar to the CCDC IP on DM355
10*4882a593Smuzhiyun * and DM6446, but with enhanced or additional ip blocks. The driver
11*4882a593Smuzhiyun * configures the ISIF upon commands from the vpfe bridge driver through
12*4882a593Smuzhiyun * ccdc_hw_device interface.
13*4882a593Smuzhiyun *
14*4882a593Smuzhiyun * TODO: 1) Raw bayer parameter settings and bayer capture
15*4882a593Smuzhiyun * 2) Add support for control ioctl
16*4882a593Smuzhiyun */
17*4882a593Smuzhiyun #include <linux/delay.h>
18*4882a593Smuzhiyun #include <linux/platform_device.h>
19*4882a593Smuzhiyun #include <linux/uaccess.h>
20*4882a593Smuzhiyun #include <linux/io.h>
21*4882a593Smuzhiyun #include <linux/videodev2.h>
22*4882a593Smuzhiyun #include <linux/err.h>
23*4882a593Smuzhiyun #include <linux/module.h>
24*4882a593Smuzhiyun
25*4882a593Smuzhiyun #include <media/davinci/isif.h>
26*4882a593Smuzhiyun #include <media/davinci/vpss.h>
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun #include "isif_regs.h"
29*4882a593Smuzhiyun #include "ccdc_hw_device.h"
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun /* Defaults for module configuration parameters */
32*4882a593Smuzhiyun static const struct isif_config_params_raw isif_config_defaults = {
33*4882a593Smuzhiyun .linearize = {
34*4882a593Smuzhiyun .en = 0,
35*4882a593Smuzhiyun .corr_shft = ISIF_NO_SHIFT,
36*4882a593Smuzhiyun .scale_fact = {1, 0},
37*4882a593Smuzhiyun },
38*4882a593Smuzhiyun .df_csc = {
39*4882a593Smuzhiyun .df_or_csc = 0,
40*4882a593Smuzhiyun .csc = {
41*4882a593Smuzhiyun .en = 0,
42*4882a593Smuzhiyun },
43*4882a593Smuzhiyun },
44*4882a593Smuzhiyun .dfc = {
45*4882a593Smuzhiyun .en = 0,
46*4882a593Smuzhiyun },
47*4882a593Smuzhiyun .bclamp = {
48*4882a593Smuzhiyun .en = 0,
49*4882a593Smuzhiyun },
50*4882a593Smuzhiyun .gain_offset = {
51*4882a593Smuzhiyun .gain = {
52*4882a593Smuzhiyun .r_ye = {1, 0},
53*4882a593Smuzhiyun .gr_cy = {1, 0},
54*4882a593Smuzhiyun .gb_g = {1, 0},
55*4882a593Smuzhiyun .b_mg = {1, 0},
56*4882a593Smuzhiyun },
57*4882a593Smuzhiyun },
58*4882a593Smuzhiyun .culling = {
59*4882a593Smuzhiyun .hcpat_odd = 0xff,
60*4882a593Smuzhiyun .hcpat_even = 0xff,
61*4882a593Smuzhiyun .vcpat = 0xff,
62*4882a593Smuzhiyun },
63*4882a593Smuzhiyun .compress = {
64*4882a593Smuzhiyun .alg = ISIF_ALAW,
65*4882a593Smuzhiyun },
66*4882a593Smuzhiyun };
67*4882a593Smuzhiyun
68*4882a593Smuzhiyun /* ISIF operation configuration */
69*4882a593Smuzhiyun static struct isif_oper_config {
70*4882a593Smuzhiyun struct device *dev;
71*4882a593Smuzhiyun enum vpfe_hw_if_type if_type;
72*4882a593Smuzhiyun struct isif_ycbcr_config ycbcr;
73*4882a593Smuzhiyun struct isif_params_raw bayer;
74*4882a593Smuzhiyun enum isif_data_pack data_pack;
75*4882a593Smuzhiyun /* ISIF base address */
76*4882a593Smuzhiyun void __iomem *base_addr;
77*4882a593Smuzhiyun /* ISIF Linear Table 0 */
78*4882a593Smuzhiyun void __iomem *linear_tbl0_addr;
79*4882a593Smuzhiyun /* ISIF Linear Table 1 */
80*4882a593Smuzhiyun void __iomem *linear_tbl1_addr;
81*4882a593Smuzhiyun } isif_cfg = {
82*4882a593Smuzhiyun .ycbcr = {
83*4882a593Smuzhiyun .pix_fmt = CCDC_PIXFMT_YCBCR_8BIT,
84*4882a593Smuzhiyun .frm_fmt = CCDC_FRMFMT_INTERLACED,
85*4882a593Smuzhiyun .win = ISIF_WIN_NTSC,
86*4882a593Smuzhiyun .fid_pol = VPFE_PINPOL_POSITIVE,
87*4882a593Smuzhiyun .vd_pol = VPFE_PINPOL_POSITIVE,
88*4882a593Smuzhiyun .hd_pol = VPFE_PINPOL_POSITIVE,
89*4882a593Smuzhiyun .pix_order = CCDC_PIXORDER_CBYCRY,
90*4882a593Smuzhiyun .buf_type = CCDC_BUFTYPE_FLD_INTERLEAVED,
91*4882a593Smuzhiyun },
92*4882a593Smuzhiyun .bayer = {
93*4882a593Smuzhiyun .pix_fmt = CCDC_PIXFMT_RAW,
94*4882a593Smuzhiyun .frm_fmt = CCDC_FRMFMT_PROGRESSIVE,
95*4882a593Smuzhiyun .win = ISIF_WIN_VGA,
96*4882a593Smuzhiyun .fid_pol = VPFE_PINPOL_POSITIVE,
97*4882a593Smuzhiyun .vd_pol = VPFE_PINPOL_POSITIVE,
98*4882a593Smuzhiyun .hd_pol = VPFE_PINPOL_POSITIVE,
99*4882a593Smuzhiyun .gain = {
100*4882a593Smuzhiyun .r_ye = {1, 0},
101*4882a593Smuzhiyun .gr_cy = {1, 0},
102*4882a593Smuzhiyun .gb_g = {1, 0},
103*4882a593Smuzhiyun .b_mg = {1, 0},
104*4882a593Smuzhiyun },
105*4882a593Smuzhiyun .cfa_pat = ISIF_CFA_PAT_MOSAIC,
106*4882a593Smuzhiyun .data_msb = ISIF_BIT_MSB_11,
107*4882a593Smuzhiyun .config_params = {
108*4882a593Smuzhiyun .data_shift = ISIF_NO_SHIFT,
109*4882a593Smuzhiyun .col_pat_field0 = {
110*4882a593Smuzhiyun .olop = ISIF_GREEN_BLUE,
111*4882a593Smuzhiyun .olep = ISIF_BLUE,
112*4882a593Smuzhiyun .elop = ISIF_RED,
113*4882a593Smuzhiyun .elep = ISIF_GREEN_RED,
114*4882a593Smuzhiyun },
115*4882a593Smuzhiyun .col_pat_field1 = {
116*4882a593Smuzhiyun .olop = ISIF_GREEN_BLUE,
117*4882a593Smuzhiyun .olep = ISIF_BLUE,
118*4882a593Smuzhiyun .elop = ISIF_RED,
119*4882a593Smuzhiyun .elep = ISIF_GREEN_RED,
120*4882a593Smuzhiyun },
121*4882a593Smuzhiyun .test_pat_gen = 0,
122*4882a593Smuzhiyun },
123*4882a593Smuzhiyun },
124*4882a593Smuzhiyun .data_pack = ISIF_DATA_PACK8,
125*4882a593Smuzhiyun };
126*4882a593Smuzhiyun
127*4882a593Smuzhiyun /* Raw Bayer formats */
128*4882a593Smuzhiyun static const u32 isif_raw_bayer_pix_formats[] = {
129*4882a593Smuzhiyun V4L2_PIX_FMT_SBGGR8, V4L2_PIX_FMT_SBGGR16};
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun /* Raw YUV formats */
132*4882a593Smuzhiyun static const u32 isif_raw_yuv_pix_formats[] = {
133*4882a593Smuzhiyun V4L2_PIX_FMT_UYVY, V4L2_PIX_FMT_YUYV};
134*4882a593Smuzhiyun
135*4882a593Smuzhiyun /* register access routines */
regr(u32 offset)136*4882a593Smuzhiyun static inline u32 regr(u32 offset)
137*4882a593Smuzhiyun {
138*4882a593Smuzhiyun return __raw_readl(isif_cfg.base_addr + offset);
139*4882a593Smuzhiyun }
140*4882a593Smuzhiyun
regw(u32 val,u32 offset)141*4882a593Smuzhiyun static inline void regw(u32 val, u32 offset)
142*4882a593Smuzhiyun {
143*4882a593Smuzhiyun __raw_writel(val, isif_cfg.base_addr + offset);
144*4882a593Smuzhiyun }
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun /* reg_modify() - read, modify and write register */
reg_modify(u32 mask,u32 val,u32 offset)147*4882a593Smuzhiyun static inline u32 reg_modify(u32 mask, u32 val, u32 offset)
148*4882a593Smuzhiyun {
149*4882a593Smuzhiyun u32 new_val = (regr(offset) & ~mask) | (val & mask);
150*4882a593Smuzhiyun
151*4882a593Smuzhiyun regw(new_val, offset);
152*4882a593Smuzhiyun return new_val;
153*4882a593Smuzhiyun }
154*4882a593Smuzhiyun
regw_lin_tbl(u32 val,u32 offset,int i)155*4882a593Smuzhiyun static inline void regw_lin_tbl(u32 val, u32 offset, int i)
156*4882a593Smuzhiyun {
157*4882a593Smuzhiyun if (!i)
158*4882a593Smuzhiyun __raw_writel(val, isif_cfg.linear_tbl0_addr + offset);
159*4882a593Smuzhiyun else
160*4882a593Smuzhiyun __raw_writel(val, isif_cfg.linear_tbl1_addr + offset);
161*4882a593Smuzhiyun }
162*4882a593Smuzhiyun
isif_disable_all_modules(void)163*4882a593Smuzhiyun static void isif_disable_all_modules(void)
164*4882a593Smuzhiyun {
165*4882a593Smuzhiyun /* disable BC */
166*4882a593Smuzhiyun regw(0, CLAMPCFG);
167*4882a593Smuzhiyun /* disable vdfc */
168*4882a593Smuzhiyun regw(0, DFCCTL);
169*4882a593Smuzhiyun /* disable CSC */
170*4882a593Smuzhiyun regw(0, CSCCTL);
171*4882a593Smuzhiyun /* disable linearization */
172*4882a593Smuzhiyun regw(0, LINCFG0);
173*4882a593Smuzhiyun /* disable other modules here as they are supported */
174*4882a593Smuzhiyun }
175*4882a593Smuzhiyun
isif_enable(int en)176*4882a593Smuzhiyun static void isif_enable(int en)
177*4882a593Smuzhiyun {
178*4882a593Smuzhiyun if (!en) {
179*4882a593Smuzhiyun /* Before disable isif, disable all ISIF modules */
180*4882a593Smuzhiyun isif_disable_all_modules();
181*4882a593Smuzhiyun /*
182*4882a593Smuzhiyun * wait for next VD. Assume lowest scan rate is 12 Hz. So
183*4882a593Smuzhiyun * 100 msec delay is good enough
184*4882a593Smuzhiyun */
185*4882a593Smuzhiyun msleep(100);
186*4882a593Smuzhiyun }
187*4882a593Smuzhiyun reg_modify(ISIF_SYNCEN_VDHDEN_MASK, en, SYNCEN);
188*4882a593Smuzhiyun }
189*4882a593Smuzhiyun
isif_enable_output_to_sdram(int en)190*4882a593Smuzhiyun static void isif_enable_output_to_sdram(int en)
191*4882a593Smuzhiyun {
192*4882a593Smuzhiyun reg_modify(ISIF_SYNCEN_WEN_MASK, en << ISIF_SYNCEN_WEN_SHIFT, SYNCEN);
193*4882a593Smuzhiyun }
194*4882a593Smuzhiyun
isif_config_culling(struct isif_cul * cul)195*4882a593Smuzhiyun static void isif_config_culling(struct isif_cul *cul)
196*4882a593Smuzhiyun {
197*4882a593Smuzhiyun u32 val;
198*4882a593Smuzhiyun
199*4882a593Smuzhiyun /* Horizontal pattern */
200*4882a593Smuzhiyun val = (cul->hcpat_even << CULL_PAT_EVEN_LINE_SHIFT) | cul->hcpat_odd;
201*4882a593Smuzhiyun regw(val, CULH);
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun /* vertical pattern */
204*4882a593Smuzhiyun regw(cul->vcpat, CULV);
205*4882a593Smuzhiyun
206*4882a593Smuzhiyun /* LPF */
207*4882a593Smuzhiyun reg_modify(ISIF_LPF_MASK << ISIF_LPF_SHIFT,
208*4882a593Smuzhiyun cul->en_lpf << ISIF_LPF_SHIFT, MODESET);
209*4882a593Smuzhiyun }
210*4882a593Smuzhiyun
isif_config_gain_offset(void)211*4882a593Smuzhiyun static void isif_config_gain_offset(void)
212*4882a593Smuzhiyun {
213*4882a593Smuzhiyun struct isif_gain_offsets_adj *gain_off_p =
214*4882a593Smuzhiyun &isif_cfg.bayer.config_params.gain_offset;
215*4882a593Smuzhiyun u32 val;
216*4882a593Smuzhiyun
217*4882a593Smuzhiyun val = (!!gain_off_p->gain_sdram_en << GAIN_SDRAM_EN_SHIFT) |
218*4882a593Smuzhiyun (!!gain_off_p->gain_ipipe_en << GAIN_IPIPE_EN_SHIFT) |
219*4882a593Smuzhiyun (!!gain_off_p->gain_h3a_en << GAIN_H3A_EN_SHIFT) |
220*4882a593Smuzhiyun (!!gain_off_p->offset_sdram_en << OFST_SDRAM_EN_SHIFT) |
221*4882a593Smuzhiyun (!!gain_off_p->offset_ipipe_en << OFST_IPIPE_EN_SHIFT) |
222*4882a593Smuzhiyun (!!gain_off_p->offset_h3a_en << OFST_H3A_EN_SHIFT);
223*4882a593Smuzhiyun
224*4882a593Smuzhiyun reg_modify(GAIN_OFFSET_EN_MASK, val, CGAMMAWD);
225*4882a593Smuzhiyun
226*4882a593Smuzhiyun val = (gain_off_p->gain.r_ye.integer << GAIN_INTEGER_SHIFT) |
227*4882a593Smuzhiyun gain_off_p->gain.r_ye.decimal;
228*4882a593Smuzhiyun regw(val, CRGAIN);
229*4882a593Smuzhiyun
230*4882a593Smuzhiyun val = (gain_off_p->gain.gr_cy.integer << GAIN_INTEGER_SHIFT) |
231*4882a593Smuzhiyun gain_off_p->gain.gr_cy.decimal;
232*4882a593Smuzhiyun regw(val, CGRGAIN);
233*4882a593Smuzhiyun
234*4882a593Smuzhiyun val = (gain_off_p->gain.gb_g.integer << GAIN_INTEGER_SHIFT) |
235*4882a593Smuzhiyun gain_off_p->gain.gb_g.decimal;
236*4882a593Smuzhiyun regw(val, CGBGAIN);
237*4882a593Smuzhiyun
238*4882a593Smuzhiyun val = (gain_off_p->gain.b_mg.integer << GAIN_INTEGER_SHIFT) |
239*4882a593Smuzhiyun gain_off_p->gain.b_mg.decimal;
240*4882a593Smuzhiyun regw(val, CBGAIN);
241*4882a593Smuzhiyun
242*4882a593Smuzhiyun regw(gain_off_p->offset, COFSTA);
243*4882a593Smuzhiyun }
244*4882a593Smuzhiyun
isif_restore_defaults(void)245*4882a593Smuzhiyun static void isif_restore_defaults(void)
246*4882a593Smuzhiyun {
247*4882a593Smuzhiyun enum vpss_ccdc_source_sel source = VPSS_CCDCIN;
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "\nstarting isif_restore_defaults...");
250*4882a593Smuzhiyun isif_cfg.bayer.config_params = isif_config_defaults;
251*4882a593Smuzhiyun /* Enable clock to ISIF, IPIPEIF and BL */
252*4882a593Smuzhiyun vpss_enable_clock(VPSS_CCDC_CLOCK, 1);
253*4882a593Smuzhiyun vpss_enable_clock(VPSS_IPIPEIF_CLOCK, 1);
254*4882a593Smuzhiyun vpss_enable_clock(VPSS_BL_CLOCK, 1);
255*4882a593Smuzhiyun /* Set default offset and gain */
256*4882a593Smuzhiyun isif_config_gain_offset();
257*4882a593Smuzhiyun vpss_select_ccdc_source(source);
258*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "\nEnd of isif_restore_defaults...");
259*4882a593Smuzhiyun }
260*4882a593Smuzhiyun
isif_open(struct device * device)261*4882a593Smuzhiyun static int isif_open(struct device *device)
262*4882a593Smuzhiyun {
263*4882a593Smuzhiyun isif_restore_defaults();
264*4882a593Smuzhiyun return 0;
265*4882a593Smuzhiyun }
266*4882a593Smuzhiyun
267*4882a593Smuzhiyun /* This function will configure the window size to be capture in ISIF reg */
isif_setwin(struct v4l2_rect * image_win,enum ccdc_frmfmt frm_fmt,int ppc)268*4882a593Smuzhiyun static void isif_setwin(struct v4l2_rect *image_win,
269*4882a593Smuzhiyun enum ccdc_frmfmt frm_fmt, int ppc)
270*4882a593Smuzhiyun {
271*4882a593Smuzhiyun int horz_start, horz_nr_pixels;
272*4882a593Smuzhiyun int vert_start, vert_nr_lines;
273*4882a593Smuzhiyun int mid_img = 0;
274*4882a593Smuzhiyun
275*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "\nStarting isif_setwin...");
276*4882a593Smuzhiyun /*
277*4882a593Smuzhiyun * ppc - per pixel count. indicates how many pixels per cell
278*4882a593Smuzhiyun * output to SDRAM. example, for ycbcr, it is one y and one c, so 2.
279*4882a593Smuzhiyun * raw capture this is 1
280*4882a593Smuzhiyun */
281*4882a593Smuzhiyun horz_start = image_win->left << (ppc - 1);
282*4882a593Smuzhiyun horz_nr_pixels = ((image_win->width) << (ppc - 1)) - 1;
283*4882a593Smuzhiyun
284*4882a593Smuzhiyun /* Writing the horizontal info into the registers */
285*4882a593Smuzhiyun regw(horz_start & START_PX_HOR_MASK, SPH);
286*4882a593Smuzhiyun regw(horz_nr_pixels & NUM_PX_HOR_MASK, LNH);
287*4882a593Smuzhiyun vert_start = image_win->top;
288*4882a593Smuzhiyun
289*4882a593Smuzhiyun if (frm_fmt == CCDC_FRMFMT_INTERLACED) {
290*4882a593Smuzhiyun vert_nr_lines = (image_win->height >> 1) - 1;
291*4882a593Smuzhiyun vert_start >>= 1;
292*4882a593Smuzhiyun /* To account for VD since line 0 doesn't have any data */
293*4882a593Smuzhiyun vert_start += 1;
294*4882a593Smuzhiyun } else {
295*4882a593Smuzhiyun /* To account for VD since line 0 doesn't have any data */
296*4882a593Smuzhiyun vert_start += 1;
297*4882a593Smuzhiyun vert_nr_lines = image_win->height - 1;
298*4882a593Smuzhiyun /* configure VDINT0 and VDINT1 */
299*4882a593Smuzhiyun mid_img = vert_start + (image_win->height / 2);
300*4882a593Smuzhiyun regw(mid_img, VDINT1);
301*4882a593Smuzhiyun }
302*4882a593Smuzhiyun
303*4882a593Smuzhiyun regw(0, VDINT0);
304*4882a593Smuzhiyun regw(vert_start & START_VER_ONE_MASK, SLV0);
305*4882a593Smuzhiyun regw(vert_start & START_VER_TWO_MASK, SLV1);
306*4882a593Smuzhiyun regw(vert_nr_lines & NUM_LINES_VER, LNV);
307*4882a593Smuzhiyun }
308*4882a593Smuzhiyun
isif_config_bclamp(struct isif_black_clamp * bc)309*4882a593Smuzhiyun static void isif_config_bclamp(struct isif_black_clamp *bc)
310*4882a593Smuzhiyun {
311*4882a593Smuzhiyun u32 val;
312*4882a593Smuzhiyun
313*4882a593Smuzhiyun /*
314*4882a593Smuzhiyun * DC Offset is always added to image data irrespective of bc enable
315*4882a593Smuzhiyun * status
316*4882a593Smuzhiyun */
317*4882a593Smuzhiyun regw(bc->dc_offset, CLDCOFST);
318*4882a593Smuzhiyun
319*4882a593Smuzhiyun if (bc->en) {
320*4882a593Smuzhiyun val = bc->bc_mode_color << ISIF_BC_MODE_COLOR_SHIFT;
321*4882a593Smuzhiyun
322*4882a593Smuzhiyun /* Enable BC and horizontal clamp calculation parameters */
323*4882a593Smuzhiyun val = val | 1 | (bc->horz.mode << ISIF_HORZ_BC_MODE_SHIFT);
324*4882a593Smuzhiyun
325*4882a593Smuzhiyun regw(val, CLAMPCFG);
326*4882a593Smuzhiyun
327*4882a593Smuzhiyun if (bc->horz.mode != ISIF_HORZ_BC_DISABLE) {
328*4882a593Smuzhiyun /*
329*4882a593Smuzhiyun * Window count for calculation
330*4882a593Smuzhiyun * Base window selection
331*4882a593Smuzhiyun * pixel limit
332*4882a593Smuzhiyun * Horizontal size of window
333*4882a593Smuzhiyun * vertical size of the window
334*4882a593Smuzhiyun * Horizontal start position of the window
335*4882a593Smuzhiyun * Vertical start position of the window
336*4882a593Smuzhiyun */
337*4882a593Smuzhiyun val = bc->horz.win_count_calc |
338*4882a593Smuzhiyun ((!!bc->horz.base_win_sel_calc) <<
339*4882a593Smuzhiyun ISIF_HORZ_BC_WIN_SEL_SHIFT) |
340*4882a593Smuzhiyun ((!!bc->horz.clamp_pix_limit) <<
341*4882a593Smuzhiyun ISIF_HORZ_BC_PIX_LIMIT_SHIFT) |
342*4882a593Smuzhiyun (bc->horz.win_h_sz_calc <<
343*4882a593Smuzhiyun ISIF_HORZ_BC_WIN_H_SIZE_SHIFT) |
344*4882a593Smuzhiyun (bc->horz.win_v_sz_calc <<
345*4882a593Smuzhiyun ISIF_HORZ_BC_WIN_V_SIZE_SHIFT);
346*4882a593Smuzhiyun regw(val, CLHWIN0);
347*4882a593Smuzhiyun
348*4882a593Smuzhiyun regw(bc->horz.win_start_h_calc, CLHWIN1);
349*4882a593Smuzhiyun regw(bc->horz.win_start_v_calc, CLHWIN2);
350*4882a593Smuzhiyun }
351*4882a593Smuzhiyun
352*4882a593Smuzhiyun /* vertical clamp calculation parameters */
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun /* Reset clamp value sel for previous line */
355*4882a593Smuzhiyun val |=
356*4882a593Smuzhiyun (bc->vert.reset_val_sel << ISIF_VERT_BC_RST_VAL_SEL_SHIFT) |
357*4882a593Smuzhiyun (bc->vert.line_ave_coef << ISIF_VERT_BC_LINE_AVE_COEF_SHIFT);
358*4882a593Smuzhiyun regw(val, CLVWIN0);
359*4882a593Smuzhiyun
360*4882a593Smuzhiyun /* Optical Black horizontal start position */
361*4882a593Smuzhiyun regw(bc->vert.ob_start_h, CLVWIN1);
362*4882a593Smuzhiyun /* Optical Black vertical start position */
363*4882a593Smuzhiyun regw(bc->vert.ob_start_v, CLVWIN2);
364*4882a593Smuzhiyun /* Optical Black vertical size for calculation */
365*4882a593Smuzhiyun regw(bc->vert.ob_v_sz_calc, CLVWIN3);
366*4882a593Smuzhiyun /* Vertical start position for BC subtraction */
367*4882a593Smuzhiyun regw(bc->vert_start_sub, CLSV);
368*4882a593Smuzhiyun }
369*4882a593Smuzhiyun }
370*4882a593Smuzhiyun
isif_config_linearization(struct isif_linearize * linearize)371*4882a593Smuzhiyun static void isif_config_linearization(struct isif_linearize *linearize)
372*4882a593Smuzhiyun {
373*4882a593Smuzhiyun u32 val, i;
374*4882a593Smuzhiyun
375*4882a593Smuzhiyun if (!linearize->en) {
376*4882a593Smuzhiyun regw(0, LINCFG0);
377*4882a593Smuzhiyun return;
378*4882a593Smuzhiyun }
379*4882a593Smuzhiyun
380*4882a593Smuzhiyun /* shift value for correction & enable linearization (set lsb) */
381*4882a593Smuzhiyun val = (linearize->corr_shft << ISIF_LIN_CORRSFT_SHIFT) | 1;
382*4882a593Smuzhiyun regw(val, LINCFG0);
383*4882a593Smuzhiyun
384*4882a593Smuzhiyun /* Scale factor */
385*4882a593Smuzhiyun val = ((!!linearize->scale_fact.integer) <<
386*4882a593Smuzhiyun ISIF_LIN_SCALE_FACT_INTEG_SHIFT) |
387*4882a593Smuzhiyun linearize->scale_fact.decimal;
388*4882a593Smuzhiyun regw(val, LINCFG1);
389*4882a593Smuzhiyun
390*4882a593Smuzhiyun for (i = 0; i < ISIF_LINEAR_TAB_SIZE; i++) {
391*4882a593Smuzhiyun if (i % 2)
392*4882a593Smuzhiyun regw_lin_tbl(linearize->table[i], ((i >> 1) << 2), 1);
393*4882a593Smuzhiyun else
394*4882a593Smuzhiyun regw_lin_tbl(linearize->table[i], ((i >> 1) << 2), 0);
395*4882a593Smuzhiyun }
396*4882a593Smuzhiyun }
397*4882a593Smuzhiyun
isif_config_dfc(struct isif_dfc * vdfc)398*4882a593Smuzhiyun static int isif_config_dfc(struct isif_dfc *vdfc)
399*4882a593Smuzhiyun {
400*4882a593Smuzhiyun /* initialize retries to loop for max ~ 250 usec */
401*4882a593Smuzhiyun u32 val, count, retries = loops_per_jiffy / (4000/HZ);
402*4882a593Smuzhiyun int i;
403*4882a593Smuzhiyun
404*4882a593Smuzhiyun if (!vdfc->en)
405*4882a593Smuzhiyun return 0;
406*4882a593Smuzhiyun
407*4882a593Smuzhiyun /* Correction mode */
408*4882a593Smuzhiyun val = (vdfc->corr_mode << ISIF_VDFC_CORR_MOD_SHIFT);
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun /* Correct whole line or partial */
411*4882a593Smuzhiyun if (vdfc->corr_whole_line)
412*4882a593Smuzhiyun val |= 1 << ISIF_VDFC_CORR_WHOLE_LN_SHIFT;
413*4882a593Smuzhiyun
414*4882a593Smuzhiyun /* level shift value */
415*4882a593Smuzhiyun val |= vdfc->def_level_shift << ISIF_VDFC_LEVEL_SHFT_SHIFT;
416*4882a593Smuzhiyun
417*4882a593Smuzhiyun regw(val, DFCCTL);
418*4882a593Smuzhiyun
419*4882a593Smuzhiyun /* Defect saturation level */
420*4882a593Smuzhiyun regw(vdfc->def_sat_level, VDFSATLV);
421*4882a593Smuzhiyun
422*4882a593Smuzhiyun regw(vdfc->table[0].pos_vert, DFCMEM0);
423*4882a593Smuzhiyun regw(vdfc->table[0].pos_horz, DFCMEM1);
424*4882a593Smuzhiyun if (vdfc->corr_mode == ISIF_VDFC_NORMAL ||
425*4882a593Smuzhiyun vdfc->corr_mode == ISIF_VDFC_HORZ_INTERPOL_IF_SAT) {
426*4882a593Smuzhiyun regw(vdfc->table[0].level_at_pos, DFCMEM2);
427*4882a593Smuzhiyun regw(vdfc->table[0].level_up_pixels, DFCMEM3);
428*4882a593Smuzhiyun regw(vdfc->table[0].level_low_pixels, DFCMEM4);
429*4882a593Smuzhiyun }
430*4882a593Smuzhiyun
431*4882a593Smuzhiyun /* set DFCMARST and set DFCMWR */
432*4882a593Smuzhiyun val = regr(DFCMEMCTL) | (1 << ISIF_DFCMEMCTL_DFCMARST_SHIFT) | 1;
433*4882a593Smuzhiyun regw(val, DFCMEMCTL);
434*4882a593Smuzhiyun
435*4882a593Smuzhiyun count = retries;
436*4882a593Smuzhiyun while (count && (regr(DFCMEMCTL) & 0x1))
437*4882a593Smuzhiyun count--;
438*4882a593Smuzhiyun
439*4882a593Smuzhiyun if (!count) {
440*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "defect table write timeout !!!\n");
441*4882a593Smuzhiyun return -1;
442*4882a593Smuzhiyun }
443*4882a593Smuzhiyun
444*4882a593Smuzhiyun for (i = 1; i < vdfc->num_vdefects; i++) {
445*4882a593Smuzhiyun regw(vdfc->table[i].pos_vert, DFCMEM0);
446*4882a593Smuzhiyun regw(vdfc->table[i].pos_horz, DFCMEM1);
447*4882a593Smuzhiyun if (vdfc->corr_mode == ISIF_VDFC_NORMAL ||
448*4882a593Smuzhiyun vdfc->corr_mode == ISIF_VDFC_HORZ_INTERPOL_IF_SAT) {
449*4882a593Smuzhiyun regw(vdfc->table[i].level_at_pos, DFCMEM2);
450*4882a593Smuzhiyun regw(vdfc->table[i].level_up_pixels, DFCMEM3);
451*4882a593Smuzhiyun regw(vdfc->table[i].level_low_pixels, DFCMEM4);
452*4882a593Smuzhiyun }
453*4882a593Smuzhiyun val = regr(DFCMEMCTL);
454*4882a593Smuzhiyun /* clear DFCMARST and set DFCMWR */
455*4882a593Smuzhiyun val &= ~BIT(ISIF_DFCMEMCTL_DFCMARST_SHIFT);
456*4882a593Smuzhiyun val |= 1;
457*4882a593Smuzhiyun regw(val, DFCMEMCTL);
458*4882a593Smuzhiyun
459*4882a593Smuzhiyun count = retries;
460*4882a593Smuzhiyun while (count && (regr(DFCMEMCTL) & 0x1))
461*4882a593Smuzhiyun count--;
462*4882a593Smuzhiyun
463*4882a593Smuzhiyun if (!count) {
464*4882a593Smuzhiyun dev_err(isif_cfg.dev,
465*4882a593Smuzhiyun "defect table write timeout !!!\n");
466*4882a593Smuzhiyun return -1;
467*4882a593Smuzhiyun }
468*4882a593Smuzhiyun }
469*4882a593Smuzhiyun if (vdfc->num_vdefects < ISIF_VDFC_TABLE_SIZE) {
470*4882a593Smuzhiyun /* Extra cycle needed */
471*4882a593Smuzhiyun regw(0, DFCMEM0);
472*4882a593Smuzhiyun regw(0x1FFF, DFCMEM1);
473*4882a593Smuzhiyun regw(1, DFCMEMCTL);
474*4882a593Smuzhiyun }
475*4882a593Smuzhiyun
476*4882a593Smuzhiyun /* enable VDFC */
477*4882a593Smuzhiyun reg_modify((1 << ISIF_VDFC_EN_SHIFT), (1 << ISIF_VDFC_EN_SHIFT),
478*4882a593Smuzhiyun DFCCTL);
479*4882a593Smuzhiyun return 0;
480*4882a593Smuzhiyun }
481*4882a593Smuzhiyun
isif_config_csc(struct isif_df_csc * df_csc)482*4882a593Smuzhiyun static void isif_config_csc(struct isif_df_csc *df_csc)
483*4882a593Smuzhiyun {
484*4882a593Smuzhiyun u32 val1 = 0, val2 = 0, i;
485*4882a593Smuzhiyun
486*4882a593Smuzhiyun if (!df_csc->csc.en) {
487*4882a593Smuzhiyun regw(0, CSCCTL);
488*4882a593Smuzhiyun return;
489*4882a593Smuzhiyun }
490*4882a593Smuzhiyun for (i = 0; i < ISIF_CSC_NUM_COEFF; i++) {
491*4882a593Smuzhiyun if ((i % 2) == 0) {
492*4882a593Smuzhiyun /* CSCM - LSB */
493*4882a593Smuzhiyun val1 = (df_csc->csc.coeff[i].integer <<
494*4882a593Smuzhiyun ISIF_CSC_COEF_INTEG_SHIFT) |
495*4882a593Smuzhiyun df_csc->csc.coeff[i].decimal;
496*4882a593Smuzhiyun } else {
497*4882a593Smuzhiyun
498*4882a593Smuzhiyun /* CSCM - MSB */
499*4882a593Smuzhiyun val2 = (df_csc->csc.coeff[i].integer <<
500*4882a593Smuzhiyun ISIF_CSC_COEF_INTEG_SHIFT) |
501*4882a593Smuzhiyun df_csc->csc.coeff[i].decimal;
502*4882a593Smuzhiyun val2 <<= ISIF_CSCM_MSB_SHIFT;
503*4882a593Smuzhiyun val2 |= val1;
504*4882a593Smuzhiyun regw(val2, (CSCM0 + ((i - 1) << 1)));
505*4882a593Smuzhiyun }
506*4882a593Smuzhiyun }
507*4882a593Smuzhiyun
508*4882a593Smuzhiyun /* program the active area */
509*4882a593Smuzhiyun regw(df_csc->start_pix, FMTSPH);
510*4882a593Smuzhiyun /*
511*4882a593Smuzhiyun * one extra pixel as required for CSC. Actually number of
512*4882a593Smuzhiyun * pixel - 1 should be configured in this register. So we
513*4882a593Smuzhiyun * need to subtract 1 before writing to FMTSPH, but we will
514*4882a593Smuzhiyun * not do this since csc requires one extra pixel
515*4882a593Smuzhiyun */
516*4882a593Smuzhiyun regw(df_csc->num_pixels, FMTLNH);
517*4882a593Smuzhiyun regw(df_csc->start_line, FMTSLV);
518*4882a593Smuzhiyun /*
519*4882a593Smuzhiyun * one extra line as required for CSC. See reason documented for
520*4882a593Smuzhiyun * num_pixels
521*4882a593Smuzhiyun */
522*4882a593Smuzhiyun regw(df_csc->num_lines, FMTLNV);
523*4882a593Smuzhiyun
524*4882a593Smuzhiyun /* Enable CSC */
525*4882a593Smuzhiyun regw(1, CSCCTL);
526*4882a593Smuzhiyun }
527*4882a593Smuzhiyun
isif_config_raw(void)528*4882a593Smuzhiyun static int isif_config_raw(void)
529*4882a593Smuzhiyun {
530*4882a593Smuzhiyun struct isif_params_raw *params = &isif_cfg.bayer;
531*4882a593Smuzhiyun struct isif_config_params_raw *module_params =
532*4882a593Smuzhiyun &isif_cfg.bayer.config_params;
533*4882a593Smuzhiyun struct vpss_pg_frame_size frame_size;
534*4882a593Smuzhiyun struct vpss_sync_pol sync;
535*4882a593Smuzhiyun u32 val;
536*4882a593Smuzhiyun
537*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "\nStarting isif_config_raw..\n");
538*4882a593Smuzhiyun
539*4882a593Smuzhiyun /*
540*4882a593Smuzhiyun * Configure CCDCFG register:-
541*4882a593Smuzhiyun * Set CCD Not to swap input since input is RAW data
542*4882a593Smuzhiyun * Set FID detection function to Latch at V-Sync
543*4882a593Smuzhiyun * Set WENLOG - isif valid area
544*4882a593Smuzhiyun * Set TRGSEL
545*4882a593Smuzhiyun * Set EXTRG
546*4882a593Smuzhiyun * Packed to 8 or 16 bits
547*4882a593Smuzhiyun */
548*4882a593Smuzhiyun
549*4882a593Smuzhiyun val = ISIF_YCINSWP_RAW | ISIF_CCDCFG_FIDMD_LATCH_VSYNC |
550*4882a593Smuzhiyun ISIF_CCDCFG_WENLOG_AND | ISIF_CCDCFG_TRGSEL_WEN |
551*4882a593Smuzhiyun ISIF_CCDCFG_EXTRG_DISABLE | isif_cfg.data_pack;
552*4882a593Smuzhiyun
553*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Writing 0x%x to ...CCDCFG \n", val);
554*4882a593Smuzhiyun regw(val, CCDCFG);
555*4882a593Smuzhiyun
556*4882a593Smuzhiyun /*
557*4882a593Smuzhiyun * Configure the vertical sync polarity(MODESET.VDPOL)
558*4882a593Smuzhiyun * Configure the horizontal sync polarity (MODESET.HDPOL)
559*4882a593Smuzhiyun * Configure frame id polarity (MODESET.FLDPOL)
560*4882a593Smuzhiyun * Configure data polarity
561*4882a593Smuzhiyun * Configure External WEN Selection
562*4882a593Smuzhiyun * Configure frame format(progressive or interlace)
563*4882a593Smuzhiyun * Configure pixel format (Input mode)
564*4882a593Smuzhiyun * Configure the data shift
565*4882a593Smuzhiyun */
566*4882a593Smuzhiyun
567*4882a593Smuzhiyun val = ISIF_VDHDOUT_INPUT | (params->vd_pol << ISIF_VD_POL_SHIFT) |
568*4882a593Smuzhiyun (params->hd_pol << ISIF_HD_POL_SHIFT) |
569*4882a593Smuzhiyun (params->fid_pol << ISIF_FID_POL_SHIFT) |
570*4882a593Smuzhiyun (ISIF_DATAPOL_NORMAL << ISIF_DATAPOL_SHIFT) |
571*4882a593Smuzhiyun (ISIF_EXWEN_DISABLE << ISIF_EXWEN_SHIFT) |
572*4882a593Smuzhiyun (params->frm_fmt << ISIF_FRM_FMT_SHIFT) |
573*4882a593Smuzhiyun (params->pix_fmt << ISIF_INPUT_SHIFT) |
574*4882a593Smuzhiyun (params->config_params.data_shift << ISIF_DATASFT_SHIFT);
575*4882a593Smuzhiyun
576*4882a593Smuzhiyun regw(val, MODESET);
577*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Writing 0x%x to MODESET...\n", val);
578*4882a593Smuzhiyun
579*4882a593Smuzhiyun /*
580*4882a593Smuzhiyun * Configure GAMMAWD register
581*4882a593Smuzhiyun * CFA pattern setting
582*4882a593Smuzhiyun */
583*4882a593Smuzhiyun val = params->cfa_pat << ISIF_GAMMAWD_CFA_SHIFT;
584*4882a593Smuzhiyun
585*4882a593Smuzhiyun /* Gamma msb */
586*4882a593Smuzhiyun if (module_params->compress.alg == ISIF_ALAW)
587*4882a593Smuzhiyun val |= ISIF_ALAW_ENABLE;
588*4882a593Smuzhiyun
589*4882a593Smuzhiyun val |= (params->data_msb << ISIF_ALAW_GAMMA_WD_SHIFT);
590*4882a593Smuzhiyun regw(val, CGAMMAWD);
591*4882a593Smuzhiyun
592*4882a593Smuzhiyun /* Configure DPCM compression settings */
593*4882a593Smuzhiyun if (module_params->compress.alg == ISIF_DPCM) {
594*4882a593Smuzhiyun val = BIT(ISIF_DPCM_EN_SHIFT) |
595*4882a593Smuzhiyun (module_params->compress.pred <<
596*4882a593Smuzhiyun ISIF_DPCM_PREDICTOR_SHIFT);
597*4882a593Smuzhiyun }
598*4882a593Smuzhiyun
599*4882a593Smuzhiyun regw(val, MISC);
600*4882a593Smuzhiyun
601*4882a593Smuzhiyun /* Configure Gain & Offset */
602*4882a593Smuzhiyun isif_config_gain_offset();
603*4882a593Smuzhiyun
604*4882a593Smuzhiyun /* Configure Color pattern */
605*4882a593Smuzhiyun val = (params->config_params.col_pat_field0.olop) |
606*4882a593Smuzhiyun (params->config_params.col_pat_field0.olep << 2) |
607*4882a593Smuzhiyun (params->config_params.col_pat_field0.elop << 4) |
608*4882a593Smuzhiyun (params->config_params.col_pat_field0.elep << 6) |
609*4882a593Smuzhiyun (params->config_params.col_pat_field1.olop << 8) |
610*4882a593Smuzhiyun (params->config_params.col_pat_field1.olep << 10) |
611*4882a593Smuzhiyun (params->config_params.col_pat_field1.elop << 12) |
612*4882a593Smuzhiyun (params->config_params.col_pat_field1.elep << 14);
613*4882a593Smuzhiyun regw(val, CCOLP);
614*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Writing %x to CCOLP ...\n", val);
615*4882a593Smuzhiyun
616*4882a593Smuzhiyun /* Configure HSIZE register */
617*4882a593Smuzhiyun val = (!!params->horz_flip_en) << ISIF_HSIZE_FLIP_SHIFT;
618*4882a593Smuzhiyun
619*4882a593Smuzhiyun /* calculate line offset in 32 bytes based on pack value */
620*4882a593Smuzhiyun if (isif_cfg.data_pack == ISIF_PACK_8BIT)
621*4882a593Smuzhiyun val |= ((params->win.width + 31) >> 5);
622*4882a593Smuzhiyun else if (isif_cfg.data_pack == ISIF_PACK_12BIT)
623*4882a593Smuzhiyun val |= (((params->win.width +
624*4882a593Smuzhiyun (params->win.width >> 2)) + 31) >> 5);
625*4882a593Smuzhiyun else
626*4882a593Smuzhiyun val |= (((params->win.width * 2) + 31) >> 5);
627*4882a593Smuzhiyun regw(val, HSIZE);
628*4882a593Smuzhiyun
629*4882a593Smuzhiyun /* Configure SDOFST register */
630*4882a593Smuzhiyun if (params->frm_fmt == CCDC_FRMFMT_INTERLACED) {
631*4882a593Smuzhiyun if (params->image_invert_en) {
632*4882a593Smuzhiyun /* For interlace inverse mode */
633*4882a593Smuzhiyun regw(0x4B6D, SDOFST);
634*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Writing 0x4B6D to SDOFST...\n");
635*4882a593Smuzhiyun } else {
636*4882a593Smuzhiyun /* For interlace non inverse mode */
637*4882a593Smuzhiyun regw(0x0B6D, SDOFST);
638*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Writing 0x0B6D to SDOFST...\n");
639*4882a593Smuzhiyun }
640*4882a593Smuzhiyun } else if (params->frm_fmt == CCDC_FRMFMT_PROGRESSIVE) {
641*4882a593Smuzhiyun if (params->image_invert_en) {
642*4882a593Smuzhiyun /* For progressive inverse mode */
643*4882a593Smuzhiyun regw(0x4000, SDOFST);
644*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Writing 0x4000 to SDOFST...\n");
645*4882a593Smuzhiyun } else {
646*4882a593Smuzhiyun /* For progressive non inverse mode */
647*4882a593Smuzhiyun regw(0x0000, SDOFST);
648*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Writing 0x0000 to SDOFST...\n");
649*4882a593Smuzhiyun }
650*4882a593Smuzhiyun }
651*4882a593Smuzhiyun
652*4882a593Smuzhiyun /* Configure video window */
653*4882a593Smuzhiyun isif_setwin(¶ms->win, params->frm_fmt, 1);
654*4882a593Smuzhiyun
655*4882a593Smuzhiyun /* Configure Black Clamp */
656*4882a593Smuzhiyun isif_config_bclamp(&module_params->bclamp);
657*4882a593Smuzhiyun
658*4882a593Smuzhiyun /* Configure Vertical Defection Pixel Correction */
659*4882a593Smuzhiyun if (isif_config_dfc(&module_params->dfc) < 0)
660*4882a593Smuzhiyun return -EFAULT;
661*4882a593Smuzhiyun
662*4882a593Smuzhiyun if (!module_params->df_csc.df_or_csc)
663*4882a593Smuzhiyun /* Configure Color Space Conversion */
664*4882a593Smuzhiyun isif_config_csc(&module_params->df_csc);
665*4882a593Smuzhiyun
666*4882a593Smuzhiyun isif_config_linearization(&module_params->linearize);
667*4882a593Smuzhiyun
668*4882a593Smuzhiyun /* Configure Culling */
669*4882a593Smuzhiyun isif_config_culling(&module_params->culling);
670*4882a593Smuzhiyun
671*4882a593Smuzhiyun /* Configure horizontal and vertical offsets(DFC,LSC,Gain) */
672*4882a593Smuzhiyun regw(module_params->horz_offset, DATAHOFST);
673*4882a593Smuzhiyun regw(module_params->vert_offset, DATAVOFST);
674*4882a593Smuzhiyun
675*4882a593Smuzhiyun /* Setup test pattern if enabled */
676*4882a593Smuzhiyun if (params->config_params.test_pat_gen) {
677*4882a593Smuzhiyun /* Use the HD/VD pol settings from user */
678*4882a593Smuzhiyun sync.ccdpg_hdpol = params->hd_pol;
679*4882a593Smuzhiyun sync.ccdpg_vdpol = params->vd_pol;
680*4882a593Smuzhiyun dm365_vpss_set_sync_pol(sync);
681*4882a593Smuzhiyun frame_size.hlpfr = isif_cfg.bayer.win.width;
682*4882a593Smuzhiyun frame_size.pplen = isif_cfg.bayer.win.height;
683*4882a593Smuzhiyun dm365_vpss_set_pg_frame_size(frame_size);
684*4882a593Smuzhiyun vpss_select_ccdc_source(VPSS_PGLPBK);
685*4882a593Smuzhiyun }
686*4882a593Smuzhiyun
687*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "\nEnd of isif_config_ycbcr...\n");
688*4882a593Smuzhiyun return 0;
689*4882a593Smuzhiyun }
690*4882a593Smuzhiyun
isif_set_buftype(enum ccdc_buftype buf_type)691*4882a593Smuzhiyun static int isif_set_buftype(enum ccdc_buftype buf_type)
692*4882a593Smuzhiyun {
693*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER)
694*4882a593Smuzhiyun isif_cfg.bayer.buf_type = buf_type;
695*4882a593Smuzhiyun else
696*4882a593Smuzhiyun isif_cfg.ycbcr.buf_type = buf_type;
697*4882a593Smuzhiyun
698*4882a593Smuzhiyun return 0;
699*4882a593Smuzhiyun
700*4882a593Smuzhiyun }
isif_get_buftype(void)701*4882a593Smuzhiyun static enum ccdc_buftype isif_get_buftype(void)
702*4882a593Smuzhiyun {
703*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER)
704*4882a593Smuzhiyun return isif_cfg.bayer.buf_type;
705*4882a593Smuzhiyun
706*4882a593Smuzhiyun return isif_cfg.ycbcr.buf_type;
707*4882a593Smuzhiyun }
708*4882a593Smuzhiyun
isif_enum_pix(u32 * pix,int i)709*4882a593Smuzhiyun static int isif_enum_pix(u32 *pix, int i)
710*4882a593Smuzhiyun {
711*4882a593Smuzhiyun int ret = -EINVAL;
712*4882a593Smuzhiyun
713*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER) {
714*4882a593Smuzhiyun if (i < ARRAY_SIZE(isif_raw_bayer_pix_formats)) {
715*4882a593Smuzhiyun *pix = isif_raw_bayer_pix_formats[i];
716*4882a593Smuzhiyun ret = 0;
717*4882a593Smuzhiyun }
718*4882a593Smuzhiyun } else {
719*4882a593Smuzhiyun if (i < ARRAY_SIZE(isif_raw_yuv_pix_formats)) {
720*4882a593Smuzhiyun *pix = isif_raw_yuv_pix_formats[i];
721*4882a593Smuzhiyun ret = 0;
722*4882a593Smuzhiyun }
723*4882a593Smuzhiyun }
724*4882a593Smuzhiyun
725*4882a593Smuzhiyun return ret;
726*4882a593Smuzhiyun }
727*4882a593Smuzhiyun
isif_set_pixel_format(unsigned int pixfmt)728*4882a593Smuzhiyun static int isif_set_pixel_format(unsigned int pixfmt)
729*4882a593Smuzhiyun {
730*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER) {
731*4882a593Smuzhiyun if (pixfmt == V4L2_PIX_FMT_SBGGR8) {
732*4882a593Smuzhiyun if ((isif_cfg.bayer.config_params.compress.alg !=
733*4882a593Smuzhiyun ISIF_ALAW) &&
734*4882a593Smuzhiyun (isif_cfg.bayer.config_params.compress.alg !=
735*4882a593Smuzhiyun ISIF_DPCM)) {
736*4882a593Smuzhiyun dev_dbg(isif_cfg.dev,
737*4882a593Smuzhiyun "Either configure A-Law or DPCM\n");
738*4882a593Smuzhiyun return -EINVAL;
739*4882a593Smuzhiyun }
740*4882a593Smuzhiyun isif_cfg.data_pack = ISIF_PACK_8BIT;
741*4882a593Smuzhiyun } else if (pixfmt == V4L2_PIX_FMT_SBGGR16) {
742*4882a593Smuzhiyun isif_cfg.bayer.config_params.compress.alg =
743*4882a593Smuzhiyun ISIF_NO_COMPRESSION;
744*4882a593Smuzhiyun isif_cfg.data_pack = ISIF_PACK_16BIT;
745*4882a593Smuzhiyun } else
746*4882a593Smuzhiyun return -EINVAL;
747*4882a593Smuzhiyun isif_cfg.bayer.pix_fmt = CCDC_PIXFMT_RAW;
748*4882a593Smuzhiyun } else {
749*4882a593Smuzhiyun if (pixfmt == V4L2_PIX_FMT_YUYV)
750*4882a593Smuzhiyun isif_cfg.ycbcr.pix_order = CCDC_PIXORDER_YCBYCR;
751*4882a593Smuzhiyun else if (pixfmt == V4L2_PIX_FMT_UYVY)
752*4882a593Smuzhiyun isif_cfg.ycbcr.pix_order = CCDC_PIXORDER_CBYCRY;
753*4882a593Smuzhiyun else
754*4882a593Smuzhiyun return -EINVAL;
755*4882a593Smuzhiyun isif_cfg.data_pack = ISIF_PACK_8BIT;
756*4882a593Smuzhiyun }
757*4882a593Smuzhiyun return 0;
758*4882a593Smuzhiyun }
759*4882a593Smuzhiyun
isif_get_pixel_format(void)760*4882a593Smuzhiyun static u32 isif_get_pixel_format(void)
761*4882a593Smuzhiyun {
762*4882a593Smuzhiyun u32 pixfmt;
763*4882a593Smuzhiyun
764*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER)
765*4882a593Smuzhiyun if (isif_cfg.bayer.config_params.compress.alg == ISIF_ALAW ||
766*4882a593Smuzhiyun isif_cfg.bayer.config_params.compress.alg == ISIF_DPCM)
767*4882a593Smuzhiyun pixfmt = V4L2_PIX_FMT_SBGGR8;
768*4882a593Smuzhiyun else
769*4882a593Smuzhiyun pixfmt = V4L2_PIX_FMT_SBGGR16;
770*4882a593Smuzhiyun else {
771*4882a593Smuzhiyun if (isif_cfg.ycbcr.pix_order == CCDC_PIXORDER_YCBYCR)
772*4882a593Smuzhiyun pixfmt = V4L2_PIX_FMT_YUYV;
773*4882a593Smuzhiyun else
774*4882a593Smuzhiyun pixfmt = V4L2_PIX_FMT_UYVY;
775*4882a593Smuzhiyun }
776*4882a593Smuzhiyun return pixfmt;
777*4882a593Smuzhiyun }
778*4882a593Smuzhiyun
isif_set_image_window(struct v4l2_rect * win)779*4882a593Smuzhiyun static int isif_set_image_window(struct v4l2_rect *win)
780*4882a593Smuzhiyun {
781*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER) {
782*4882a593Smuzhiyun isif_cfg.bayer.win.top = win->top;
783*4882a593Smuzhiyun isif_cfg.bayer.win.left = win->left;
784*4882a593Smuzhiyun isif_cfg.bayer.win.width = win->width;
785*4882a593Smuzhiyun isif_cfg.bayer.win.height = win->height;
786*4882a593Smuzhiyun } else {
787*4882a593Smuzhiyun isif_cfg.ycbcr.win.top = win->top;
788*4882a593Smuzhiyun isif_cfg.ycbcr.win.left = win->left;
789*4882a593Smuzhiyun isif_cfg.ycbcr.win.width = win->width;
790*4882a593Smuzhiyun isif_cfg.ycbcr.win.height = win->height;
791*4882a593Smuzhiyun }
792*4882a593Smuzhiyun return 0;
793*4882a593Smuzhiyun }
794*4882a593Smuzhiyun
isif_get_image_window(struct v4l2_rect * win)795*4882a593Smuzhiyun static void isif_get_image_window(struct v4l2_rect *win)
796*4882a593Smuzhiyun {
797*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER)
798*4882a593Smuzhiyun *win = isif_cfg.bayer.win;
799*4882a593Smuzhiyun else
800*4882a593Smuzhiyun *win = isif_cfg.ycbcr.win;
801*4882a593Smuzhiyun }
802*4882a593Smuzhiyun
isif_get_line_length(void)803*4882a593Smuzhiyun static unsigned int isif_get_line_length(void)
804*4882a593Smuzhiyun {
805*4882a593Smuzhiyun unsigned int len;
806*4882a593Smuzhiyun
807*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER) {
808*4882a593Smuzhiyun if (isif_cfg.data_pack == ISIF_PACK_8BIT)
809*4882a593Smuzhiyun len = ((isif_cfg.bayer.win.width));
810*4882a593Smuzhiyun else if (isif_cfg.data_pack == ISIF_PACK_12BIT)
811*4882a593Smuzhiyun len = (((isif_cfg.bayer.win.width * 2) +
812*4882a593Smuzhiyun (isif_cfg.bayer.win.width >> 2)));
813*4882a593Smuzhiyun else
814*4882a593Smuzhiyun len = (((isif_cfg.bayer.win.width * 2)));
815*4882a593Smuzhiyun } else
816*4882a593Smuzhiyun len = (((isif_cfg.ycbcr.win.width * 2)));
817*4882a593Smuzhiyun return ALIGN(len, 32);
818*4882a593Smuzhiyun }
819*4882a593Smuzhiyun
isif_set_frame_format(enum ccdc_frmfmt frm_fmt)820*4882a593Smuzhiyun static int isif_set_frame_format(enum ccdc_frmfmt frm_fmt)
821*4882a593Smuzhiyun {
822*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER)
823*4882a593Smuzhiyun isif_cfg.bayer.frm_fmt = frm_fmt;
824*4882a593Smuzhiyun else
825*4882a593Smuzhiyun isif_cfg.ycbcr.frm_fmt = frm_fmt;
826*4882a593Smuzhiyun return 0;
827*4882a593Smuzhiyun }
isif_get_frame_format(void)828*4882a593Smuzhiyun static enum ccdc_frmfmt isif_get_frame_format(void)
829*4882a593Smuzhiyun {
830*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER)
831*4882a593Smuzhiyun return isif_cfg.bayer.frm_fmt;
832*4882a593Smuzhiyun return isif_cfg.ycbcr.frm_fmt;
833*4882a593Smuzhiyun }
834*4882a593Smuzhiyun
isif_getfid(void)835*4882a593Smuzhiyun static int isif_getfid(void)
836*4882a593Smuzhiyun {
837*4882a593Smuzhiyun return (regr(MODESET) >> 15) & 0x1;
838*4882a593Smuzhiyun }
839*4882a593Smuzhiyun
840*4882a593Smuzhiyun /* misc operations */
isif_setfbaddr(unsigned long addr)841*4882a593Smuzhiyun static void isif_setfbaddr(unsigned long addr)
842*4882a593Smuzhiyun {
843*4882a593Smuzhiyun regw((addr >> 21) & 0x07ff, CADU);
844*4882a593Smuzhiyun regw((addr >> 5) & 0x0ffff, CADL);
845*4882a593Smuzhiyun }
846*4882a593Smuzhiyun
isif_set_hw_if_params(struct vpfe_hw_if_param * params)847*4882a593Smuzhiyun static int isif_set_hw_if_params(struct vpfe_hw_if_param *params)
848*4882a593Smuzhiyun {
849*4882a593Smuzhiyun isif_cfg.if_type = params->if_type;
850*4882a593Smuzhiyun
851*4882a593Smuzhiyun switch (params->if_type) {
852*4882a593Smuzhiyun case VPFE_BT656:
853*4882a593Smuzhiyun case VPFE_BT656_10BIT:
854*4882a593Smuzhiyun case VPFE_YCBCR_SYNC_8:
855*4882a593Smuzhiyun isif_cfg.ycbcr.pix_fmt = CCDC_PIXFMT_YCBCR_8BIT;
856*4882a593Smuzhiyun isif_cfg.ycbcr.pix_order = CCDC_PIXORDER_CBYCRY;
857*4882a593Smuzhiyun break;
858*4882a593Smuzhiyun case VPFE_BT1120:
859*4882a593Smuzhiyun case VPFE_YCBCR_SYNC_16:
860*4882a593Smuzhiyun isif_cfg.ycbcr.pix_fmt = CCDC_PIXFMT_YCBCR_16BIT;
861*4882a593Smuzhiyun isif_cfg.ycbcr.pix_order = CCDC_PIXORDER_CBYCRY;
862*4882a593Smuzhiyun break;
863*4882a593Smuzhiyun case VPFE_RAW_BAYER:
864*4882a593Smuzhiyun isif_cfg.bayer.pix_fmt = CCDC_PIXFMT_RAW;
865*4882a593Smuzhiyun break;
866*4882a593Smuzhiyun default:
867*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Invalid interface type\n");
868*4882a593Smuzhiyun return -EINVAL;
869*4882a593Smuzhiyun }
870*4882a593Smuzhiyun
871*4882a593Smuzhiyun return 0;
872*4882a593Smuzhiyun }
873*4882a593Smuzhiyun
874*4882a593Smuzhiyun /* This function will configure ISIF for YCbCr parameters. */
isif_config_ycbcr(void)875*4882a593Smuzhiyun static int isif_config_ycbcr(void)
876*4882a593Smuzhiyun {
877*4882a593Smuzhiyun struct isif_ycbcr_config *params = &isif_cfg.ycbcr;
878*4882a593Smuzhiyun u32 modeset = 0, ccdcfg = 0;
879*4882a593Smuzhiyun
880*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "\nStarting isif_config_ycbcr...");
881*4882a593Smuzhiyun
882*4882a593Smuzhiyun /* configure pixel format or input mode */
883*4882a593Smuzhiyun modeset = modeset | (params->pix_fmt << ISIF_INPUT_SHIFT) |
884*4882a593Smuzhiyun (params->frm_fmt << ISIF_FRM_FMT_SHIFT) |
885*4882a593Smuzhiyun (params->fid_pol << ISIF_FID_POL_SHIFT) |
886*4882a593Smuzhiyun (params->hd_pol << ISIF_HD_POL_SHIFT) |
887*4882a593Smuzhiyun (params->vd_pol << ISIF_VD_POL_SHIFT);
888*4882a593Smuzhiyun
889*4882a593Smuzhiyun /* pack the data to 8-bit ISIFCFG */
890*4882a593Smuzhiyun switch (isif_cfg.if_type) {
891*4882a593Smuzhiyun case VPFE_BT656:
892*4882a593Smuzhiyun if (params->pix_fmt != CCDC_PIXFMT_YCBCR_8BIT) {
893*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Invalid pix_fmt(input mode)\n");
894*4882a593Smuzhiyun return -EINVAL;
895*4882a593Smuzhiyun }
896*4882a593Smuzhiyun modeset |= (VPFE_PINPOL_NEGATIVE << ISIF_VD_POL_SHIFT);
897*4882a593Smuzhiyun regw(3, REC656IF);
898*4882a593Smuzhiyun ccdcfg = ccdcfg | ISIF_DATA_PACK8 | ISIF_YCINSWP_YCBCR;
899*4882a593Smuzhiyun break;
900*4882a593Smuzhiyun case VPFE_BT656_10BIT:
901*4882a593Smuzhiyun if (params->pix_fmt != CCDC_PIXFMT_YCBCR_8BIT) {
902*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Invalid pix_fmt(input mode)\n");
903*4882a593Smuzhiyun return -EINVAL;
904*4882a593Smuzhiyun }
905*4882a593Smuzhiyun /* setup BT.656, embedded sync */
906*4882a593Smuzhiyun regw(3, REC656IF);
907*4882a593Smuzhiyun /* enable 10 bit mode in ccdcfg */
908*4882a593Smuzhiyun ccdcfg = ccdcfg | ISIF_DATA_PACK8 | ISIF_YCINSWP_YCBCR |
909*4882a593Smuzhiyun ISIF_BW656_ENABLE;
910*4882a593Smuzhiyun break;
911*4882a593Smuzhiyun case VPFE_BT1120:
912*4882a593Smuzhiyun if (params->pix_fmt != CCDC_PIXFMT_YCBCR_16BIT) {
913*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Invalid pix_fmt(input mode)\n");
914*4882a593Smuzhiyun return -EINVAL;
915*4882a593Smuzhiyun }
916*4882a593Smuzhiyun regw(3, REC656IF);
917*4882a593Smuzhiyun break;
918*4882a593Smuzhiyun
919*4882a593Smuzhiyun case VPFE_YCBCR_SYNC_8:
920*4882a593Smuzhiyun ccdcfg |= ISIF_DATA_PACK8;
921*4882a593Smuzhiyun ccdcfg |= ISIF_YCINSWP_YCBCR;
922*4882a593Smuzhiyun if (params->pix_fmt != CCDC_PIXFMT_YCBCR_8BIT) {
923*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Invalid pix_fmt(input mode)\n");
924*4882a593Smuzhiyun return -EINVAL;
925*4882a593Smuzhiyun }
926*4882a593Smuzhiyun break;
927*4882a593Smuzhiyun case VPFE_YCBCR_SYNC_16:
928*4882a593Smuzhiyun if (params->pix_fmt != CCDC_PIXFMT_YCBCR_16BIT) {
929*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Invalid pix_fmt(input mode)\n");
930*4882a593Smuzhiyun return -EINVAL;
931*4882a593Smuzhiyun }
932*4882a593Smuzhiyun break;
933*4882a593Smuzhiyun default:
934*4882a593Smuzhiyun /* should never come here */
935*4882a593Smuzhiyun dev_dbg(isif_cfg.dev, "Invalid interface type\n");
936*4882a593Smuzhiyun return -EINVAL;
937*4882a593Smuzhiyun }
938*4882a593Smuzhiyun
939*4882a593Smuzhiyun regw(modeset, MODESET);
940*4882a593Smuzhiyun
941*4882a593Smuzhiyun /* Set up pix order */
942*4882a593Smuzhiyun ccdcfg |= params->pix_order << ISIF_PIX_ORDER_SHIFT;
943*4882a593Smuzhiyun
944*4882a593Smuzhiyun regw(ccdcfg, CCDCFG);
945*4882a593Smuzhiyun
946*4882a593Smuzhiyun /* configure video window */
947*4882a593Smuzhiyun if ((isif_cfg.if_type == VPFE_BT1120) ||
948*4882a593Smuzhiyun (isif_cfg.if_type == VPFE_YCBCR_SYNC_16))
949*4882a593Smuzhiyun isif_setwin(¶ms->win, params->frm_fmt, 1);
950*4882a593Smuzhiyun else
951*4882a593Smuzhiyun isif_setwin(¶ms->win, params->frm_fmt, 2);
952*4882a593Smuzhiyun
953*4882a593Smuzhiyun /*
954*4882a593Smuzhiyun * configure the horizontal line offset
955*4882a593Smuzhiyun * this is done by rounding up width to a multiple of 16 pixels
956*4882a593Smuzhiyun * and multiply by two to account for y:cb:cr 4:2:2 data
957*4882a593Smuzhiyun */
958*4882a593Smuzhiyun regw(((((params->win.width * 2) + 31) & 0xffffffe0) >> 5), HSIZE);
959*4882a593Smuzhiyun
960*4882a593Smuzhiyun /* configure the memory line offset */
961*4882a593Smuzhiyun if ((params->frm_fmt == CCDC_FRMFMT_INTERLACED) &&
962*4882a593Smuzhiyun (params->buf_type == CCDC_BUFTYPE_FLD_INTERLEAVED))
963*4882a593Smuzhiyun /* two fields are interleaved in memory */
964*4882a593Smuzhiyun regw(0x00000249, SDOFST);
965*4882a593Smuzhiyun
966*4882a593Smuzhiyun return 0;
967*4882a593Smuzhiyun }
968*4882a593Smuzhiyun
isif_configure(void)969*4882a593Smuzhiyun static int isif_configure(void)
970*4882a593Smuzhiyun {
971*4882a593Smuzhiyun if (isif_cfg.if_type == VPFE_RAW_BAYER)
972*4882a593Smuzhiyun return isif_config_raw();
973*4882a593Smuzhiyun return isif_config_ycbcr();
974*4882a593Smuzhiyun }
975*4882a593Smuzhiyun
isif_close(struct device * device)976*4882a593Smuzhiyun static int isif_close(struct device *device)
977*4882a593Smuzhiyun {
978*4882a593Smuzhiyun /* copy defaults to module params */
979*4882a593Smuzhiyun isif_cfg.bayer.config_params = isif_config_defaults;
980*4882a593Smuzhiyun return 0;
981*4882a593Smuzhiyun }
982*4882a593Smuzhiyun
983*4882a593Smuzhiyun static const struct ccdc_hw_device isif_hw_dev = {
984*4882a593Smuzhiyun .name = "ISIF",
985*4882a593Smuzhiyun .owner = THIS_MODULE,
986*4882a593Smuzhiyun .hw_ops = {
987*4882a593Smuzhiyun .open = isif_open,
988*4882a593Smuzhiyun .close = isif_close,
989*4882a593Smuzhiyun .enable = isif_enable,
990*4882a593Smuzhiyun .enable_out_to_sdram = isif_enable_output_to_sdram,
991*4882a593Smuzhiyun .set_hw_if_params = isif_set_hw_if_params,
992*4882a593Smuzhiyun .configure = isif_configure,
993*4882a593Smuzhiyun .set_buftype = isif_set_buftype,
994*4882a593Smuzhiyun .get_buftype = isif_get_buftype,
995*4882a593Smuzhiyun .enum_pix = isif_enum_pix,
996*4882a593Smuzhiyun .set_pixel_format = isif_set_pixel_format,
997*4882a593Smuzhiyun .get_pixel_format = isif_get_pixel_format,
998*4882a593Smuzhiyun .set_frame_format = isif_set_frame_format,
999*4882a593Smuzhiyun .get_frame_format = isif_get_frame_format,
1000*4882a593Smuzhiyun .set_image_window = isif_set_image_window,
1001*4882a593Smuzhiyun .get_image_window = isif_get_image_window,
1002*4882a593Smuzhiyun .get_line_length = isif_get_line_length,
1003*4882a593Smuzhiyun .setfbaddr = isif_setfbaddr,
1004*4882a593Smuzhiyun .getfid = isif_getfid,
1005*4882a593Smuzhiyun },
1006*4882a593Smuzhiyun };
1007*4882a593Smuzhiyun
isif_probe(struct platform_device * pdev)1008*4882a593Smuzhiyun static int isif_probe(struct platform_device *pdev)
1009*4882a593Smuzhiyun {
1010*4882a593Smuzhiyun void (*setup_pinmux)(void);
1011*4882a593Smuzhiyun struct resource *res;
1012*4882a593Smuzhiyun void __iomem *addr;
1013*4882a593Smuzhiyun int status = 0, i;
1014*4882a593Smuzhiyun
1015*4882a593Smuzhiyun /* Platform data holds setup_pinmux function ptr */
1016*4882a593Smuzhiyun if (!pdev->dev.platform_data)
1017*4882a593Smuzhiyun return -ENODEV;
1018*4882a593Smuzhiyun
1019*4882a593Smuzhiyun /*
1020*4882a593Smuzhiyun * first try to register with vpfe. If not correct platform, then we
1021*4882a593Smuzhiyun * don't have to iomap
1022*4882a593Smuzhiyun */
1023*4882a593Smuzhiyun status = vpfe_register_ccdc_device(&isif_hw_dev);
1024*4882a593Smuzhiyun if (status < 0)
1025*4882a593Smuzhiyun return status;
1026*4882a593Smuzhiyun
1027*4882a593Smuzhiyun setup_pinmux = pdev->dev.platform_data;
1028*4882a593Smuzhiyun /*
1029*4882a593Smuzhiyun * setup Mux configuration for ccdc which may be different for
1030*4882a593Smuzhiyun * different SoCs using this CCDC
1031*4882a593Smuzhiyun */
1032*4882a593Smuzhiyun setup_pinmux();
1033*4882a593Smuzhiyun
1034*4882a593Smuzhiyun i = 0;
1035*4882a593Smuzhiyun /* Get the ISIF base address, linearization table0 and table1 addr. */
1036*4882a593Smuzhiyun while (i < 3) {
1037*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM, i);
1038*4882a593Smuzhiyun if (!res) {
1039*4882a593Smuzhiyun status = -ENODEV;
1040*4882a593Smuzhiyun goto fail_nobase_res;
1041*4882a593Smuzhiyun }
1042*4882a593Smuzhiyun res = request_mem_region(res->start, resource_size(res),
1043*4882a593Smuzhiyun res->name);
1044*4882a593Smuzhiyun if (!res) {
1045*4882a593Smuzhiyun status = -EBUSY;
1046*4882a593Smuzhiyun goto fail_nobase_res;
1047*4882a593Smuzhiyun }
1048*4882a593Smuzhiyun addr = ioremap(res->start, resource_size(res));
1049*4882a593Smuzhiyun if (!addr) {
1050*4882a593Smuzhiyun status = -ENOMEM;
1051*4882a593Smuzhiyun goto fail_base_iomap;
1052*4882a593Smuzhiyun }
1053*4882a593Smuzhiyun switch (i) {
1054*4882a593Smuzhiyun case 0:
1055*4882a593Smuzhiyun /* ISIF base address */
1056*4882a593Smuzhiyun isif_cfg.base_addr = addr;
1057*4882a593Smuzhiyun break;
1058*4882a593Smuzhiyun case 1:
1059*4882a593Smuzhiyun /* ISIF linear tbl0 address */
1060*4882a593Smuzhiyun isif_cfg.linear_tbl0_addr = addr;
1061*4882a593Smuzhiyun break;
1062*4882a593Smuzhiyun default:
1063*4882a593Smuzhiyun /* ISIF linear tbl0 address */
1064*4882a593Smuzhiyun isif_cfg.linear_tbl1_addr = addr;
1065*4882a593Smuzhiyun break;
1066*4882a593Smuzhiyun }
1067*4882a593Smuzhiyun i++;
1068*4882a593Smuzhiyun }
1069*4882a593Smuzhiyun isif_cfg.dev = &pdev->dev;
1070*4882a593Smuzhiyun
1071*4882a593Smuzhiyun printk(KERN_NOTICE "%s is registered with vpfe.\n",
1072*4882a593Smuzhiyun isif_hw_dev.name);
1073*4882a593Smuzhiyun return 0;
1074*4882a593Smuzhiyun fail_base_iomap:
1075*4882a593Smuzhiyun release_mem_region(res->start, resource_size(res));
1076*4882a593Smuzhiyun i--;
1077*4882a593Smuzhiyun fail_nobase_res:
1078*4882a593Smuzhiyun if (isif_cfg.base_addr)
1079*4882a593Smuzhiyun iounmap(isif_cfg.base_addr);
1080*4882a593Smuzhiyun if (isif_cfg.linear_tbl0_addr)
1081*4882a593Smuzhiyun iounmap(isif_cfg.linear_tbl0_addr);
1082*4882a593Smuzhiyun
1083*4882a593Smuzhiyun while (i >= 0) {
1084*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM, i);
1085*4882a593Smuzhiyun if (res)
1086*4882a593Smuzhiyun release_mem_region(res->start, resource_size(res));
1087*4882a593Smuzhiyun i--;
1088*4882a593Smuzhiyun }
1089*4882a593Smuzhiyun vpfe_unregister_ccdc_device(&isif_hw_dev);
1090*4882a593Smuzhiyun return status;
1091*4882a593Smuzhiyun }
1092*4882a593Smuzhiyun
isif_remove(struct platform_device * pdev)1093*4882a593Smuzhiyun static int isif_remove(struct platform_device *pdev)
1094*4882a593Smuzhiyun {
1095*4882a593Smuzhiyun struct resource *res;
1096*4882a593Smuzhiyun int i = 0;
1097*4882a593Smuzhiyun
1098*4882a593Smuzhiyun iounmap(isif_cfg.base_addr);
1099*4882a593Smuzhiyun iounmap(isif_cfg.linear_tbl0_addr);
1100*4882a593Smuzhiyun iounmap(isif_cfg.linear_tbl1_addr);
1101*4882a593Smuzhiyun while (i < 3) {
1102*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM, i);
1103*4882a593Smuzhiyun if (res)
1104*4882a593Smuzhiyun release_mem_region(res->start, resource_size(res));
1105*4882a593Smuzhiyun i++;
1106*4882a593Smuzhiyun }
1107*4882a593Smuzhiyun vpfe_unregister_ccdc_device(&isif_hw_dev);
1108*4882a593Smuzhiyun return 0;
1109*4882a593Smuzhiyun }
1110*4882a593Smuzhiyun
1111*4882a593Smuzhiyun static struct platform_driver isif_driver = {
1112*4882a593Smuzhiyun .driver = {
1113*4882a593Smuzhiyun .name = "isif",
1114*4882a593Smuzhiyun },
1115*4882a593Smuzhiyun .remove = isif_remove,
1116*4882a593Smuzhiyun .probe = isif_probe,
1117*4882a593Smuzhiyun };
1118*4882a593Smuzhiyun
1119*4882a593Smuzhiyun module_platform_driver(isif_driver);
1120*4882a593Smuzhiyun
1121*4882a593Smuzhiyun MODULE_LICENSE("GPL");
1122