2009-08-05 20:18:06 +07:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2009 Nokia Corporation
|
2018-05-24 18:46:19 +07:00
|
|
|
* Author: Tomi Valkeinen <tomi.valkeinen@ti.com>
|
2009-08-05 20:18:06 +07:00
|
|
|
*
|
|
|
|
* VENC settings from TI's DSS driver
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 as published by
|
|
|
|
* the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with
|
|
|
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define DSS_SUBSYS_NAME "VENC"
|
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/clk.h>
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <linux/mutex.h>
|
|
|
|
#include <linux/completion.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/string.h>
|
|
|
|
#include <linux/seq_file.h>
|
|
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <linux/regulator/consumer.h>
|
2011-05-27 14:52:19 +07:00
|
|
|
#include <linux/pm_runtime.h>
|
2013-12-16 20:14:15 +07:00
|
|
|
#include <linux/of.h>
|
2017-03-22 20:26:08 +07:00
|
|
|
#include <linux/of_graph.h>
|
OMAPDSS: componentize omapdss
omapdss kernel module contains drivers for multiple devices, one for
each DSS submodule. The probing we have at the moment is a mess, and
doesn't give us proper deferred probing nor ensure that all the devices
are probed before omapfb/omapdrm start using omapdss.
This patch solves the mess by using the component system for DSS
submodules.
The changes to all DSS submodules (dispc, dpi, dsi, hdmi4/5, rfbi, sdi,
venc) are the same: probe & remove functions are changed to bind &
unbind, and new probe & remove functions are added which call
component_add/del.
The dss_core driver (dss.c) acts as a component master. Adding and
matching the components is simple: all dss device's child devices are
added as components.
However, we do have some dependencies between the drivers. The order in
which they should be probed is reflected by the list in core.c
(dss_output_drv_reg_funcs). The drivers are registered in that order,
which causes the components to be added in that order, which makes the
components to be bound in that order. This feels a bit fragile, and we
probably should improve the code to manage binds in random order.
However, for now, this works fine.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Acked-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
2015-06-04 19:22:23 +07:00
|
|
|
#include <linux/component.h>
|
2017-08-05 05:44:09 +07:00
|
|
|
#include <linux/sys_soc.h>
|
2009-08-05 20:18:06 +07:00
|
|
|
|
2016-05-27 18:40:49 +07:00
|
|
|
#include "omapdss.h"
|
2009-08-05 20:18:06 +07:00
|
|
|
#include "dss.h"
|
|
|
|
|
|
|
|
/* Venc registers */
|
|
|
|
#define VENC_REV_ID 0x00
|
|
|
|
#define VENC_STATUS 0x04
|
|
|
|
#define VENC_F_CONTROL 0x08
|
|
|
|
#define VENC_VIDOUT_CTRL 0x10
|
|
|
|
#define VENC_SYNC_CTRL 0x14
|
|
|
|
#define VENC_LLEN 0x1C
|
|
|
|
#define VENC_FLENS 0x20
|
|
|
|
#define VENC_HFLTR_CTRL 0x24
|
|
|
|
#define VENC_CC_CARR_WSS_CARR 0x28
|
|
|
|
#define VENC_C_PHASE 0x2C
|
|
|
|
#define VENC_GAIN_U 0x30
|
|
|
|
#define VENC_GAIN_V 0x34
|
|
|
|
#define VENC_GAIN_Y 0x38
|
|
|
|
#define VENC_BLACK_LEVEL 0x3C
|
|
|
|
#define VENC_BLANK_LEVEL 0x40
|
|
|
|
#define VENC_X_COLOR 0x44
|
|
|
|
#define VENC_M_CONTROL 0x48
|
|
|
|
#define VENC_BSTAMP_WSS_DATA 0x4C
|
|
|
|
#define VENC_S_CARR 0x50
|
|
|
|
#define VENC_LINE21 0x54
|
|
|
|
#define VENC_LN_SEL 0x58
|
|
|
|
#define VENC_L21__WC_CTL 0x5C
|
|
|
|
#define VENC_HTRIGGER_VTRIGGER 0x60
|
|
|
|
#define VENC_SAVID__EAVID 0x64
|
|
|
|
#define VENC_FLEN__FAL 0x68
|
|
|
|
#define VENC_LAL__PHASE_RESET 0x6C
|
|
|
|
#define VENC_HS_INT_START_STOP_X 0x70
|
|
|
|
#define VENC_HS_EXT_START_STOP_X 0x74
|
|
|
|
#define VENC_VS_INT_START_X 0x78
|
|
|
|
#define VENC_VS_INT_STOP_X__VS_INT_START_Y 0x7C
|
|
|
|
#define VENC_VS_INT_STOP_Y__VS_EXT_START_X 0x80
|
|
|
|
#define VENC_VS_EXT_STOP_X__VS_EXT_START_Y 0x84
|
|
|
|
#define VENC_VS_EXT_STOP_Y 0x88
|
|
|
|
#define VENC_AVID_START_STOP_X 0x90
|
|
|
|
#define VENC_AVID_START_STOP_Y 0x94
|
|
|
|
#define VENC_FID_INT_START_X__FID_INT_START_Y 0xA0
|
|
|
|
#define VENC_FID_INT_OFFSET_Y__FID_EXT_START_X 0xA4
|
|
|
|
#define VENC_FID_EXT_START_Y__FID_EXT_OFFSET_Y 0xA8
|
|
|
|
#define VENC_TVDETGP_INT_START_STOP_X 0xB0
|
|
|
|
#define VENC_TVDETGP_INT_START_STOP_Y 0xB4
|
|
|
|
#define VENC_GEN_CTRL 0xB8
|
|
|
|
#define VENC_OUTPUT_CONTROL 0xC4
|
|
|
|
#define VENC_OUTPUT_TEST 0xC8
|
|
|
|
#define VENC_DAC_B__DAC_C 0xC8
|
|
|
|
|
|
|
|
struct venc_config {
|
|
|
|
u32 f_control;
|
|
|
|
u32 vidout_ctrl;
|
|
|
|
u32 sync_ctrl;
|
|
|
|
u32 llen;
|
|
|
|
u32 flens;
|
|
|
|
u32 hfltr_ctrl;
|
|
|
|
u32 cc_carr_wss_carr;
|
|
|
|
u32 c_phase;
|
|
|
|
u32 gain_u;
|
|
|
|
u32 gain_v;
|
|
|
|
u32 gain_y;
|
|
|
|
u32 black_level;
|
|
|
|
u32 blank_level;
|
|
|
|
u32 x_color;
|
|
|
|
u32 m_control;
|
|
|
|
u32 bstamp_wss_data;
|
|
|
|
u32 s_carr;
|
|
|
|
u32 line21;
|
|
|
|
u32 ln_sel;
|
|
|
|
u32 l21__wc_ctl;
|
|
|
|
u32 htrigger_vtrigger;
|
|
|
|
u32 savid__eavid;
|
|
|
|
u32 flen__fal;
|
|
|
|
u32 lal__phase_reset;
|
|
|
|
u32 hs_int_start_stop_x;
|
|
|
|
u32 hs_ext_start_stop_x;
|
|
|
|
u32 vs_int_start_x;
|
|
|
|
u32 vs_int_stop_x__vs_int_start_y;
|
|
|
|
u32 vs_int_stop_y__vs_ext_start_x;
|
|
|
|
u32 vs_ext_stop_x__vs_ext_start_y;
|
|
|
|
u32 vs_ext_stop_y;
|
|
|
|
u32 avid_start_stop_x;
|
|
|
|
u32 avid_start_stop_y;
|
|
|
|
u32 fid_int_start_x__fid_int_start_y;
|
|
|
|
u32 fid_int_offset_y__fid_ext_start_x;
|
|
|
|
u32 fid_ext_start_y__fid_ext_offset_y;
|
|
|
|
u32 tvdetgp_int_start_stop_x;
|
|
|
|
u32 tvdetgp_int_start_stop_y;
|
|
|
|
u32 gen_ctrl;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* from TRM */
|
|
|
|
static const struct venc_config venc_config_pal_trm = {
|
|
|
|
.f_control = 0,
|
|
|
|
.vidout_ctrl = 1,
|
|
|
|
.sync_ctrl = 0x40,
|
|
|
|
.llen = 0x35F, /* 863 */
|
|
|
|
.flens = 0x270, /* 624 */
|
|
|
|
.hfltr_ctrl = 0,
|
|
|
|
.cc_carr_wss_carr = 0x2F7225ED,
|
|
|
|
.c_phase = 0,
|
|
|
|
.gain_u = 0x111,
|
|
|
|
.gain_v = 0x181,
|
|
|
|
.gain_y = 0x140,
|
|
|
|
.black_level = 0x3B,
|
|
|
|
.blank_level = 0x3B,
|
|
|
|
.x_color = 0x7,
|
|
|
|
.m_control = 0x2,
|
|
|
|
.bstamp_wss_data = 0x3F,
|
|
|
|
.s_carr = 0x2A098ACB,
|
|
|
|
.line21 = 0,
|
|
|
|
.ln_sel = 0x01290015,
|
|
|
|
.l21__wc_ctl = 0x0000F603,
|
|
|
|
.htrigger_vtrigger = 0,
|
|
|
|
|
|
|
|
.savid__eavid = 0x06A70108,
|
|
|
|
.flen__fal = 0x00180270,
|
|
|
|
.lal__phase_reset = 0x00040135,
|
|
|
|
.hs_int_start_stop_x = 0x00880358,
|
|
|
|
.hs_ext_start_stop_x = 0x000F035F,
|
|
|
|
.vs_int_start_x = 0x01A70000,
|
|
|
|
.vs_int_stop_x__vs_int_start_y = 0x000001A7,
|
|
|
|
.vs_int_stop_y__vs_ext_start_x = 0x01AF0000,
|
|
|
|
.vs_ext_stop_x__vs_ext_start_y = 0x000101AF,
|
|
|
|
.vs_ext_stop_y = 0x00000025,
|
|
|
|
.avid_start_stop_x = 0x03530083,
|
|
|
|
.avid_start_stop_y = 0x026C002E,
|
|
|
|
.fid_int_start_x__fid_int_start_y = 0x0001008A,
|
|
|
|
.fid_int_offset_y__fid_ext_start_x = 0x002E0138,
|
|
|
|
.fid_ext_start_y__fid_ext_offset_y = 0x01380001,
|
|
|
|
|
|
|
|
.tvdetgp_int_start_stop_x = 0x00140001,
|
|
|
|
.tvdetgp_int_start_stop_y = 0x00010001,
|
|
|
|
.gen_ctrl = 0x00FF0000,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* from TRM */
|
|
|
|
static const struct venc_config venc_config_ntsc_trm = {
|
|
|
|
.f_control = 0,
|
|
|
|
.vidout_ctrl = 1,
|
|
|
|
.sync_ctrl = 0x8040,
|
|
|
|
.llen = 0x359,
|
|
|
|
.flens = 0x20C,
|
|
|
|
.hfltr_ctrl = 0,
|
|
|
|
.cc_carr_wss_carr = 0x043F2631,
|
|
|
|
.c_phase = 0,
|
|
|
|
.gain_u = 0x102,
|
|
|
|
.gain_v = 0x16C,
|
|
|
|
.gain_y = 0x12F,
|
|
|
|
.black_level = 0x43,
|
|
|
|
.blank_level = 0x38,
|
|
|
|
.x_color = 0x7,
|
|
|
|
.m_control = 0x1,
|
|
|
|
.bstamp_wss_data = 0x38,
|
|
|
|
.s_carr = 0x21F07C1F,
|
|
|
|
.line21 = 0,
|
|
|
|
.ln_sel = 0x01310011,
|
|
|
|
.l21__wc_ctl = 0x0000F003,
|
|
|
|
.htrigger_vtrigger = 0,
|
|
|
|
|
|
|
|
.savid__eavid = 0x069300F4,
|
|
|
|
.flen__fal = 0x0016020C,
|
|
|
|
.lal__phase_reset = 0x00060107,
|
|
|
|
.hs_int_start_stop_x = 0x008E0350,
|
|
|
|
.hs_ext_start_stop_x = 0x000F0359,
|
|
|
|
.vs_int_start_x = 0x01A00000,
|
|
|
|
.vs_int_stop_x__vs_int_start_y = 0x020701A0,
|
|
|
|
.vs_int_stop_y__vs_ext_start_x = 0x01AC0024,
|
|
|
|
.vs_ext_stop_x__vs_ext_start_y = 0x020D01AC,
|
|
|
|
.vs_ext_stop_y = 0x00000006,
|
|
|
|
.avid_start_stop_x = 0x03480078,
|
|
|
|
.avid_start_stop_y = 0x02060024,
|
|
|
|
.fid_int_start_x__fid_int_start_y = 0x0001008A,
|
|
|
|
.fid_int_offset_y__fid_ext_start_x = 0x01AC0106,
|
|
|
|
.fid_ext_start_y__fid_ext_offset_y = 0x01060006,
|
|
|
|
|
|
|
|
.tvdetgp_int_start_stop_x = 0x00140001,
|
|
|
|
.tvdetgp_int_start_stop_y = 0x00010001,
|
|
|
|
.gen_ctrl = 0x00F90000,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct venc_config venc_config_pal_bdghi = {
|
|
|
|
.f_control = 0,
|
|
|
|
.vidout_ctrl = 0,
|
|
|
|
.sync_ctrl = 0,
|
|
|
|
.hfltr_ctrl = 0,
|
|
|
|
.x_color = 0,
|
|
|
|
.line21 = 0,
|
|
|
|
.ln_sel = 21,
|
|
|
|
.htrigger_vtrigger = 0,
|
|
|
|
.tvdetgp_int_start_stop_x = 0x00140001,
|
|
|
|
.tvdetgp_int_start_stop_y = 0x00010001,
|
|
|
|
.gen_ctrl = 0x00FB0000,
|
|
|
|
|
|
|
|
.llen = 864-1,
|
|
|
|
.flens = 625-1,
|
|
|
|
.cc_carr_wss_carr = 0x2F7625ED,
|
|
|
|
.c_phase = 0xDF,
|
|
|
|
.gain_u = 0x111,
|
|
|
|
.gain_v = 0x181,
|
|
|
|
.gain_y = 0x140,
|
|
|
|
.black_level = 0x3e,
|
|
|
|
.blank_level = 0x3e,
|
|
|
|
.m_control = 0<<2 | 1<<1,
|
|
|
|
.bstamp_wss_data = 0x42,
|
|
|
|
.s_carr = 0x2a098acb,
|
|
|
|
.l21__wc_ctl = 0<<13 | 0x16<<8 | 0<<0,
|
|
|
|
.savid__eavid = 0x06A70108,
|
|
|
|
.flen__fal = 23<<16 | 624<<0,
|
|
|
|
.lal__phase_reset = 2<<17 | 310<<0,
|
|
|
|
.hs_int_start_stop_x = 0x00920358,
|
|
|
|
.hs_ext_start_stop_x = 0x000F035F,
|
|
|
|
.vs_int_start_x = 0x1a7<<16,
|
|
|
|
.vs_int_stop_x__vs_int_start_y = 0x000601A7,
|
|
|
|
.vs_int_stop_y__vs_ext_start_x = 0x01AF0036,
|
|
|
|
.vs_ext_stop_x__vs_ext_start_y = 0x27101af,
|
|
|
|
.vs_ext_stop_y = 0x05,
|
|
|
|
.avid_start_stop_x = 0x03530082,
|
|
|
|
.avid_start_stop_y = 0x0270002E,
|
|
|
|
.fid_int_start_x__fid_int_start_y = 0x0005008A,
|
|
|
|
.fid_int_offset_y__fid_ext_start_x = 0x002E0138,
|
|
|
|
.fid_ext_start_y__fid_ext_offset_y = 0x01380005,
|
|
|
|
};
|
|
|
|
|
2017-06-13 16:02:09 +07:00
|
|
|
enum venc_videomode {
|
|
|
|
VENC_MODE_UNKNOWN,
|
|
|
|
VENC_MODE_PAL,
|
|
|
|
VENC_MODE_NTSC,
|
|
|
|
};
|
|
|
|
|
2017-08-05 05:43:50 +07:00
|
|
|
static const struct videomode omap_dss_pal_vm = {
|
2016-09-22 18:06:46 +07:00
|
|
|
.hactive = 720,
|
2016-09-22 18:06:47 +07:00
|
|
|
.vactive = 574,
|
2013-04-10 18:12:14 +07:00
|
|
|
.pixelclock = 13500000,
|
2016-09-22 18:06:48 +07:00
|
|
|
.hsync_len = 64,
|
2016-09-22 18:06:49 +07:00
|
|
|
.hfront_porch = 12,
|
2016-09-22 18:06:50 +07:00
|
|
|
.hback_porch = 68,
|
2016-09-22 18:06:51 +07:00
|
|
|
.vsync_len = 5,
|
2016-09-22 18:06:52 +07:00
|
|
|
.vfront_porch = 5,
|
2016-09-22 18:06:53 +07:00
|
|
|
.vback_porch = 41,
|
2012-06-28 12:45:51 +07:00
|
|
|
|
2016-09-22 18:06:57 +07:00
|
|
|
.flags = DISPLAY_FLAGS_INTERLACED | DISPLAY_FLAGS_HSYNC_LOW |
|
2016-09-22 18:07:00 +07:00
|
|
|
DISPLAY_FLAGS_VSYNC_LOW | DISPLAY_FLAGS_DE_HIGH |
|
2016-09-22 18:07:01 +07:00
|
|
|
DISPLAY_FLAGS_PIXDATA_POSEDGE |
|
|
|
|
DISPLAY_FLAGS_SYNC_NEGEDGE,
|
2009-08-05 20:18:06 +07:00
|
|
|
};
|
|
|
|
|
2017-08-05 05:43:50 +07:00
|
|
|
static const struct videomode omap_dss_ntsc_vm = {
|
2016-09-22 18:06:46 +07:00
|
|
|
.hactive = 720,
|
2016-09-22 18:06:47 +07:00
|
|
|
.vactive = 482,
|
2013-04-10 18:12:14 +07:00
|
|
|
.pixelclock = 13500000,
|
2016-09-22 18:06:48 +07:00
|
|
|
.hsync_len = 64,
|
2016-09-22 18:06:49 +07:00
|
|
|
.hfront_porch = 16,
|
2016-09-22 18:06:50 +07:00
|
|
|
.hback_porch = 58,
|
2016-09-22 18:06:51 +07:00
|
|
|
.vsync_len = 6,
|
2016-09-22 18:06:52 +07:00
|
|
|
.vfront_porch = 6,
|
2016-09-22 18:06:53 +07:00
|
|
|
.vback_porch = 31,
|
2012-06-28 12:45:51 +07:00
|
|
|
|
2016-09-22 18:06:57 +07:00
|
|
|
.flags = DISPLAY_FLAGS_INTERLACED | DISPLAY_FLAGS_HSYNC_LOW |
|
2016-09-22 18:07:00 +07:00
|
|
|
DISPLAY_FLAGS_VSYNC_LOW | DISPLAY_FLAGS_DE_HIGH |
|
2016-09-22 18:07:01 +07:00
|
|
|
DISPLAY_FLAGS_PIXDATA_POSEDGE |
|
|
|
|
DISPLAY_FLAGS_SYNC_NEGEDGE,
|
2009-08-05 20:18:06 +07:00
|
|
|
};
|
|
|
|
|
2017-06-13 16:02:09 +07:00
|
|
|
static enum venc_videomode venc_get_videomode(const struct videomode *vm)
|
|
|
|
{
|
|
|
|
if (!(vm->flags & DISPLAY_FLAGS_INTERLACED))
|
|
|
|
return VENC_MODE_UNKNOWN;
|
|
|
|
|
|
|
|
if (vm->pixelclock == omap_dss_pal_vm.pixelclock &&
|
|
|
|
vm->hactive == omap_dss_pal_vm.hactive &&
|
|
|
|
vm->vactive == omap_dss_pal_vm.vactive)
|
|
|
|
return VENC_MODE_PAL;
|
|
|
|
|
|
|
|
if (vm->pixelclock == omap_dss_ntsc_vm.pixelclock &&
|
|
|
|
vm->hactive == omap_dss_ntsc_vm.hactive &&
|
|
|
|
vm->vactive == omap_dss_ntsc_vm.vactive)
|
|
|
|
return VENC_MODE_NTSC;
|
|
|
|
|
|
|
|
return VENC_MODE_UNKNOWN;
|
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device {
|
2011-01-24 13:22:01 +07:00
|
|
|
struct platform_device *pdev;
|
2009-08-05 20:18:06 +07:00
|
|
|
void __iomem *base;
|
|
|
|
struct mutex venc_lock;
|
|
|
|
struct regulator *vdda_dac_reg;
|
2018-02-13 19:00:27 +07:00
|
|
|
struct dss_device *dss;
|
2011-05-27 14:52:19 +07:00
|
|
|
|
2018-02-13 19:00:29 +07:00
|
|
|
struct dss_debugfs_entry *debugfs;
|
|
|
|
|
2011-05-27 14:52:19 +07:00
|
|
|
struct clk *tv_dac_clk;
|
2012-07-20 17:45:44 +07:00
|
|
|
|
2016-09-22 18:07:04 +07:00
|
|
|
struct videomode vm;
|
2012-08-16 13:25:15 +07:00
|
|
|
enum omap_dss_venc_type type;
|
2012-08-16 13:26:31 +07:00
|
|
|
bool invert_polarity;
|
2017-08-05 05:44:09 +07:00
|
|
|
bool requires_tv_dac_clk;
|
2012-09-26 18:00:49 +07:00
|
|
|
|
OMAPDSS: combine omap_dss_output into omap_dss_device
We currently have omap_dss_device, which represents an external display
device, sometimes an external encoder, sometimes a panel. Then we have
omap_dss_output, which represents DSS's output encoder.
In the future with new display device model, we construct a video
pipeline from the display blocks. To accomplish this, all the blocks
need to be presented by the same entity.
Thus, this patch combines omap_dss_output into omap_dss_device. Some of
the fields in omap_dss_output are already found in omap_dss_device, but
some are not. This means we'll have DSS output specific fields in
omap_dss_device, which is not very nice. However, it is easier to just
keep those output specific fields there for now, and after transition to
new display device model is made, they can be cleaned up easier than
could be done now.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-04-19 19:09:34 +07:00
|
|
|
struct omap_dss_device output;
|
2018-02-13 19:00:48 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
#define dssdev_to_venc(dssdev) container_of(dssdev, struct venc_device, output)
|
2009-08-05 20:18:06 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static inline void venc_write_reg(struct venc_device *venc, int idx, u32 val)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
__raw_writel(val, venc->base + idx);
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static inline u32 venc_read_reg(struct venc_device *venc, int idx)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
u32 l = __raw_readl(venc->base + idx);
|
2009-08-05 20:18:06 +07:00
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static void venc_write_config(struct venc_device *venc,
|
|
|
|
const struct venc_config *config)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
|
|
|
DSSDBG("write venc conf\n");
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_LLEN, config->llen);
|
|
|
|
venc_write_reg(venc, VENC_FLENS, config->flens);
|
|
|
|
venc_write_reg(venc, VENC_CC_CARR_WSS_CARR, config->cc_carr_wss_carr);
|
|
|
|
venc_write_reg(venc, VENC_C_PHASE, config->c_phase);
|
|
|
|
venc_write_reg(venc, VENC_GAIN_U, config->gain_u);
|
|
|
|
venc_write_reg(venc, VENC_GAIN_V, config->gain_v);
|
|
|
|
venc_write_reg(venc, VENC_GAIN_Y, config->gain_y);
|
|
|
|
venc_write_reg(venc, VENC_BLACK_LEVEL, config->black_level);
|
|
|
|
venc_write_reg(venc, VENC_BLANK_LEVEL, config->blank_level);
|
|
|
|
venc_write_reg(venc, VENC_M_CONTROL, config->m_control);
|
2018-09-20 05:19:32 +07:00
|
|
|
venc_write_reg(venc, VENC_BSTAMP_WSS_DATA, config->bstamp_wss_data);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_S_CARR, config->s_carr);
|
|
|
|
venc_write_reg(venc, VENC_L21__WC_CTL, config->l21__wc_ctl);
|
|
|
|
venc_write_reg(venc, VENC_SAVID__EAVID, config->savid__eavid);
|
|
|
|
venc_write_reg(venc, VENC_FLEN__FAL, config->flen__fal);
|
|
|
|
venc_write_reg(venc, VENC_LAL__PHASE_RESET, config->lal__phase_reset);
|
|
|
|
venc_write_reg(venc, VENC_HS_INT_START_STOP_X,
|
|
|
|
config->hs_int_start_stop_x);
|
|
|
|
venc_write_reg(venc, VENC_HS_EXT_START_STOP_X,
|
|
|
|
config->hs_ext_start_stop_x);
|
|
|
|
venc_write_reg(venc, VENC_VS_INT_START_X, config->vs_int_start_x);
|
|
|
|
venc_write_reg(venc, VENC_VS_INT_STOP_X__VS_INT_START_Y,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->vs_int_stop_x__vs_int_start_y);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_VS_INT_STOP_Y__VS_EXT_START_X,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->vs_int_stop_y__vs_ext_start_x);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_VS_EXT_STOP_X__VS_EXT_START_Y,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->vs_ext_stop_x__vs_ext_start_y);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_VS_EXT_STOP_Y, config->vs_ext_stop_y);
|
|
|
|
venc_write_reg(venc, VENC_AVID_START_STOP_X, config->avid_start_stop_x);
|
|
|
|
venc_write_reg(venc, VENC_AVID_START_STOP_Y, config->avid_start_stop_y);
|
|
|
|
venc_write_reg(venc, VENC_FID_INT_START_X__FID_INT_START_Y,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->fid_int_start_x__fid_int_start_y);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_FID_INT_OFFSET_Y__FID_EXT_START_X,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->fid_int_offset_y__fid_ext_start_x);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_FID_EXT_START_Y__FID_EXT_OFFSET_Y,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->fid_ext_start_y__fid_ext_offset_y);
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_DAC_B__DAC_C,
|
|
|
|
venc_read_reg(venc, VENC_DAC_B__DAC_C));
|
|
|
|
venc_write_reg(venc, VENC_VIDOUT_CTRL, config->vidout_ctrl);
|
|
|
|
venc_write_reg(venc, VENC_HFLTR_CTRL, config->hfltr_ctrl);
|
|
|
|
venc_write_reg(venc, VENC_X_COLOR, config->x_color);
|
|
|
|
venc_write_reg(venc, VENC_LINE21, config->line21);
|
|
|
|
venc_write_reg(venc, VENC_LN_SEL, config->ln_sel);
|
|
|
|
venc_write_reg(venc, VENC_HTRIGGER_VTRIGGER, config->htrigger_vtrigger);
|
|
|
|
venc_write_reg(venc, VENC_TVDETGP_INT_START_STOP_X,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->tvdetgp_int_start_stop_x);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_TVDETGP_INT_START_STOP_Y,
|
2009-08-05 20:18:06 +07:00
|
|
|
config->tvdetgp_int_start_stop_y);
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_GEN_CTRL, config->gen_ctrl);
|
|
|
|
venc_write_reg(venc, VENC_F_CONTROL, config->f_control);
|
|
|
|
venc_write_reg(venc, VENC_SYNC_CTRL, config->sync_ctrl);
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static void venc_reset(struct venc_device *venc)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
|
|
|
int t = 1000;
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_F_CONTROL, 1<<8);
|
|
|
|
while (venc_read_reg(venc, VENC_F_CONTROL) & (1<<8)) {
|
2009-08-05 20:18:06 +07:00
|
|
|
if (--t == 0) {
|
|
|
|
DSSERR("Failed to reset venc\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-02 21:48:22 +07:00
|
|
|
#ifdef CONFIG_OMAP2_DSS_SLEEP_AFTER_VENC_RESET
|
2009-08-05 20:18:06 +07:00
|
|
|
/* the magical sleep that makes things work */
|
2010-06-02 21:48:22 +07:00
|
|
|
/* XXX more info? What bug this circumvents? */
|
2009-08-05 20:18:06 +07:00
|
|
|
msleep(20);
|
2010-06-02 21:48:22 +07:00
|
|
|
#endif
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static int venc_runtime_get(struct venc_device *venc)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
2011-05-27 14:52:19 +07:00
|
|
|
int r;
|
|
|
|
|
|
|
|
DSSDBG("venc_runtime_get\n");
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
r = pm_runtime_get_sync(&venc->pdev->dev);
|
2011-05-27 14:52:19 +07:00
|
|
|
WARN_ON(r < 0);
|
|
|
|
return r < 0 ? r : 0;
|
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static void venc_runtime_put(struct venc_device *venc)
|
2011-05-27 14:52:19 +07:00
|
|
|
{
|
|
|
|
int r;
|
|
|
|
|
|
|
|
DSSDBG("venc_runtime_put\n");
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
r = pm_runtime_put_sync(&venc->pdev->dev);
|
2012-06-27 20:37:18 +07:00
|
|
|
WARN_ON(r < 0 && r != -ENOSYS);
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
|
|
|
|
2018-06-08 19:59:31 +07:00
|
|
|
static const struct venc_config *venc_timings_to_config(const struct videomode *vm)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
2017-06-13 16:02:09 +07:00
|
|
|
switch (venc_get_videomode(vm)) {
|
|
|
|
default:
|
|
|
|
WARN_ON_ONCE(1);
|
|
|
|
case VENC_MODE_PAL:
|
2009-08-05 20:18:06 +07:00
|
|
|
return &venc_config_pal_trm;
|
2017-06-13 16:02:09 +07:00
|
|
|
case VENC_MODE_NTSC:
|
2009-08-05 20:18:06 +07:00
|
|
|
return &venc_config_ntsc_trm;
|
2017-06-13 16:02:09 +07:00
|
|
|
}
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static int venc_power_on(struct venc_device *venc)
|
2010-01-12 20:12:07 +07:00
|
|
|
{
|
|
|
|
u32 l;
|
2011-11-21 18:42:58 +07:00
|
|
|
int r;
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
r = venc_runtime_get(venc);
|
2012-07-06 22:22:37 +07:00
|
|
|
if (r)
|
|
|
|
goto err0;
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_reset(venc);
|
|
|
|
venc_write_config(venc, venc_timings_to_config(&venc->vm));
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
dss_set_venc_output(venc->dss, venc->type);
|
|
|
|
dss_set_dac_pwrdn_bgz(venc->dss, 1);
|
2010-01-12 20:12:07 +07:00
|
|
|
|
|
|
|
l = 0;
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
if (venc->type == OMAP_DSS_VENC_TYPE_COMPOSITE)
|
2010-01-12 20:12:07 +07:00
|
|
|
l |= 1 << 1;
|
|
|
|
else /* S-Video */
|
|
|
|
l |= (1 << 0) | (1 << 2);
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
if (venc->invert_polarity == false)
|
2010-01-12 20:12:07 +07:00
|
|
|
l |= 1 << 3;
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_OUTPUT_CONTROL, l);
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
r = regulator_enable(venc->vdda_dac_reg);
|
2012-03-19 21:56:39 +07:00
|
|
|
if (r)
|
2012-07-06 22:22:37 +07:00
|
|
|
goto err1;
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
r = dss_mgr_enable(&venc->output);
|
2011-11-21 18:42:58 +07:00
|
|
|
if (r)
|
2012-07-06 22:22:37 +07:00
|
|
|
goto err2;
|
2011-11-21 18:42:58 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2012-07-06 22:22:37 +07:00
|
|
|
err2:
|
2018-02-13 19:00:48 +07:00
|
|
|
regulator_disable(venc->vdda_dac_reg);
|
2012-07-06 22:22:37 +07:00
|
|
|
err1:
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_OUTPUT_CONTROL, 0);
|
|
|
|
dss_set_dac_pwrdn_bgz(venc->dss, 0);
|
2011-11-21 18:42:58 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_runtime_put(venc);
|
2012-07-06 22:22:37 +07:00
|
|
|
err0:
|
2011-11-21 18:42:58 +07:00
|
|
|
return r;
|
2010-01-12 20:12:07 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static void venc_power_off(struct venc_device *venc)
|
2010-01-12 20:12:07 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_write_reg(venc, VENC_OUTPUT_CONTROL, 0);
|
|
|
|
dss_set_dac_pwrdn_bgz(venc->dss, 0);
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
dss_mgr_disable(&venc->output);
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
regulator_disable(venc->vdda_dac_reg);
|
2012-07-06 22:22:37 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_runtime_put(venc);
|
2010-01-12 20:12:07 +07:00
|
|
|
}
|
|
|
|
|
2018-08-24 23:38:07 +07:00
|
|
|
static void venc_display_enable(struct omap_dss_device *dssdev)
|
2012-04-24 04:08:54 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device *venc = dssdev_to_venc(dssdev);
|
2012-04-24 04:08:54 +07:00
|
|
|
|
2012-07-06 22:22:37 +07:00
|
|
|
DSSDBG("venc_display_enable\n");
|
2012-04-24 04:08:54 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
mutex_lock(&venc->venc_lock);
|
2012-04-24 04:08:54 +07:00
|
|
|
|
2018-08-24 23:38:07 +07:00
|
|
|
venc_power_on(venc);
|
2012-07-06 22:22:37 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
mutex_unlock(&venc->venc_lock);
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
|
|
|
|
2013-05-22 17:19:04 +07:00
|
|
|
static void venc_display_disable(struct omap_dss_device *dssdev)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device *venc = dssdev_to_venc(dssdev);
|
|
|
|
|
2012-07-06 22:22:37 +07:00
|
|
|
DSSDBG("venc_display_disable\n");
|
2009-08-05 20:18:06 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
mutex_lock(&venc->venc_lock);
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_power_off(venc);
|
2010-01-12 20:12:07 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
mutex_unlock(&venc->venc_lock);
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
|
|
|
|
2018-06-06 21:49:37 +07:00
|
|
|
static void venc_get_timings(struct omap_dss_device *dssdev,
|
|
|
|
struct videomode *vm)
|
|
|
|
{
|
|
|
|
struct venc_device *venc = dssdev_to_venc(dssdev);
|
|
|
|
|
|
|
|
mutex_lock(&venc->venc_lock);
|
|
|
|
*vm = venc->vm;
|
|
|
|
mutex_unlock(&venc->venc_lock);
|
|
|
|
}
|
|
|
|
|
2013-05-22 17:19:04 +07:00
|
|
|
static void venc_set_timings(struct omap_dss_device *dssdev,
|
2018-06-04 22:29:01 +07:00
|
|
|
const struct videomode *vm)
|
2010-01-20 17:11:25 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device *venc = dssdev_to_venc(dssdev);
|
2017-06-13 16:02:09 +07:00
|
|
|
|
2010-01-20 17:11:25 +07:00
|
|
|
DSSDBG("venc_set_timings\n");
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
mutex_lock(&venc->venc_lock);
|
2012-07-06 22:22:37 +07:00
|
|
|
|
2018-06-08 19:59:31 +07:00
|
|
|
venc->vm = *vm;
|
2012-07-06 22:22:37 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
dispc_set_tv_pclk(venc->dss->dispc, 13500000);
|
2013-05-16 14:44:13 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
mutex_unlock(&venc->venc_lock);
|
2010-01-20 17:11:25 +07:00
|
|
|
}
|
|
|
|
|
2013-05-22 17:19:04 +07:00
|
|
|
static int venc_check_timings(struct omap_dss_device *dssdev,
|
2016-09-22 18:07:04 +07:00
|
|
|
struct videomode *vm)
|
2010-01-20 17:11:25 +07:00
|
|
|
{
|
|
|
|
DSSDBG("venc_check_timings\n");
|
|
|
|
|
2017-06-13 16:02:09 +07:00
|
|
|
switch (venc_get_videomode(vm)) {
|
|
|
|
case VENC_MODE_PAL:
|
2018-06-08 19:59:31 +07:00
|
|
|
*vm = omap_dss_pal_vm;
|
|
|
|
return 0;
|
|
|
|
|
2017-06-13 16:02:09 +07:00
|
|
|
case VENC_MODE_NTSC:
|
2018-06-08 19:59:31 +07:00
|
|
|
*vm = omap_dss_ntsc_vm;
|
2010-01-20 17:11:25 +07:00
|
|
|
return 0;
|
2018-06-08 19:59:31 +07:00
|
|
|
|
2017-06-13 16:02:09 +07:00
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2010-01-20 17:11:25 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:29 +07:00
|
|
|
static int venc_dump_regs(struct seq_file *s, void *p)
|
2009-08-05 20:18:06 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device *venc = s->private;
|
|
|
|
|
|
|
|
#define DUMPREG(venc, r) \
|
|
|
|
seq_printf(s, "%-35s %08x\n", #r, venc_read_reg(venc, r))
|
2009-08-05 20:18:06 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
if (venc_runtime_get(venc))
|
2018-02-13 19:00:29 +07:00
|
|
|
return 0;
|
2009-08-05 20:18:06 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
DUMPREG(venc, VENC_F_CONTROL);
|
|
|
|
DUMPREG(venc, VENC_VIDOUT_CTRL);
|
|
|
|
DUMPREG(venc, VENC_SYNC_CTRL);
|
|
|
|
DUMPREG(venc, VENC_LLEN);
|
|
|
|
DUMPREG(venc, VENC_FLENS);
|
|
|
|
DUMPREG(venc, VENC_HFLTR_CTRL);
|
|
|
|
DUMPREG(venc, VENC_CC_CARR_WSS_CARR);
|
|
|
|
DUMPREG(venc, VENC_C_PHASE);
|
|
|
|
DUMPREG(venc, VENC_GAIN_U);
|
|
|
|
DUMPREG(venc, VENC_GAIN_V);
|
|
|
|
DUMPREG(venc, VENC_GAIN_Y);
|
|
|
|
DUMPREG(venc, VENC_BLACK_LEVEL);
|
|
|
|
DUMPREG(venc, VENC_BLANK_LEVEL);
|
|
|
|
DUMPREG(venc, VENC_X_COLOR);
|
|
|
|
DUMPREG(venc, VENC_M_CONTROL);
|
|
|
|
DUMPREG(venc, VENC_BSTAMP_WSS_DATA);
|
|
|
|
DUMPREG(venc, VENC_S_CARR);
|
|
|
|
DUMPREG(venc, VENC_LINE21);
|
|
|
|
DUMPREG(venc, VENC_LN_SEL);
|
|
|
|
DUMPREG(venc, VENC_L21__WC_CTL);
|
|
|
|
DUMPREG(venc, VENC_HTRIGGER_VTRIGGER);
|
|
|
|
DUMPREG(venc, VENC_SAVID__EAVID);
|
|
|
|
DUMPREG(venc, VENC_FLEN__FAL);
|
|
|
|
DUMPREG(venc, VENC_LAL__PHASE_RESET);
|
|
|
|
DUMPREG(venc, VENC_HS_INT_START_STOP_X);
|
|
|
|
DUMPREG(venc, VENC_HS_EXT_START_STOP_X);
|
|
|
|
DUMPREG(venc, VENC_VS_INT_START_X);
|
|
|
|
DUMPREG(venc, VENC_VS_INT_STOP_X__VS_INT_START_Y);
|
|
|
|
DUMPREG(venc, VENC_VS_INT_STOP_Y__VS_EXT_START_X);
|
|
|
|
DUMPREG(venc, VENC_VS_EXT_STOP_X__VS_EXT_START_Y);
|
|
|
|
DUMPREG(venc, VENC_VS_EXT_STOP_Y);
|
|
|
|
DUMPREG(venc, VENC_AVID_START_STOP_X);
|
|
|
|
DUMPREG(venc, VENC_AVID_START_STOP_Y);
|
|
|
|
DUMPREG(venc, VENC_FID_INT_START_X__FID_INT_START_Y);
|
|
|
|
DUMPREG(venc, VENC_FID_INT_OFFSET_Y__FID_EXT_START_X);
|
|
|
|
DUMPREG(venc, VENC_FID_EXT_START_Y__FID_EXT_OFFSET_Y);
|
|
|
|
DUMPREG(venc, VENC_TVDETGP_INT_START_STOP_X);
|
|
|
|
DUMPREG(venc, VENC_TVDETGP_INT_START_STOP_Y);
|
|
|
|
DUMPREG(venc, VENC_GEN_CTRL);
|
|
|
|
DUMPREG(venc, VENC_OUTPUT_CONTROL);
|
|
|
|
DUMPREG(venc, VENC_OUTPUT_TEST);
|
|
|
|
|
|
|
|
venc_runtime_put(venc);
|
2009-08-05 20:18:06 +07:00
|
|
|
|
|
|
|
#undef DUMPREG
|
2018-02-13 19:00:29 +07:00
|
|
|
return 0;
|
2009-08-05 20:18:06 +07:00
|
|
|
}
|
2011-01-24 13:22:01 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static int venc_get_clocks(struct venc_device *venc)
|
2011-05-27 14:52:19 +07:00
|
|
|
{
|
|
|
|
struct clk *clk;
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
if (venc->requires_tv_dac_clk) {
|
|
|
|
clk = devm_clk_get(&venc->pdev->dev, "tv_dac_clk");
|
2011-05-27 14:52:19 +07:00
|
|
|
if (IS_ERR(clk)) {
|
|
|
|
DSSERR("can't get tv_dac_clk\n");
|
|
|
|
return PTR_ERR(clk);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
clk = NULL;
|
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc->tv_dac_clk = clk;
|
2011-05-27 14:52:19 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-03-05 04:42:36 +07:00
|
|
|
static int venc_connect(struct omap_dss_device *src,
|
|
|
|
struct omap_dss_device *dst)
|
2013-05-24 17:19:50 +07:00
|
|
|
{
|
2018-09-05 03:53:34 +07:00
|
|
|
return omapdss_device_connect(dst->dss, dst, dst->next);
|
2013-05-24 17:19:50 +07:00
|
|
|
}
|
|
|
|
|
2018-03-05 04:42:36 +07:00
|
|
|
static void venc_disconnect(struct omap_dss_device *src,
|
|
|
|
struct omap_dss_device *dst)
|
2013-05-24 17:19:50 +07:00
|
|
|
{
|
2018-03-05 04:42:36 +07:00
|
|
|
omapdss_device_disconnect(dst, dst->next);
|
2013-05-24 17:19:50 +07:00
|
|
|
}
|
|
|
|
|
2018-02-28 20:58:13 +07:00
|
|
|
static const struct omap_dss_device_ops venc_ops = {
|
2013-05-24 17:19:50 +07:00
|
|
|
.connect = venc_connect,
|
|
|
|
.disconnect = venc_disconnect,
|
|
|
|
|
2013-05-22 17:19:04 +07:00
|
|
|
.enable = venc_display_enable,
|
|
|
|
.disable = venc_display_disable,
|
2013-05-24 17:19:50 +07:00
|
|
|
|
2013-05-22 17:19:04 +07:00
|
|
|
.check_timings = venc_check_timings,
|
2018-06-06 21:49:37 +07:00
|
|
|
.get_timings = venc_get_timings,
|
2013-05-22 17:19:04 +07:00
|
|
|
.set_timings = venc_set_timings,
|
2013-05-24 17:19:50 +07:00
|
|
|
};
|
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
/* -----------------------------------------------------------------------------
|
|
|
|
* Component Bind & Unbind
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int venc_bind(struct device *dev, struct device *master, void *data)
|
|
|
|
{
|
|
|
|
struct dss_device *dss = dss_get_device(master);
|
|
|
|
struct venc_device *venc = dev_get_drvdata(dev);
|
|
|
|
u8 rev_id;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
venc->dss = dss;
|
|
|
|
|
|
|
|
r = venc_runtime_get(venc);
|
|
|
|
if (r)
|
|
|
|
return r;
|
|
|
|
|
|
|
|
rev_id = (u8)(venc_read_reg(venc, VENC_REV_ID) & 0xff);
|
|
|
|
dev_dbg(dev, "OMAP VENC rev %d\n", rev_id);
|
|
|
|
|
|
|
|
venc_runtime_put(venc);
|
|
|
|
|
|
|
|
venc->debugfs = dss_debugfs_create_file(dss, "venc", venc_dump_regs,
|
|
|
|
venc);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void venc_unbind(struct device *dev, struct device *master, void *data)
|
|
|
|
{
|
|
|
|
struct venc_device *venc = dev_get_drvdata(dev);
|
|
|
|
|
|
|
|
dss_debugfs_remove_file(venc->debugfs);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct component_ops venc_component_ops = {
|
|
|
|
.bind = venc_bind,
|
|
|
|
.unbind = venc_unbind,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
|
|
* Probe & Remove, Suspend & Resume
|
|
|
|
*/
|
|
|
|
|
2018-03-03 03:13:06 +07:00
|
|
|
static int venc_init_output(struct venc_device *venc)
|
2012-09-26 18:00:49 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct omap_dss_device *out = &venc->output;
|
2018-03-06 06:25:13 +07:00
|
|
|
int r;
|
2012-09-26 18:00:49 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
out->dev = &venc->pdev->dev;
|
2012-09-26 18:00:49 +07:00
|
|
|
out->id = OMAP_DSS_OUTPUT_VENC;
|
OMAPDSS: combine omap_dss_output into omap_dss_device
We currently have omap_dss_device, which represents an external display
device, sometimes an external encoder, sometimes a panel. Then we have
omap_dss_output, which represents DSS's output encoder.
In the future with new display device model, we construct a video
pipeline from the display blocks. To accomplish this, all the blocks
need to be presented by the same entity.
Thus, this patch combines omap_dss_output into omap_dss_device. Some of
the fields in omap_dss_output are already found in omap_dss_device, but
some are not. This means we'll have DSS output specific fields in
omap_dss_device, which is not very nice. However, it is easier to just
keep those output specific fields there for now, and after transition to
new display device model is made, they can be cleaned up easier than
could be done now.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-04-19 19:09:34 +07:00
|
|
|
out->output_type = OMAP_DISPLAY_TYPE_VENC;
|
2013-02-18 18:06:01 +07:00
|
|
|
out->name = "venc.0";
|
2013-02-13 16:23:54 +07:00
|
|
|
out->dispc_channel = OMAP_DSS_CHANNEL_DIGIT;
|
2018-02-28 20:58:13 +07:00
|
|
|
out->ops = &venc_ops;
|
2013-05-03 15:42:18 +07:00
|
|
|
out->owner = THIS_MODULE;
|
2018-03-05 02:49:28 +07:00
|
|
|
out->of_ports = BIT(0);
|
2012-09-26 18:00:49 +07:00
|
|
|
|
2018-09-12 23:41:31 +07:00
|
|
|
r = omapdss_device_init_output(out);
|
|
|
|
if (r < 0)
|
2018-03-06 06:25:13 +07:00
|
|
|
return r;
|
|
|
|
|
2018-03-02 06:25:32 +07:00
|
|
|
omapdss_device_register(out);
|
2018-03-03 03:13:06 +07:00
|
|
|
|
|
|
|
return 0;
|
2012-09-26 18:00:49 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static void venc_uninit_output(struct venc_device *venc)
|
2012-09-26 18:00:49 +07:00
|
|
|
{
|
2018-03-02 06:25:32 +07:00
|
|
|
omapdss_device_unregister(&venc->output);
|
2018-09-12 23:41:31 +07:00
|
|
|
omapdss_device_cleanup_output(&venc->output);
|
2012-09-26 18:00:49 +07:00
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
static int venc_probe_of(struct venc_device *venc)
|
2013-12-16 20:14:15 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct device_node *node = venc->pdev->dev.of_node;
|
2013-12-16 20:14:15 +07:00
|
|
|
struct device_node *ep;
|
|
|
|
u32 channels;
|
|
|
|
int r;
|
|
|
|
|
2017-03-22 20:26:08 +07:00
|
|
|
ep = of_graph_get_endpoint_by_regs(node, 0, 0);
|
2013-12-16 20:14:15 +07:00
|
|
|
if (!ep)
|
|
|
|
return 0;
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc->invert_polarity = of_property_read_bool(ep, "ti,invert-polarity");
|
2013-12-16 20:14:15 +07:00
|
|
|
|
|
|
|
r = of_property_read_u32(ep, "ti,channels", &channels);
|
|
|
|
if (r) {
|
2018-02-13 19:00:48 +07:00
|
|
|
dev_err(&venc->pdev->dev,
|
2013-12-16 20:14:15 +07:00
|
|
|
"failed to read property 'ti,channels': %d\n", r);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (channels) {
|
|
|
|
case 1:
|
2018-02-13 19:00:48 +07:00
|
|
|
venc->type = OMAP_DSS_VENC_TYPE_COMPOSITE;
|
2013-12-16 20:14:15 +07:00
|
|
|
break;
|
|
|
|
case 2:
|
2018-02-13 19:00:48 +07:00
|
|
|
venc->type = OMAP_DSS_VENC_TYPE_SVIDEO;
|
2013-12-16 20:14:15 +07:00
|
|
|
break;
|
|
|
|
default:
|
2018-02-13 19:00:48 +07:00
|
|
|
dev_err(&venc->pdev->dev, "bad channel propert '%d'\n",
|
|
|
|
channels);
|
2013-12-16 20:14:15 +07:00
|
|
|
r = -EINVAL;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
of_node_put(ep);
|
|
|
|
|
|
|
|
return 0;
|
2017-10-13 21:59:04 +07:00
|
|
|
|
2013-12-16 20:14:15 +07:00
|
|
|
err:
|
|
|
|
of_node_put(ep);
|
2017-10-13 21:59:04 +07:00
|
|
|
return r;
|
2013-12-16 20:14:15 +07:00
|
|
|
}
|
|
|
|
|
2017-08-05 05:44:09 +07:00
|
|
|
static const struct soc_device_attribute venc_soc_devices[] = {
|
|
|
|
{ .machine = "OMAP3[45]*" },
|
|
|
|
{ .machine = "AM35*" },
|
|
|
|
{ /* sentinel */ }
|
|
|
|
};
|
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
static int venc_probe(struct platform_device *pdev)
|
2011-01-24 13:22:01 +07:00
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device *venc;
|
2011-01-24 13:22:04 +07:00
|
|
|
struct resource *venc_mem;
|
2012-05-02 18:55:12 +07:00
|
|
|
int r;
|
2011-01-24 13:22:04 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc = kzalloc(sizeof(*venc), GFP_KERNEL);
|
|
|
|
if (!venc)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
venc->pdev = pdev;
|
2018-03-03 23:52:59 +07:00
|
|
|
|
|
|
|
platform_set_drvdata(pdev, venc);
|
2011-01-24 13:22:01 +07:00
|
|
|
|
2017-08-05 05:44:09 +07:00
|
|
|
/* The OMAP34xx, OMAP35xx and AM35xx VENC require the TV DAC clock. */
|
|
|
|
if (soc_device_match(venc_soc_devices))
|
2018-02-13 19:00:48 +07:00
|
|
|
venc->requires_tv_dac_clk = true;
|
2017-08-05 05:44:09 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
mutex_init(&venc->venc_lock);
|
2011-01-24 13:22:01 +07:00
|
|
|
|
2018-06-06 21:49:37 +07:00
|
|
|
venc->vm = omap_dss_pal_vm;
|
2011-01-24 13:22:01 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_mem = platform_get_resource(venc->pdev, IORESOURCE_MEM, 0);
|
|
|
|
venc->base = devm_ioremap_resource(&pdev->dev, venc_mem);
|
|
|
|
if (IS_ERR(venc->base)) {
|
|
|
|
r = PTR_ERR(venc->base);
|
|
|
|
goto err_free;
|
|
|
|
}
|
2011-01-24 13:22:01 +07:00
|
|
|
|
2018-03-05 05:10:55 +07:00
|
|
|
venc->vdda_dac_reg = devm_regulator_get(&pdev->dev, "vdda");
|
|
|
|
if (IS_ERR(venc->vdda_dac_reg)) {
|
|
|
|
r = PTR_ERR(venc->vdda_dac_reg);
|
|
|
|
if (r != -EPROBE_DEFER)
|
|
|
|
DSSERR("can't get VDDA_DAC regulator\n");
|
|
|
|
goto err_free;
|
|
|
|
}
|
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
r = venc_get_clocks(venc);
|
2011-05-27 14:52:19 +07:00
|
|
|
if (r)
|
2018-02-13 19:00:48 +07:00
|
|
|
goto err_free;
|
2011-05-27 14:52:19 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
r = venc_probe_of(venc);
|
2018-03-03 23:52:59 +07:00
|
|
|
if (r)
|
|
|
|
goto err_free;
|
2013-12-16 20:14:15 +07:00
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
pm_runtime_enable(&pdev->dev);
|
2012-03-02 23:01:07 +07:00
|
|
|
|
2018-03-03 03:13:06 +07:00
|
|
|
r = venc_init_output(venc);
|
|
|
|
if (r)
|
|
|
|
goto err_pm_disable;
|
2012-09-26 18:00:49 +07:00
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
r = component_add(&pdev->dev, &venc_component_ops);
|
|
|
|
if (r)
|
|
|
|
goto err_uninit_output;
|
|
|
|
|
2012-01-25 18:31:04 +07:00
|
|
|
return 0;
|
2011-05-27 14:52:19 +07:00
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
err_uninit_output:
|
|
|
|
venc_uninit_output(venc);
|
2018-03-03 03:13:06 +07:00
|
|
|
err_pm_disable:
|
2011-05-27 14:52:19 +07:00
|
|
|
pm_runtime_disable(&pdev->dev);
|
2018-02-13 19:00:48 +07:00
|
|
|
err_free:
|
|
|
|
kfree(venc);
|
2011-05-27 14:52:19 +07:00
|
|
|
return r;
|
2011-01-24 13:22:01 +07:00
|
|
|
}
|
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
static int venc_remove(struct platform_device *pdev)
|
2011-01-24 13:22:01 +07:00
|
|
|
{
|
2018-03-03 23:52:59 +07:00
|
|
|
struct venc_device *venc = platform_get_drvdata(pdev);
|
OMAPDSS: componentize omapdss
omapdss kernel module contains drivers for multiple devices, one for
each DSS submodule. The probing we have at the moment is a mess, and
doesn't give us proper deferred probing nor ensure that all the devices
are probed before omapfb/omapdrm start using omapdss.
This patch solves the mess by using the component system for DSS
submodules.
The changes to all DSS submodules (dispc, dpi, dsi, hdmi4/5, rfbi, sdi,
venc) are the same: probe & remove functions are changed to bind &
unbind, and new probe & remove functions are added which call
component_add/del.
The dss_core driver (dss.c) acts as a component master. Adding and
matching the components is simple: all dss device's child devices are
added as components.
However, we do have some dependencies between the drivers. The order in
which they should be probed is reflected by the list in core.c
(dss_output_drv_reg_funcs). The drivers are registered in that order,
which causes the components to be added in that order, which makes the
components to be bound in that order. This feels a bit fragile, and we
probably should improve the code to manage binds in random order.
However, for now, this works fine.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Acked-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
2015-06-04 19:22:23 +07:00
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
component_del(&pdev->dev, &venc_component_ops);
|
2018-02-13 19:00:29 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
venc_uninit_output(venc);
|
2012-09-26 18:00:49 +07:00
|
|
|
|
2018-03-03 23:52:59 +07:00
|
|
|
pm_runtime_disable(&pdev->dev);
|
2018-02-13 19:00:48 +07:00
|
|
|
|
|
|
|
kfree(venc);
|
2011-01-24 13:22:01 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-05-27 14:52:19 +07:00
|
|
|
static int venc_runtime_suspend(struct device *dev)
|
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device *venc = dev_get_drvdata(dev);
|
|
|
|
|
|
|
|
if (venc->tv_dac_clk)
|
|
|
|
clk_disable_unprepare(venc->tv_dac_clk);
|
2011-05-27 14:52:19 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int venc_runtime_resume(struct device *dev)
|
|
|
|
{
|
2018-02-13 19:00:48 +07:00
|
|
|
struct venc_device *venc = dev_get_drvdata(dev);
|
2011-05-27 14:52:19 +07:00
|
|
|
|
2018-02-13 19:00:48 +07:00
|
|
|
if (venc->tv_dac_clk)
|
|
|
|
clk_prepare_enable(venc->tv_dac_clk);
|
2011-05-27 14:52:19 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct dev_pm_ops venc_pm_ops = {
|
|
|
|
.runtime_suspend = venc_runtime_suspend,
|
|
|
|
.runtime_resume = venc_runtime_resume,
|
|
|
|
};
|
|
|
|
|
2013-12-16 20:14:15 +07:00
|
|
|
static const struct of_device_id venc_of_match[] = {
|
|
|
|
{ .compatible = "ti,omap2-venc", },
|
|
|
|
{ .compatible = "ti,omap3-venc", },
|
|
|
|
{ .compatible = "ti,omap4-venc", },
|
|
|
|
{},
|
|
|
|
};
|
|
|
|
|
2017-12-06 03:29:32 +07:00
|
|
|
struct platform_driver omap_venchw_driver = {
|
OMAPDSS: componentize omapdss
omapdss kernel module contains drivers for multiple devices, one for
each DSS submodule. The probing we have at the moment is a mess, and
doesn't give us proper deferred probing nor ensure that all the devices
are probed before omapfb/omapdrm start using omapdss.
This patch solves the mess by using the component system for DSS
submodules.
The changes to all DSS submodules (dispc, dpi, dsi, hdmi4/5, rfbi, sdi,
venc) are the same: probe & remove functions are changed to bind &
unbind, and new probe & remove functions are added which call
component_add/del.
The dss_core driver (dss.c) acts as a component master. Adding and
matching the components is simple: all dss device's child devices are
added as components.
However, we do have some dependencies between the drivers. The order in
which they should be probed is reflected by the list in core.c
(dss_output_drv_reg_funcs). The drivers are registered in that order,
which causes the components to be added in that order, which makes the
components to be bound in that order. This feels a bit fragile, and we
probably should improve the code to manage binds in random order.
However, for now, this works fine.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Acked-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
2015-06-04 19:22:23 +07:00
|
|
|
.probe = venc_probe,
|
|
|
|
.remove = venc_remove,
|
2011-01-24 13:22:01 +07:00
|
|
|
.driver = {
|
|
|
|
.name = "omapdss_venc",
|
2011-05-27 14:52:19 +07:00
|
|
|
.pm = &venc_pm_ops,
|
2013-12-16 20:14:15 +07:00
|
|
|
.of_match_table = venc_of_match,
|
2014-10-16 13:54:25 +07:00
|
|
|
.suppress_bind_attrs = true,
|
2011-01-24 13:22:01 +07:00
|
|
|
},
|
|
|
|
};
|