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