mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-15 04:56:43 +07:00
8da55c94dd
Change register write and read routines to similar which are typically used. We have to add processor core as a part of register address in order to simplify register access. Chip has two cores, called link and ofdm. As for now, use address bit 24 to address used core. Bits 15:0 are register address in given core. Signed-off-by: Antti Palosaari <crope@iki.fi> Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
473 lines
10 KiB
C
473 lines
10 KiB
C
/*
|
|
* ITE Tech IT9137 silicon tuner driver
|
|
*
|
|
* Copyright (C) 2011 Malcolm Priestley (tvboxspy@gmail.com)
|
|
* IT9137 Copyright (C) ITE Tech Inc.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.=
|
|
*/
|
|
|
|
#include "it913x_priv.h"
|
|
|
|
struct it913x_state {
|
|
struct i2c_client *client;
|
|
struct dvb_frontend *fe;
|
|
u8 chip_ver;
|
|
u8 firmware_ver;
|
|
u16 tun_xtal;
|
|
u8 tun_fdiv;
|
|
u8 tun_clk_mode;
|
|
u32 tun_fn_min;
|
|
};
|
|
|
|
/* read multiple registers */
|
|
static int it913x_rd_regs(struct it913x_state *state,
|
|
u32 reg, u8 *data, u8 count)
|
|
{
|
|
int ret;
|
|
u8 b[3];
|
|
struct i2c_msg msg[2] = {
|
|
{ .addr = state->client->addr, .flags = 0,
|
|
.buf = b, .len = sizeof(b) },
|
|
{ .addr = state->client->addr, .flags = I2C_M_RD,
|
|
.buf = data, .len = count }
|
|
};
|
|
|
|
b[0] = (u8)(reg >> 16) & 0xff;
|
|
b[1] = (u8)(reg >> 8) & 0xff;
|
|
b[2] = (u8) reg & 0xff;
|
|
|
|
ret = i2c_transfer(state->client->adapter, msg, 2);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* read single register */
|
|
static int it913x_rd_reg(struct it913x_state *state, u32 reg, u8 *val)
|
|
{
|
|
int ret;
|
|
u8 b[1];
|
|
|
|
ret = it913x_rd_regs(state, reg, &b[0], sizeof(b));
|
|
if (ret < 0)
|
|
return -ENODEV;
|
|
*val = b[0];
|
|
return 0;
|
|
}
|
|
|
|
/* write multiple registers */
|
|
static int it913x_wr_regs(struct it913x_state *state,
|
|
u32 reg, u8 buf[], u8 count)
|
|
{
|
|
u8 b[256];
|
|
struct i2c_msg msg[1] = {
|
|
{ .addr = state->client->addr, .flags = 0,
|
|
.buf = b, .len = 3 + count }
|
|
};
|
|
int ret;
|
|
|
|
b[0] = (u8)(reg >> 16) & 0xff;
|
|
b[1] = (u8)(reg >> 8) & 0xff;
|
|
b[2] = (u8) reg & 0xff;
|
|
memcpy(&b[3], buf, count);
|
|
|
|
ret = i2c_transfer(state->client->adapter, msg, 1);
|
|
|
|
if (ret < 0)
|
|
return -EIO;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* write single register */
|
|
static int it913x_wr_reg(struct it913x_state *state,
|
|
u32 reg, u32 data)
|
|
{
|
|
int ret;
|
|
u8 b[4];
|
|
u8 s;
|
|
|
|
b[0] = data >> 24;
|
|
b[1] = (data >> 16) & 0xff;
|
|
b[2] = (data >> 8) & 0xff;
|
|
b[3] = data & 0xff;
|
|
/* expand write as needed */
|
|
if (data < 0x100)
|
|
s = 3;
|
|
else if (data < 0x1000)
|
|
s = 2;
|
|
else if (data < 0x100000)
|
|
s = 1;
|
|
else
|
|
s = 0;
|
|
|
|
ret = it913x_wr_regs(state, reg, &b[s], sizeof(b) - s);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int it913x_script_loader(struct it913x_state *state,
|
|
struct it913xset *loadscript)
|
|
{
|
|
int ret, i;
|
|
|
|
if (loadscript == NULL)
|
|
return -EINVAL;
|
|
|
|
for (i = 0; i < 1000; ++i) {
|
|
if (loadscript[i].address == 0x000000)
|
|
break;
|
|
ret = it913x_wr_regs(state,
|
|
loadscript[i].address,
|
|
loadscript[i].reg, loadscript[i].count);
|
|
if (ret < 0)
|
|
return -ENODEV;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int it913x_init(struct dvb_frontend *fe)
|
|
{
|
|
struct it913x_state *state = fe->tuner_priv;
|
|
int ret, i;
|
|
u8 reg = 0;
|
|
u8 val, nv_val;
|
|
u8 nv[] = {48, 32, 24, 16, 12, 8, 6, 4, 2};
|
|
u8 b[2];
|
|
|
|
ret = it913x_rd_reg(state, 0x80ec86, ®);
|
|
switch (reg) {
|
|
case 0:
|
|
state->tun_clk_mode = reg;
|
|
state->tun_xtal = 2000;
|
|
state->tun_fdiv = 3;
|
|
val = 16;
|
|
break;
|
|
case 1:
|
|
default: /* I/O error too */
|
|
state->tun_clk_mode = reg;
|
|
state->tun_xtal = 640;
|
|
state->tun_fdiv = 1;
|
|
val = 6;
|
|
break;
|
|
}
|
|
|
|
ret = it913x_rd_reg(state, 0x80ed03, ®);
|
|
|
|
if (reg < 0)
|
|
return -ENODEV;
|
|
else if (reg < ARRAY_SIZE(nv))
|
|
nv_val = nv[reg];
|
|
else
|
|
nv_val = 2;
|
|
|
|
for (i = 0; i < 50; i++) {
|
|
ret = it913x_rd_regs(state, 0x80ed23, &b[0], sizeof(b));
|
|
reg = (b[1] << 8) + b[0];
|
|
if (reg > 0)
|
|
break;
|
|
if (ret < 0)
|
|
return -ENODEV;
|
|
udelay(2000);
|
|
}
|
|
state->tun_fn_min = state->tun_xtal * reg;
|
|
state->tun_fn_min /= (state->tun_fdiv * nv_val);
|
|
dev_dbg(&state->client->dev, "Tuner fn_min %d\n", state->tun_fn_min);
|
|
|
|
if (state->chip_ver > 1)
|
|
msleep(50);
|
|
else {
|
|
for (i = 0; i < 50; i++) {
|
|
ret = it913x_rd_reg(state, 0x80ec82, ®);
|
|
if (ret < 0)
|
|
return -ENODEV;
|
|
if (reg > 0)
|
|
break;
|
|
udelay(2000);
|
|
}
|
|
}
|
|
|
|
/* Power Up Tuner - common all versions */
|
|
ret = it913x_wr_reg(state, 0x80ec40, 0x1);
|
|
ret |= it913x_wr_reg(state, 0x80ec57, 0x0);
|
|
ret |= it913x_wr_reg(state, 0x80ec58, 0x0);
|
|
|
|
return it913x_wr_reg(state, 0x80ed81, val);
|
|
}
|
|
|
|
static int it9137_set_params(struct dvb_frontend *fe)
|
|
{
|
|
struct it913x_state *state = fe->tuner_priv;
|
|
struct it913xset *set_tuner = set_it9137_template;
|
|
struct dtv_frontend_properties *p = &fe->dtv_property_cache;
|
|
u32 bandwidth = p->bandwidth_hz;
|
|
u32 frequency_m = p->frequency;
|
|
int ret;
|
|
u8 reg = 0;
|
|
u32 frequency = frequency_m / 1000;
|
|
u32 freq, temp_f, tmp;
|
|
u16 iqik_m_cal;
|
|
u16 n_div;
|
|
u8 n;
|
|
u8 l_band;
|
|
u8 lna_band;
|
|
u8 bw;
|
|
|
|
if (state->firmware_ver == 1)
|
|
set_tuner = set_it9135_template;
|
|
else
|
|
set_tuner = set_it9137_template;
|
|
|
|
dev_dbg(&state->client->dev, "Tuner Frequency %d Bandwidth %d\n",
|
|
frequency, bandwidth);
|
|
|
|
if (frequency >= 51000 && frequency <= 440000) {
|
|
l_band = 0;
|
|
lna_band = 0;
|
|
} else if (frequency > 440000 && frequency <= 484000) {
|
|
l_band = 1;
|
|
lna_band = 1;
|
|
} else if (frequency > 484000 && frequency <= 533000) {
|
|
l_band = 1;
|
|
lna_band = 2;
|
|
} else if (frequency > 533000 && frequency <= 587000) {
|
|
l_band = 1;
|
|
lna_band = 3;
|
|
} else if (frequency > 587000 && frequency <= 645000) {
|
|
l_band = 1;
|
|
lna_band = 4;
|
|
} else if (frequency > 645000 && frequency <= 710000) {
|
|
l_band = 1;
|
|
lna_band = 5;
|
|
} else if (frequency > 710000 && frequency <= 782000) {
|
|
l_band = 1;
|
|
lna_band = 6;
|
|
} else if (frequency > 782000 && frequency <= 860000) {
|
|
l_band = 1;
|
|
lna_band = 7;
|
|
} else if (frequency > 1450000 && frequency <= 1492000) {
|
|
l_band = 1;
|
|
lna_band = 0;
|
|
} else if (frequency > 1660000 && frequency <= 1685000) {
|
|
l_band = 1;
|
|
lna_band = 1;
|
|
} else
|
|
return -EINVAL;
|
|
set_tuner[0].reg[0] = lna_band;
|
|
|
|
switch (bandwidth) {
|
|
case 5000000:
|
|
bw = 0;
|
|
break;
|
|
case 6000000:
|
|
bw = 2;
|
|
break;
|
|
case 7000000:
|
|
bw = 4;
|
|
break;
|
|
default:
|
|
case 8000000:
|
|
bw = 6;
|
|
break;
|
|
}
|
|
|
|
set_tuner[1].reg[0] = bw;
|
|
set_tuner[2].reg[0] = 0xa0 | (l_band << 3);
|
|
|
|
if (frequency > 53000 && frequency <= 74000) {
|
|
n_div = 48;
|
|
n = 0;
|
|
} else if (frequency > 74000 && frequency <= 111000) {
|
|
n_div = 32;
|
|
n = 1;
|
|
} else if (frequency > 111000 && frequency <= 148000) {
|
|
n_div = 24;
|
|
n = 2;
|
|
} else if (frequency > 148000 && frequency <= 222000) {
|
|
n_div = 16;
|
|
n = 3;
|
|
} else if (frequency > 222000 && frequency <= 296000) {
|
|
n_div = 12;
|
|
n = 4;
|
|
} else if (frequency > 296000 && frequency <= 445000) {
|
|
n_div = 8;
|
|
n = 5;
|
|
} else if (frequency > 445000 && frequency <= state->tun_fn_min) {
|
|
n_div = 6;
|
|
n = 6;
|
|
} else if (frequency > state->tun_fn_min && frequency <= 950000) {
|
|
n_div = 4;
|
|
n = 7;
|
|
} else if (frequency > 1450000 && frequency <= 1680000) {
|
|
n_div = 2;
|
|
n = 0;
|
|
} else
|
|
return -EINVAL;
|
|
|
|
ret = it913x_rd_reg(state, 0x80ed81, ®);
|
|
iqik_m_cal = (u16)reg * n_div;
|
|
|
|
if (reg < 0x20) {
|
|
if (state->tun_clk_mode == 0)
|
|
iqik_m_cal = (iqik_m_cal * 9) >> 5;
|
|
else
|
|
iqik_m_cal >>= 1;
|
|
} else {
|
|
iqik_m_cal = 0x40 - iqik_m_cal;
|
|
if (state->tun_clk_mode == 0)
|
|
iqik_m_cal = ~((iqik_m_cal * 9) >> 5);
|
|
else
|
|
iqik_m_cal = ~(iqik_m_cal >> 1);
|
|
}
|
|
|
|
temp_f = frequency * (u32)n_div * (u32)state->tun_fdiv;
|
|
freq = temp_f / state->tun_xtal;
|
|
tmp = freq * state->tun_xtal;
|
|
|
|
if ((temp_f - tmp) >= (state->tun_xtal >> 1))
|
|
freq++;
|
|
|
|
freq += (u32) n << 13;
|
|
/* Frequency OMEGA_IQIK_M_CAL_MID*/
|
|
temp_f = freq + (u32)iqik_m_cal;
|
|
|
|
set_tuner[3].reg[0] = temp_f & 0xff;
|
|
set_tuner[4].reg[0] = (temp_f >> 8) & 0xff;
|
|
|
|
dev_dbg(&state->client->dev, "High Frequency = %04x\n", temp_f);
|
|
|
|
/* Lower frequency */
|
|
set_tuner[5].reg[0] = freq & 0xff;
|
|
set_tuner[6].reg[0] = (freq >> 8) & 0xff;
|
|
|
|
dev_dbg(&state->client->dev, "low Frequency = %04x\n", freq);
|
|
|
|
ret = it913x_script_loader(state, set_tuner);
|
|
|
|
return (ret < 0) ? -ENODEV : 0;
|
|
}
|
|
|
|
/* Power sequence */
|
|
/* Power Up Tuner on -> Frontend suspend off -> Tuner clk on */
|
|
/* Power Down Frontend suspend on -> Tuner clk off -> Tuner off */
|
|
|
|
static int it913x_sleep(struct dvb_frontend *fe)
|
|
{
|
|
struct it913x_state *state = fe->tuner_priv;
|
|
|
|
if (state->chip_ver == 0x01)
|
|
return it913x_script_loader(state, it9135ax_tuner_off);
|
|
else
|
|
return it913x_script_loader(state, it9137_tuner_off);
|
|
}
|
|
|
|
static const struct dvb_tuner_ops it913x_tuner_ops = {
|
|
.info = {
|
|
.name = "ITE Tech IT913X",
|
|
.frequency_min = 174000000,
|
|
.frequency_max = 862000000,
|
|
},
|
|
|
|
.init = it913x_init,
|
|
.sleep = it913x_sleep,
|
|
.set_params = it9137_set_params,
|
|
};
|
|
|
|
static int it913x_probe(struct i2c_client *client,
|
|
const struct i2c_device_id *id)
|
|
{
|
|
struct it913x_config *cfg = client->dev.platform_data;
|
|
struct dvb_frontend *fe = cfg->fe;
|
|
struct it913x_state *state;
|
|
int ret;
|
|
char *chip_ver_str;
|
|
|
|
state = kzalloc(sizeof(struct it913x_state), GFP_KERNEL);
|
|
if (state == NULL) {
|
|
ret = -ENOMEM;
|
|
dev_err(&client->dev, "kzalloc() failed\n");
|
|
goto err;
|
|
}
|
|
|
|
state->client = client;
|
|
state->fe = cfg->fe;
|
|
state->chip_ver = cfg->chip_ver;
|
|
state->firmware_ver = 1;
|
|
|
|
/* tuner RF initial */
|
|
ret = it913x_wr_reg(state, 0x80ec4c, 0x68);
|
|
if (ret < 0)
|
|
goto err;
|
|
|
|
fe->tuner_priv = state;
|
|
memcpy(&fe->ops.tuner_ops, &it913x_tuner_ops,
|
|
sizeof(struct dvb_tuner_ops));
|
|
i2c_set_clientdata(client, state);
|
|
|
|
if (state->chip_ver == 1)
|
|
chip_ver_str = "AX";
|
|
else if (state->chip_ver == 2)
|
|
chip_ver_str = "BX";
|
|
else
|
|
chip_ver_str = "??";
|
|
|
|
dev_info(&state->client->dev, "ITE IT913X %s successfully attached\n",
|
|
chip_ver_str);
|
|
dev_dbg(&state->client->dev, "chip_ver=%02x\n", state->chip_ver);
|
|
return 0;
|
|
err:
|
|
dev_dbg(&client->dev, "failed %d\n", ret);
|
|
kfree(state);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int it913x_remove(struct i2c_client *client)
|
|
{
|
|
struct it913x_state *state = i2c_get_clientdata(client);
|
|
struct dvb_frontend *fe = state->fe;
|
|
|
|
dev_dbg(&client->dev, "\n");
|
|
|
|
memset(&fe->ops.tuner_ops, 0, sizeof(struct dvb_tuner_ops));
|
|
fe->tuner_priv = NULL;
|
|
kfree(state);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct i2c_device_id it913x_id_table[] = {
|
|
{"it913x", 0},
|
|
{}
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, it913x_id_table);
|
|
|
|
static struct i2c_driver it913x_driver = {
|
|
.driver = {
|
|
.owner = THIS_MODULE,
|
|
.name = "it913x",
|
|
},
|
|
.probe = it913x_probe,
|
|
.remove = it913x_remove,
|
|
.id_table = it913x_id_table,
|
|
};
|
|
|
|
module_i2c_driver(it913x_driver);
|
|
|
|
MODULE_DESCRIPTION("ITE Tech IT913X silicon tuner driver");
|
|
MODULE_AUTHOR("Antti Palosaari <crope@iki.fi>");
|
|
MODULE_LICENSE("GPL");
|