[media] m88ds3103: use regmap for I2C register access
Use regmap for I2C register access. Remove own I2C repeated mutex as it should not be needed. I2C adapter lock is already taken when I2C mux adapter is called, no need for double locking. Signed-off-by: Antti Palosaari <crope@iki.fi> Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
This commit is contained in:
parent
7978b8a1bc
commit
478932b160
|
@ -38,6 +38,7 @@ config DVB_STV6110x
|
||||||
config DVB_M88DS3103
|
config DVB_M88DS3103
|
||||||
tristate "Montage Technology M88DS3103"
|
tristate "Montage Technology M88DS3103"
|
||||||
depends on DVB_CORE && I2C && I2C_MUX
|
depends on DVB_CORE && I2C && I2C_MUX
|
||||||
|
select REGMAP_I2C
|
||||||
default m if !MEDIA_SUBDRV_AUTOSELECT
|
default m if !MEDIA_SUBDRV_AUTOSELECT
|
||||||
help
|
help
|
||||||
Say Y when you want to support this frontend.
|
Say Y when you want to support this frontend.
|
||||||
|
|
|
@ -18,141 +18,6 @@
|
||||||
|
|
||||||
static struct dvb_frontend_ops m88ds3103_ops;
|
static struct dvb_frontend_ops m88ds3103_ops;
|
||||||
|
|
||||||
/* write multiple registers */
|
|
||||||
static int m88ds3103_wr_regs(struct m88ds3103_dev *dev,
|
|
||||||
u8 reg, const u8 *val, int len)
|
|
||||||
{
|
|
||||||
#define MAX_WR_LEN 32
|
|
||||||
#define MAX_WR_XFER_LEN (MAX_WR_LEN + 1)
|
|
||||||
struct i2c_client *client = dev->client;
|
|
||||||
int ret;
|
|
||||||
u8 buf[MAX_WR_XFER_LEN];
|
|
||||||
struct i2c_msg msg[1] = {
|
|
||||||
{
|
|
||||||
.addr = client->addr,
|
|
||||||
.flags = 0,
|
|
||||||
.len = 1 + len,
|
|
||||||
.buf = buf,
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (WARN_ON(len > MAX_WR_LEN))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
buf[0] = reg;
|
|
||||||
memcpy(&buf[1], val, len);
|
|
||||||
|
|
||||||
mutex_lock(&dev->i2c_mutex);
|
|
||||||
ret = i2c_transfer(client->adapter, msg, 1);
|
|
||||||
mutex_unlock(&dev->i2c_mutex);
|
|
||||||
if (ret == 1) {
|
|
||||||
ret = 0;
|
|
||||||
} else {
|
|
||||||
dev_warn(&client->dev, "i2c wr failed=%d reg=%02x len=%d\n",
|
|
||||||
ret, reg, len);
|
|
||||||
ret = -EREMOTEIO;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* read multiple registers */
|
|
||||||
static int m88ds3103_rd_regs(struct m88ds3103_dev *dev,
|
|
||||||
u8 reg, u8 *val, int len)
|
|
||||||
{
|
|
||||||
#define MAX_RD_LEN 3
|
|
||||||
#define MAX_RD_XFER_LEN (MAX_RD_LEN)
|
|
||||||
struct i2c_client *client = dev->client;
|
|
||||||
int ret;
|
|
||||||
u8 buf[MAX_RD_XFER_LEN];
|
|
||||||
struct i2c_msg msg[2] = {
|
|
||||||
{
|
|
||||||
.addr = client->addr,
|
|
||||||
.flags = 0,
|
|
||||||
.len = 1,
|
|
||||||
.buf = ®,
|
|
||||||
}, {
|
|
||||||
.addr = client->addr,
|
|
||||||
.flags = I2C_M_RD,
|
|
||||||
.len = len,
|
|
||||||
.buf = buf,
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (WARN_ON(len > MAX_RD_LEN))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
mutex_lock(&dev->i2c_mutex);
|
|
||||||
ret = i2c_transfer(client->adapter, msg, 2);
|
|
||||||
mutex_unlock(&dev->i2c_mutex);
|
|
||||||
if (ret == 2) {
|
|
||||||
memcpy(val, buf, len);
|
|
||||||
ret = 0;
|
|
||||||
} else {
|
|
||||||
dev_warn(&client->dev, "i2c rd failed=%d reg=%02x len=%d\n",
|
|
||||||
ret, reg, len);
|
|
||||||
ret = -EREMOTEIO;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* write single register */
|
|
||||||
static int m88ds3103_wr_reg(struct m88ds3103_dev *dev, u8 reg, u8 val)
|
|
||||||
{
|
|
||||||
return m88ds3103_wr_regs(dev, reg, &val, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* read single register */
|
|
||||||
static int m88ds3103_rd_reg(struct m88ds3103_dev *dev, u8 reg, u8 *val)
|
|
||||||
{
|
|
||||||
return m88ds3103_rd_regs(dev, reg, val, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* write single register with mask */
|
|
||||||
static int m88ds3103_wr_reg_mask(struct m88ds3103_dev *dev,
|
|
||||||
u8 reg, u8 val, u8 mask)
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
u8 u8tmp;
|
|
||||||
|
|
||||||
/* no need for read if whole reg is written */
|
|
||||||
if (mask != 0xff) {
|
|
||||||
ret = m88ds3103_rd_regs(dev, reg, &u8tmp, 1);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
val &= mask;
|
|
||||||
u8tmp &= ~mask;
|
|
||||||
val |= u8tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
return m88ds3103_wr_regs(dev, reg, &val, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* read single register with mask */
|
|
||||||
static int m88ds3103_rd_reg_mask(struct m88ds3103_dev *dev,
|
|
||||||
u8 reg, u8 *val, u8 mask)
|
|
||||||
{
|
|
||||||
int ret, i;
|
|
||||||
u8 u8tmp;
|
|
||||||
|
|
||||||
ret = m88ds3103_rd_regs(dev, reg, &u8tmp, 1);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
u8tmp &= mask;
|
|
||||||
|
|
||||||
/* find position of the first bit */
|
|
||||||
for (i = 0; i < 8; i++) {
|
|
||||||
if ((mask >> i) & 0x01)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
*val = u8tmp >> i;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* write reg val table using reg addr auto increment */
|
/* write reg val table using reg addr auto increment */
|
||||||
static int m88ds3103_wr_reg_val_tab(struct m88ds3103_dev *dev,
|
static int m88ds3103_wr_reg_val_tab(struct m88ds3103_dev *dev,
|
||||||
const struct m88ds3103_reg_val *tab, int tab_len)
|
const struct m88ds3103_reg_val *tab, int tab_len)
|
||||||
|
@ -173,7 +38,7 @@ static int m88ds3103_wr_reg_val_tab(struct m88ds3103_dev *dev,
|
||||||
|
|
||||||
if (i == tab_len - 1 || tab[i].reg != tab[i + 1].reg - 1 ||
|
if (i == tab_len - 1 || tab[i].reg != tab[i + 1].reg - 1 ||
|
||||||
!((j + 1) % (dev->cfg->i2c_wr_max - 1))) {
|
!((j + 1) % (dev->cfg->i2c_wr_max - 1))) {
|
||||||
ret = m88ds3103_wr_regs(dev, tab[i].reg - j, buf, j + 1);
|
ret = regmap_bulk_write(dev->regmap, tab[i].reg - j, buf, j + 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -194,7 +59,7 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
struct dtv_frontend_properties *c = &fe->dtv_property_cache;
|
struct dtv_frontend_properties *c = &fe->dtv_property_cache;
|
||||||
int ret, i, itmp;
|
int ret, i, itmp;
|
||||||
u8 u8tmp;
|
unsigned int utmp;
|
||||||
u8 buf[3];
|
u8 buf[3];
|
||||||
|
|
||||||
*status = 0;
|
*status = 0;
|
||||||
|
@ -206,21 +71,21 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
|
|
||||||
switch (c->delivery_system) {
|
switch (c->delivery_system) {
|
||||||
case SYS_DVBS:
|
case SYS_DVBS:
|
||||||
ret = m88ds3103_rd_reg_mask(dev, 0xd1, &u8tmp, 0x07);
|
ret = regmap_read(dev->regmap, 0xd1, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
if (u8tmp == 0x07)
|
if ((utmp & 0x07) == 0x07)
|
||||||
*status = FE_HAS_SIGNAL | FE_HAS_CARRIER |
|
*status = FE_HAS_SIGNAL | FE_HAS_CARRIER |
|
||||||
FE_HAS_VITERBI | FE_HAS_SYNC |
|
FE_HAS_VITERBI | FE_HAS_SYNC |
|
||||||
FE_HAS_LOCK;
|
FE_HAS_LOCK;
|
||||||
break;
|
break;
|
||||||
case SYS_DVBS2:
|
case SYS_DVBS2:
|
||||||
ret = m88ds3103_rd_reg_mask(dev, 0x0d, &u8tmp, 0x8f);
|
ret = regmap_read(dev->regmap, 0x0d, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
if (u8tmp == 0x8f)
|
if ((utmp & 0x8f) == 0x8f)
|
||||||
*status = FE_HAS_SIGNAL | FE_HAS_CARRIER |
|
*status = FE_HAS_SIGNAL | FE_HAS_CARRIER |
|
||||||
FE_HAS_VITERBI | FE_HAS_SYNC |
|
FE_HAS_VITERBI | FE_HAS_SYNC |
|
||||||
FE_HAS_LOCK;
|
FE_HAS_LOCK;
|
||||||
|
@ -232,8 +97,7 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
}
|
}
|
||||||
|
|
||||||
dev->fe_status = *status;
|
dev->fe_status = *status;
|
||||||
|
dev_dbg(&client->dev, "lock=%02x status=%02x\n", utmp, *status);
|
||||||
dev_dbg(&client->dev, "lock=%02x status=%02x\n", u8tmp, *status);
|
|
||||||
|
|
||||||
/* CNR */
|
/* CNR */
|
||||||
if (dev->fe_status & FE_HAS_VITERBI) {
|
if (dev->fe_status & FE_HAS_VITERBI) {
|
||||||
|
@ -248,11 +112,11 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
itmp = 0;
|
itmp = 0;
|
||||||
|
|
||||||
for (i = 0; i < M88DS3103_SNR_ITERATIONS; i++) {
|
for (i = 0; i < M88DS3103_SNR_ITERATIONS; i++) {
|
||||||
ret = m88ds3103_rd_reg(dev, 0xff, &buf[0]);
|
ret = regmap_read(dev->regmap, 0xff, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
itmp += buf[0];
|
itmp += utmp;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* use of single register limits max value to 15 dB */
|
/* use of single register limits max value to 15 dB */
|
||||||
|
@ -266,7 +130,7 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
signal_tot = 0;
|
signal_tot = 0;
|
||||||
|
|
||||||
for (i = 0; i < M88DS3103_SNR_ITERATIONS; i++) {
|
for (i = 0; i < M88DS3103_SNR_ITERATIONS; i++) {
|
||||||
ret = m88ds3103_rd_regs(dev, 0x8c, buf, 3);
|
ret = regmap_bulk_read(dev->regmap, 0x8c, buf, 3);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -311,17 +175,17 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
|
|
||||||
switch (c->delivery_system) {
|
switch (c->delivery_system) {
|
||||||
case SYS_DVBS:
|
case SYS_DVBS:
|
||||||
ret = m88ds3103_wr_reg(dev, 0xf9, 0x04);
|
ret = regmap_write(dev->regmap, 0xf9, 0x04);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_rd_reg(dev, 0xf8, &u8tmp);
|
ret = regmap_read(dev->regmap, 0xf8, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
/* measurement ready? */
|
/* measurement ready? */
|
||||||
if (!(u8tmp & 0x10)) {
|
if (!(utmp & 0x10)) {
|
||||||
ret = m88ds3103_rd_regs(dev, 0xf6, buf, 2);
|
ret = regmap_bulk_read(dev->regmap, 0xf6, buf, 2);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -332,14 +196,14 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
dev->dvbv3_ber = post_bit_error;
|
dev->dvbv3_ber = post_bit_error;
|
||||||
|
|
||||||
/* restart measurement */
|
/* restart measurement */
|
||||||
u8tmp |= 0x10;
|
utmp |= 0x10;
|
||||||
ret = m88ds3103_wr_reg(dev, 0xf8, u8tmp);
|
ret = regmap_write(dev->regmap, 0xf8, utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case SYS_DVBS2:
|
case SYS_DVBS2:
|
||||||
ret = m88ds3103_rd_regs(dev, 0xd5, buf, 3);
|
ret = regmap_bulk_read(dev->regmap, 0xd5, buf, 3);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -347,7 +211,7 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
|
|
||||||
/* enough data? */
|
/* enough data? */
|
||||||
if (utmp > 4000) {
|
if (utmp > 4000) {
|
||||||
ret = m88ds3103_rd_regs(dev, 0xf7, buf, 2);
|
ret = regmap_bulk_read(dev->regmap, 0xf7, buf, 2);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -358,19 +222,19 @@ static int m88ds3103_read_status(struct dvb_frontend *fe,
|
||||||
dev->dvbv3_ber = post_bit_error;
|
dev->dvbv3_ber = post_bit_error;
|
||||||
|
|
||||||
/* restart measurement */
|
/* restart measurement */
|
||||||
ret = m88ds3103_wr_reg(dev, 0xd1, 0x01);
|
ret = regmap_write(dev->regmap, 0xd1, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xf9, 0x01);
|
ret = regmap_write(dev->regmap, 0xf9, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xf9, 0x00);
|
ret = regmap_write(dev->regmap, 0xf9, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xd1, 0x00);
|
ret = regmap_write(dev->regmap, 0xd1, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
@ -420,17 +284,17 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* reset */
|
/* reset */
|
||||||
ret = m88ds3103_wr_reg(dev, 0x07, 0x80);
|
ret = regmap_write(dev->regmap, 0x07, 0x80);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0x07, 0x00);
|
ret = regmap_write(dev->regmap, 0x07, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
/* Disable demod clock path */
|
/* Disable demod clock path */
|
||||||
if (dev->chip_id == M88RS6000_CHIP_ID) {
|
if (dev->chip_id == M88RS6000_CHIP_ID) {
|
||||||
ret = m88ds3103_wr_reg(dev, 0x06, 0xe0);
|
ret = regmap_write(dev->regmap, 0x06, 0xe0);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
@ -468,7 +332,7 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
target_mclk = 144000;
|
target_mclk = 144000;
|
||||||
|
|
||||||
/* Enable demod clock path */
|
/* Enable demod clock path */
|
||||||
ret = m88ds3103_wr_reg(dev, 0x06, 0x00);
|
ret = regmap_write(dev->regmap, 0x06, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
usleep_range(10000, 20000);
|
usleep_range(10000, 20000);
|
||||||
|
@ -514,19 +378,19 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
u8tmp2 = 0x00; /* 0b00 */
|
u8tmp2 = 0x00; /* 0b00 */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x22, u8tmp1 << 6, 0xc0);
|
ret = regmap_update_bits(dev->regmap, 0x22, 0xc0, u8tmp1 << 6);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x24, u8tmp2 << 6, 0xc0);
|
ret = regmap_update_bits(dev->regmap, 0x24, 0xc0, u8tmp2 << 6);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xb2, 0x01);
|
ret = regmap_write(dev->regmap, 0xb2, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0x00, 0x01);
|
ret = regmap_write(dev->regmap, 0x00, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -565,23 +429,23 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
if (dev->chip_id == M88RS6000_CHIP_ID) {
|
if (dev->chip_id == M88RS6000_CHIP_ID) {
|
||||||
if ((c->delivery_system == SYS_DVBS2)
|
if ((c->delivery_system == SYS_DVBS2)
|
||||||
&& ((c->symbol_rate / 1000) <= 5000)) {
|
&& ((c->symbol_rate / 1000) <= 5000)) {
|
||||||
ret = m88ds3103_wr_reg(dev, 0xc0, 0x04);
|
ret = regmap_write(dev->regmap, 0xc0, 0x04);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
buf[0] = 0x09;
|
buf[0] = 0x09;
|
||||||
buf[1] = 0x22;
|
buf[1] = 0x22;
|
||||||
buf[2] = 0x88;
|
buf[2] = 0x88;
|
||||||
ret = m88ds3103_wr_regs(dev, 0x8a, buf, 3);
|
ret = regmap_bulk_write(dev->regmap, 0x8a, buf, 3);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x9d, 0x08, 0x08);
|
ret = regmap_update_bits(dev->regmap, 0x9d, 0x08, 0x08);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
ret = m88ds3103_wr_reg(dev, 0xf1, 0x01);
|
ret = regmap_write(dev->regmap, 0xf1, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x30, 0x80, 0x80);
|
ret = regmap_update_bits(dev->regmap, 0x30, 0x80, 0x80);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
@ -611,14 +475,14 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
u8tmp |= 0x40;
|
u8tmp |= 0x40;
|
||||||
|
|
||||||
/* TS mode */
|
/* TS mode */
|
||||||
ret = m88ds3103_wr_reg(dev, 0xfd, u8tmp);
|
ret = regmap_write(dev->regmap, 0xfd, u8tmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
switch (dev->cfg->ts_mode) {
|
switch (dev->cfg->ts_mode) {
|
||||||
case M88DS3103_TS_SERIAL:
|
case M88DS3103_TS_SERIAL:
|
||||||
case M88DS3103_TS_SERIAL_D7:
|
case M88DS3103_TS_SERIAL_D7:
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x29, u8tmp1, 0x20);
|
ret = regmap_update_bits(dev->regmap, 0x29, 0x20, u8tmp1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
u8tmp1 = 0;
|
u8tmp1 = 0;
|
||||||
|
@ -643,17 +507,17 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
/* u8tmp2[5:0] => ea[5:0] */
|
/* u8tmp2[5:0] => ea[5:0] */
|
||||||
u8tmp2 &= 0x3f;
|
u8tmp2 &= 0x3f;
|
||||||
|
|
||||||
ret = m88ds3103_rd_reg(dev, 0xfe, &u8tmp);
|
ret = regmap_bulk_read(dev->regmap, 0xfe, &u8tmp, 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
u8tmp = ((u8tmp & 0xf0) << 0) | u8tmp1 >> 2;
|
u8tmp = ((u8tmp & 0xf0) << 0) | u8tmp1 >> 2;
|
||||||
ret = m88ds3103_wr_reg(dev, 0xfe, u8tmp);
|
ret = regmap_write(dev->regmap, 0xfe, u8tmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
u8tmp = ((u8tmp1 & 0x03) << 6) | u8tmp2 >> 0;
|
u8tmp = ((u8tmp1 & 0x03) << 6) | u8tmp2 >> 0;
|
||||||
ret = m88ds3103_wr_reg(dev, 0xea, u8tmp);
|
ret = regmap_write(dev->regmap, 0xea, u8tmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -664,38 +528,38 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
else
|
else
|
||||||
u8tmp = 0x06;
|
u8tmp = 0x06;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xc3, 0x08);
|
ret = regmap_write(dev->regmap, 0xc3, 0x08);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xc8, u8tmp);
|
ret = regmap_write(dev->regmap, 0xc8, u8tmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xc4, 0x08);
|
ret = regmap_write(dev->regmap, 0xc4, 0x08);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xc7, 0x00);
|
ret = regmap_write(dev->regmap, 0xc7, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
u16tmp = DIV_ROUND_CLOSEST((c->symbol_rate / 1000) << 15, dev->mclk_khz / 2);
|
u16tmp = DIV_ROUND_CLOSEST((c->symbol_rate / 1000) << 15, dev->mclk_khz / 2);
|
||||||
buf[0] = (u16tmp >> 0) & 0xff;
|
buf[0] = (u16tmp >> 0) & 0xff;
|
||||||
buf[1] = (u16tmp >> 8) & 0xff;
|
buf[1] = (u16tmp >> 8) & 0xff;
|
||||||
ret = m88ds3103_wr_regs(dev, 0x61, buf, 2);
|
ret = regmap_bulk_write(dev->regmap, 0x61, buf, 2);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x4d, dev->cfg->spec_inv << 1, 0x02);
|
ret = regmap_update_bits(dev->regmap, 0x4d, 0x02, dev->cfg->spec_inv << 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x30, dev->cfg->agc_inv << 4, 0x10);
|
ret = regmap_update_bits(dev->regmap, 0x30, 0x10, dev->cfg->agc_inv << 4);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0x33, dev->cfg->agc);
|
ret = regmap_write(dev->regmap, 0x33, dev->cfg->agc);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -709,15 +573,15 @@ static int m88ds3103_set_frontend(struct dvb_frontend *fe)
|
||||||
|
|
||||||
buf[0] = (s32tmp >> 0) & 0xff;
|
buf[0] = (s32tmp >> 0) & 0xff;
|
||||||
buf[1] = (s32tmp >> 8) & 0xff;
|
buf[1] = (s32tmp >> 8) & 0xff;
|
||||||
ret = m88ds3103_wr_regs(dev, 0x5e, buf, 2);
|
ret = regmap_bulk_write(dev->regmap, 0x5e, buf, 2);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0x00, 0x00);
|
ret = regmap_write(dev->regmap, 0x00, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xb2, 0x00);
|
ret = regmap_write(dev->regmap, 0xb2, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -735,9 +599,9 @@ static int m88ds3103_init(struct dvb_frontend *fe)
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
struct dtv_frontend_properties *c = &fe->dtv_property_cache;
|
struct dtv_frontend_properties *c = &fe->dtv_property_cache;
|
||||||
int ret, len, remaining;
|
int ret, len, remaining;
|
||||||
|
unsigned int utmp;
|
||||||
const struct firmware *fw = NULL;
|
const struct firmware *fw = NULL;
|
||||||
u8 *fw_file;
|
u8 *fw_file;
|
||||||
u8 u8tmp;
|
|
||||||
|
|
||||||
dev_dbg(&client->dev, "\n");
|
dev_dbg(&client->dev, "\n");
|
||||||
|
|
||||||
|
@ -745,34 +609,31 @@ static int m88ds3103_init(struct dvb_frontend *fe)
|
||||||
dev->warm = false;
|
dev->warm = false;
|
||||||
|
|
||||||
/* wake up device from sleep */
|
/* wake up device from sleep */
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x08, 0x01, 0x01);
|
ret = regmap_update_bits(dev->regmap, 0x08, 0x01, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
ret = regmap_update_bits(dev->regmap, 0x04, 0x01, 0x00);
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x04, 0x00, 0x01);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
ret = regmap_update_bits(dev->regmap, 0x23, 0x10, 0x00);
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x23, 0x00, 0x10);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
/* firmware status */
|
/* firmware status */
|
||||||
ret = m88ds3103_rd_reg(dev, 0xb9, &u8tmp);
|
ret = regmap_read(dev->regmap, 0xb9, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
dev_dbg(&client->dev, "firmware=%02x\n", u8tmp);
|
dev_dbg(&client->dev, "firmware=%02x\n", utmp);
|
||||||
|
|
||||||
if (u8tmp)
|
if (utmp)
|
||||||
goto skip_fw_download;
|
goto skip_fw_download;
|
||||||
|
|
||||||
/* global reset, global diseqc reset, golbal fec reset */
|
/* global reset, global diseqc reset, golbal fec reset */
|
||||||
ret = m88ds3103_wr_reg(dev, 0x07, 0xe0);
|
ret = regmap_write(dev->regmap, 0x07, 0xe0);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
ret = regmap_write(dev->regmap, 0x07, 0x00);
|
||||||
ret = m88ds3103_wr_reg(dev, 0x07, 0x00);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -794,7 +655,7 @@ static int m88ds3103_init(struct dvb_frontend *fe)
|
||||||
dev_info(&client->dev, "downloading firmware from file '%s'\n",
|
dev_info(&client->dev, "downloading firmware from file '%s'\n",
|
||||||
fw_file);
|
fw_file);
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xb2, 0x01);
|
ret = regmap_write(dev->regmap, 0xb2, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_fw_release;
|
goto error_fw_release;
|
||||||
|
|
||||||
|
@ -804,7 +665,7 @@ static int m88ds3103_init(struct dvb_frontend *fe)
|
||||||
if (len > (dev->cfg->i2c_wr_max - 1))
|
if (len > (dev->cfg->i2c_wr_max - 1))
|
||||||
len = (dev->cfg->i2c_wr_max - 1);
|
len = (dev->cfg->i2c_wr_max - 1);
|
||||||
|
|
||||||
ret = m88ds3103_wr_regs(dev, 0xb0,
|
ret = regmap_bulk_write(dev->regmap, 0xb0,
|
||||||
&fw->data[fw->size - remaining], len);
|
&fw->data[fw->size - remaining], len);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(&client->dev, "firmware download failed=%d\n",
|
dev_err(&client->dev, "firmware download failed=%d\n",
|
||||||
|
@ -813,18 +674,18 @@ static int m88ds3103_init(struct dvb_frontend *fe)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xb2, 0x00);
|
ret = regmap_write(dev->regmap, 0xb2, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_fw_release;
|
goto error_fw_release;
|
||||||
|
|
||||||
release_firmware(fw);
|
release_firmware(fw);
|
||||||
fw = NULL;
|
fw = NULL;
|
||||||
|
|
||||||
ret = m88ds3103_rd_reg(dev, 0xb9, &u8tmp);
|
ret = regmap_read(dev->regmap, 0xb9, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
if (!u8tmp) {
|
if (!utmp) {
|
||||||
dev_info(&client->dev, "firmware did not run\n");
|
dev_info(&client->dev, "firmware did not run\n");
|
||||||
ret = -EFAULT;
|
ret = -EFAULT;
|
||||||
goto err;
|
goto err;
|
||||||
|
@ -833,7 +694,7 @@ static int m88ds3103_init(struct dvb_frontend *fe)
|
||||||
dev_info(&client->dev, "found a '%s' in warm state\n",
|
dev_info(&client->dev, "found a '%s' in warm state\n",
|
||||||
m88ds3103_ops.info.name);
|
m88ds3103_ops.info.name);
|
||||||
dev_info(&client->dev, "firmware version: %X.%X\n",
|
dev_info(&client->dev, "firmware version: %X.%X\n",
|
||||||
(u8tmp >> 4) & 0xf, (u8tmp >> 0 & 0xf));
|
(utmp >> 4) & 0xf, (utmp >> 0 & 0xf));
|
||||||
|
|
||||||
skip_fw_download:
|
skip_fw_download:
|
||||||
/* warm state */
|
/* warm state */
|
||||||
|
@ -860,7 +721,7 @@ static int m88ds3103_sleep(struct dvb_frontend *fe)
|
||||||
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
int ret;
|
int ret;
|
||||||
u8 u8tmp;
|
unsigned int utmp;
|
||||||
|
|
||||||
dev_dbg(&client->dev, "\n");
|
dev_dbg(&client->dev, "\n");
|
||||||
|
|
||||||
|
@ -869,23 +730,21 @@ static int m88ds3103_sleep(struct dvb_frontend *fe)
|
||||||
|
|
||||||
/* TS Hi-Z */
|
/* TS Hi-Z */
|
||||||
if (dev->chip_id == M88RS6000_CHIP_ID)
|
if (dev->chip_id == M88RS6000_CHIP_ID)
|
||||||
u8tmp = 0x29;
|
utmp = 0x29;
|
||||||
else
|
else
|
||||||
u8tmp = 0x27;
|
utmp = 0x27;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, u8tmp, 0x00, 0x01);
|
ret = regmap_update_bits(dev->regmap, utmp, 0x01, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
/* sleep */
|
/* sleep */
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x08, 0x00, 0x01);
|
ret = regmap_update_bits(dev->regmap, 0x08, 0x01, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
ret = regmap_update_bits(dev->regmap, 0x04, 0x01, 0x01);
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x04, 0x01, 0x01);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
ret = regmap_update_bits(dev->regmap, 0x23, 0x10, 0x10);
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x23, 0x10, 0x10);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -912,11 +771,11 @@ static int m88ds3103_get_frontend(struct dvb_frontend *fe)
|
||||||
|
|
||||||
switch (c->delivery_system) {
|
switch (c->delivery_system) {
|
||||||
case SYS_DVBS:
|
case SYS_DVBS:
|
||||||
ret = m88ds3103_rd_reg(dev, 0xe0, &buf[0]);
|
ret = regmap_bulk_read(dev->regmap, 0xe0, &buf[0], 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_rd_reg(dev, 0xe6, &buf[1]);
|
ret = regmap_bulk_read(dev->regmap, 0xe6, &buf[1], 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -953,15 +812,15 @@ static int m88ds3103_get_frontend(struct dvb_frontend *fe)
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case SYS_DVBS2:
|
case SYS_DVBS2:
|
||||||
ret = m88ds3103_rd_reg(dev, 0x7e, &buf[0]);
|
ret = regmap_bulk_read(dev->regmap, 0x7e, &buf[0], 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_rd_reg(dev, 0x89, &buf[1]);
|
ret = regmap_bulk_read(dev->regmap, 0x89, &buf[1], 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_rd_reg(dev, 0xf2, &buf[2]);
|
ret = regmap_bulk_read(dev->regmap, 0xf2, &buf[2], 1);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -1052,7 +911,7 @@ static int m88ds3103_get_frontend(struct dvb_frontend *fe)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = m88ds3103_rd_regs(dev, 0x6d, buf, 2);
|
ret = regmap_bulk_read(dev->regmap, 0x6d, buf, 2);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -1092,7 +951,7 @@ static int m88ds3103_set_tone(struct dvb_frontend *fe,
|
||||||
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
int ret;
|
int ret;
|
||||||
u8 u8tmp, tone, reg_a1_mask;
|
unsigned int utmp, tone, reg_a1_mask;
|
||||||
|
|
||||||
dev_dbg(&client->dev, "fe_sec_tone_mode=%d\n", fe_sec_tone_mode);
|
dev_dbg(&client->dev, "fe_sec_tone_mode=%d\n", fe_sec_tone_mode);
|
||||||
|
|
||||||
|
@ -1116,13 +975,13 @@ static int m88ds3103_set_tone(struct dvb_frontend *fe,
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
u8tmp = tone << 7 | dev->cfg->envelope_mode << 5;
|
utmp = tone << 7 | dev->cfg->envelope_mode << 5;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa2, u8tmp, 0xe0);
|
ret = regmap_update_bits(dev->regmap, 0xa2, 0xe0, utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
u8tmp = 1 << 2;
|
utmp = 1 << 2;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa1, u8tmp, reg_a1_mask);
|
ret = regmap_update_bits(dev->regmap, 0xa1, reg_a1_mask, utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -1138,7 +997,7 @@ static int m88ds3103_set_voltage(struct dvb_frontend *fe,
|
||||||
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
int ret;
|
int ret;
|
||||||
u8 u8tmp;
|
unsigned int utmp;
|
||||||
bool voltage_sel, voltage_dis;
|
bool voltage_sel, voltage_dis;
|
||||||
|
|
||||||
dev_dbg(&client->dev, "fe_sec_voltage=%d\n", fe_sec_voltage);
|
dev_dbg(&client->dev, "fe_sec_voltage=%d\n", fe_sec_voltage);
|
||||||
|
@ -1171,8 +1030,8 @@ static int m88ds3103_set_voltage(struct dvb_frontend *fe,
|
||||||
voltage_sel ^= dev->cfg->lnb_hv_pol;
|
voltage_sel ^= dev->cfg->lnb_hv_pol;
|
||||||
voltage_dis ^= dev->cfg->lnb_en_pol;
|
voltage_dis ^= dev->cfg->lnb_en_pol;
|
||||||
|
|
||||||
u8tmp = voltage_dis << 1 | voltage_sel << 0;
|
utmp = voltage_dis << 1 | voltage_sel << 0;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa2, u8tmp, 0x03);
|
ret = regmap_update_bits(dev->regmap, 0xa2, 0x03, utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -1188,8 +1047,8 @@ static int m88ds3103_diseqc_send_master_cmd(struct dvb_frontend *fe,
|
||||||
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
int ret;
|
int ret;
|
||||||
|
unsigned int utmp;
|
||||||
unsigned long timeout;
|
unsigned long timeout;
|
||||||
u8 u8tmp;
|
|
||||||
|
|
||||||
dev_dbg(&client->dev, "msg=%*ph\n",
|
dev_dbg(&client->dev, "msg=%*ph\n",
|
||||||
diseqc_cmd->msg_len, diseqc_cmd->msg);
|
diseqc_cmd->msg_len, diseqc_cmd->msg);
|
||||||
|
@ -1204,17 +1063,17 @@ static int m88ds3103_diseqc_send_master_cmd(struct dvb_frontend *fe,
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
u8tmp = dev->cfg->envelope_mode << 5;
|
utmp = dev->cfg->envelope_mode << 5;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa2, u8tmp, 0xe0);
|
ret = regmap_update_bits(dev->regmap, 0xa2, 0xe0, utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_regs(dev, 0xa3, diseqc_cmd->msg,
|
ret = regmap_bulk_write(dev->regmap, 0xa3, diseqc_cmd->msg,
|
||||||
diseqc_cmd->msg_len);
|
diseqc_cmd->msg_len);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xa1,
|
ret = regmap_write(dev->regmap, 0xa1,
|
||||||
(diseqc_cmd->msg_len - 1) << 3 | 0x07);
|
(diseqc_cmd->msg_len - 1) << 3 | 0x07);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
@ -1226,29 +1085,30 @@ static int m88ds3103_diseqc_send_master_cmd(struct dvb_frontend *fe,
|
||||||
/* DiSEqC message typical period is 54 ms */
|
/* DiSEqC message typical period is 54 ms */
|
||||||
usleep_range(50000, 54000);
|
usleep_range(50000, 54000);
|
||||||
|
|
||||||
for (u8tmp = 1; !time_after(jiffies, timeout) && u8tmp;) {
|
for (utmp = 1; !time_after(jiffies, timeout) && utmp;) {
|
||||||
ret = m88ds3103_rd_reg_mask(dev, 0xa1, &u8tmp, 0x40);
|
ret = regmap_read(dev->regmap, 0xa1, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
utmp = (utmp >> 6) & 0x1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (u8tmp == 0) {
|
if (utmp == 0) {
|
||||||
dev_dbg(&client->dev, "diseqc tx took %u ms\n",
|
dev_dbg(&client->dev, "diseqc tx took %u ms\n",
|
||||||
jiffies_to_msecs(jiffies) -
|
jiffies_to_msecs(jiffies) -
|
||||||
(jiffies_to_msecs(timeout) - SEND_MASTER_CMD_TIMEOUT));
|
(jiffies_to_msecs(timeout) - SEND_MASTER_CMD_TIMEOUT));
|
||||||
} else {
|
} else {
|
||||||
dev_dbg(&client->dev, "diseqc tx timeout\n");
|
dev_dbg(&client->dev, "diseqc tx timeout\n");
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa1, 0x40, 0xc0);
|
ret = regmap_update_bits(dev->regmap, 0xa1, 0xc0, 0x40);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa2, 0x80, 0xc0);
|
ret = regmap_update_bits(dev->regmap, 0xa2, 0xc0, 0x80);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
if (u8tmp == 1) {
|
if (utmp == 1) {
|
||||||
ret = -ETIMEDOUT;
|
ret = -ETIMEDOUT;
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
@ -1265,8 +1125,8 @@ static int m88ds3103_diseqc_send_burst(struct dvb_frontend *fe,
|
||||||
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
struct m88ds3103_dev *dev = fe->demodulator_priv;
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
int ret;
|
int ret;
|
||||||
|
unsigned int utmp, burst;
|
||||||
unsigned long timeout;
|
unsigned long timeout;
|
||||||
u8 u8tmp, burst;
|
|
||||||
|
|
||||||
dev_dbg(&client->dev, "fe_sec_mini_cmd=%d\n", fe_sec_mini_cmd);
|
dev_dbg(&client->dev, "fe_sec_mini_cmd=%d\n", fe_sec_mini_cmd);
|
||||||
|
|
||||||
|
@ -1275,8 +1135,8 @@ static int m88ds3103_diseqc_send_burst(struct dvb_frontend *fe,
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
u8tmp = dev->cfg->envelope_mode << 5;
|
utmp = dev->cfg->envelope_mode << 5;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa2, u8tmp, 0xe0);
|
ret = regmap_update_bits(dev->regmap, 0xa2, 0xe0, utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -1293,7 +1153,7 @@ static int m88ds3103_diseqc_send_burst(struct dvb_frontend *fe,
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0xa1, burst);
|
ret = regmap_write(dev->regmap, 0xa1, burst);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -1304,29 +1164,30 @@ static int m88ds3103_diseqc_send_burst(struct dvb_frontend *fe,
|
||||||
/* DiSEqC ToneBurst period is 12.5 ms */
|
/* DiSEqC ToneBurst period is 12.5 ms */
|
||||||
usleep_range(8500, 12500);
|
usleep_range(8500, 12500);
|
||||||
|
|
||||||
for (u8tmp = 1; !time_after(jiffies, timeout) && u8tmp;) {
|
for (utmp = 1; !time_after(jiffies, timeout) && utmp;) {
|
||||||
ret = m88ds3103_rd_reg_mask(dev, 0xa1, &u8tmp, 0x40);
|
ret = regmap_read(dev->regmap, 0xa1, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
utmp = (utmp >> 6) & 0x1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (u8tmp == 0) {
|
if (utmp == 0) {
|
||||||
dev_dbg(&client->dev, "diseqc tx took %u ms\n",
|
dev_dbg(&client->dev, "diseqc tx took %u ms\n",
|
||||||
jiffies_to_msecs(jiffies) -
|
jiffies_to_msecs(jiffies) -
|
||||||
(jiffies_to_msecs(timeout) - SEND_BURST_TIMEOUT));
|
(jiffies_to_msecs(timeout) - SEND_BURST_TIMEOUT));
|
||||||
} else {
|
} else {
|
||||||
dev_dbg(&client->dev, "diseqc tx timeout\n");
|
dev_dbg(&client->dev, "diseqc tx timeout\n");
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa1, 0x40, 0xc0);
|
ret = regmap_update_bits(dev->regmap, 0xa1, 0xc0, 0x40);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0xa2, 0x80, 0xc0);
|
ret = regmap_update_bits(dev->regmap, 0xa2, 0xc0, 0x80);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
if (u8tmp == 1) {
|
if (utmp == 1) {
|
||||||
ret = -ETIMEDOUT;
|
ret = -ETIMEDOUT;
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
@ -1358,40 +1219,25 @@ static int m88ds3103_select(struct i2c_adapter *adap, void *mux_priv, u32 chan)
|
||||||
struct m88ds3103_dev *dev = mux_priv;
|
struct m88ds3103_dev *dev = mux_priv;
|
||||||
struct i2c_client *client = dev->client;
|
struct i2c_client *client = dev->client;
|
||||||
int ret;
|
int ret;
|
||||||
struct i2c_msg gate_open_msg[1] = {
|
struct i2c_msg msg = {
|
||||||
{
|
.addr = client->addr,
|
||||||
.addr = client->addr,
|
.flags = 0,
|
||||||
.flags = 0,
|
.len = 2,
|
||||||
.len = 2,
|
.buf = "\x03\x11",
|
||||||
.buf = "\x03\x11",
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
mutex_lock(&dev->i2c_mutex);
|
/* Open tuner I2C repeater for 1 xfer, closes automatically */
|
||||||
|
ret = __i2c_transfer(client->adapter, &msg, 1);
|
||||||
/* open tuner I2C repeater for 1 xfer, closes automatically */
|
|
||||||
ret = __i2c_transfer(client->adapter, gate_open_msg, 1);
|
|
||||||
if (ret != 1) {
|
if (ret != 1) {
|
||||||
dev_warn(&client->dev, "i2c wr failed=%d\n", ret);
|
dev_warn(&client->dev, "i2c wr failed=%d\n", ret);
|
||||||
if (ret >= 0)
|
if (ret >= 0)
|
||||||
ret = -EREMOTEIO;
|
ret = -EREMOTEIO;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int m88ds3103_deselect(struct i2c_adapter *adap, void *mux_priv,
|
|
||||||
u32 chan)
|
|
||||||
{
|
|
||||||
struct m88ds3103_dev *dev = mux_priv;
|
|
||||||
|
|
||||||
mutex_unlock(&dev->i2c_mutex);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* XXX: That is wrapper to m88ds3103_probe() via driver core in order to provide
|
* XXX: That is wrapper to m88ds3103_probe() via driver core in order to provide
|
||||||
* proper I2C client for legacy media attach binding.
|
* proper I2C client for legacy media attach binding.
|
||||||
|
@ -1500,7 +1346,7 @@ static int m88ds3103_probe(struct i2c_client *client,
|
||||||
struct m88ds3103_dev *dev;
|
struct m88ds3103_dev *dev;
|
||||||
struct m88ds3103_platform_data *pdata = client->dev.platform_data;
|
struct m88ds3103_platform_data *pdata = client->dev.platform_data;
|
||||||
int ret;
|
int ret;
|
||||||
u8 chip_id, u8tmp;
|
unsigned int utmp;
|
||||||
|
|
||||||
dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
||||||
if (!dev) {
|
if (!dev) {
|
||||||
|
@ -1522,34 +1368,41 @@ static int m88ds3103_probe(struct i2c_client *client,
|
||||||
dev->config.lnb_hv_pol = pdata->lnb_hv_pol;
|
dev->config.lnb_hv_pol = pdata->lnb_hv_pol;
|
||||||
dev->config.lnb_en_pol = pdata->lnb_en_pol;
|
dev->config.lnb_en_pol = pdata->lnb_en_pol;
|
||||||
dev->cfg = &dev->config;
|
dev->cfg = &dev->config;
|
||||||
mutex_init(&dev->i2c_mutex);
|
/* create regmap */
|
||||||
|
dev->regmap_config.reg_bits = 8,
|
||||||
|
dev->regmap_config.val_bits = 8,
|
||||||
|
dev->regmap_config.lock_arg = dev,
|
||||||
|
dev->regmap = devm_regmap_init_i2c(client, &dev->regmap_config);
|
||||||
|
if (IS_ERR(dev->regmap)) {
|
||||||
|
ret = PTR_ERR(dev->regmap);
|
||||||
|
goto err_kfree;
|
||||||
|
}
|
||||||
|
|
||||||
/* 0x00: chip id[6:0], 0x01: chip ver[7:0], 0x02: chip ver[15:8] */
|
/* 0x00: chip id[6:0], 0x01: chip ver[7:0], 0x02: chip ver[15:8] */
|
||||||
ret = m88ds3103_rd_reg(dev, 0x00, &chip_id);
|
ret = regmap_read(dev->regmap, 0x00, &utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_kfree;
|
goto err_kfree;
|
||||||
|
|
||||||
chip_id >>= 1;
|
dev->chip_id = utmp >> 1;
|
||||||
dev_dbg(&client->dev, "chip_id=%02x\n", chip_id);
|
dev_dbg(&client->dev, "chip_id=%02x\n", dev->chip_id);
|
||||||
|
|
||||||
switch (chip_id) {
|
switch (dev->chip_id) {
|
||||||
case M88RS6000_CHIP_ID:
|
case M88RS6000_CHIP_ID:
|
||||||
case M88DS3103_CHIP_ID:
|
case M88DS3103_CHIP_ID:
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
goto err_kfree;
|
goto err_kfree;
|
||||||
}
|
}
|
||||||
dev->chip_id = chip_id;
|
|
||||||
|
|
||||||
switch (dev->cfg->clock_out) {
|
switch (dev->cfg->clock_out) {
|
||||||
case M88DS3103_CLOCK_OUT_DISABLED:
|
case M88DS3103_CLOCK_OUT_DISABLED:
|
||||||
u8tmp = 0x80;
|
utmp = 0x80;
|
||||||
break;
|
break;
|
||||||
case M88DS3103_CLOCK_OUT_ENABLED:
|
case M88DS3103_CLOCK_OUT_ENABLED:
|
||||||
u8tmp = 0x00;
|
utmp = 0x00;
|
||||||
break;
|
break;
|
||||||
case M88DS3103_CLOCK_OUT_ENABLED_DIV2:
|
case M88DS3103_CLOCK_OUT_ENABLED_DIV2:
|
||||||
u8tmp = 0x10;
|
utmp = 0x10;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
|
@ -1558,28 +1411,28 @@ static int m88ds3103_probe(struct i2c_client *client,
|
||||||
|
|
||||||
/* 0x29 register is defined differently for m88rs6000. */
|
/* 0x29 register is defined differently for m88rs6000. */
|
||||||
/* set internal tuner address to 0x21 */
|
/* set internal tuner address to 0x21 */
|
||||||
if (chip_id == M88RS6000_CHIP_ID)
|
if (dev->chip_id == M88RS6000_CHIP_ID)
|
||||||
u8tmp = 0x00;
|
utmp = 0x00;
|
||||||
|
|
||||||
ret = m88ds3103_wr_reg(dev, 0x29, u8tmp);
|
ret = regmap_write(dev->regmap, 0x29, utmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_kfree;
|
goto err_kfree;
|
||||||
|
|
||||||
/* sleep */
|
/* sleep */
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x08, 0x00, 0x01);
|
ret = regmap_update_bits(dev->regmap, 0x08, 0x01, 0x00);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_kfree;
|
goto err_kfree;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x04, 0x01, 0x01);
|
ret = regmap_update_bits(dev->regmap, 0x04, 0x01, 0x01);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_kfree;
|
goto err_kfree;
|
||||||
ret = m88ds3103_wr_reg_mask(dev, 0x23, 0x10, 0x10);
|
ret = regmap_update_bits(dev->regmap, 0x23, 0x10, 0x10);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_kfree;
|
goto err_kfree;
|
||||||
|
|
||||||
/* create mux i2c adapter for tuner */
|
/* create mux i2c adapter for tuner */
|
||||||
dev->i2c_adapter = i2c_add_mux_adapter(client->adapter, &client->dev,
|
dev->i2c_adapter = i2c_add_mux_adapter(client->adapter, &client->dev,
|
||||||
dev, 0, 0, 0, m88ds3103_select,
|
dev, 0, 0, 0, m88ds3103_select,
|
||||||
m88ds3103_deselect);
|
NULL);
|
||||||
if (dev->i2c_adapter == NULL) {
|
if (dev->i2c_adapter == NULL) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto err_kfree;
|
goto err_kfree;
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
#include "dvb_math.h"
|
#include "dvb_math.h"
|
||||||
#include <linux/firmware.h>
|
#include <linux/firmware.h>
|
||||||
#include <linux/i2c-mux.h>
|
#include <linux/i2c-mux.h>
|
||||||
|
#include <linux/regmap.h>
|
||||||
#include <linux/math64.h>
|
#include <linux/math64.h>
|
||||||
|
|
||||||
#define M88DS3103_FIRMWARE "dvb-demod-m88ds3103.fw"
|
#define M88DS3103_FIRMWARE "dvb-demod-m88ds3103.fw"
|
||||||
|
@ -32,8 +33,8 @@
|
||||||
|
|
||||||
struct m88ds3103_dev {
|
struct m88ds3103_dev {
|
||||||
struct i2c_client *client;
|
struct i2c_client *client;
|
||||||
/* mutex needed due to own tuner I2C adapter */
|
struct regmap_config regmap_config;
|
||||||
struct mutex i2c_mutex;
|
struct regmap *regmap;
|
||||||
struct m88ds3103_config config;
|
struct m88ds3103_config config;
|
||||||
const struct m88ds3103_config *cfg;
|
const struct m88ds3103_config *cfg;
|
||||||
struct dvb_frontend fe;
|
struct dvb_frontend fe;
|
||||||
|
|
Loading…
Reference in New Issue