2009-08-05 13:18:06 +00:00
|
|
|
/*
|
|
|
|
* linux/drivers/video/omap2/dss/venc.c
|
|
|
|
*
|
|
|
|
* Copyright (C) 2009 Nokia Corporation
|
|
|
|
* Author: Tomi Valkeinen <tomi.valkeinen@nokia.com>
|
|
|
|
*
|
|
|
|
* 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 07:52:19 +00:00
|
|
|
#include <linux/pm_runtime.h>
|
2009-08-05 13:18:06 +00:00
|
|
|
|
2011-05-11 11:05:07 +00:00
|
|
|
#include <video/omapdss.h>
|
2009-08-05 13:18:06 +00:00
|
|
|
|
|
|
|
#include "dss.h"
|
2011-05-18 08:59:21 +00:00
|
|
|
#include "dss_features.h"
|
2009-08-05 13:18:06 +00:00
|
|
|
|
|
|
|
/* 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,
|
|
|
|
};
|
|
|
|
|
|
|
|
const struct omap_video_timings omap_dss_pal_timings = {
|
|
|
|
.x_res = 720,
|
|
|
|
.y_res = 574,
|
|
|
|
.pixel_clock = 13500,
|
|
|
|
.hsw = 64,
|
|
|
|
.hfp = 12,
|
|
|
|
.hbp = 68,
|
|
|
|
.vsw = 5,
|
|
|
|
.vfp = 5,
|
|
|
|
.vbp = 41,
|
2012-06-28 05:45:51 +00:00
|
|
|
|
|
|
|
.interlace = true,
|
2009-08-05 13:18:06 +00:00
|
|
|
};
|
|
|
|
EXPORT_SYMBOL(omap_dss_pal_timings);
|
|
|
|
|
|
|
|
const struct omap_video_timings omap_dss_ntsc_timings = {
|
|
|
|
.x_res = 720,
|
|
|
|
.y_res = 482,
|
|
|
|
.pixel_clock = 13500,
|
|
|
|
.hsw = 64,
|
|
|
|
.hfp = 16,
|
|
|
|
.hbp = 58,
|
|
|
|
.vsw = 6,
|
|
|
|
.vfp = 6,
|
|
|
|
.vbp = 31,
|
2012-06-28 05:45:51 +00:00
|
|
|
|
|
|
|
.interlace = true,
|
2009-08-05 13:18:06 +00:00
|
|
|
};
|
|
|
|
EXPORT_SYMBOL(omap_dss_ntsc_timings);
|
|
|
|
|
|
|
|
static struct {
|
2011-01-24 06:22:01 +00:00
|
|
|
struct platform_device *pdev;
|
2009-08-05 13:18:06 +00:00
|
|
|
void __iomem *base;
|
|
|
|
struct mutex venc_lock;
|
|
|
|
u32 wss_data;
|
|
|
|
struct regulator *vdda_dac_reg;
|
2011-05-27 07:52:19 +00:00
|
|
|
|
|
|
|
struct clk *tv_dac_clk;
|
2012-07-20 10:45:44 +00:00
|
|
|
|
|
|
|
struct omap_video_timings timings;
|
2012-08-16 06:25:15 +00:00
|
|
|
enum omap_dss_venc_type type;
|
2012-08-16 06:26:31 +00:00
|
|
|
bool invert_polarity;
|
2012-09-26 11:00:49 +00:00
|
|
|
|
|
|
|
struct omap_dss_output output;
|
2009-08-05 13:18:06 +00:00
|
|
|
} venc;
|
|
|
|
|
|
|
|
static inline void venc_write_reg(int idx, u32 val)
|
|
|
|
{
|
|
|
|
__raw_writel(val, venc.base + idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u32 venc_read_reg(int idx)
|
|
|
|
{
|
|
|
|
u32 l = __raw_readl(venc.base + idx);
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void venc_write_config(const struct venc_config *config)
|
|
|
|
{
|
|
|
|
DSSDBG("write venc conf\n");
|
|
|
|
|
|
|
|
venc_write_reg(VENC_LLEN, config->llen);
|
|
|
|
venc_write_reg(VENC_FLENS, config->flens);
|
|
|
|
venc_write_reg(VENC_CC_CARR_WSS_CARR, config->cc_carr_wss_carr);
|
|
|
|
venc_write_reg(VENC_C_PHASE, config->c_phase);
|
|
|
|
venc_write_reg(VENC_GAIN_U, config->gain_u);
|
|
|
|
venc_write_reg(VENC_GAIN_V, config->gain_v);
|
|
|
|
venc_write_reg(VENC_GAIN_Y, config->gain_y);
|
|
|
|
venc_write_reg(VENC_BLACK_LEVEL, config->black_level);
|
|
|
|
venc_write_reg(VENC_BLANK_LEVEL, config->blank_level);
|
|
|
|
venc_write_reg(VENC_M_CONTROL, config->m_control);
|
|
|
|
venc_write_reg(VENC_BSTAMP_WSS_DATA, config->bstamp_wss_data |
|
|
|
|
venc.wss_data);
|
|
|
|
venc_write_reg(VENC_S_CARR, config->s_carr);
|
|
|
|
venc_write_reg(VENC_L21__WC_CTL, config->l21__wc_ctl);
|
|
|
|
venc_write_reg(VENC_SAVID__EAVID, config->savid__eavid);
|
|
|
|
venc_write_reg(VENC_FLEN__FAL, config->flen__fal);
|
|
|
|
venc_write_reg(VENC_LAL__PHASE_RESET, config->lal__phase_reset);
|
|
|
|
venc_write_reg(VENC_HS_INT_START_STOP_X, config->hs_int_start_stop_x);
|
|
|
|
venc_write_reg(VENC_HS_EXT_START_STOP_X, config->hs_ext_start_stop_x);
|
|
|
|
venc_write_reg(VENC_VS_INT_START_X, config->vs_int_start_x);
|
|
|
|
venc_write_reg(VENC_VS_INT_STOP_X__VS_INT_START_Y,
|
|
|
|
config->vs_int_stop_x__vs_int_start_y);
|
|
|
|
venc_write_reg(VENC_VS_INT_STOP_Y__VS_EXT_START_X,
|
|
|
|
config->vs_int_stop_y__vs_ext_start_x);
|
|
|
|
venc_write_reg(VENC_VS_EXT_STOP_X__VS_EXT_START_Y,
|
|
|
|
config->vs_ext_stop_x__vs_ext_start_y);
|
|
|
|
venc_write_reg(VENC_VS_EXT_STOP_Y, config->vs_ext_stop_y);
|
|
|
|
venc_write_reg(VENC_AVID_START_STOP_X, config->avid_start_stop_x);
|
|
|
|
venc_write_reg(VENC_AVID_START_STOP_Y, config->avid_start_stop_y);
|
|
|
|
venc_write_reg(VENC_FID_INT_START_X__FID_INT_START_Y,
|
|
|
|
config->fid_int_start_x__fid_int_start_y);
|
|
|
|
venc_write_reg(VENC_FID_INT_OFFSET_Y__FID_EXT_START_X,
|
|
|
|
config->fid_int_offset_y__fid_ext_start_x);
|
|
|
|
venc_write_reg(VENC_FID_EXT_START_Y__FID_EXT_OFFSET_Y,
|
|
|
|
config->fid_ext_start_y__fid_ext_offset_y);
|
|
|
|
|
|
|
|
venc_write_reg(VENC_DAC_B__DAC_C, venc_read_reg(VENC_DAC_B__DAC_C));
|
|
|
|
venc_write_reg(VENC_VIDOUT_CTRL, config->vidout_ctrl);
|
|
|
|
venc_write_reg(VENC_HFLTR_CTRL, config->hfltr_ctrl);
|
|
|
|
venc_write_reg(VENC_X_COLOR, config->x_color);
|
|
|
|
venc_write_reg(VENC_LINE21, config->line21);
|
|
|
|
venc_write_reg(VENC_LN_SEL, config->ln_sel);
|
|
|
|
venc_write_reg(VENC_HTRIGGER_VTRIGGER, config->htrigger_vtrigger);
|
|
|
|
venc_write_reg(VENC_TVDETGP_INT_START_STOP_X,
|
|
|
|
config->tvdetgp_int_start_stop_x);
|
|
|
|
venc_write_reg(VENC_TVDETGP_INT_START_STOP_Y,
|
|
|
|
config->tvdetgp_int_start_stop_y);
|
|
|
|
venc_write_reg(VENC_GEN_CTRL, config->gen_ctrl);
|
|
|
|
venc_write_reg(VENC_F_CONTROL, config->f_control);
|
|
|
|
venc_write_reg(VENC_SYNC_CTRL, config->sync_ctrl);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void venc_reset(void)
|
|
|
|
{
|
|
|
|
int t = 1000;
|
|
|
|
|
|
|
|
venc_write_reg(VENC_F_CONTROL, 1<<8);
|
|
|
|
while (venc_read_reg(VENC_F_CONTROL) & (1<<8)) {
|
|
|
|
if (--t == 0) {
|
|
|
|
DSSERR("Failed to reset venc\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-02 14:48:22 +00:00
|
|
|
#ifdef CONFIG_OMAP2_DSS_SLEEP_AFTER_VENC_RESET
|
2009-08-05 13:18:06 +00:00
|
|
|
/* the magical sleep that makes things work */
|
2010-06-02 14:48:22 +00:00
|
|
|
/* XXX more info? What bug this circumvents? */
|
2009-08-05 13:18:06 +00:00
|
|
|
msleep(20);
|
2010-06-02 14:48:22 +00:00
|
|
|
#endif
|
2009-08-05 13:18:06 +00:00
|
|
|
}
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
static int venc_runtime_get(void)
|
2009-08-05 13:18:06 +00:00
|
|
|
{
|
2011-05-27 07:52:19 +00:00
|
|
|
int r;
|
|
|
|
|
|
|
|
DSSDBG("venc_runtime_get\n");
|
|
|
|
|
|
|
|
r = pm_runtime_get_sync(&venc.pdev->dev);
|
|
|
|
WARN_ON(r < 0);
|
|
|
|
return r < 0 ? r : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void venc_runtime_put(void)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
|
|
|
|
DSSDBG("venc_runtime_put\n");
|
|
|
|
|
2012-01-23 11:23:08 +00:00
|
|
|
r = pm_runtime_put_sync(&venc.pdev->dev);
|
2012-06-27 13:37:18 +00:00
|
|
|
WARN_ON(r < 0 && r != -ENOSYS);
|
2009-08-05 13:18:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct venc_config *venc_timings_to_config(
|
|
|
|
struct omap_video_timings *timings)
|
|
|
|
{
|
|
|
|
if (memcmp(&omap_dss_pal_timings, timings, sizeof(*timings)) == 0)
|
|
|
|
return &venc_config_pal_trm;
|
|
|
|
|
|
|
|
if (memcmp(&omap_dss_ntsc_timings, timings, sizeof(*timings)) == 0)
|
|
|
|
return &venc_config_ntsc_trm;
|
|
|
|
|
|
|
|
BUG();
|
2012-05-18 08:47:02 +00:00
|
|
|
return NULL;
|
2009-08-05 13:18:06 +00:00
|
|
|
}
|
|
|
|
|
2011-11-21 11:42:58 +00:00
|
|
|
static int venc_power_on(struct omap_dss_device *dssdev)
|
2010-01-12 13:12:07 +00:00
|
|
|
{
|
2012-09-07 12:24:27 +00:00
|
|
|
struct omap_overlay_manager *mgr = dssdev->output->manager;
|
2010-01-12 13:12:07 +00:00
|
|
|
u32 l;
|
2011-11-21 11:42:58 +00:00
|
|
|
int r;
|
2010-01-12 13:12:07 +00:00
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
r = venc_runtime_get();
|
|
|
|
if (r)
|
|
|
|
goto err0;
|
|
|
|
|
2010-01-12 13:12:07 +00:00
|
|
|
venc_reset();
|
2012-07-20 10:45:44 +00:00
|
|
|
venc_write_config(venc_timings_to_config(&venc.timings));
|
2010-01-12 13:12:07 +00:00
|
|
|
|
2012-08-16 06:25:15 +00:00
|
|
|
dss_set_venc_output(venc.type);
|
2010-01-12 13:12:07 +00:00
|
|
|
dss_set_dac_pwrdn_bgz(1);
|
|
|
|
|
|
|
|
l = 0;
|
|
|
|
|
2012-08-16 06:25:15 +00:00
|
|
|
if (venc.type == OMAP_DSS_VENC_TYPE_COMPOSITE)
|
2010-01-12 13:12:07 +00:00
|
|
|
l |= 1 << 1;
|
|
|
|
else /* S-Video */
|
|
|
|
l |= (1 << 0) | (1 << 2);
|
|
|
|
|
2012-08-16 06:26:31 +00:00
|
|
|
if (venc.invert_polarity == false)
|
2010-01-12 13:12:07 +00:00
|
|
|
l |= 1 << 3;
|
|
|
|
|
|
|
|
venc_write_reg(VENC_OUTPUT_CONTROL, l);
|
|
|
|
|
2012-09-07 12:24:27 +00:00
|
|
|
dss_mgr_set_timings(mgr, &venc.timings);
|
2010-01-12 13:12:07 +00:00
|
|
|
|
2012-03-19 14:56:39 +00:00
|
|
|
r = regulator_enable(venc.vdda_dac_reg);
|
|
|
|
if (r)
|
2012-07-06 15:22:37 +00:00
|
|
|
goto err1;
|
2010-01-12 13:12:07 +00:00
|
|
|
|
2012-09-07 12:24:27 +00:00
|
|
|
r = dss_mgr_enable(mgr);
|
2011-11-21 11:42:58 +00:00
|
|
|
if (r)
|
2012-07-06 15:22:37 +00:00
|
|
|
goto err2;
|
2011-11-21 11:42:58 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
err2:
|
|
|
|
regulator_disable(venc.vdda_dac_reg);
|
|
|
|
err1:
|
2011-11-21 11:42:58 +00:00
|
|
|
venc_write_reg(VENC_OUTPUT_CONTROL, 0);
|
|
|
|
dss_set_dac_pwrdn_bgz(0);
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
venc_runtime_put();
|
|
|
|
err0:
|
2011-11-21 11:42:58 +00:00
|
|
|
return r;
|
2010-01-12 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void venc_power_off(struct omap_dss_device *dssdev)
|
|
|
|
{
|
2012-09-07 12:24:27 +00:00
|
|
|
struct omap_overlay_manager *mgr = dssdev->output->manager;
|
|
|
|
|
2010-01-12 13:12:07 +00:00
|
|
|
venc_write_reg(VENC_OUTPUT_CONTROL, 0);
|
|
|
|
dss_set_dac_pwrdn_bgz(0);
|
|
|
|
|
2012-09-07 12:24:27 +00:00
|
|
|
dss_mgr_disable(mgr);
|
2010-01-12 13:12:07 +00:00
|
|
|
|
|
|
|
regulator_disable(venc.vdda_dac_reg);
|
2012-07-06 15:22:37 +00:00
|
|
|
|
|
|
|
venc_runtime_put();
|
2010-01-12 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
2011-09-13 12:58:41 +00:00
|
|
|
unsigned long venc_get_pixel_clock(void)
|
|
|
|
{
|
|
|
|
/* VENC Pixel Clock in Mhz */
|
|
|
|
return 13500000;
|
|
|
|
}
|
2009-08-05 13:18:06 +00:00
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
int omapdss_venc_display_enable(struct omap_dss_device *dssdev)
|
2012-04-23 21:08:54 +00:00
|
|
|
{
|
2012-09-07 12:24:27 +00:00
|
|
|
struct omap_dss_output *out = dssdev->output;
|
2012-07-06 15:22:37 +00:00
|
|
|
int r;
|
2012-04-23 21:08:54 +00:00
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
DSSDBG("venc_display_enable\n");
|
2012-04-23 21:08:54 +00:00
|
|
|
|
|
|
|
mutex_lock(&venc.venc_lock);
|
|
|
|
|
2012-09-07 12:24:27 +00:00
|
|
|
if (out == NULL || out->manager == NULL) {
|
|
|
|
DSSERR("Failed to enable display: no output/manager\n");
|
2012-07-06 15:22:37 +00:00
|
|
|
r = -ENODEV;
|
|
|
|
goto err0;
|
2012-04-23 21:08:54 +00:00
|
|
|
}
|
|
|
|
|
2010-10-28 13:46:38 +00:00
|
|
|
r = omap_dss_start_device(dssdev);
|
|
|
|
if (r) {
|
|
|
|
DSSERR("failed to start device\n");
|
|
|
|
goto err0;
|
|
|
|
}
|
|
|
|
|
2011-11-21 11:42:58 +00:00
|
|
|
r = venc_power_on(dssdev);
|
|
|
|
if (r)
|
2012-07-06 15:22:37 +00:00
|
|
|
goto err1;
|
2010-01-12 13:12:07 +00:00
|
|
|
|
|
|
|
venc.wss_data = 0;
|
|
|
|
|
2010-10-28 13:46:38 +00:00
|
|
|
mutex_unlock(&venc.venc_lock);
|
2012-07-06 15:22:37 +00:00
|
|
|
|
2010-10-28 13:46:38 +00:00
|
|
|
return 0;
|
2010-01-12 13:12:07 +00:00
|
|
|
err1:
|
2010-10-28 13:46:38 +00:00
|
|
|
omap_dss_stop_device(dssdev);
|
|
|
|
err0:
|
2010-01-12 13:12:07 +00:00
|
|
|
mutex_unlock(&venc.venc_lock);
|
2009-08-05 13:18:06 +00:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
void omapdss_venc_display_disable(struct omap_dss_device *dssdev)
|
2009-08-05 13:18:06 +00:00
|
|
|
{
|
2012-07-06 15:22:37 +00:00
|
|
|
DSSDBG("venc_display_disable\n");
|
2009-08-05 13:18:06 +00:00
|
|
|
|
2010-01-12 13:12:07 +00:00
|
|
|
mutex_lock(&venc.venc_lock);
|
|
|
|
|
|
|
|
venc_power_off(dssdev);
|
|
|
|
|
2010-10-28 13:46:38 +00:00
|
|
|
omap_dss_stop_device(dssdev);
|
2009-08-05 13:18:06 +00:00
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
mutex_unlock(&venc.venc_lock);
|
2009-08-05 13:18:06 +00:00
|
|
|
}
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
void omapdss_venc_set_timings(struct omap_dss_device *dssdev,
|
|
|
|
struct omap_video_timings *timings)
|
2010-01-20 10:11:25 +00:00
|
|
|
{
|
|
|
|
DSSDBG("venc_set_timings\n");
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
mutex_lock(&venc.venc_lock);
|
|
|
|
|
2010-01-20 10:11:25 +00:00
|
|
|
/* Reset WSS data when the TV standard changes. */
|
2012-07-20 10:45:44 +00:00
|
|
|
if (memcmp(&venc.timings, timings, sizeof(*timings)))
|
2010-01-20 10:11:25 +00:00
|
|
|
venc.wss_data = 0;
|
|
|
|
|
2012-07-20 10:45:44 +00:00
|
|
|
venc.timings = *timings;
|
2012-07-06 15:22:37 +00:00
|
|
|
|
|
|
|
mutex_unlock(&venc.venc_lock);
|
2010-01-20 10:11:25 +00:00
|
|
|
}
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
int omapdss_venc_check_timings(struct omap_dss_device *dssdev,
|
|
|
|
struct omap_video_timings *timings)
|
2010-01-20 10:11:25 +00:00
|
|
|
{
|
|
|
|
DSSDBG("venc_check_timings\n");
|
|
|
|
|
|
|
|
if (memcmp(&omap_dss_pal_timings, timings, sizeof(*timings)) == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (memcmp(&omap_dss_ntsc_timings, timings, sizeof(*timings)) == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
u32 omapdss_venc_get_wss(struct omap_dss_device *dssdev)
|
2010-01-19 13:53:16 +00:00
|
|
|
{
|
|
|
|
/* Invert due to VENC_L21_WC_CTL:INV=1 */
|
|
|
|
return (venc.wss_data >> 8) ^ 0xfffff;
|
|
|
|
}
|
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
int omapdss_venc_set_wss(struct omap_dss_device *dssdev, u32 wss)
|
2010-01-19 13:53:16 +00:00
|
|
|
{
|
|
|
|
const struct venc_config *config;
|
2011-05-27 07:52:19 +00:00
|
|
|
int r;
|
2010-01-19 13:53:16 +00:00
|
|
|
|
|
|
|
DSSDBG("venc_set_wss\n");
|
|
|
|
|
|
|
|
mutex_lock(&venc.venc_lock);
|
|
|
|
|
2012-07-20 10:45:44 +00:00
|
|
|
config = venc_timings_to_config(&venc.timings);
|
2010-01-19 13:53:16 +00:00
|
|
|
|
|
|
|
/* Invert due to VENC_L21_WC_CTL:INV=1 */
|
|
|
|
venc.wss_data = (wss ^ 0xfffff) << 8;
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
r = venc_runtime_get();
|
|
|
|
if (r)
|
|
|
|
goto err;
|
2010-01-19 13:53:16 +00:00
|
|
|
|
|
|
|
venc_write_reg(VENC_BSTAMP_WSS_DATA, config->bstamp_wss_data |
|
|
|
|
venc.wss_data);
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
venc_runtime_put();
|
2010-01-19 13:53:16 +00:00
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
err:
|
2010-01-19 13:53:16 +00:00
|
|
|
mutex_unlock(&venc.venc_lock);
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
return r;
|
2010-01-19 13:53:16 +00:00
|
|
|
}
|
|
|
|
|
2012-08-16 06:25:15 +00:00
|
|
|
void omapdss_venc_set_type(struct omap_dss_device *dssdev,
|
|
|
|
enum omap_dss_venc_type type)
|
|
|
|
{
|
|
|
|
mutex_lock(&venc.venc_lock);
|
|
|
|
|
|
|
|
venc.type = type;
|
|
|
|
|
|
|
|
mutex_unlock(&venc.venc_lock);
|
|
|
|
}
|
|
|
|
|
2012-08-16 06:26:31 +00:00
|
|
|
void omapdss_venc_invert_vid_out_polarity(struct omap_dss_device *dssdev,
|
|
|
|
bool invert_polarity)
|
|
|
|
{
|
|
|
|
mutex_lock(&venc.venc_lock);
|
|
|
|
|
|
|
|
venc.invert_polarity = invert_polarity;
|
|
|
|
|
|
|
|
mutex_unlock(&venc.venc_lock);
|
|
|
|
}
|
|
|
|
|
2013-05-17 09:48:55 +00:00
|
|
|
static int venc_init_regulator(void)
|
2009-08-05 13:18:06 +00:00
|
|
|
{
|
2013-05-17 09:48:55 +00:00
|
|
|
struct regulator *vdda_dac;
|
2009-08-05 13:18:06 +00:00
|
|
|
|
2013-05-17 09:48:55 +00:00
|
|
|
if (venc.vdda_dac_reg != NULL)
|
|
|
|
return 0;
|
2011-02-22 13:53:46 +00:00
|
|
|
|
2013-05-17 09:48:55 +00:00
|
|
|
vdda_dac = devm_regulator_get(&venc.pdev->dev, "vdda_dac");
|
2011-02-22 13:53:46 +00:00
|
|
|
|
2013-05-17 09:48:55 +00:00
|
|
|
if (IS_ERR(vdda_dac)) {
|
|
|
|
DSSERR("can't get VDDA_DAC regulator\n");
|
|
|
|
return PTR_ERR(vdda_dac);
|
2011-02-22 13:53:46 +00:00
|
|
|
}
|
|
|
|
|
2013-05-17 09:48:55 +00:00
|
|
|
venc.vdda_dac_reg = vdda_dac;
|
|
|
|
|
2009-08-05 13:18:06 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-03-02 16:01:07 +00:00
|
|
|
static void venc_dump_regs(struct seq_file *s)
|
2009-08-05 13:18:06 +00:00
|
|
|
{
|
|
|
|
#define DUMPREG(r) seq_printf(s, "%-35s %08x\n", #r, venc_read_reg(r))
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
if (venc_runtime_get())
|
|
|
|
return;
|
2009-08-05 13:18:06 +00:00
|
|
|
|
|
|
|
DUMPREG(VENC_F_CONTROL);
|
|
|
|
DUMPREG(VENC_VIDOUT_CTRL);
|
|
|
|
DUMPREG(VENC_SYNC_CTRL);
|
|
|
|
DUMPREG(VENC_LLEN);
|
|
|
|
DUMPREG(VENC_FLENS);
|
|
|
|
DUMPREG(VENC_HFLTR_CTRL);
|
|
|
|
DUMPREG(VENC_CC_CARR_WSS_CARR);
|
|
|
|
DUMPREG(VENC_C_PHASE);
|
|
|
|
DUMPREG(VENC_GAIN_U);
|
|
|
|
DUMPREG(VENC_GAIN_V);
|
|
|
|
DUMPREG(VENC_GAIN_Y);
|
|
|
|
DUMPREG(VENC_BLACK_LEVEL);
|
|
|
|
DUMPREG(VENC_BLANK_LEVEL);
|
|
|
|
DUMPREG(VENC_X_COLOR);
|
|
|
|
DUMPREG(VENC_M_CONTROL);
|
|
|
|
DUMPREG(VENC_BSTAMP_WSS_DATA);
|
|
|
|
DUMPREG(VENC_S_CARR);
|
|
|
|
DUMPREG(VENC_LINE21);
|
|
|
|
DUMPREG(VENC_LN_SEL);
|
|
|
|
DUMPREG(VENC_L21__WC_CTL);
|
|
|
|
DUMPREG(VENC_HTRIGGER_VTRIGGER);
|
|
|
|
DUMPREG(VENC_SAVID__EAVID);
|
|
|
|
DUMPREG(VENC_FLEN__FAL);
|
|
|
|
DUMPREG(VENC_LAL__PHASE_RESET);
|
|
|
|
DUMPREG(VENC_HS_INT_START_STOP_X);
|
|
|
|
DUMPREG(VENC_HS_EXT_START_STOP_X);
|
|
|
|
DUMPREG(VENC_VS_INT_START_X);
|
|
|
|
DUMPREG(VENC_VS_INT_STOP_X__VS_INT_START_Y);
|
|
|
|
DUMPREG(VENC_VS_INT_STOP_Y__VS_EXT_START_X);
|
|
|
|
DUMPREG(VENC_VS_EXT_STOP_X__VS_EXT_START_Y);
|
|
|
|
DUMPREG(VENC_VS_EXT_STOP_Y);
|
|
|
|
DUMPREG(VENC_AVID_START_STOP_X);
|
|
|
|
DUMPREG(VENC_AVID_START_STOP_Y);
|
|
|
|
DUMPREG(VENC_FID_INT_START_X__FID_INT_START_Y);
|
|
|
|
DUMPREG(VENC_FID_INT_OFFSET_Y__FID_EXT_START_X);
|
|
|
|
DUMPREG(VENC_FID_EXT_START_Y__FID_EXT_OFFSET_Y);
|
|
|
|
DUMPREG(VENC_TVDETGP_INT_START_STOP_X);
|
|
|
|
DUMPREG(VENC_TVDETGP_INT_START_STOP_Y);
|
|
|
|
DUMPREG(VENC_GEN_CTRL);
|
|
|
|
DUMPREG(VENC_OUTPUT_CONTROL);
|
|
|
|
DUMPREG(VENC_OUTPUT_TEST);
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
venc_runtime_put();
|
2009-08-05 13:18:06 +00:00
|
|
|
|
|
|
|
#undef DUMPREG
|
|
|
|
}
|
2011-01-24 06:22:01 +00:00
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
static int venc_get_clocks(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct clk *clk;
|
|
|
|
|
|
|
|
if (dss_has_feature(FEAT_VENC_REQUIRES_TV_DAC_CLK)) {
|
2013-04-08 08:55:00 +00:00
|
|
|
clk = devm_clk_get(&pdev->dev, "tv_dac_clk");
|
2011-05-27 07:52:19 +00:00
|
|
|
if (IS_ERR(clk)) {
|
|
|
|
DSSERR("can't get tv_dac_clk\n");
|
|
|
|
return PTR_ERR(clk);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
clk = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
venc.tv_dac_clk = clk;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-05-02 09:18:20 +00:00
|
|
|
static struct omap_dss_device *venc_find_dssdev(struct platform_device *pdev)
|
2012-05-02 11:55:12 +00:00
|
|
|
{
|
|
|
|
struct omap_dss_board_info *pdata = pdev->dev.platform_data;
|
2012-10-29 10:40:46 +00:00
|
|
|
const char *def_disp_name = omapdss_get_default_display_name();
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
struct omap_dss_device *def_dssdev;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
def_dssdev = NULL;
|
2012-05-02 11:55:12 +00:00
|
|
|
|
|
|
|
for (i = 0; i < pdata->num_devices; ++i) {
|
|
|
|
struct omap_dss_device *dssdev = pdata->devices[i];
|
|
|
|
|
|
|
|
if (dssdev->type != OMAP_DISPLAY_TYPE_VENC)
|
|
|
|
continue;
|
|
|
|
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
if (def_dssdev == NULL)
|
|
|
|
def_dssdev = dssdev;
|
|
|
|
|
|
|
|
if (def_disp_name != NULL &&
|
|
|
|
strcmp(dssdev->name, def_disp_name) == 0) {
|
|
|
|
def_dssdev = dssdev;
|
|
|
|
break;
|
2012-05-02 11:55:12 +00:00
|
|
|
}
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return def_dssdev;
|
|
|
|
}
|
|
|
|
|
2013-05-02 09:19:54 +00:00
|
|
|
static int venc_probe_pdata(struct platform_device *vencdev)
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
{
|
2012-09-10 10:58:29 +00:00
|
|
|
struct omap_dss_device *plat_dssdev;
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
struct omap_dss_device *dssdev;
|
|
|
|
int r;
|
|
|
|
|
2012-09-10 10:58:29 +00:00
|
|
|
plat_dssdev = venc_find_dssdev(vencdev);
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
|
2012-09-10 10:58:29 +00:00
|
|
|
if (!plat_dssdev)
|
2013-05-02 09:19:54 +00:00
|
|
|
return 0;
|
2012-09-10 10:58:29 +00:00
|
|
|
|
2013-05-17 09:48:55 +00:00
|
|
|
r = venc_init_regulator();
|
|
|
|
if (r)
|
|
|
|
return r;
|
|
|
|
|
2012-09-10 10:58:29 +00:00
|
|
|
dssdev = dss_alloc_and_init_device(&vencdev->dev);
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
if (!dssdev)
|
2013-05-02 09:19:54 +00:00
|
|
|
return -ENOMEM;
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
|
2012-09-10 10:58:29 +00:00
|
|
|
dss_copy_device_pdata(dssdev, plat_dssdev);
|
|
|
|
|
2012-12-07 10:50:08 +00:00
|
|
|
r = omapdss_output_set_device(&venc.output, dssdev);
|
|
|
|
if (r) {
|
|
|
|
DSSERR("failed to connect output to new device: %s\n",
|
|
|
|
dssdev->name);
|
|
|
|
dss_put_device(dssdev);
|
2013-05-02 09:19:54 +00:00
|
|
|
return r;
|
2012-12-07 10:50:08 +00:00
|
|
|
}
|
|
|
|
|
2012-09-10 10:58:29 +00:00
|
|
|
r = dss_add_device(dssdev);
|
OMAPDSS: register only one display device per output
We have boards with multiple panel devices connected to the same
physical output, of which only one panel can be enabled at one time.
Examples of these are Overo, where you can use different daughter boards
that have different LCDs, and 3430SDP which has an LCD and a DVI output
and a physical switch to select the active display.
These are supported by omapdss so that we add all the possible display
devices at probe, but the displays are inactive until somebody enables
one. At this point the panel driver starts using the DSS, thus reserving
the physcal resource and excluding the other panels.
This is problematic:
- Panel drivers can't allocate their resources properly at probe(),
because the resources can be shared with other panels. Thus they can
be only reserved at enable time.
- Managing this in omapdss is confusing. It's not natural to have
child devices, which may not even exist (for example, a daughterboard
that is not connected).
Only some boards have multiple displays per output, and of those, only
very few have possibility of switching the display during runtime.
Because of the above points:
- We don't want to make omapdss and all the panel drivers more complex
just because some boards have complex setups.
- Only few boards support runtime switching, and afaik even then it's
not required. So we don't need to support runtime switching.
Thus we'll change to a model where we will have only one display device
per output and this cannot be (currently) changed at runtime. We'll
still have the possibility to select the display from multiple options
during boot with the default display option.
This patch accomplishes the above by changing how the output drivers
register the display device. Instead of registering all the devices
given from the board file, we'll only register one. If the default
display option is set, the output driver selects that display from its
displays. If the default display is not set, or the default display is
not one of the output's displays, the output driver selects the first
display.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-09-06 11:29:31 +00:00
|
|
|
if (r) {
|
|
|
|
DSSERR("device %s register failed: %d\n", dssdev->name, r);
|
2012-12-07 10:50:08 +00:00
|
|
|
omapdss_output_unset_device(&venc.output);
|
2012-09-10 10:58:29 +00:00
|
|
|
dss_put_device(dssdev);
|
2013-05-02 09:19:54 +00:00
|
|
|
return r;
|
2012-05-02 11:55:12 +00:00
|
|
|
}
|
2013-05-02 09:19:54 +00:00
|
|
|
|
|
|
|
return 0;
|
2012-05-02 11:55:12 +00:00
|
|
|
}
|
|
|
|
|
2013-05-02 09:18:20 +00:00
|
|
|
static void venc_init_output(struct platform_device *pdev)
|
2012-09-26 11:00:49 +00:00
|
|
|
{
|
|
|
|
struct omap_dss_output *out = &venc.output;
|
|
|
|
|
|
|
|
out->pdev = pdev;
|
|
|
|
out->id = OMAP_DSS_OUTPUT_VENC;
|
|
|
|
out->type = OMAP_DISPLAY_TYPE_VENC;
|
2013-02-18 11:06:01 +00:00
|
|
|
out->name = "venc.0";
|
2013-02-13 09:23:54 +00:00
|
|
|
out->dispc_channel = OMAP_DSS_CHANNEL_DIGIT;
|
2012-09-26 11:00:49 +00:00
|
|
|
|
|
|
|
dss_register_output(out);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit venc_uninit_output(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct omap_dss_output *out = &venc.output;
|
|
|
|
|
|
|
|
dss_unregister_output(out);
|
|
|
|
}
|
|
|
|
|
2011-01-24 06:22:01 +00:00
|
|
|
/* VENC HW IP initialisation */
|
2013-05-02 09:18:20 +00:00
|
|
|
static int omap_venchw_probe(struct platform_device *pdev)
|
2011-01-24 06:22:01 +00:00
|
|
|
{
|
|
|
|
u8 rev_id;
|
2011-01-24 06:22:04 +00:00
|
|
|
struct resource *venc_mem;
|
2012-05-02 11:55:12 +00:00
|
|
|
int r;
|
2011-01-24 06:22:04 +00:00
|
|
|
|
2011-01-24 06:22:01 +00:00
|
|
|
venc.pdev = pdev;
|
|
|
|
|
|
|
|
mutex_init(&venc.venc_lock);
|
|
|
|
|
|
|
|
venc.wss_data = 0;
|
|
|
|
|
2011-01-24 06:22:04 +00:00
|
|
|
venc_mem = platform_get_resource(venc.pdev, IORESOURCE_MEM, 0);
|
|
|
|
if (!venc_mem) {
|
|
|
|
DSSERR("can't get IORESOURCE_MEM VENC\n");
|
2012-01-25 11:31:04 +00:00
|
|
|
return -EINVAL;
|
2011-01-24 06:22:04 +00:00
|
|
|
}
|
2012-01-25 11:31:04 +00:00
|
|
|
|
2012-01-24 13:00:45 +00:00
|
|
|
venc.base = devm_ioremap(&pdev->dev, venc_mem->start,
|
|
|
|
resource_size(venc_mem));
|
2011-01-24 06:22:01 +00:00
|
|
|
if (!venc.base) {
|
|
|
|
DSSERR("can't ioremap VENC\n");
|
2012-01-25 11:31:04 +00:00
|
|
|
return -ENOMEM;
|
2011-01-24 06:22:01 +00:00
|
|
|
}
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
r = venc_get_clocks(pdev);
|
|
|
|
if (r)
|
2012-01-25 11:31:04 +00:00
|
|
|
return r;
|
2011-05-27 07:52:19 +00:00
|
|
|
|
|
|
|
pm_runtime_enable(&pdev->dev);
|
|
|
|
|
|
|
|
r = venc_runtime_get();
|
|
|
|
if (r)
|
2012-01-25 11:31:04 +00:00
|
|
|
goto err_runtime_get;
|
2011-01-24 06:22:01 +00:00
|
|
|
|
|
|
|
rev_id = (u8)(venc_read_reg(VENC_REV_ID) & 0xff);
|
2011-01-24 06:22:03 +00:00
|
|
|
dev_dbg(&pdev->dev, "OMAP VENC rev %d\n", rev_id);
|
2011-01-24 06:22:01 +00:00
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
venc_runtime_put();
|
2011-01-24 06:22:01 +00:00
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
r = venc_panel_init();
|
2012-01-25 11:31:04 +00:00
|
|
|
if (r)
|
2012-07-06 15:22:37 +00:00
|
|
|
goto err_panel_init;
|
2012-01-25 11:31:04 +00:00
|
|
|
|
2012-03-02 16:01:07 +00:00
|
|
|
dss_debugfs_create_file("venc", venc_dump_regs);
|
|
|
|
|
2012-09-26 11:00:49 +00:00
|
|
|
venc_init_output(pdev);
|
|
|
|
|
2013-03-14 13:47:29 +00:00
|
|
|
if (pdev->dev.platform_data) {
|
|
|
|
r = venc_probe_pdata(pdev);
|
|
|
|
if (r)
|
|
|
|
goto err_probe;
|
2013-05-02 09:19:54 +00:00
|
|
|
}
|
OMAPDSS: interface drivers register their panel devices
Currently the higher level omapdss platform driver gets the list of
displays in its platform data, and uses that list to create the
omap_dss_device for each display.
With DT, the logical way to do the above is to list the displays under
each individual output, i.e. we'd have "dpi" node, under which we would
have the display that uses DPI. In other words, each output driver
handles the displays that use that particular output.
To make the current code ready for DT, this patch modifies the output
drivers so that each of them creates the display devices which use that
output. However, instead of changing the platform data to suit this
method, each output driver is passed the full list of displays, and the
drivers pick the displays that are meant for them. This allows us to
keep the old platform data, and thus we avoid the need to change the
board files.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-03-01 13:45:53 +00:00
|
|
|
|
2012-01-25 11:31:04 +00:00
|
|
|
return 0;
|
2011-05-27 07:52:19 +00:00
|
|
|
|
2013-03-14 13:47:29 +00:00
|
|
|
err_probe:
|
|
|
|
venc_panel_exit();
|
|
|
|
venc_uninit_output(pdev);
|
2012-07-06 15:22:37 +00:00
|
|
|
err_panel_init:
|
2012-01-25 11:31:04 +00:00
|
|
|
err_runtime_get:
|
2011-05-27 07:52:19 +00:00
|
|
|
pm_runtime_disable(&pdev->dev);
|
|
|
|
return r;
|
2011-01-24 06:22:01 +00:00
|
|
|
}
|
|
|
|
|
2012-02-17 15:41:13 +00:00
|
|
|
static int __exit omap_venchw_remove(struct platform_device *pdev)
|
2011-01-24 06:22:01 +00:00
|
|
|
{
|
2012-09-10 10:58:29 +00:00
|
|
|
dss_unregister_child_devices(&pdev->dev);
|
OMAPDSS: interface drivers register their panel devices
Currently the higher level omapdss platform driver gets the list of
displays in its platform data, and uses that list to create the
omap_dss_device for each display.
With DT, the logical way to do the above is to list the displays under
each individual output, i.e. we'd have "dpi" node, under which we would
have the display that uses DPI. In other words, each output driver
handles the displays that use that particular output.
To make the current code ready for DT, this patch modifies the output
drivers so that each of them creates the display devices which use that
output. However, instead of changing the platform data to suit this
method, each output driver is passed the full list of displays, and the
drivers pick the displays that are meant for them. This allows us to
keep the old platform data, and thus we avoid the need to change the
board files.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-03-01 13:45:53 +00:00
|
|
|
|
2012-07-06 15:22:37 +00:00
|
|
|
venc_panel_exit();
|
2011-01-24 06:22:01 +00:00
|
|
|
|
2012-09-26 11:00:49 +00:00
|
|
|
venc_uninit_output(pdev);
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
pm_runtime_disable(&pdev->dev);
|
|
|
|
|
2011-01-24 06:22:01 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-05-27 07:52:19 +00:00
|
|
|
static int venc_runtime_suspend(struct device *dev)
|
|
|
|
{
|
|
|
|
if (venc.tv_dac_clk)
|
2012-06-27 08:51:26 +00:00
|
|
|
clk_disable_unprepare(venc.tv_dac_clk);
|
2011-05-27 07:52:19 +00:00
|
|
|
|
|
|
|
dispc_runtime_put();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int venc_runtime_resume(struct device *dev)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
|
|
|
|
r = dispc_runtime_get();
|
|
|
|
if (r < 0)
|
2012-02-17 15:58:04 +00:00
|
|
|
return r;
|
2011-05-27 07:52:19 +00:00
|
|
|
|
|
|
|
if (venc.tv_dac_clk)
|
2012-06-27 08:51:26 +00:00
|
|
|
clk_prepare_enable(venc.tv_dac_clk);
|
2011-05-27 07:52:19 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct dev_pm_ops venc_pm_ops = {
|
|
|
|
.runtime_suspend = venc_runtime_suspend,
|
|
|
|
.runtime_resume = venc_runtime_resume,
|
|
|
|
};
|
|
|
|
|
2011-01-24 06:22:01 +00:00
|
|
|
static struct platform_driver omap_venchw_driver = {
|
2013-05-02 09:18:20 +00:00
|
|
|
.probe = omap_venchw_probe,
|
2012-02-17 15:41:13 +00:00
|
|
|
.remove = __exit_p(omap_venchw_remove),
|
2011-01-24 06:22:01 +00:00
|
|
|
.driver = {
|
|
|
|
.name = "omapdss_venc",
|
|
|
|
.owner = THIS_MODULE,
|
2011-05-27 07:52:19 +00:00
|
|
|
.pm = &venc_pm_ops,
|
2011-01-24 06:22:01 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2012-02-17 15:41:13 +00:00
|
|
|
int __init venc_init_platform_driver(void)
|
2011-01-24 06:22:01 +00:00
|
|
|
{
|
2013-05-02 09:18:20 +00:00
|
|
|
return platform_driver_register(&omap_venchw_driver);
|
2011-01-24 06:22:01 +00:00
|
|
|
}
|
|
|
|
|
2012-02-17 15:41:13 +00:00
|
|
|
void __exit venc_uninit_platform_driver(void)
|
2011-01-24 06:22:01 +00:00
|
|
|
{
|
2012-02-23 13:32:37 +00:00
|
|
|
platform_driver_unregister(&omap_venchw_driver);
|
2011-01-24 06:22:01 +00:00
|
|
|
}
|