]> Git Repo - linux.git/blob - drivers/gpu/drm/meson/meson_venc.c
Merge branch 'linus' of git://git.kernel.org/pub/scm/linux/kernel/git/herbert/crypto-2.6
[linux.git] / drivers / gpu / drm / meson / meson_venc.c
1 /*
2  * Copyright (C) 2016 BayLibre, SAS
3  * Author: Neil Armstrong <[email protected]>
4  * Copyright (C) 2015 Amlogic, Inc. All rights reserved.
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License as
8  * published by the Free Software Foundation; either version 2 of the
9  * License, or (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful, but
12  * WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, see <http://www.gnu.org/licenses/>.
18  */
19
20 #include <linux/kernel.h>
21 #include <linux/module.h>
22 #include <drm/drmP.h>
23 #include "meson_drv.h"
24 #include "meson_venc.h"
25 #include "meson_vpp.h"
26 #include "meson_vclk.h"
27 #include "meson_registers.h"
28
29 /*
30  * VENC Handle the pixels encoding to the output formats.
31  * We handle the following encodings :
32  * - CVBS Encoding via the ENCI encoder and VDAC digital to analog converter
33  *
34  * What is missing :
35  * - TMDS/HDMI Encoding via ENCI_DIV and ENCP
36  * - Setup of more clock rates for HDMI modes
37  * - LCD Panel encoding via ENCL
38  * - TV Panel encoding via ENCT
39  */
40
41 struct meson_cvbs_enci_mode meson_cvbs_enci_pal = {
42         .mode_tag = MESON_VENC_MODE_CVBS_PAL,
43         .hso_begin = 3,
44         .hso_end = 129,
45         .vso_even = 3,
46         .vso_odd = 260,
47         .macv_max_amp = 7,
48         .video_prog_mode = 0xff,
49         .video_mode = 0x13,
50         .sch_adjust = 0x28,
51         .yc_delay = 0x343,
52         .pixel_start = 251,
53         .pixel_end = 1691,
54         .top_field_line_start = 22,
55         .top_field_line_end = 310,
56         .bottom_field_line_start = 23,
57         .bottom_field_line_end = 311,
58         .video_saturation = 9,
59         .video_contrast = 0,
60         .video_brightness = 0,
61         .video_hue = 0,
62         .analog_sync_adj = 0x8080,
63 };
64
65 struct meson_cvbs_enci_mode meson_cvbs_enci_ntsc = {
66         .mode_tag = MESON_VENC_MODE_CVBS_NTSC,
67         .hso_begin = 5,
68         .hso_end = 129,
69         .vso_even = 3,
70         .vso_odd = 260,
71         .macv_max_amp = 0xb,
72         .video_prog_mode = 0xf0,
73         .video_mode = 0x8,
74         .sch_adjust = 0x20,
75         .yc_delay = 0x333,
76         .pixel_start = 227,
77         .pixel_end = 1667,
78         .top_field_line_start = 18,
79         .top_field_line_end = 258,
80         .bottom_field_line_start = 19,
81         .bottom_field_line_end = 259,
82         .video_saturation = 18,
83         .video_contrast = 3,
84         .video_brightness = 0,
85         .video_hue = 0,
86         .analog_sync_adj = 0x9c00,
87 };
88
89 void meson_venci_cvbs_mode_set(struct meson_drm *priv,
90                                struct meson_cvbs_enci_mode *mode)
91 {
92         if (mode->mode_tag == priv->venc.current_mode)
93                 return;
94
95         /* CVBS Filter settings */
96         writel_relaxed(0x12, priv->io_base + _REG(ENCI_CFILT_CTRL));
97         writel_relaxed(0x12, priv->io_base + _REG(ENCI_CFILT_CTRL2));
98
99         /* Digital Video Select : Interlace, clk27 clk, external */
100         writel_relaxed(0, priv->io_base + _REG(VENC_DVI_SETTING));
101
102         /* Reset Video Mode */
103         writel_relaxed(0, priv->io_base + _REG(ENCI_VIDEO_MODE));
104         writel_relaxed(0, priv->io_base + _REG(ENCI_VIDEO_MODE_ADV));
105
106         /* Horizontal sync signal output */
107         writel_relaxed(mode->hso_begin,
108                         priv->io_base + _REG(ENCI_SYNC_HSO_BEGIN));
109         writel_relaxed(mode->hso_end,
110                         priv->io_base + _REG(ENCI_SYNC_HSO_END));
111
112         /* Vertical Sync lines */
113         writel_relaxed(mode->vso_even,
114                         priv->io_base + _REG(ENCI_SYNC_VSO_EVNLN));
115         writel_relaxed(mode->vso_odd,
116                         priv->io_base + _REG(ENCI_SYNC_VSO_ODDLN));
117
118         /* Macrovision max amplitude change */
119         writel_relaxed(0x8100 + mode->macv_max_amp,
120                         priv->io_base + _REG(ENCI_MACV_MAX_AMP));
121
122         /* Video mode */
123         writel_relaxed(mode->video_prog_mode,
124                         priv->io_base + _REG(VENC_VIDEO_PROG_MODE));
125         writel_relaxed(mode->video_mode,
126                         priv->io_base + _REG(ENCI_VIDEO_MODE));
127
128         /* Advanced Video Mode :
129          * Demux shifting 0x2
130          * Blank line end at line17/22
131          * High bandwidth Luma Filter
132          * Low bandwidth Chroma Filter
133          * Bypass luma low pass filter
134          * No macrovision on CSYNC
135          */
136         writel_relaxed(0x26, priv->io_base + _REG(ENCI_VIDEO_MODE_ADV));
137
138         writel(mode->sch_adjust, priv->io_base + _REG(ENCI_VIDEO_SCH));
139
140         /* Sync mode : MASTER Master mode, free run, send HSO/VSO out */
141         writel_relaxed(0x07, priv->io_base + _REG(ENCI_SYNC_MODE));
142
143         /* 0x3 Y, C, and Component Y delay */
144         writel_relaxed(mode->yc_delay, priv->io_base + _REG(ENCI_YC_DELAY));
145
146         /* Timings */
147         writel_relaxed(mode->pixel_start,
148                         priv->io_base + _REG(ENCI_VFIFO2VD_PIXEL_START));
149         writel_relaxed(mode->pixel_end,
150                         priv->io_base + _REG(ENCI_VFIFO2VD_PIXEL_END));
151
152         writel_relaxed(mode->top_field_line_start,
153                         priv->io_base + _REG(ENCI_VFIFO2VD_LINE_TOP_START));
154         writel_relaxed(mode->top_field_line_end,
155                         priv->io_base + _REG(ENCI_VFIFO2VD_LINE_TOP_END));
156
157         writel_relaxed(mode->bottom_field_line_start,
158                         priv->io_base + _REG(ENCI_VFIFO2VD_LINE_BOT_START));
159         writel_relaxed(mode->bottom_field_line_end,
160                         priv->io_base + _REG(ENCI_VFIFO2VD_LINE_BOT_END));
161
162         /* Internal Venc, Internal VIU Sync, Internal Vencoder */
163         writel_relaxed(0, priv->io_base + _REG(VENC_SYNC_ROUTE));
164
165         /* UNreset Interlaced TV Encoder */
166         writel_relaxed(0, priv->io_base + _REG(ENCI_DBG_PX_RST));
167
168         /* Enable Vfifo2vd, Y_Cb_Y_Cr select */
169         writel_relaxed(0x4e01, priv->io_base + _REG(ENCI_VFIFO2VD_CTL));
170
171         /* Power UP Dacs */
172         writel_relaxed(0, priv->io_base + _REG(VENC_VDAC_SETTING));
173
174         /* Video Upsampling */
175         writel_relaxed(0x0061, priv->io_base + _REG(VENC_UPSAMPLE_CTRL0));
176         writel_relaxed(0x4061, priv->io_base + _REG(VENC_UPSAMPLE_CTRL1));
177         writel_relaxed(0x5061, priv->io_base + _REG(VENC_UPSAMPLE_CTRL2));
178
179         /* Select Interlace Y DACs */
180         writel_relaxed(0, priv->io_base + _REG(VENC_VDAC_DACSEL0));
181         writel_relaxed(0, priv->io_base + _REG(VENC_VDAC_DACSEL1));
182         writel_relaxed(0, priv->io_base + _REG(VENC_VDAC_DACSEL2));
183         writel_relaxed(0, priv->io_base + _REG(VENC_VDAC_DACSEL3));
184         writel_relaxed(0, priv->io_base + _REG(VENC_VDAC_DACSEL4));
185         writel_relaxed(0, priv->io_base + _REG(VENC_VDAC_DACSEL5));
186
187         /* Select ENCI for VIU */
188         meson_vpp_setup_mux(priv, MESON_VIU_VPP_MUX_ENCI);
189
190         /* Enable ENCI FIFO */
191         writel_relaxed(0x2000, priv->io_base + _REG(VENC_VDAC_FIFO_CTRL));
192
193         /* Select ENCI DACs 0, 1, 4, and 5 */
194         writel_relaxed(0x11, priv->io_base + _REG(ENCI_DACSEL_0));
195         writel_relaxed(0x11, priv->io_base + _REG(ENCI_DACSEL_1));
196
197         /* Interlace video enable */
198         writel_relaxed(1, priv->io_base + _REG(ENCI_VIDEO_EN));
199
200         /* Configure Video Saturation / Contrast / Brightness / Hue */
201         writel_relaxed(mode->video_saturation,
202                         priv->io_base + _REG(ENCI_VIDEO_SAT));
203         writel_relaxed(mode->video_contrast,
204                         priv->io_base + _REG(ENCI_VIDEO_CONT));
205         writel_relaxed(mode->video_brightness,
206                         priv->io_base + _REG(ENCI_VIDEO_BRIGHT));
207         writel_relaxed(mode->video_hue,
208                         priv->io_base + _REG(ENCI_VIDEO_HUE));
209
210         /* Enable DAC0 Filter */
211         writel_relaxed(0x1, priv->io_base + _REG(VENC_VDAC_DAC0_FILT_CTRL0));
212         writel_relaxed(0xfc48, priv->io_base + _REG(VENC_VDAC_DAC0_FILT_CTRL1));
213
214         /* 0 in Macrovision register 0 */
215         writel_relaxed(0, priv->io_base + _REG(ENCI_MACV_N0));
216
217         /* Analog Synchronization and color burst value adjust */
218         writel_relaxed(mode->analog_sync_adj,
219                         priv->io_base + _REG(ENCI_SYNC_ADJ));
220
221         /* Setup 27MHz vclk2 for ENCI and VDAC */
222         meson_vclk_setup(priv, MESON_VCLK_TARGET_CVBS, MESON_VCLK_CVBS);
223
224         priv->venc.current_mode = mode->mode_tag;
225 }
226
227 /* Returns the current ENCI field polarity */
228 unsigned int meson_venci_get_field(struct meson_drm *priv)
229 {
230         return readl_relaxed(priv->io_base + _REG(ENCI_INFO_READ)) & BIT(29);
231 }
232
233 void meson_venc_enable_vsync(struct meson_drm *priv)
234 {
235         writel_relaxed(2, priv->io_base + _REG(VENC_INTCTRL));
236 }
237
238 void meson_venc_disable_vsync(struct meson_drm *priv)
239 {
240         writel_relaxed(0, priv->io_base + _REG(VENC_INTCTRL));
241 }
242
243 void meson_venc_init(struct meson_drm *priv)
244 {
245         /* Disable all encoders */
246         writel_relaxed(0, priv->io_base + _REG(ENCI_VIDEO_EN));
247         writel_relaxed(0, priv->io_base + _REG(ENCP_VIDEO_EN));
248         writel_relaxed(0, priv->io_base + _REG(ENCL_VIDEO_EN));
249
250         /* Disable VSync IRQ */
251         meson_venc_disable_vsync(priv);
252
253         priv->venc.current_mode = MESON_VENC_MODE_NONE;
254 }
This page took 0.051208 seconds and 4 git commands to generate.