Commit 2f7f730a authored by Jean-Francois Moine's avatar Jean-Francois Moine Committed by Russell King

drm/i2c: tda998x: simplify the i2c read/write functions

This patch simplifies the i2c read/write functions and permits them to
be easily called in more contexts.
Tested-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
Acked-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
Signed-off-by: default avatarJean-Francois Moine <moinejf@free.fr>
Signed-off-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
parent f0b33b28
...@@ -31,6 +31,7 @@ ...@@ -31,6 +31,7 @@
struct tda998x_priv { struct tda998x_priv {
struct i2c_client *cec; struct i2c_client *cec;
struct i2c_client *hdmi;
uint16_t rev; uint16_t rev;
uint8_t current_page; uint8_t current_page;
int dpms; int dpms;
...@@ -329,9 +330,9 @@ struct tda998x_priv { ...@@ -329,9 +330,9 @@ struct tda998x_priv {
#define TDA19988 0x0301 #define TDA19988 0x0301
static void static void
cec_write(struct drm_encoder *encoder, uint16_t addr, uint8_t val) cec_write(struct tda998x_priv *priv, uint16_t addr, uint8_t val)
{ {
struct i2c_client *client = to_tda998x_priv(encoder)->cec; struct i2c_client *client = priv->cec;
uint8_t buf[] = {addr, val}; uint8_t buf[] = {addr, val};
int ret; int ret;
...@@ -341,9 +342,9 @@ cec_write(struct drm_encoder *encoder, uint16_t addr, uint8_t val) ...@@ -341,9 +342,9 @@ cec_write(struct drm_encoder *encoder, uint16_t addr, uint8_t val)
} }
static uint8_t static uint8_t
cec_read(struct drm_encoder *encoder, uint8_t addr) cec_read(struct tda998x_priv *priv, uint8_t addr)
{ {
struct i2c_client *client = to_tda998x_priv(encoder)->cec; struct i2c_client *client = priv->cec;
uint8_t val; uint8_t val;
int ret; int ret;
...@@ -363,12 +364,10 @@ cec_read(struct drm_encoder *encoder, uint8_t addr) ...@@ -363,12 +364,10 @@ cec_read(struct drm_encoder *encoder, uint8_t addr)
} }
static void static void
set_page(struct drm_encoder *encoder, uint16_t reg) set_page(struct tda998x_priv *priv, uint16_t reg)
{ {
struct tda998x_priv *priv = to_tda998x_priv(encoder);
if (REG2PAGE(reg) != priv->current_page) { if (REG2PAGE(reg) != priv->current_page) {
struct i2c_client *client = drm_i2c_encoder_get_client(encoder); struct i2c_client *client = priv->hdmi;
uint8_t buf[] = { uint8_t buf[] = {
REG_CURPAGE, REG2PAGE(reg) REG_CURPAGE, REG2PAGE(reg)
}; };
...@@ -381,13 +380,13 @@ set_page(struct drm_encoder *encoder, uint16_t reg) ...@@ -381,13 +380,13 @@ set_page(struct drm_encoder *encoder, uint16_t reg)
} }
static int static int
reg_read_range(struct drm_encoder *encoder, uint16_t reg, char *buf, int cnt) reg_read_range(struct tda998x_priv *priv, uint16_t reg, char *buf, int cnt)
{ {
struct i2c_client *client = drm_i2c_encoder_get_client(encoder); struct i2c_client *client = priv->hdmi;
uint8_t addr = REG2ADDR(reg); uint8_t addr = REG2ADDR(reg);
int ret; int ret;
set_page(encoder, reg); set_page(priv, reg);
ret = i2c_master_send(client, &addr, sizeof(addr)); ret = i2c_master_send(client, &addr, sizeof(addr));
if (ret < 0) if (ret < 0)
...@@ -405,16 +404,16 @@ reg_read_range(struct drm_encoder *encoder, uint16_t reg, char *buf, int cnt) ...@@ -405,16 +404,16 @@ reg_read_range(struct drm_encoder *encoder, uint16_t reg, char *buf, int cnt)
} }
static void static void
reg_write_range(struct drm_encoder *encoder, uint16_t reg, uint8_t *p, int cnt) reg_write_range(struct tda998x_priv *priv, uint16_t reg, uint8_t *p, int cnt)
{ {
struct i2c_client *client = drm_i2c_encoder_get_client(encoder); struct i2c_client *client = priv->hdmi;
uint8_t buf[cnt+1]; uint8_t buf[cnt+1];
int ret; int ret;
buf[0] = REG2ADDR(reg); buf[0] = REG2ADDR(reg);
memcpy(&buf[1], p, cnt); memcpy(&buf[1], p, cnt);
set_page(encoder, reg); set_page(priv, reg);
ret = i2c_master_send(client, buf, cnt + 1); ret = i2c_master_send(client, buf, cnt + 1);
if (ret < 0) if (ret < 0)
...@@ -422,21 +421,21 @@ reg_write_range(struct drm_encoder *encoder, uint16_t reg, uint8_t *p, int cnt) ...@@ -422,21 +421,21 @@ reg_write_range(struct drm_encoder *encoder, uint16_t reg, uint8_t *p, int cnt)
} }
static uint8_t static uint8_t
reg_read(struct drm_encoder *encoder, uint16_t reg) reg_read(struct tda998x_priv *priv, uint16_t reg)
{ {
uint8_t val = 0; uint8_t val = 0;
reg_read_range(encoder, reg, &val, sizeof(val)); reg_read_range(priv, reg, &val, sizeof(val));
return val; return val;
} }
static void static void
reg_write(struct drm_encoder *encoder, uint16_t reg, uint8_t val) reg_write(struct tda998x_priv *priv, uint16_t reg, uint8_t val)
{ {
struct i2c_client *client = drm_i2c_encoder_get_client(encoder); struct i2c_client *client = priv->hdmi;
uint8_t buf[] = {REG2ADDR(reg), val}; uint8_t buf[] = {REG2ADDR(reg), val};
int ret; int ret;
set_page(encoder, reg); set_page(priv, reg);
ret = i2c_master_send(client, buf, ARRAY_SIZE(buf)); ret = i2c_master_send(client, buf, ARRAY_SIZE(buf));
if (ret < 0) if (ret < 0)
...@@ -444,13 +443,13 @@ reg_write(struct drm_encoder *encoder, uint16_t reg, uint8_t val) ...@@ -444,13 +443,13 @@ reg_write(struct drm_encoder *encoder, uint16_t reg, uint8_t val)
} }
static void static void
reg_write16(struct drm_encoder *encoder, uint16_t reg, uint16_t val) reg_write16(struct tda998x_priv *priv, uint16_t reg, uint16_t val)
{ {
struct i2c_client *client = drm_i2c_encoder_get_client(encoder); struct i2c_client *client = priv->hdmi;
uint8_t buf[] = {REG2ADDR(reg), val >> 8, val}; uint8_t buf[] = {REG2ADDR(reg), val >> 8, val};
int ret; int ret;
set_page(encoder, reg); set_page(priv, reg);
ret = i2c_master_send(client, buf, ARRAY_SIZE(buf)); ret = i2c_master_send(client, buf, ARRAY_SIZE(buf));
if (ret < 0) if (ret < 0)
...@@ -458,47 +457,47 @@ reg_write16(struct drm_encoder *encoder, uint16_t reg, uint16_t val) ...@@ -458,47 +457,47 @@ reg_write16(struct drm_encoder *encoder, uint16_t reg, uint16_t val)
} }
static void static void
reg_set(struct drm_encoder *encoder, uint16_t reg, uint8_t val) reg_set(struct tda998x_priv *priv, uint16_t reg, uint8_t val)
{ {
reg_write(encoder, reg, reg_read(encoder, reg) | val); reg_write(priv, reg, reg_read(priv, reg) | val);
} }
static void static void
reg_clear(struct drm_encoder *encoder, uint16_t reg, uint8_t val) reg_clear(struct tda998x_priv *priv, uint16_t reg, uint8_t val)
{ {
reg_write(encoder, reg, reg_read(encoder, reg) & ~val); reg_write(priv, reg, reg_read(priv, reg) & ~val);
} }
static void static void
tda998x_reset(struct drm_encoder *encoder) tda998x_reset(struct tda998x_priv *priv)
{ {
/* reset audio and i2c master: */ /* reset audio and i2c master: */
reg_set(encoder, REG_SOFTRESET, SOFTRESET_AUDIO | SOFTRESET_I2C_MASTER); reg_set(priv, REG_SOFTRESET, SOFTRESET_AUDIO | SOFTRESET_I2C_MASTER);
msleep(50); msleep(50);
reg_clear(encoder, REG_SOFTRESET, SOFTRESET_AUDIO | SOFTRESET_I2C_MASTER); reg_clear(priv, REG_SOFTRESET, SOFTRESET_AUDIO | SOFTRESET_I2C_MASTER);
msleep(50); msleep(50);
/* reset transmitter: */ /* reset transmitter: */
reg_set(encoder, REG_MAIN_CNTRL0, MAIN_CNTRL0_SR); reg_set(priv, REG_MAIN_CNTRL0, MAIN_CNTRL0_SR);
reg_clear(encoder, REG_MAIN_CNTRL0, MAIN_CNTRL0_SR); reg_clear(priv, REG_MAIN_CNTRL0, MAIN_CNTRL0_SR);
/* PLL registers common configuration */ /* PLL registers common configuration */
reg_write(encoder, REG_PLL_SERIAL_1, 0x00); reg_write(priv, REG_PLL_SERIAL_1, 0x00);
reg_write(encoder, REG_PLL_SERIAL_2, PLL_SERIAL_2_SRL_NOSC(1)); reg_write(priv, REG_PLL_SERIAL_2, PLL_SERIAL_2_SRL_NOSC(1));
reg_write(encoder, REG_PLL_SERIAL_3, 0x00); reg_write(priv, REG_PLL_SERIAL_3, 0x00);
reg_write(encoder, REG_SERIALIZER, 0x00); reg_write(priv, REG_SERIALIZER, 0x00);
reg_write(encoder, REG_BUFFER_OUT, 0x00); reg_write(priv, REG_BUFFER_OUT, 0x00);
reg_write(encoder, REG_PLL_SCG1, 0x00); reg_write(priv, REG_PLL_SCG1, 0x00);
reg_write(encoder, REG_AUDIO_DIV, AUDIO_DIV_SERCLK_8); reg_write(priv, REG_AUDIO_DIV, AUDIO_DIV_SERCLK_8);
reg_write(encoder, REG_SEL_CLK, SEL_CLK_SEL_CLK1 | SEL_CLK_ENA_SC_CLK); reg_write(priv, REG_SEL_CLK, SEL_CLK_SEL_CLK1 | SEL_CLK_ENA_SC_CLK);
reg_write(encoder, REG_PLL_SCGN1, 0xfa); reg_write(priv, REG_PLL_SCGN1, 0xfa);
reg_write(encoder, REG_PLL_SCGN2, 0x00); reg_write(priv, REG_PLL_SCGN2, 0x00);
reg_write(encoder, REG_PLL_SCGR1, 0x5b); reg_write(priv, REG_PLL_SCGR1, 0x5b);
reg_write(encoder, REG_PLL_SCGR2, 0x00); reg_write(priv, REG_PLL_SCGR2, 0x00);
reg_write(encoder, REG_PLL_SCG2, 0x10); reg_write(priv, REG_PLL_SCG2, 0x10);
/* Write the default value MUX register */ /* Write the default value MUX register */
reg_write(encoder, REG_MUX_VP_VIP_OUT, 0x24); reg_write(priv, REG_MUX_VP_VIP_OUT, 0x24);
} }
static uint8_t tda998x_cksum(uint8_t *buf, size_t bytes) static uint8_t tda998x_cksum(uint8_t *buf, size_t bytes)
...@@ -514,18 +513,18 @@ static uint8_t tda998x_cksum(uint8_t *buf, size_t bytes) ...@@ -514,18 +513,18 @@ static uint8_t tda998x_cksum(uint8_t *buf, size_t bytes)
#define PB(x) (HB(2) + 1 + (x)) #define PB(x) (HB(2) + 1 + (x))
static void static void
tda998x_write_if(struct drm_encoder *encoder, uint8_t bit, uint16_t addr, tda998x_write_if(struct tda998x_priv *priv, uint8_t bit, uint16_t addr,
uint8_t *buf, size_t size) uint8_t *buf, size_t size)
{ {
buf[PB(0)] = tda998x_cksum(buf, size); buf[PB(0)] = tda998x_cksum(buf, size);
reg_clear(encoder, REG_DIP_IF_FLAGS, bit); reg_clear(priv, REG_DIP_IF_FLAGS, bit);
reg_write_range(encoder, addr, buf, size); reg_write_range(priv, addr, buf, size);
reg_set(encoder, REG_DIP_IF_FLAGS, bit); reg_set(priv, REG_DIP_IF_FLAGS, bit);
} }
static void static void
tda998x_write_aif(struct drm_encoder *encoder, struct tda998x_encoder_params *p) tda998x_write_aif(struct tda998x_priv *priv, struct tda998x_encoder_params *p)
{ {
u8 buf[PB(HDMI_AUDIO_INFOFRAME_SIZE) + 1]; u8 buf[PB(HDMI_AUDIO_INFOFRAME_SIZE) + 1];
...@@ -538,12 +537,12 @@ tda998x_write_aif(struct drm_encoder *encoder, struct tda998x_encoder_params *p) ...@@ -538,12 +537,12 @@ tda998x_write_aif(struct drm_encoder *encoder, struct tda998x_encoder_params *p)
buf[PB(4)] = p->audio_frame[4]; buf[PB(4)] = p->audio_frame[4];
buf[PB(5)] = p->audio_frame[5] & 0xf8; /* DM_INH + LSV */ buf[PB(5)] = p->audio_frame[5] & 0xf8; /* DM_INH + LSV */
tda998x_write_if(encoder, DIP_IF_FLAGS_IF4, REG_IF4_HB0, buf, tda998x_write_if(priv, DIP_IF_FLAGS_IF4, REG_IF4_HB0, buf,
sizeof(buf)); sizeof(buf));
} }
static void static void
tda998x_write_avi(struct drm_encoder *encoder, struct drm_display_mode *mode) tda998x_write_avi(struct tda998x_priv *priv, struct drm_display_mode *mode)
{ {
u8 buf[PB(HDMI_AVI_INFOFRAME_SIZE) + 1]; u8 buf[PB(HDMI_AVI_INFOFRAME_SIZE) + 1];
...@@ -556,36 +555,36 @@ tda998x_write_avi(struct drm_encoder *encoder, struct drm_display_mode *mode) ...@@ -556,36 +555,36 @@ tda998x_write_avi(struct drm_encoder *encoder, struct drm_display_mode *mode)
buf[PB(3)] = HDMI_QUANTIZATION_RANGE_FULL << 2; buf[PB(3)] = HDMI_QUANTIZATION_RANGE_FULL << 2;
buf[PB(4)] = drm_match_cea_mode(mode); buf[PB(4)] = drm_match_cea_mode(mode);
tda998x_write_if(encoder, DIP_IF_FLAGS_IF2, REG_IF2_HB0, buf, tda998x_write_if(priv, DIP_IF_FLAGS_IF2, REG_IF2_HB0, buf,
sizeof(buf)); sizeof(buf));
} }
static void tda998x_audio_mute(struct drm_encoder *encoder, bool on) static void tda998x_audio_mute(struct tda998x_priv *priv, bool on)
{ {
if (on) { if (on) {
reg_set(encoder, REG_SOFTRESET, SOFTRESET_AUDIO); reg_set(priv, REG_SOFTRESET, SOFTRESET_AUDIO);
reg_clear(encoder, REG_SOFTRESET, SOFTRESET_AUDIO); reg_clear(priv, REG_SOFTRESET, SOFTRESET_AUDIO);
reg_set(encoder, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_FIFO); reg_set(priv, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_FIFO);
} else { } else {
reg_clear(encoder, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_FIFO); reg_clear(priv, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_FIFO);
} }
} }
static void static void
tda998x_configure_audio(struct drm_encoder *encoder, tda998x_configure_audio(struct tda998x_priv *priv,
struct drm_display_mode *mode, struct tda998x_encoder_params *p) struct drm_display_mode *mode, struct tda998x_encoder_params *p)
{ {
uint8_t buf[6], clksel_aip, clksel_fs, ca_i2s, cts_n, adiv; uint8_t buf[6], clksel_aip, clksel_fs, ca_i2s, cts_n, adiv;
uint32_t n; uint32_t n;
/* Enable audio ports */ /* Enable audio ports */
reg_write(encoder, REG_ENA_AP, p->audio_cfg); reg_write(priv, REG_ENA_AP, p->audio_cfg);
reg_write(encoder, REG_ENA_ACLK, p->audio_clk_cfg); reg_write(priv, REG_ENA_ACLK, p->audio_clk_cfg);
/* Set audio input source */ /* Set audio input source */
switch (p->audio_format) { switch (p->audio_format) {
case AFMT_SPDIF: case AFMT_SPDIF:
reg_write(encoder, REG_MUX_AP, 0x40); reg_write(priv, REG_MUX_AP, 0x40);
clksel_aip = AIP_CLKSEL_AIP(0); clksel_aip = AIP_CLKSEL_AIP(0);
/* FS64SPDIF */ /* FS64SPDIF */
clksel_fs = AIP_CLKSEL_FS(2); clksel_fs = AIP_CLKSEL_FS(2);
...@@ -594,7 +593,7 @@ tda998x_configure_audio(struct drm_encoder *encoder, ...@@ -594,7 +593,7 @@ tda998x_configure_audio(struct drm_encoder *encoder,
break; break;
case AFMT_I2S: case AFMT_I2S:
reg_write(encoder, REG_MUX_AP, 0x64); reg_write(priv, REG_MUX_AP, 0x64);
clksel_aip = AIP_CLKSEL_AIP(1); clksel_aip = AIP_CLKSEL_AIP(1);
/* ACLK */ /* ACLK */
clksel_fs = AIP_CLKSEL_FS(0); clksel_fs = AIP_CLKSEL_FS(0);
...@@ -607,12 +606,12 @@ tda998x_configure_audio(struct drm_encoder *encoder, ...@@ -607,12 +606,12 @@ tda998x_configure_audio(struct drm_encoder *encoder,
return; return;
} }
reg_write(encoder, REG_AIP_CLKSEL, clksel_aip); reg_write(priv, REG_AIP_CLKSEL, clksel_aip);
reg_clear(encoder, REG_AIP_CNTRL_0, AIP_CNTRL_0_LAYOUT); reg_clear(priv, REG_AIP_CNTRL_0, AIP_CNTRL_0_LAYOUT);
/* Enable automatic CTS generation */ /* Enable automatic CTS generation */
reg_clear(encoder, REG_AIP_CNTRL_0, AIP_CNTRL_0_ACR_MAN); reg_clear(priv, REG_AIP_CNTRL_0, AIP_CNTRL_0_ACR_MAN);
reg_write(encoder, REG_CTS_N, cts_n); reg_write(priv, REG_CTS_N, cts_n);
/* /*
* Audio input somehow depends on HDMI line rate which is * Audio input somehow depends on HDMI line rate which is
...@@ -625,7 +624,7 @@ tda998x_configure_audio(struct drm_encoder *encoder, ...@@ -625,7 +624,7 @@ tda998x_configure_audio(struct drm_encoder *encoder,
adiv = AUDIO_DIV_SERCLK_16; adiv = AUDIO_DIV_SERCLK_16;
else else
adiv = AUDIO_DIV_SERCLK_8; adiv = AUDIO_DIV_SERCLK_8;
reg_write(encoder, REG_AUDIO_DIV, adiv); reg_write(priv, REG_AUDIO_DIV, adiv);
/* /*
* This is the approximate value of N, which happens to be * This is the approximate value of N, which happens to be
...@@ -640,14 +639,14 @@ tda998x_configure_audio(struct drm_encoder *encoder, ...@@ -640,14 +639,14 @@ tda998x_configure_audio(struct drm_encoder *encoder,
buf[3] = n; buf[3] = n;
buf[4] = n >> 8; buf[4] = n >> 8;
buf[5] = n >> 16; buf[5] = n >> 16;
reg_write_range(encoder, REG_ACR_CTS_0, buf, 6); reg_write_range(priv, REG_ACR_CTS_0, buf, 6);
/* Set CTS clock reference */ /* Set CTS clock reference */
reg_write(encoder, REG_AIP_CLKSEL, clksel_aip | clksel_fs); reg_write(priv, REG_AIP_CLKSEL, clksel_aip | clksel_fs);
/* Reset CTS generator */ /* Reset CTS generator */
reg_set(encoder, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_CTS); reg_set(priv, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_CTS);
reg_clear(encoder, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_CTS); reg_clear(priv, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_CTS);
/* Write the channel status */ /* Write the channel status */
buf[0] = IEC958_AES0_CON_NOT_COPYRIGHT; buf[0] = IEC958_AES0_CON_NOT_COPYRIGHT;
...@@ -655,14 +654,14 @@ tda998x_configure_audio(struct drm_encoder *encoder, ...@@ -655,14 +654,14 @@ tda998x_configure_audio(struct drm_encoder *encoder,
buf[2] = IEC958_AES3_CON_FS_NOTID; buf[2] = IEC958_AES3_CON_FS_NOTID;
buf[3] = IEC958_AES4_CON_ORIGFS_NOTID | buf[3] = IEC958_AES4_CON_ORIGFS_NOTID |
IEC958_AES4_CON_MAX_WORDLEN_24; IEC958_AES4_CON_MAX_WORDLEN_24;
reg_write_range(encoder, REG_CH_STAT_B(0), buf, 4); reg_write_range(priv, REG_CH_STAT_B(0), buf, 4);
tda998x_audio_mute(encoder, true); tda998x_audio_mute(priv, true);
mdelay(20); mdelay(20);
tda998x_audio_mute(encoder, false); tda998x_audio_mute(priv, false);
/* Write the audio information packet */ /* Write the audio information packet */
tda998x_write_aif(encoder, p); tda998x_write_aif(priv, p);
} }
/* DRM encoder functions */ /* DRM encoder functions */
...@@ -704,19 +703,19 @@ tda998x_encoder_dpms(struct drm_encoder *encoder, int mode) ...@@ -704,19 +703,19 @@ tda998x_encoder_dpms(struct drm_encoder *encoder, int mode)
switch (mode) { switch (mode) {
case DRM_MODE_DPMS_ON: case DRM_MODE_DPMS_ON:
/* enable video ports, audio will be enabled later */ /* enable video ports, audio will be enabled later */
reg_write(encoder, REG_ENA_VP_0, 0xff); reg_write(priv, REG_ENA_VP_0, 0xff);
reg_write(encoder, REG_ENA_VP_1, 0xff); reg_write(priv, REG_ENA_VP_1, 0xff);
reg_write(encoder, REG_ENA_VP_2, 0xff); reg_write(priv, REG_ENA_VP_2, 0xff);
/* set muxing after enabling ports: */ /* set muxing after enabling ports: */
reg_write(encoder, REG_VIP_CNTRL_0, priv->vip_cntrl_0); reg_write(priv, REG_VIP_CNTRL_0, priv->vip_cntrl_0);
reg_write(encoder, REG_VIP_CNTRL_1, priv->vip_cntrl_1); reg_write(priv, REG_VIP_CNTRL_1, priv->vip_cntrl_1);
reg_write(encoder, REG_VIP_CNTRL_2, priv->vip_cntrl_2); reg_write(priv, REG_VIP_CNTRL_2, priv->vip_cntrl_2);
break; break;
case DRM_MODE_DPMS_OFF: case DRM_MODE_DPMS_OFF:
/* disable video ports */ /* disable video ports */
reg_write(encoder, REG_ENA_VP_0, 0x00); reg_write(priv, REG_ENA_VP_0, 0x00);
reg_write(encoder, REG_ENA_VP_1, 0x00); reg_write(priv, REG_ENA_VP_1, 0x00);
reg_write(encoder, REG_ENA_VP_2, 0x00); reg_write(priv, REG_ENA_VP_2, 0x00);
break; break;
} }
...@@ -834,57 +833,57 @@ tda998x_encoder_mode_set(struct drm_encoder *encoder, ...@@ -834,57 +833,57 @@ tda998x_encoder_mode_set(struct drm_encoder *encoder,
} }
/* mute the audio FIFO: */ /* mute the audio FIFO: */
reg_set(encoder, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_FIFO); reg_set(priv, REG_AIP_CNTRL_0, AIP_CNTRL_0_RST_FIFO);
/* set HDMI HDCP mode off: */ /* set HDMI HDCP mode off: */
reg_set(encoder, REG_TBG_CNTRL_1, TBG_CNTRL_1_DWIN_DIS); reg_set(priv, REG_TBG_CNTRL_1, TBG_CNTRL_1_DWIN_DIS);
reg_clear(encoder, REG_TX33, TX33_HDMI); reg_clear(priv, REG_TX33, TX33_HDMI);
reg_write(priv, REG_ENC_CNTRL, ENC_CNTRL_CTL_CODE(0));
reg_write(encoder, REG_ENC_CNTRL, ENC_CNTRL_CTL_CODE(0));
/* no pre-filter or interpolator: */ /* no pre-filter or interpolator: */
reg_write(encoder, REG_HVF_CNTRL_0, HVF_CNTRL_0_PREFIL(0) | reg_write(priv, REG_HVF_CNTRL_0, HVF_CNTRL_0_PREFIL(0) |
HVF_CNTRL_0_INTPOL(0)); HVF_CNTRL_0_INTPOL(0));
reg_write(encoder, REG_VIP_CNTRL_5, VIP_CNTRL_5_SP_CNT(0)); reg_write(priv, REG_VIP_CNTRL_5, VIP_CNTRL_5_SP_CNT(0));
reg_write(encoder, REG_VIP_CNTRL_4, VIP_CNTRL_4_BLANKIT(0) | reg_write(priv, REG_VIP_CNTRL_4, VIP_CNTRL_4_BLANKIT(0) |
VIP_CNTRL_4_BLC(0)); VIP_CNTRL_4_BLC(0));
reg_clear(encoder, REG_PLL_SERIAL_3, PLL_SERIAL_3_SRL_CCIR); reg_clear(priv, REG_PLL_SERIAL_3, PLL_SERIAL_3_SRL_CCIR);
reg_clear(encoder, REG_PLL_SERIAL_1, PLL_SERIAL_1_SRL_MAN_IZ); reg_clear(priv, REG_PLL_SERIAL_1, PLL_SERIAL_1_SRL_MAN_IZ);
reg_clear(encoder, REG_PLL_SERIAL_3, PLL_SERIAL_3_SRL_DE); reg_clear(priv, REG_PLL_SERIAL_3, PLL_SERIAL_3_SRL_DE);
reg_write(encoder, REG_SERIALIZER, 0); reg_write(priv, REG_SERIALIZER, 0);
reg_write(encoder, REG_HVF_CNTRL_1, HVF_CNTRL_1_VQR(0)); reg_write(priv, REG_HVF_CNTRL_1, HVF_CNTRL_1_VQR(0));
/* TODO enable pixel repeat for pixel rates less than 25Msamp/s */ /* TODO enable pixel repeat for pixel rates less than 25Msamp/s */
rep = 0; rep = 0;
reg_write(encoder, REG_RPT_CNTRL, 0); reg_write(priv, REG_RPT_CNTRL, 0);
reg_write(encoder, REG_SEL_CLK, SEL_CLK_SEL_VRF_CLK(0) | reg_write(priv, REG_SEL_CLK, SEL_CLK_SEL_VRF_CLK(0) |
SEL_CLK_SEL_CLK1 | SEL_CLK_ENA_SC_CLK); SEL_CLK_SEL_CLK1 | SEL_CLK_ENA_SC_CLK);
reg_write(encoder, REG_PLL_SERIAL_2, PLL_SERIAL_2_SRL_NOSC(div) | reg_write(priv, REG_PLL_SERIAL_2, PLL_SERIAL_2_SRL_NOSC(div) |
PLL_SERIAL_2_SRL_PR(rep)); PLL_SERIAL_2_SRL_PR(rep));
/* set color matrix bypass flag: */ /* set color matrix bypass flag: */
reg_set(encoder, REG_MAT_CONTRL, MAT_CONTRL_MAT_BP); reg_set(priv, REG_MAT_CONTRL, MAT_CONTRL_MAT_BP);
/* set BIAS tmds value: */ /* set BIAS tmds value: */
reg_write(encoder, REG_ANA_GENERAL, 0x09); reg_write(priv, REG_ANA_GENERAL, 0x09);
reg_clear(encoder, REG_TBG_CNTRL_0, TBG_CNTRL_0_SYNC_MTHD); reg_clear(priv, REG_TBG_CNTRL_0, TBG_CNTRL_0_SYNC_MTHD);
/* /*
* Sync on rising HSYNC/VSYNC * Sync on rising HSYNC/VSYNC
*/ */
reg_write(encoder, REG_VIP_CNTRL_3, 0); reg_write(priv, REG_VIP_CNTRL_3, 0);
reg_set(encoder, REG_VIP_CNTRL_3, VIP_CNTRL_3_SYNC_HS); reg_set(priv, REG_VIP_CNTRL_3, VIP_CNTRL_3_SYNC_HS);
/* /*
* TDA19988 requires high-active sync at input stage, * TDA19988 requires high-active sync at input stage,
* so invert low-active sync provided by master encoder here * so invert low-active sync provided by master encoder here
*/ */
if (mode->flags & DRM_MODE_FLAG_NHSYNC) if (mode->flags & DRM_MODE_FLAG_NHSYNC)
reg_set(encoder, REG_VIP_CNTRL_3, VIP_CNTRL_3_H_TGL); reg_set(priv, REG_VIP_CNTRL_3, VIP_CNTRL_3_H_TGL);
if (mode->flags & DRM_MODE_FLAG_NVSYNC) if (mode->flags & DRM_MODE_FLAG_NVSYNC)
reg_set(encoder, REG_VIP_CNTRL_3, VIP_CNTRL_3_V_TGL); reg_set(priv, REG_VIP_CNTRL_3, VIP_CNTRL_3_V_TGL);
/* /*
* Always generate sync polarity relative to input sync and * Always generate sync polarity relative to input sync and
...@@ -895,49 +894,49 @@ tda998x_encoder_mode_set(struct drm_encoder *encoder, ...@@ -895,49 +894,49 @@ tda998x_encoder_mode_set(struct drm_encoder *encoder,
reg |= TBG_CNTRL_1_H_TGL; reg |= TBG_CNTRL_1_H_TGL;
if (mode->flags & DRM_MODE_FLAG_NVSYNC) if (mode->flags & DRM_MODE_FLAG_NVSYNC)
reg |= TBG_CNTRL_1_V_TGL; reg |= TBG_CNTRL_1_V_TGL;
reg_write(encoder, REG_TBG_CNTRL_1, reg); reg_write(priv, REG_TBG_CNTRL_1, reg);
reg_write(encoder, REG_VIDFORMAT, 0x00); reg_write(priv, REG_VIDFORMAT, 0x00);
reg_write16(encoder, REG_REFPIX_MSB, ref_pix); reg_write16(priv, REG_REFPIX_MSB, ref_pix);
reg_write16(encoder, REG_REFLINE_MSB, ref_line); reg_write16(priv, REG_REFLINE_MSB, ref_line);
reg_write16(encoder, REG_NPIX_MSB, n_pix); reg_write16(priv, REG_NPIX_MSB, n_pix);
reg_write16(encoder, REG_NLINE_MSB, n_line); reg_write16(priv, REG_NLINE_MSB, n_line);
reg_write16(encoder, REG_VS_LINE_STRT_1_MSB, vs1_line_s); reg_write16(priv, REG_VS_LINE_STRT_1_MSB, vs1_line_s);
reg_write16(encoder, REG_VS_PIX_STRT_1_MSB, vs1_pix_s); reg_write16(priv, REG_VS_PIX_STRT_1_MSB, vs1_pix_s);
reg_write16(encoder, REG_VS_LINE_END_1_MSB, vs1_line_e); reg_write16(priv, REG_VS_LINE_END_1_MSB, vs1_line_e);
reg_write16(encoder, REG_VS_PIX_END_1_MSB, vs1_pix_e); reg_write16(priv, REG_VS_PIX_END_1_MSB, vs1_pix_e);
reg_write16(encoder, REG_VS_LINE_STRT_2_MSB, vs2_line_s); reg_write16(priv, REG_VS_LINE_STRT_2_MSB, vs2_line_s);
reg_write16(encoder, REG_VS_PIX_STRT_2_MSB, vs2_pix_s); reg_write16(priv, REG_VS_PIX_STRT_2_MSB, vs2_pix_s);
reg_write16(encoder, REG_VS_LINE_END_2_MSB, vs2_line_e); reg_write16(priv, REG_VS_LINE_END_2_MSB, vs2_line_e);
reg_write16(encoder, REG_VS_PIX_END_2_MSB, vs2_pix_e); reg_write16(priv, REG_VS_PIX_END_2_MSB, vs2_pix_e);
reg_write16(encoder, REG_HS_PIX_START_MSB, hs_pix_s); reg_write16(priv, REG_HS_PIX_START_MSB, hs_pix_s);
reg_write16(encoder, REG_HS_PIX_STOP_MSB, hs_pix_e); reg_write16(priv, REG_HS_PIX_STOP_MSB, hs_pix_e);
reg_write16(encoder, REG_VWIN_START_1_MSB, vwin1_line_s); reg_write16(priv, REG_VWIN_START_1_MSB, vwin1_line_s);
reg_write16(encoder, REG_VWIN_END_1_MSB, vwin1_line_e); reg_write16(priv, REG_VWIN_END_1_MSB, vwin1_line_e);
reg_write16(encoder, REG_VWIN_START_2_MSB, vwin2_line_s); reg_write16(priv, REG_VWIN_START_2_MSB, vwin2_line_s);
reg_write16(encoder, REG_VWIN_END_2_MSB, vwin2_line_e); reg_write16(priv, REG_VWIN_END_2_MSB, vwin2_line_e);
reg_write16(encoder, REG_DE_START_MSB, de_pix_s); reg_write16(priv, REG_DE_START_MSB, de_pix_s);
reg_write16(encoder, REG_DE_STOP_MSB, de_pix_e); reg_write16(priv, REG_DE_STOP_MSB, de_pix_e);
if (priv->rev == TDA19988) { if (priv->rev == TDA19988) {
/* let incoming pixels fill the active space (if any) */ /* let incoming pixels fill the active space (if any) */
reg_write(encoder, REG_ENABLE_SPACE, 0x00); reg_write(priv, REG_ENABLE_SPACE, 0x00);
} }
/* must be last register set: */ /* must be last register set: */
reg_clear(encoder, REG_TBG_CNTRL_0, TBG_CNTRL_0_SYNC_ONCE); reg_clear(priv, REG_TBG_CNTRL_0, TBG_CNTRL_0_SYNC_ONCE);
/* Only setup the info frames if the sink is HDMI */ /* Only setup the info frames if the sink is HDMI */
if (priv->is_hdmi_sink) { if (priv->is_hdmi_sink) {
/* We need to turn HDMI HDCP stuff on to get audio through */ /* We need to turn HDMI HDCP stuff on to get audio through */
reg_clear(encoder, REG_TBG_CNTRL_1, TBG_CNTRL_1_DWIN_DIS); reg_clear(priv, REG_TBG_CNTRL_1, TBG_CNTRL_1_DWIN_DIS);
reg_write(encoder, REG_ENC_CNTRL, ENC_CNTRL_CTL_CODE(1)); reg_write(priv, REG_ENC_CNTRL, ENC_CNTRL_CTL_CODE(1));
reg_set(encoder, REG_TX33, TX33_HDMI); reg_set(priv, REG_TX33, TX33_HDMI);
tda998x_write_avi(encoder, adjusted_mode); tda998x_write_avi(priv, adjusted_mode);
if (priv->params.audio_cfg) if (priv->params.audio_cfg)
tda998x_configure_audio(encoder, adjusted_mode, tda998x_configure_audio(priv, adjusted_mode,
&priv->params); &priv->params);
} }
} }
...@@ -946,7 +945,9 @@ static enum drm_connector_status ...@@ -946,7 +945,9 @@ static enum drm_connector_status
tda998x_encoder_detect(struct drm_encoder *encoder, tda998x_encoder_detect(struct drm_encoder *encoder,
struct drm_connector *connector) struct drm_connector *connector)
{ {
uint8_t val = cec_read(encoder, REG_CEC_RXSHPDLEV); struct tda998x_priv *priv = to_tda998x_priv(encoder);
uint8_t val = cec_read(priv, REG_CEC_RXSHPDLEV);
return (val & CEC_RXSHPDLEV_HPD) ? connector_status_connected : return (val & CEC_RXSHPDLEV_HPD) ? connector_status_connected :
connector_status_disconnected; connector_status_disconnected;
} }
...@@ -954,29 +955,30 @@ tda998x_encoder_detect(struct drm_encoder *encoder, ...@@ -954,29 +955,30 @@ tda998x_encoder_detect(struct drm_encoder *encoder,
static int static int
read_edid_block(struct drm_encoder *encoder, uint8_t *buf, int blk) read_edid_block(struct drm_encoder *encoder, uint8_t *buf, int blk)
{ {
struct tda998x_priv *priv = to_tda998x_priv(encoder);
uint8_t offset, segptr; uint8_t offset, segptr;
int ret, i; int ret, i;
/* enable EDID read irq: */ /* enable EDID read irq: */
reg_set(encoder, REG_INT_FLAGS_2, INT_FLAGS_2_EDID_BLK_RD); reg_set(priv, REG_INT_FLAGS_2, INT_FLAGS_2_EDID_BLK_RD);
offset = (blk & 1) ? 128 : 0; offset = (blk & 1) ? 128 : 0;
segptr = blk / 2; segptr = blk / 2;
reg_write(encoder, REG_DDC_ADDR, 0xa0); reg_write(priv, REG_DDC_ADDR, 0xa0);
reg_write(encoder, REG_DDC_OFFS, offset); reg_write(priv, REG_DDC_OFFS, offset);
reg_write(encoder, REG_DDC_SEGM_ADDR, 0x60); reg_write(priv, REG_DDC_SEGM_ADDR, 0x60);
reg_write(encoder, REG_DDC_SEGM, segptr); reg_write(priv, REG_DDC_SEGM, segptr);
/* enable reading EDID: */ /* enable reading EDID: */
reg_write(encoder, REG_EDID_CTRL, 0x1); reg_write(priv, REG_EDID_CTRL, 0x1);
/* flag must be cleared by sw: */ /* flag must be cleared by sw: */
reg_write(encoder, REG_EDID_CTRL, 0x0); reg_write(priv, REG_EDID_CTRL, 0x0);
/* wait for block read to complete: */ /* wait for block read to complete: */
for (i = 100; i > 0; i--) { for (i = 100; i > 0; i--) {
uint8_t val = reg_read(encoder, REG_INT_FLAGS_2); uint8_t val = reg_read(priv, REG_INT_FLAGS_2);
if (val & INT_FLAGS_2_EDID_BLK_RD) if (val & INT_FLAGS_2_EDID_BLK_RD)
break; break;
msleep(1); msleep(1);
...@@ -985,14 +987,14 @@ read_edid_block(struct drm_encoder *encoder, uint8_t *buf, int blk) ...@@ -985,14 +987,14 @@ read_edid_block(struct drm_encoder *encoder, uint8_t *buf, int blk)
if (i == 0) if (i == 0)
return -ETIMEDOUT; return -ETIMEDOUT;
ret = reg_read_range(encoder, REG_EDID_DATA_0, buf, EDID_LENGTH); ret = reg_read_range(priv, REG_EDID_DATA_0, buf, EDID_LENGTH);
if (ret != EDID_LENGTH) { if (ret != EDID_LENGTH) {
dev_err(encoder->dev->dev, "failed to read edid block %d: %d", dev_err(encoder->dev->dev, "failed to read edid block %d: %d",
blk, ret); blk, ret);
return ret; return ret;
} }
reg_clear(encoder, REG_INT_FLAGS_2, INT_FLAGS_2_EDID_BLK_RD); reg_clear(priv, REG_INT_FLAGS_2, INT_FLAGS_2_EDID_BLK_RD);
return 0; return 0;
} }
...@@ -1009,7 +1011,7 @@ do_get_edid(struct drm_encoder *encoder) ...@@ -1009,7 +1011,7 @@ do_get_edid(struct drm_encoder *encoder)
return NULL; return NULL;
if (priv->rev == TDA19988) if (priv->rev == TDA19988)
reg_clear(encoder, REG_TX4, TX4_PD_RAM); reg_clear(priv, REG_TX4, TX4_PD_RAM);
/* base block fetch */ /* base block fetch */
if (read_edid_block(encoder, block, 0)) if (read_edid_block(encoder, block, 0))
...@@ -1049,13 +1051,13 @@ do_get_edid(struct drm_encoder *encoder) ...@@ -1049,13 +1051,13 @@ do_get_edid(struct drm_encoder *encoder)
done: done:
if (priv->rev == TDA19988) if (priv->rev == TDA19988)
reg_set(encoder, REG_TX4, TX4_PD_RAM); reg_set(priv, REG_TX4, TX4_PD_RAM);
return block; return block;
fail: fail:
if (priv->rev == TDA19988) if (priv->rev == TDA19988)
reg_set(encoder, REG_TX4, TX4_PD_RAM); reg_set(priv, REG_TX4, TX4_PD_RAM);
dev_warn(encoder->dev->dev, "failed to read EDID\n"); dev_warn(encoder->dev->dev, "failed to read EDID\n");
kfree(block); kfree(block);
return NULL; return NULL;
...@@ -1141,7 +1143,6 @@ tda998x_encoder_init(struct i2c_client *client, ...@@ -1141,7 +1143,6 @@ tda998x_encoder_init(struct i2c_client *client,
struct drm_device *dev, struct drm_device *dev,
struct drm_encoder_slave *encoder_slave) struct drm_encoder_slave *encoder_slave)
{ {
struct drm_encoder *encoder = &encoder_slave->base;
struct tda998x_priv *priv; struct tda998x_priv *priv;
priv = kzalloc(sizeof(*priv), GFP_KERNEL); priv = kzalloc(sizeof(*priv), GFP_KERNEL);
...@@ -1153,6 +1154,7 @@ tda998x_encoder_init(struct i2c_client *client, ...@@ -1153,6 +1154,7 @@ tda998x_encoder_init(struct i2c_client *client,
priv->vip_cntrl_2 = VIP_CNTRL_2_SWAP_E(4) | VIP_CNTRL_2_SWAP_F(5); priv->vip_cntrl_2 = VIP_CNTRL_2_SWAP_E(4) | VIP_CNTRL_2_SWAP_F(5);
priv->current_page = 0xff; priv->current_page = 0xff;
priv->hdmi = client;
priv->cec = i2c_new_dummy(client->adapter, 0x34); priv->cec = i2c_new_dummy(client->adapter, 0x34);
if (!priv->cec) { if (!priv->cec) {
kfree(priv); kfree(priv);
...@@ -1164,14 +1166,14 @@ tda998x_encoder_init(struct i2c_client *client, ...@@ -1164,14 +1166,14 @@ tda998x_encoder_init(struct i2c_client *client,
encoder_slave->slave_funcs = &tda998x_encoder_funcs; encoder_slave->slave_funcs = &tda998x_encoder_funcs;
/* wake up the device: */ /* wake up the device: */
cec_write(encoder, REG_CEC_ENAMODS, cec_write(priv, REG_CEC_ENAMODS,
CEC_ENAMODS_EN_RXSENS | CEC_ENAMODS_EN_HDMI); CEC_ENAMODS_EN_RXSENS | CEC_ENAMODS_EN_HDMI);
tda998x_reset(encoder); tda998x_reset(priv);
/* read version: */ /* read version: */
priv->rev = reg_read(encoder, REG_VERSION_LSB) | priv->rev = reg_read(priv, REG_VERSION_LSB) |
reg_read(encoder, REG_VERSION_MSB) << 8; reg_read(priv, REG_VERSION_MSB) << 8;
/* mask off feature bits: */ /* mask off feature bits: */
priv->rev &= ~0x30; /* not-hdcp and not-scalar bit */ priv->rev &= ~0x30; /* not-hdcp and not-scalar bit */
...@@ -1187,16 +1189,16 @@ tda998x_encoder_init(struct i2c_client *client, ...@@ -1187,16 +1189,16 @@ tda998x_encoder_init(struct i2c_client *client,
} }
/* after reset, enable DDC: */ /* after reset, enable DDC: */
reg_write(encoder, REG_DDC_DISABLE, 0x00); reg_write(priv, REG_DDC_DISABLE, 0x00);
/* set clock on DDC channel: */ /* set clock on DDC channel: */
reg_write(encoder, REG_TX3, 39); reg_write(priv, REG_TX3, 39);
/* if necessary, disable multi-master: */ /* if necessary, disable multi-master: */
if (priv->rev == TDA19989) if (priv->rev == TDA19989)
reg_set(encoder, REG_I2C_MASTER, I2C_MASTER_DIS_MM); reg_set(priv, REG_I2C_MASTER, I2C_MASTER_DIS_MM);
cec_write(encoder, REG_CEC_FRO_IM_CLK_CTRL, cec_write(priv, REG_CEC_FRO_IM_CLK_CTRL,
CEC_FRO_IM_CLK_CTRL_GHOST_DIS | CEC_FRO_IM_CLK_CTRL_IMCLK_SEL); CEC_FRO_IM_CLK_CTRL_GHOST_DIS | CEC_FRO_IM_CLK_CTRL_IMCLK_SEL);
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment