1*c1080626SJernej Skrabec /* 2*c1080626SJernej Skrabec * Sunxi TV encoder register and constant defines 3*c1080626SJernej Skrabec * 4*c1080626SJernej Skrabec * (C) Copyright 2014 Hans de Goede <hdegoede@redhat.com> 5*c1080626SJernej Skrabec * (C) Copyright 2017 Jernej Skrabec <jernej.skrabec@siol.net> 6*c1080626SJernej Skrabec * 7*c1080626SJernej Skrabec * SPDX-License-Identifier: GPL-2.0+ 8*c1080626SJernej Skrabec */ 9*c1080626SJernej Skrabec 10*c1080626SJernej Skrabec #ifndef _TVE_H 11*c1080626SJernej Skrabec #define _TVE_H 12*c1080626SJernej Skrabec 13*c1080626SJernej Skrabec enum tve_mode { 14*c1080626SJernej Skrabec tve_mode_vga, 15*c1080626SJernej Skrabec tve_mode_composite_pal, 16*c1080626SJernej Skrabec tve_mode_composite_ntsc, 17*c1080626SJernej Skrabec tve_mode_composite_pal_m, 18*c1080626SJernej Skrabec tve_mode_composite_pal_nc, 19*c1080626SJernej Skrabec }; 20*c1080626SJernej Skrabec 21*c1080626SJernej Skrabec /* 22*c1080626SJernej Skrabec * This is based on the A10s User Manual, and the A10s only supports 23*c1080626SJernej Skrabec * composite video and not vga like the A10 / A20 does, still other 24*c1080626SJernej Skrabec * than the removed vga out capability the tvencoder seems to be the same. 25*c1080626SJernej Skrabec * "unknown#" registers are registers which are used in the A10 kernel code, 26*c1080626SJernej Skrabec * but not documented in the A10s User Manual. 27*c1080626SJernej Skrabec */ 28*c1080626SJernej Skrabec struct sunxi_tve_reg { 29*c1080626SJernej Skrabec u32 gctrl; /* 0x000 */ 30*c1080626SJernej Skrabec u32 cfg0; /* 0x004 */ 31*c1080626SJernej Skrabec u32 dac_cfg0; /* 0x008 */ 32*c1080626SJernej Skrabec u32 filter; /* 0x00c */ 33*c1080626SJernej Skrabec u32 chroma_freq; /* 0x010 */ 34*c1080626SJernej Skrabec u32 porch_num; /* 0x014 */ 35*c1080626SJernej Skrabec u32 unknown0; /* 0x018 */ 36*c1080626SJernej Skrabec u32 line_num; /* 0x01c */ 37*c1080626SJernej Skrabec u32 blank_black_level; /* 0x020 */ 38*c1080626SJernej Skrabec u32 unknown1; /* 0x024, seems to be 1 byte per dac */ 39*c1080626SJernej Skrabec u8 res0[0x08]; /* 0x028 */ 40*c1080626SJernej Skrabec u32 auto_detect_en; /* 0x030 */ 41*c1080626SJernej Skrabec u32 auto_detect_int_status; /* 0x034 */ 42*c1080626SJernej Skrabec u32 auto_detect_status; /* 0x038 */ 43*c1080626SJernej Skrabec u32 auto_detect_debounce; /* 0x03c */ 44*c1080626SJernej Skrabec u32 csc_reg0; /* 0x040 */ 45*c1080626SJernej Skrabec u32 csc_reg1; /* 0x044 */ 46*c1080626SJernej Skrabec u32 csc_reg2; /* 0x048 */ 47*c1080626SJernej Skrabec u32 csc_reg3; /* 0x04c */ 48*c1080626SJernej Skrabec u8 res1[0xb0]; /* 0x050 */ 49*c1080626SJernej Skrabec u32 color_burst; /* 0x100 */ 50*c1080626SJernej Skrabec u32 vsync_num; /* 0x104 */ 51*c1080626SJernej Skrabec u32 notch_freq; /* 0x108 */ 52*c1080626SJernej Skrabec u32 cbr_level; /* 0x10c */ 53*c1080626SJernej Skrabec u32 burst_phase; /* 0x110 */ 54*c1080626SJernej Skrabec u32 burst_width; /* 0x114 */ 55*c1080626SJernej Skrabec u32 unknown2; /* 0x118 */ 56*c1080626SJernej Skrabec u32 sync_vbi_level; /* 0x11c */ 57*c1080626SJernej Skrabec u32 white_level; /* 0x120 */ 58*c1080626SJernej Skrabec u32 active_num; /* 0x124 */ 59*c1080626SJernej Skrabec u32 chroma_bw_gain; /* 0x128 */ 60*c1080626SJernej Skrabec u32 notch_width; /* 0x12c */ 61*c1080626SJernej Skrabec u32 resync_num; /* 0x130 */ 62*c1080626SJernej Skrabec u32 slave_para; /* 0x134 */ 63*c1080626SJernej Skrabec u32 cfg1; /* 0x138 */ 64*c1080626SJernej Skrabec u32 cfg2; /* 0x13c */ 65*c1080626SJernej Skrabec }; 66*c1080626SJernej Skrabec 67*c1080626SJernej Skrabec /* 68*c1080626SJernej Skrabec * TVE register constants. 69*c1080626SJernej Skrabec */ 70*c1080626SJernej Skrabec #define SUNXI_TVE_GCTRL_ENABLE (1 << 0) 71*c1080626SJernej Skrabec /* 72*c1080626SJernej Skrabec * Select input 0 to disable dac, 1 - 4 to feed dac from tve0, 5 - 8 to feed 73*c1080626SJernej Skrabec * dac from tve1. When using tve1 the mux value must be written to both tve0's 74*c1080626SJernej Skrabec * and tve1's gctrl reg. 75*c1080626SJernej Skrabec */ 76*c1080626SJernej Skrabec #define SUNXI_TVE_GCTRL_DAC_INPUT_MASK(dac) (0xf << (((dac) + 1) * 4)) 77*c1080626SJernej Skrabec #define SUNXI_TVE_GCTRL_DAC_INPUT(dac, sel) ((sel) << (((dac) + 1) * 4)) 78*c1080626SJernej Skrabec #define SUNXI_TVE_CFG0_VGA 0x20000000 79*c1080626SJernej Skrabec #define SUNXI_TVE_CFG0_PAL 0x07030001 80*c1080626SJernej Skrabec #define SUNXI_TVE_CFG0_NTSC 0x07030000 81*c1080626SJernej Skrabec #define SUNXI_TVE_DAC_CFG0_VGA 0x403e1ac7 82*c1080626SJernej Skrabec #ifdef CONFIG_MACH_SUN5I 83*c1080626SJernej Skrabec #define SUNXI_TVE_DAC_CFG0_COMPOSITE 0x433f0009 84*c1080626SJernej Skrabec #else 85*c1080626SJernej Skrabec #define SUNXI_TVE_DAC_CFG0_COMPOSITE 0x403f0008 86*c1080626SJernej Skrabec #endif 87*c1080626SJernej Skrabec #define SUNXI_TVE_FILTER_COMPOSITE 0x00000120 88*c1080626SJernej Skrabec #define SUNXI_TVE_CHROMA_FREQ_PAL_M 0x21e6efe3 89*c1080626SJernej Skrabec #define SUNXI_TVE_CHROMA_FREQ_PAL_NC 0x21f69446 90*c1080626SJernej Skrabec #define SUNXI_TVE_PORCH_NUM_PAL 0x008a0018 91*c1080626SJernej Skrabec #define SUNXI_TVE_PORCH_NUM_NTSC 0x00760020 92*c1080626SJernej Skrabec #define SUNXI_TVE_LINE_NUM_PAL 0x00160271 93*c1080626SJernej Skrabec #define SUNXI_TVE_LINE_NUM_NTSC 0x0016020d 94*c1080626SJernej Skrabec #define SUNXI_TVE_BLANK_BLACK_LEVEL_PAL 0x00fc00fc 95*c1080626SJernej Skrabec #define SUNXI_TVE_BLANK_BLACK_LEVEL_NTSC 0x00f0011a 96*c1080626SJernej Skrabec #define SUNXI_TVE_UNKNOWN1_VGA 0x00000000 97*c1080626SJernej Skrabec #define SUNXI_TVE_UNKNOWN1_COMPOSITE 0x18181818 98*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_EN_DET_EN(dac) (1 << ((dac) + 0)) 99*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_EN_INT_EN(dac) (1 << ((dac) + 16)) 100*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_INT_STATUS(dac) (1 << ((dac) + 0)) 101*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_STATUS_SHIFT(dac) ((dac) * 8) 102*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_STATUS_MASK(dac) (3 << ((dac) * 8)) 103*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_STATUS_NONE 0 104*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_STATUS_CONNECTED 1 105*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_STATUS_SHORT_GND 3 106*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_DEBOUNCE_SHIFT(d) ((d) * 8) 107*c1080626SJernej Skrabec #define SUNXI_TVE_AUTO_DETECT_DEBOUNCE_MASK(d) (0xf << ((d) * 8)) 108*c1080626SJernej Skrabec #define SUNXI_TVE_CSC_REG0_ENABLE (1 << 31) 109*c1080626SJernej Skrabec #define SUNXI_TVE_CSC_REG0 0x08440832 110*c1080626SJernej Skrabec #define SUNXI_TVE_CSC_REG1 0x3b6dace1 111*c1080626SJernej Skrabec #define SUNXI_TVE_CSC_REG2 0x0e1d13dc 112*c1080626SJernej Skrabec #define SUNXI_TVE_CSC_REG3 0x00108080 113*c1080626SJernej Skrabec #define SUNXI_TVE_COLOR_BURST_PAL_M 0x00000000 114*c1080626SJernej Skrabec #define SUNXI_TVE_CBR_LEVEL_PAL 0x00002828 115*c1080626SJernej Skrabec #define SUNXI_TVE_CBR_LEVEL_NTSC 0x0000004f 116*c1080626SJernej Skrabec #define SUNXI_TVE_BURST_PHASE_NTSC 0x00000000 117*c1080626SJernej Skrabec #define SUNXI_TVE_BURST_WIDTH_COMPOSITE 0x0016447e 118*c1080626SJernej Skrabec #define SUNXI_TVE_UNKNOWN2_PAL 0x0000e0e0 119*c1080626SJernej Skrabec #define SUNXI_TVE_UNKNOWN2_NTSC 0x0000a0a0 120*c1080626SJernej Skrabec #define SUNXI_TVE_SYNC_VBI_LEVEL_NTSC 0x001000f0 121*c1080626SJernej Skrabec #define SUNXI_TVE_ACTIVE_NUM_COMPOSITE 0x000005a0 122*c1080626SJernej Skrabec #define SUNXI_TVE_CHROMA_BW_GAIN_COMP 0x00000002 123*c1080626SJernej Skrabec #define SUNXI_TVE_NOTCH_WIDTH_COMPOSITE 0x00000101 124*c1080626SJernej Skrabec #define SUNXI_TVE_RESYNC_NUM_PAL 0x800d000c 125*c1080626SJernej Skrabec #define SUNXI_TVE_RESYNC_NUM_NTSC 0x000e000c 126*c1080626SJernej Skrabec #define SUNXI_TVE_SLAVE_PARA_COMPOSITE 0x00000000 127*c1080626SJernej Skrabec 128*c1080626SJernej Skrabec void tvencoder_mode_set(struct sunxi_tve_reg * const tve, enum tve_mode mode); 129*c1080626SJernej Skrabec void tvencoder_enable(struct sunxi_tve_reg * const tve); 130*c1080626SJernej Skrabec 131*c1080626SJernej Skrabec #endif /* _TVE_H */ 132