2020-04-24 23:35:39 +02:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* Chrontel CH7033 Video Encoder Driver
*
* Copyright ( C ) 2019 , 2020 Lubomir Rintel
*/
# include <linux/gpio/consumer.h>
# include <linux/module.h>
# include <linux/regmap.h>
# include <drm/drm_atomic_helper.h>
# include <drm/drm_bridge.h>
# include <drm/drm_edid.h>
# include <drm/drm_of.h>
# include <drm/drm_print.h>
# include <drm/drm_probe_helper.h>
/* Page 0, Register 0x07 */
enum {
DRI_PD = BIT ( 3 ) ,
IO_PD = BIT ( 5 ) ,
} ;
/* Page 0, Register 0x08 */
enum {
DRI_PDDRI = GENMASK ( 7 , 4 ) ,
PDDAC = GENMASK ( 3 , 1 ) ,
PANEN = BIT ( 0 ) ,
} ;
/* Page 0, Register 0x09 */
enum {
DPD = BIT ( 7 ) ,
GCKOFF = BIT ( 6 ) ,
TV_BP = BIT ( 5 ) ,
SCLPD = BIT ( 4 ) ,
SDPD = BIT ( 3 ) ,
VGA_PD = BIT ( 2 ) ,
HDBKPD = BIT ( 1 ) ,
HDMI_PD = BIT ( 0 ) ,
} ;
/* Page 0, Register 0x0a */
enum {
MEMINIT = BIT ( 7 ) ,
MEMIDLE = BIT ( 6 ) ,
MEMPD = BIT ( 5 ) ,
STOP = BIT ( 4 ) ,
LVDS_PD = BIT ( 3 ) ,
HD_DVIB = BIT ( 2 ) ,
HDCP_PD = BIT ( 1 ) ,
MCU_PD = BIT ( 0 ) ,
} ;
/* Page 0, Register 0x18 */
enum {
IDF = GENMASK ( 7 , 4 ) ,
INTEN = BIT ( 3 ) ,
SWAP = GENMASK ( 2 , 0 ) ,
} ;
enum {
BYTE_SWAP_RGB = 0 ,
BYTE_SWAP_RBG = 1 ,
BYTE_SWAP_GRB = 2 ,
BYTE_SWAP_GBR = 3 ,
BYTE_SWAP_BRG = 4 ,
BYTE_SWAP_BGR = 5 ,
} ;
/* Page 0, Register 0x19 */
enum {
HPO_I = BIT ( 5 ) ,
VPO_I = BIT ( 4 ) ,
DEPO_I = BIT ( 3 ) ,
CRYS_EN = BIT ( 2 ) ,
GCLKFREQ = GENMASK ( 2 , 0 ) ,
} ;
/* Page 0, Register 0x2e */
enum {
HFLIP = BIT ( 7 ) ,
VFLIP = BIT ( 6 ) ,
DEPO_O = BIT ( 5 ) ,
HPO_O = BIT ( 4 ) ,
VPO_O = BIT ( 3 ) ,
TE = GENMASK ( 2 , 0 ) ,
} ;
/* Page 0, Register 0x2b */
enum {
SWAPS = GENMASK ( 7 , 4 ) ,
VFMT = GENMASK ( 3 , 0 ) ,
} ;
/* Page 0, Register 0x54 */
enum {
COMP_BP = BIT ( 7 ) ,
DAC_EN_T = BIT ( 6 ) ,
HWO_HDMI_HI = GENMASK ( 5 , 3 ) ,
HOO_HDMI_HI = GENMASK ( 2 , 0 ) ,
} ;
/* Page 0, Register 0x57 */
enum {
FLDSEN = BIT ( 7 ) ,
VWO_HDMI_HI = GENMASK ( 5 , 3 ) ,
VOO_HDMI_HI = GENMASK ( 2 , 0 ) ,
} ;
/* Page 0, Register 0x7e */
enum {
HDMI_LVDS_SEL = BIT ( 7 ) ,
DE_GEN = BIT ( 6 ) ,
PWM_INDEX_HI = BIT ( 5 ) ,
USE_DE = BIT ( 4 ) ,
R_INT = GENMASK ( 3 , 0 ) ,
} ;
/* Page 1, Register 0x07 */
enum {
BPCKSEL = BIT ( 7 ) ,
DRI_CMFB_EN = BIT ( 6 ) ,
CEC_PUEN = BIT ( 5 ) ,
CEC_T = BIT ( 3 ) ,
CKINV = BIT ( 2 ) ,
CK_TVINV = BIT ( 1 ) ,
DRI_CKS2 = BIT ( 0 ) ,
} ;
/* Page 1, Register 0x08 */
enum {
DACG = BIT ( 6 ) ,
DACKTST = BIT ( 5 ) ,
DEDGEB = BIT ( 4 ) ,
SYO = BIT ( 3 ) ,
DRI_IT_LVDS = GENMASK ( 2 , 1 ) ,
DISPON = BIT ( 0 ) ,
} ;
/* Page 1, Register 0x0c */
enum {
DRI_PLL_CP = GENMASK ( 7 , 6 ) ,
DRI_PLL_DIVSEL = BIT ( 5 ) ,
DRI_PLL_N1_1 = BIT ( 4 ) ,
DRI_PLL_N1_0 = BIT ( 3 ) ,
DRI_PLL_N3_1 = BIT ( 2 ) ,
DRI_PLL_N3_0 = BIT ( 1 ) ,
DRI_PLL_CKTSTEN = BIT ( 0 ) ,
} ;
/* Page 1, Register 0x6b */
enum {
VCO3CS = GENMASK ( 7 , 6 ) ,
ICPGBK2_0 = GENMASK ( 5 , 3 ) ,
DRI_VCO357SC = BIT ( 2 ) ,
PDPLL2 = BIT ( 1 ) ,
DRI_PD_SER = BIT ( 0 ) ,
} ;
/* Page 1, Register 0x6c */
enum {
PLL2N11 = GENMASK ( 7 , 4 ) ,
PLL2N5_4 = BIT ( 3 ) ,
PLL2N5_TOP = BIT ( 2 ) ,
DRI_PLL_PD = BIT ( 1 ) ,
PD_I2CM = BIT ( 0 ) ,
} ;
/* Page 3, Register 0x28 */
enum {
DIFF_EN = GENMASK ( 7 , 6 ) ,
CORREC_EN = GENMASK ( 5 , 4 ) ,
VGACLK_BP = BIT ( 3 ) ,
HM_LV_SEL = BIT ( 2 ) ,
HD_VGA_SEL = BIT ( 1 ) ,
} ;
/* Page 3, Register 0x2a */
enum {
LVDSCLK_BP = BIT ( 7 ) ,
HDTVCLK_BP = BIT ( 6 ) ,
HDMICLK_BP = BIT ( 5 ) ,
HDTV_BP = BIT ( 4 ) ,
HDMI_BP = BIT ( 3 ) ,
THRWL = GENMASK ( 2 , 0 ) ,
} ;
/* Page 4, Register 0x52 */
enum {
PGM_ARSTB = BIT ( 7 ) ,
MCU_ARSTB = BIT ( 6 ) ,
MCU_RETB = BIT ( 2 ) ,
RESETIB = BIT ( 1 ) ,
RESETDB = BIT ( 0 ) ,
} ;
struct ch7033_priv {
struct regmap * regmap ;
struct drm_bridge * next_bridge ;
struct drm_bridge bridge ;
struct drm_connector connector ;
} ;
# define conn_to_ch7033_priv(x) \
container_of ( x , struct ch7033_priv , connector )
# define bridge_to_ch7033_priv(x) \
container_of ( x , struct ch7033_priv , bridge )
static enum drm_connector_status ch7033_connector_detect (
struct drm_connector * connector , bool force )
{
struct ch7033_priv * priv = conn_to_ch7033_priv ( connector ) ;
return drm_bridge_detect ( priv - > next_bridge ) ;
}
static const struct drm_connector_funcs ch7033_connector_funcs = {
. reset = drm_atomic_helper_connector_reset ,
. fill_modes = drm_helper_probe_single_connector_modes ,
. detect = ch7033_connector_detect ,
. destroy = drm_connector_cleanup ,
. atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state ,
. atomic_destroy_state = drm_atomic_helper_connector_destroy_state ,
} ;
static int ch7033_connector_get_modes ( struct drm_connector * connector )
{
struct ch7033_priv * priv = conn_to_ch7033_priv ( connector ) ;
struct edid * edid ;
int ret ;
edid = drm_bridge_get_edid ( priv - > next_bridge , connector ) ;
drm_connector_update_edid_property ( connector , edid ) ;
if ( edid ) {
ret = drm_add_edid_modes ( connector , edid ) ;
kfree ( edid ) ;
} else {
ret = drm_add_modes_noedid ( connector , 1920 , 1080 ) ;
drm_set_preferred_mode ( connector , 1024 , 768 ) ;
}
return ret ;
}
static struct drm_encoder * ch7033_connector_best_encoder (
struct drm_connector * connector )
{
struct ch7033_priv * priv = conn_to_ch7033_priv ( connector ) ;
return priv - > bridge . encoder ;
}
static const struct drm_connector_helper_funcs ch7033_connector_helper_funcs = {
. get_modes = ch7033_connector_get_modes ,
. best_encoder = ch7033_connector_best_encoder ,
} ;
static void ch7033_hpd_event ( void * arg , enum drm_connector_status status )
{
struct ch7033_priv * priv = arg ;
if ( priv - > bridge . dev )
drm_helper_hpd_irq_event ( priv - > connector . dev ) ;
}
static int ch7033_bridge_attach ( struct drm_bridge * bridge ,
enum drm_bridge_attach_flags flags )
{
struct ch7033_priv * priv = bridge_to_ch7033_priv ( bridge ) ;
struct drm_connector * connector = & priv - > connector ;
int ret ;
ret = drm_bridge_attach ( bridge - > encoder , priv - > next_bridge , bridge ,
DRM_BRIDGE_ATTACH_NO_CONNECTOR ) ;
if ( ret )
return ret ;
if ( flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR )
return 0 ;
if ( priv - > next_bridge - > ops & DRM_BRIDGE_OP_DETECT ) {
connector - > polled = DRM_CONNECTOR_POLL_HPD ;
} else {
connector - > polled = DRM_CONNECTOR_POLL_CONNECT |
DRM_CONNECTOR_POLL_DISCONNECT ;
}
if ( priv - > next_bridge - > ops & DRM_BRIDGE_OP_HPD ) {
drm_bridge_hpd_enable ( priv - > next_bridge , ch7033_hpd_event ,
priv ) ;
}
drm_connector_helper_add ( connector ,
& ch7033_connector_helper_funcs ) ;
ret = drm_connector_init_with_ddc ( bridge - > dev , & priv - > connector ,
& ch7033_connector_funcs ,
priv - > next_bridge - > type ,
priv - > next_bridge - > ddc ) ;
if ( ret ) {
DRM_ERROR ( " Failed to initialize connector \n " ) ;
return ret ;
}
return drm_connector_attach_encoder ( & priv - > connector , bridge - > encoder ) ;
}
static void ch7033_bridge_detach ( struct drm_bridge * bridge )
{
struct ch7033_priv * priv = bridge_to_ch7033_priv ( bridge ) ;
if ( priv - > next_bridge - > ops & DRM_BRIDGE_OP_HPD )
drm_bridge_hpd_disable ( priv - > next_bridge ) ;
drm_connector_cleanup ( & priv - > connector ) ;
}
static enum drm_mode_status ch7033_bridge_mode_valid ( struct drm_bridge * bridge ,
2020-05-26 04:14:48 +03:00
const struct drm_display_info * info ,
2020-04-24 23:35:39 +02:00
const struct drm_display_mode * mode )
{
if ( mode - > clock > 165000 )
return MODE_CLOCK_HIGH ;
if ( mode - > hdisplay > = 1920 )
return MODE_BAD_HVALUE ;
if ( mode - > vdisplay > = 1080 )
return MODE_BAD_VVALUE ;
return MODE_OK ;
}
static void ch7033_bridge_disable ( struct drm_bridge * bridge )
{
struct ch7033_priv * priv = bridge_to_ch7033_priv ( bridge ) ;
regmap_write ( priv - > regmap , 0x03 , 0x04 ) ;
regmap_update_bits ( priv - > regmap , 0x52 , RESETDB , 0x00 ) ;
}
static void ch7033_bridge_enable ( struct drm_bridge * bridge )
{
struct ch7033_priv * priv = bridge_to_ch7033_priv ( bridge ) ;
regmap_write ( priv - > regmap , 0x03 , 0x04 ) ;
regmap_update_bits ( priv - > regmap , 0x52 , RESETDB , RESETDB ) ;
}
static void ch7033_bridge_mode_set ( struct drm_bridge * bridge ,
const struct drm_display_mode * mode ,
const struct drm_display_mode * adjusted_mode )
{
struct ch7033_priv * priv = bridge_to_ch7033_priv ( bridge ) ;
int hbporch = mode - > hsync_start - mode - > hdisplay ;
int hsynclen = mode - > hsync_end - mode - > hsync_start ;
int vbporch = mode - > vsync_start - mode - > vdisplay ;
int vsynclen = mode - > vsync_end - mode - > vsync_start ;
/*
* Page 4
*/
regmap_write ( priv - > regmap , 0x03 , 0x04 ) ;
/* Turn everything off to set all the registers to their defaults. */
regmap_write ( priv - > regmap , 0x52 , 0x00 ) ;
/* Bring I/O block up. */
regmap_write ( priv - > regmap , 0x52 , RESETIB ) ;
/*
* Page 0
*/
regmap_write ( priv - > regmap , 0x03 , 0x00 ) ;
/* Bring up parts we need from the power down. */
regmap_update_bits ( priv - > regmap , 0x07 , DRI_PD | IO_PD , 0 ) ;
regmap_update_bits ( priv - > regmap , 0x08 , DRI_PDDRI | PDDAC | PANEN , 0 ) ;
regmap_update_bits ( priv - > regmap , 0x09 , DPD | GCKOFF |
HDMI_PD | VGA_PD , 0 ) ;
regmap_update_bits ( priv - > regmap , 0x0a , HD_DVIB , 0 ) ;
/* Horizontal input timing. */
regmap_write ( priv - > regmap , 0x0b , ( mode - > htotal > > 8 ) < < 3 |
( mode - > hdisplay > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x0c , mode - > hdisplay ) ;
regmap_write ( priv - > regmap , 0x0d , mode - > htotal ) ;
regmap_write ( priv - > regmap , 0x0e , ( hsynclen > > 8 ) < < 3 |
( hbporch > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x0f , hbporch ) ;
regmap_write ( priv - > regmap , 0x10 , hsynclen ) ;
/* Vertical input timing. */
regmap_write ( priv - > regmap , 0x11 , ( mode - > vtotal > > 8 ) < < 3 |
( mode - > vdisplay > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x12 , mode - > vdisplay ) ;
regmap_write ( priv - > regmap , 0x13 , mode - > vtotal ) ;
regmap_write ( priv - > regmap , 0x14 , ( ( vsynclen > > 8 ) < < 3 ) |
( vbporch > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x15 , vbporch ) ;
regmap_write ( priv - > regmap , 0x16 , vsynclen ) ;
/* Input color swap. */
regmap_update_bits ( priv - > regmap , 0x18 , SWAP , BYTE_SWAP_BGR ) ;
/* Input clock and sync polarity. */
regmap_update_bits ( priv - > regmap , 0x19 , 0x1 , mode - > clock > > 16 ) ;
regmap_update_bits ( priv - > regmap , 0x19 , HPO_I | VPO_I | GCLKFREQ ,
( mode - > flags & DRM_MODE_FLAG_PHSYNC ) ? HPO_I : 0 |
( mode - > flags & DRM_MODE_FLAG_PVSYNC ) ? VPO_I : 0 |
mode - > clock > > 16 ) ;
regmap_write ( priv - > regmap , 0x1a , mode - > clock > > 8 ) ;
regmap_write ( priv - > regmap , 0x1b , mode - > clock ) ;
/* Horizontal output timing. */
regmap_write ( priv - > regmap , 0x1f , ( mode - > htotal > > 8 ) < < 3 |
( mode - > hdisplay > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x20 , mode - > hdisplay ) ;
regmap_write ( priv - > regmap , 0x21 , mode - > htotal ) ;
/* Vertical output timing. */
regmap_write ( priv - > regmap , 0x25 , ( mode - > vtotal > > 8 ) < < 3 |
( mode - > vdisplay > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x26 , mode - > vdisplay ) ;
regmap_write ( priv - > regmap , 0x27 , mode - > vtotal ) ;
/* VGA channel bypass */
regmap_update_bits ( priv - > regmap , 0x2b , VFMT , 9 ) ;
/* Output sync polarity. */
regmap_update_bits ( priv - > regmap , 0x2e , HPO_O | VPO_O ,
( mode - > flags & DRM_MODE_FLAG_PHSYNC ) ? HPO_O : 0 |
( mode - > flags & DRM_MODE_FLAG_PVSYNC ) ? VPO_O : 0 ) ;
/* HDMI horizontal output timing. */
regmap_update_bits ( priv - > regmap , 0x54 , HWO_HDMI_HI | HOO_HDMI_HI ,
( hsynclen > > 8 ) < < 3 |
( hbporch > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x55 , hbporch ) ;
regmap_write ( priv - > regmap , 0x56 , hsynclen ) ;
/* HDMI vertical output timing. */
regmap_update_bits ( priv - > regmap , 0x57 , VWO_HDMI_HI | VOO_HDMI_HI ,
( vsynclen > > 8 ) < < 3 |
( vbporch > > 8 ) ) ;
regmap_write ( priv - > regmap , 0x58 , vbporch ) ;
regmap_write ( priv - > regmap , 0x59 , vsynclen ) ;
/* Pick HDMI, not LVDS. */
regmap_update_bits ( priv - > regmap , 0x7e , HDMI_LVDS_SEL , HDMI_LVDS_SEL ) ;
/*
* Page 1
*/
regmap_write ( priv - > regmap , 0x03 , 0x01 ) ;
/* No idea what these do, but VGA is wobbly and blinky without them. */
regmap_update_bits ( priv - > regmap , 0x07 , CKINV , CKINV ) ;
regmap_update_bits ( priv - > regmap , 0x08 , DISPON , DISPON ) ;
/* DRI PLL */
regmap_update_bits ( priv - > regmap , 0x0c , DRI_PLL_DIVSEL , DRI_PLL_DIVSEL ) ;
if ( mode - > clock < = 40000 ) {
regmap_update_bits ( priv - > regmap , 0x0c , DRI_PLL_N1_1 |
DRI_PLL_N1_0 |
DRI_PLL_N3_1 |
DRI_PLL_N3_0 ,
0 ) ;
} else if ( mode - > clock < 80000 ) {
regmap_update_bits ( priv - > regmap , 0x0c , DRI_PLL_N1_1 |
DRI_PLL_N1_0 |
DRI_PLL_N3_1 |
DRI_PLL_N3_0 ,
DRI_PLL_N3_0 |
DRI_PLL_N1_0 ) ;
} else {
regmap_update_bits ( priv - > regmap , 0x0c , DRI_PLL_N1_1 |
DRI_PLL_N1_0 |
DRI_PLL_N3_1 |
DRI_PLL_N3_0 ,
DRI_PLL_N3_1 |
DRI_PLL_N1_1 ) ;
}
/* This seems to be color calibration for VGA. */
regmap_write ( priv - > regmap , 0x64 , 0x29 ) ; /* LSB Blue */
regmap_write ( priv - > regmap , 0x65 , 0x29 ) ; /* LSB Green */
regmap_write ( priv - > regmap , 0x66 , 0x29 ) ; /* LSB Red */
regmap_write ( priv - > regmap , 0x67 , 0x00 ) ; /* MSB Blue */
regmap_write ( priv - > regmap , 0x68 , 0x00 ) ; /* MSB Green */
regmap_write ( priv - > regmap , 0x69 , 0x00 ) ; /* MSB Red */
regmap_update_bits ( priv - > regmap , 0x6b , DRI_PD_SER , 0x00 ) ;
regmap_update_bits ( priv - > regmap , 0x6c , DRI_PLL_PD , 0x00 ) ;
/*
* Page 3
*/
regmap_write ( priv - > regmap , 0x03 , 0x03 ) ;
/* More bypasses and apparently another HDMI/LVDS selector. */
regmap_update_bits ( priv - > regmap , 0x28 , VGACLK_BP | HM_LV_SEL ,
VGACLK_BP | HM_LV_SEL ) ;
regmap_update_bits ( priv - > regmap , 0x2a , HDMICLK_BP | HDMI_BP ,
HDMICLK_BP | HDMI_BP ) ;
/*
* Page 4
*/
regmap_write ( priv - > regmap , 0x03 , 0x04 ) ;
/* Output clock. */
regmap_write ( priv - > regmap , 0x10 , mode - > clock > > 16 ) ;
regmap_write ( priv - > regmap , 0x11 , mode - > clock > > 8 ) ;
regmap_write ( priv - > regmap , 0x12 , mode - > clock ) ;
}
static const struct drm_bridge_funcs ch7033_bridge_funcs = {
. attach = ch7033_bridge_attach ,
. detach = ch7033_bridge_detach ,
. mode_valid = ch7033_bridge_mode_valid ,
. disable = ch7033_bridge_disable ,
. enable = ch7033_bridge_enable ,
. mode_set = ch7033_bridge_mode_set ,
} ;
static const struct regmap_config ch7033_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = 0x7f ,
} ;
static int ch7033_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
{
struct device * dev = & client - > dev ;
struct ch7033_priv * priv ;
unsigned int val ;
int ret ;
priv = devm_kzalloc ( dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
dev_set_drvdata ( dev , priv ) ;
ret = drm_of_find_panel_or_bridge ( dev - > of_node , 1 , - 1 , NULL ,
& priv - > next_bridge ) ;
if ( ret )
return ret ;
priv - > regmap = devm_regmap_init_i2c ( client , & ch7033_regmap_config ) ;
if ( IS_ERR ( priv - > regmap ) ) {
dev_err ( & client - > dev , " regmap init failed \n " ) ;
return PTR_ERR ( priv - > regmap ) ;
}
ret = regmap_read ( priv - > regmap , 0x00 , & val ) ;
if ( ret < 0 ) {
dev_err ( & client - > dev , " error reading the model id: %d \n " , ret ) ;
return ret ;
}
if ( ( val & 0xf7 ) ! = 0x56 ) {
dev_err ( & client - > dev , " the device is not a ch7033 \n " ) ;
return - ENODEV ;
}
regmap_write ( priv - > regmap , 0x03 , 0x04 ) ;
ret = regmap_read ( priv - > regmap , 0x51 , & val ) ;
if ( ret < 0 ) {
dev_err ( & client - > dev , " error reading the model id: %d \n " , ret ) ;
return ret ;
}
if ( ( val & 0x0f ) ! = 3 ) {
dev_err ( & client - > dev , " unknown revision %u \n " , val ) ;
return - ENODEV ;
}
INIT_LIST_HEAD ( & priv - > bridge . list ) ;
priv - > bridge . funcs = & ch7033_bridge_funcs ;
priv - > bridge . of_node = dev - > of_node ;
drm_bridge_add ( & priv - > bridge ) ;
dev_info ( dev , " Chrontel CH7033 Video Encoder \n " ) ;
return 0 ;
}
static int ch7033_remove ( struct i2c_client * client )
{
struct device * dev = & client - > dev ;
struct ch7033_priv * priv = dev_get_drvdata ( dev ) ;
drm_bridge_remove ( & priv - > bridge ) ;
return 0 ;
}
static const struct of_device_id ch7033_dt_ids [ ] = {
{ . compatible = " chrontel,ch7033 " , } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , ch7033_dt_ids ) ;
static const struct i2c_device_id ch7033_ids [ ] = {
{ " ch7033 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , ch7033_ids ) ;
static struct i2c_driver ch7033_driver = {
. probe = ch7033_probe ,
. remove = ch7033_remove ,
. driver = {
. name = " ch7033 " ,
. of_match_table = of_match_ptr ( ch7033_dt_ids ) ,
} ,
. id_table = ch7033_ids ,
} ;
module_i2c_driver ( ch7033_driver ) ;
MODULE_AUTHOR ( " Lubomir Rintel <lkundrak@v3.sk> " ) ;
MODULE_DESCRIPTION ( " Chrontel CH7033 Video Encoder Driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;