162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Realtek RTL2830 DVB-T demodulator driver
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2011 Antti Palosaari <crope@iki.fi>
662306a36Sopenharmony_ci */
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci#include "rtl2830_priv.h"
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci/* Our regmap is bypassing I2C adapter lock, thus we do it! */
1162306a36Sopenharmony_cistatic int rtl2830_bulk_write(struct i2c_client *client, unsigned int reg,
1262306a36Sopenharmony_ci			      const void *val, size_t val_count)
1362306a36Sopenharmony_ci{
1462306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
1562306a36Sopenharmony_ci	int ret;
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci	i2c_lock_bus(client->adapter, I2C_LOCK_SEGMENT);
1862306a36Sopenharmony_ci	ret = regmap_bulk_write(dev->regmap, reg, val, val_count);
1962306a36Sopenharmony_ci	i2c_unlock_bus(client->adapter, I2C_LOCK_SEGMENT);
2062306a36Sopenharmony_ci	return ret;
2162306a36Sopenharmony_ci}
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_cistatic int rtl2830_update_bits(struct i2c_client *client, unsigned int reg,
2462306a36Sopenharmony_ci			       unsigned int mask, unsigned int val)
2562306a36Sopenharmony_ci{
2662306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
2762306a36Sopenharmony_ci	int ret;
2862306a36Sopenharmony_ci
2962306a36Sopenharmony_ci	i2c_lock_bus(client->adapter, I2C_LOCK_SEGMENT);
3062306a36Sopenharmony_ci	ret = regmap_update_bits(dev->regmap, reg, mask, val);
3162306a36Sopenharmony_ci	i2c_unlock_bus(client->adapter, I2C_LOCK_SEGMENT);
3262306a36Sopenharmony_ci	return ret;
3362306a36Sopenharmony_ci}
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_cistatic int rtl2830_bulk_read(struct i2c_client *client, unsigned int reg,
3662306a36Sopenharmony_ci			     void *val, size_t val_count)
3762306a36Sopenharmony_ci{
3862306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
3962306a36Sopenharmony_ci	int ret;
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ci	i2c_lock_bus(client->adapter, I2C_LOCK_SEGMENT);
4262306a36Sopenharmony_ci	ret = regmap_bulk_read(dev->regmap, reg, val, val_count);
4362306a36Sopenharmony_ci	i2c_unlock_bus(client->adapter, I2C_LOCK_SEGMENT);
4462306a36Sopenharmony_ci	return ret;
4562306a36Sopenharmony_ci}
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_cistatic int rtl2830_init(struct dvb_frontend *fe)
4862306a36Sopenharmony_ci{
4962306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
5062306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
5162306a36Sopenharmony_ci	struct dtv_frontend_properties *c = &dev->fe.dtv_property_cache;
5262306a36Sopenharmony_ci	int ret, i;
5362306a36Sopenharmony_ci	struct rtl2830_reg_val_mask tab[] = {
5462306a36Sopenharmony_ci		{0x00d, 0x01, 0x03},
5562306a36Sopenharmony_ci		{0x00d, 0x10, 0x10},
5662306a36Sopenharmony_ci		{0x104, 0x00, 0x1e},
5762306a36Sopenharmony_ci		{0x105, 0x80, 0x80},
5862306a36Sopenharmony_ci		{0x110, 0x02, 0x03},
5962306a36Sopenharmony_ci		{0x110, 0x08, 0x0c},
6062306a36Sopenharmony_ci		{0x17b, 0x00, 0x40},
6162306a36Sopenharmony_ci		{0x17d, 0x05, 0x0f},
6262306a36Sopenharmony_ci		{0x17d, 0x50, 0xf0},
6362306a36Sopenharmony_ci		{0x18c, 0x08, 0x0f},
6462306a36Sopenharmony_ci		{0x18d, 0x00, 0xc0},
6562306a36Sopenharmony_ci		{0x188, 0x05, 0x0f},
6662306a36Sopenharmony_ci		{0x189, 0x00, 0xfc},
6762306a36Sopenharmony_ci		{0x2d5, 0x02, 0x02},
6862306a36Sopenharmony_ci		{0x2f1, 0x02, 0x06},
6962306a36Sopenharmony_ci		{0x2f1, 0x20, 0xf8},
7062306a36Sopenharmony_ci		{0x16d, 0x00, 0x01},
7162306a36Sopenharmony_ci		{0x1a6, 0x00, 0x80},
7262306a36Sopenharmony_ci		{0x106, dev->pdata->vtop, 0x3f},
7362306a36Sopenharmony_ci		{0x107, dev->pdata->krf, 0x3f},
7462306a36Sopenharmony_ci		{0x112, 0x28, 0xff},
7562306a36Sopenharmony_ci		{0x103, dev->pdata->agc_targ_val, 0xff},
7662306a36Sopenharmony_ci		{0x00a, 0x02, 0x07},
7762306a36Sopenharmony_ci		{0x140, 0x0c, 0x3c},
7862306a36Sopenharmony_ci		{0x140, 0x40, 0xc0},
7962306a36Sopenharmony_ci		{0x15b, 0x05, 0x07},
8062306a36Sopenharmony_ci		{0x15b, 0x28, 0x38},
8162306a36Sopenharmony_ci		{0x15c, 0x05, 0x07},
8262306a36Sopenharmony_ci		{0x15c, 0x28, 0x38},
8362306a36Sopenharmony_ci		{0x115, dev->pdata->spec_inv, 0x01},
8462306a36Sopenharmony_ci		{0x16f, 0x01, 0x07},
8562306a36Sopenharmony_ci		{0x170, 0x18, 0x38},
8662306a36Sopenharmony_ci		{0x172, 0x0f, 0x0f},
8762306a36Sopenharmony_ci		{0x173, 0x08, 0x38},
8862306a36Sopenharmony_ci		{0x175, 0x01, 0x07},
8962306a36Sopenharmony_ci		{0x176, 0x00, 0xc0},
9062306a36Sopenharmony_ci	};
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci	for (i = 0; i < ARRAY_SIZE(tab); i++) {
9362306a36Sopenharmony_ci		ret = rtl2830_update_bits(client, tab[i].reg, tab[i].mask,
9462306a36Sopenharmony_ci					  tab[i].val);
9562306a36Sopenharmony_ci		if (ret)
9662306a36Sopenharmony_ci			goto err;
9762306a36Sopenharmony_ci	}
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x18f, "\x28\x00", 2);
10062306a36Sopenharmony_ci	if (ret)
10162306a36Sopenharmony_ci		goto err;
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x195,
10462306a36Sopenharmony_ci				 "\x04\x06\x0a\x12\x0a\x12\x1e\x28", 8);
10562306a36Sopenharmony_ci	if (ret)
10662306a36Sopenharmony_ci		goto err;
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_ci	/* TODO: spec init */
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci	/* soft reset */
11162306a36Sopenharmony_ci	ret = rtl2830_update_bits(client, 0x101, 0x04, 0x04);
11262306a36Sopenharmony_ci	if (ret)
11362306a36Sopenharmony_ci		goto err;
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci	ret = rtl2830_update_bits(client, 0x101, 0x04, 0x00);
11662306a36Sopenharmony_ci	if (ret)
11762306a36Sopenharmony_ci		goto err;
11862306a36Sopenharmony_ci
11962306a36Sopenharmony_ci	/* init stats here in order signal app which stats are supported */
12062306a36Sopenharmony_ci	c->strength.len = 1;
12162306a36Sopenharmony_ci	c->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12262306a36Sopenharmony_ci	c->cnr.len = 1;
12362306a36Sopenharmony_ci	c->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12462306a36Sopenharmony_ci	c->post_bit_error.len = 1;
12562306a36Sopenharmony_ci	c->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12662306a36Sopenharmony_ci	c->post_bit_count.len = 1;
12762306a36Sopenharmony_ci	c->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
12862306a36Sopenharmony_ci
12962306a36Sopenharmony_ci	dev->sleeping = false;
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	return ret;
13262306a36Sopenharmony_cierr:
13362306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
13462306a36Sopenharmony_ci	return ret;
13562306a36Sopenharmony_ci}
13662306a36Sopenharmony_ci
13762306a36Sopenharmony_cistatic int rtl2830_sleep(struct dvb_frontend *fe)
13862306a36Sopenharmony_ci{
13962306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
14062306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
14162306a36Sopenharmony_ci
14262306a36Sopenharmony_ci	dev->sleeping = true;
14362306a36Sopenharmony_ci	dev->fe_status = 0;
14462306a36Sopenharmony_ci
14562306a36Sopenharmony_ci	return 0;
14662306a36Sopenharmony_ci}
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_cistatic int rtl2830_get_tune_settings(struct dvb_frontend *fe,
14962306a36Sopenharmony_ci				     struct dvb_frontend_tune_settings *s)
15062306a36Sopenharmony_ci{
15162306a36Sopenharmony_ci	s->min_delay_ms = 500;
15262306a36Sopenharmony_ci	s->step_size = fe->ops.info.frequency_stepsize_hz * 2;
15362306a36Sopenharmony_ci	s->max_drift = (fe->ops.info.frequency_stepsize_hz * 2) + 1;
15462306a36Sopenharmony_ci
15562306a36Sopenharmony_ci	return 0;
15662306a36Sopenharmony_ci}
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_cistatic int rtl2830_set_frontend(struct dvb_frontend *fe)
15962306a36Sopenharmony_ci{
16062306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
16162306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
16262306a36Sopenharmony_ci	struct dtv_frontend_properties *c = &fe->dtv_property_cache;
16362306a36Sopenharmony_ci	int ret, i;
16462306a36Sopenharmony_ci	u64 num;
16562306a36Sopenharmony_ci	u8 buf[3], u8tmp;
16662306a36Sopenharmony_ci	u32 if_ctl, if_frequency;
16762306a36Sopenharmony_ci	static const u8 bw_params1[3][34] = {
16862306a36Sopenharmony_ci		{
16962306a36Sopenharmony_ci		0x1f, 0xf0, 0x1f, 0xf0, 0x1f, 0xfa, 0x00, 0x17, 0x00, 0x41,
17062306a36Sopenharmony_ci		0x00, 0x64, 0x00, 0x67, 0x00, 0x38, 0x1f, 0xde, 0x1f, 0x7a,
17162306a36Sopenharmony_ci		0x1f, 0x47, 0x1f, 0x7c, 0x00, 0x30, 0x01, 0x4b, 0x02, 0x82,
17262306a36Sopenharmony_ci		0x03, 0x73, 0x03, 0xcf, /* 6 MHz */
17362306a36Sopenharmony_ci		}, {
17462306a36Sopenharmony_ci		0x1f, 0xfa, 0x1f, 0xda, 0x1f, 0xc1, 0x1f, 0xb3, 0x1f, 0xca,
17562306a36Sopenharmony_ci		0x00, 0x07, 0x00, 0x4d, 0x00, 0x6d, 0x00, 0x40, 0x1f, 0xca,
17662306a36Sopenharmony_ci		0x1f, 0x4d, 0x1f, 0x2a, 0x1f, 0xb2, 0x00, 0xec, 0x02, 0x7e,
17762306a36Sopenharmony_ci		0x03, 0xd0, 0x04, 0x53, /* 7 MHz */
17862306a36Sopenharmony_ci		}, {
17962306a36Sopenharmony_ci		0x00, 0x10, 0x00, 0x0e, 0x1f, 0xf7, 0x1f, 0xc9, 0x1f, 0xa0,
18062306a36Sopenharmony_ci		0x1f, 0xa6, 0x1f, 0xec, 0x00, 0x4e, 0x00, 0x7d, 0x00, 0x3a,
18162306a36Sopenharmony_ci		0x1f, 0x98, 0x1f, 0x10, 0x1f, 0x40, 0x00, 0x75, 0x02, 0x5f,
18262306a36Sopenharmony_ci		0x04, 0x24, 0x04, 0xdb, /* 8 MHz */
18362306a36Sopenharmony_ci		},
18462306a36Sopenharmony_ci	};
18562306a36Sopenharmony_ci	static const u8 bw_params2[3][6] = {
18662306a36Sopenharmony_ci		{0xc3, 0x0c, 0x44, 0x33, 0x33, 0x30}, /* 6 MHz */
18762306a36Sopenharmony_ci		{0xb8, 0xe3, 0x93, 0x99, 0x99, 0x98}, /* 7 MHz */
18862306a36Sopenharmony_ci		{0xae, 0xba, 0xf3, 0x26, 0x66, 0x64}, /* 8 MHz */
18962306a36Sopenharmony_ci	};
19062306a36Sopenharmony_ci
19162306a36Sopenharmony_ci	dev_dbg(&client->dev, "frequency=%u bandwidth_hz=%u inversion=%u\n",
19262306a36Sopenharmony_ci		c->frequency, c->bandwidth_hz, c->inversion);
19362306a36Sopenharmony_ci
19462306a36Sopenharmony_ci	/* program tuner */
19562306a36Sopenharmony_ci	if (fe->ops.tuner_ops.set_params)
19662306a36Sopenharmony_ci		fe->ops.tuner_ops.set_params(fe);
19762306a36Sopenharmony_ci
19862306a36Sopenharmony_ci	switch (c->bandwidth_hz) {
19962306a36Sopenharmony_ci	case 6000000:
20062306a36Sopenharmony_ci		i = 0;
20162306a36Sopenharmony_ci		break;
20262306a36Sopenharmony_ci	case 7000000:
20362306a36Sopenharmony_ci		i = 1;
20462306a36Sopenharmony_ci		break;
20562306a36Sopenharmony_ci	case 8000000:
20662306a36Sopenharmony_ci		i = 2;
20762306a36Sopenharmony_ci		break;
20862306a36Sopenharmony_ci	default:
20962306a36Sopenharmony_ci		dev_err(&client->dev, "invalid bandwidth_hz %u\n",
21062306a36Sopenharmony_ci			c->bandwidth_hz);
21162306a36Sopenharmony_ci		return -EINVAL;
21262306a36Sopenharmony_ci	}
21362306a36Sopenharmony_ci
21462306a36Sopenharmony_ci	ret = rtl2830_update_bits(client, 0x008, 0x06, i << 1);
21562306a36Sopenharmony_ci	if (ret)
21662306a36Sopenharmony_ci		goto err;
21762306a36Sopenharmony_ci
21862306a36Sopenharmony_ci	/* program if frequency */
21962306a36Sopenharmony_ci	if (fe->ops.tuner_ops.get_if_frequency)
22062306a36Sopenharmony_ci		ret = fe->ops.tuner_ops.get_if_frequency(fe, &if_frequency);
22162306a36Sopenharmony_ci	else
22262306a36Sopenharmony_ci		ret = -EINVAL;
22362306a36Sopenharmony_ci	if (ret)
22462306a36Sopenharmony_ci		goto err;
22562306a36Sopenharmony_ci
22662306a36Sopenharmony_ci	num = if_frequency % dev->pdata->clk;
22762306a36Sopenharmony_ci	num *= 0x400000;
22862306a36Sopenharmony_ci	num = div_u64(num, dev->pdata->clk);
22962306a36Sopenharmony_ci	num = -num;
23062306a36Sopenharmony_ci	if_ctl = num & 0x3fffff;
23162306a36Sopenharmony_ci	dev_dbg(&client->dev, "if_frequency=%d if_ctl=%08x\n",
23262306a36Sopenharmony_ci		if_frequency, if_ctl);
23362306a36Sopenharmony_ci
23462306a36Sopenharmony_ci	buf[0] = (if_ctl >> 16) & 0x3f;
23562306a36Sopenharmony_ci	buf[1] = (if_ctl >>  8) & 0xff;
23662306a36Sopenharmony_ci	buf[2] = (if_ctl >>  0) & 0xff;
23762306a36Sopenharmony_ci
23862306a36Sopenharmony_ci	ret = rtl2830_bulk_read(client, 0x119, &u8tmp, 1);
23962306a36Sopenharmony_ci	if (ret)
24062306a36Sopenharmony_ci		goto err;
24162306a36Sopenharmony_ci
24262306a36Sopenharmony_ci	buf[0] |= u8tmp & 0xc0;  /* [7:6] */
24362306a36Sopenharmony_ci
24462306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x119, buf, 3);
24562306a36Sopenharmony_ci	if (ret)
24662306a36Sopenharmony_ci		goto err;
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_ci	/* 1/2 split I2C write */
24962306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x11c, &bw_params1[i][0], 17);
25062306a36Sopenharmony_ci	if (ret)
25162306a36Sopenharmony_ci		goto err;
25262306a36Sopenharmony_ci
25362306a36Sopenharmony_ci	/* 2/2 split I2C write */
25462306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x12d, &bw_params1[i][17], 17);
25562306a36Sopenharmony_ci	if (ret)
25662306a36Sopenharmony_ci		goto err;
25762306a36Sopenharmony_ci
25862306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x19d, bw_params2[i], 6);
25962306a36Sopenharmony_ci	if (ret)
26062306a36Sopenharmony_ci		goto err;
26162306a36Sopenharmony_ci
26262306a36Sopenharmony_ci	return ret;
26362306a36Sopenharmony_cierr:
26462306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
26562306a36Sopenharmony_ci	return ret;
26662306a36Sopenharmony_ci}
26762306a36Sopenharmony_ci
26862306a36Sopenharmony_cistatic int rtl2830_get_frontend(struct dvb_frontend *fe,
26962306a36Sopenharmony_ci				struct dtv_frontend_properties *c)
27062306a36Sopenharmony_ci{
27162306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
27262306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
27362306a36Sopenharmony_ci	int ret;
27462306a36Sopenharmony_ci	u8 buf[3];
27562306a36Sopenharmony_ci
27662306a36Sopenharmony_ci	if (dev->sleeping)
27762306a36Sopenharmony_ci		return 0;
27862306a36Sopenharmony_ci
27962306a36Sopenharmony_ci	ret = rtl2830_bulk_read(client, 0x33c, buf, 2);
28062306a36Sopenharmony_ci	if (ret)
28162306a36Sopenharmony_ci		goto err;
28262306a36Sopenharmony_ci
28362306a36Sopenharmony_ci	ret = rtl2830_bulk_read(client, 0x351, &buf[2], 1);
28462306a36Sopenharmony_ci	if (ret)
28562306a36Sopenharmony_ci		goto err;
28662306a36Sopenharmony_ci
28762306a36Sopenharmony_ci	dev_dbg(&client->dev, "TPS=%*ph\n", 3, buf);
28862306a36Sopenharmony_ci
28962306a36Sopenharmony_ci	switch ((buf[0] >> 2) & 3) {
29062306a36Sopenharmony_ci	case 0:
29162306a36Sopenharmony_ci		c->modulation = QPSK;
29262306a36Sopenharmony_ci		break;
29362306a36Sopenharmony_ci	case 1:
29462306a36Sopenharmony_ci		c->modulation = QAM_16;
29562306a36Sopenharmony_ci		break;
29662306a36Sopenharmony_ci	case 2:
29762306a36Sopenharmony_ci		c->modulation = QAM_64;
29862306a36Sopenharmony_ci		break;
29962306a36Sopenharmony_ci	}
30062306a36Sopenharmony_ci
30162306a36Sopenharmony_ci	switch ((buf[2] >> 2) & 1) {
30262306a36Sopenharmony_ci	case 0:
30362306a36Sopenharmony_ci		c->transmission_mode = TRANSMISSION_MODE_2K;
30462306a36Sopenharmony_ci		break;
30562306a36Sopenharmony_ci	case 1:
30662306a36Sopenharmony_ci		c->transmission_mode = TRANSMISSION_MODE_8K;
30762306a36Sopenharmony_ci	}
30862306a36Sopenharmony_ci
30962306a36Sopenharmony_ci	switch ((buf[2] >> 0) & 3) {
31062306a36Sopenharmony_ci	case 0:
31162306a36Sopenharmony_ci		c->guard_interval = GUARD_INTERVAL_1_32;
31262306a36Sopenharmony_ci		break;
31362306a36Sopenharmony_ci	case 1:
31462306a36Sopenharmony_ci		c->guard_interval = GUARD_INTERVAL_1_16;
31562306a36Sopenharmony_ci		break;
31662306a36Sopenharmony_ci	case 2:
31762306a36Sopenharmony_ci		c->guard_interval = GUARD_INTERVAL_1_8;
31862306a36Sopenharmony_ci		break;
31962306a36Sopenharmony_ci	case 3:
32062306a36Sopenharmony_ci		c->guard_interval = GUARD_INTERVAL_1_4;
32162306a36Sopenharmony_ci		break;
32262306a36Sopenharmony_ci	}
32362306a36Sopenharmony_ci
32462306a36Sopenharmony_ci	switch ((buf[0] >> 4) & 7) {
32562306a36Sopenharmony_ci	case 0:
32662306a36Sopenharmony_ci		c->hierarchy = HIERARCHY_NONE;
32762306a36Sopenharmony_ci		break;
32862306a36Sopenharmony_ci	case 1:
32962306a36Sopenharmony_ci		c->hierarchy = HIERARCHY_1;
33062306a36Sopenharmony_ci		break;
33162306a36Sopenharmony_ci	case 2:
33262306a36Sopenharmony_ci		c->hierarchy = HIERARCHY_2;
33362306a36Sopenharmony_ci		break;
33462306a36Sopenharmony_ci	case 3:
33562306a36Sopenharmony_ci		c->hierarchy = HIERARCHY_4;
33662306a36Sopenharmony_ci		break;
33762306a36Sopenharmony_ci	}
33862306a36Sopenharmony_ci
33962306a36Sopenharmony_ci	switch ((buf[1] >> 3) & 7) {
34062306a36Sopenharmony_ci	case 0:
34162306a36Sopenharmony_ci		c->code_rate_HP = FEC_1_2;
34262306a36Sopenharmony_ci		break;
34362306a36Sopenharmony_ci	case 1:
34462306a36Sopenharmony_ci		c->code_rate_HP = FEC_2_3;
34562306a36Sopenharmony_ci		break;
34662306a36Sopenharmony_ci	case 2:
34762306a36Sopenharmony_ci		c->code_rate_HP = FEC_3_4;
34862306a36Sopenharmony_ci		break;
34962306a36Sopenharmony_ci	case 3:
35062306a36Sopenharmony_ci		c->code_rate_HP = FEC_5_6;
35162306a36Sopenharmony_ci		break;
35262306a36Sopenharmony_ci	case 4:
35362306a36Sopenharmony_ci		c->code_rate_HP = FEC_7_8;
35462306a36Sopenharmony_ci		break;
35562306a36Sopenharmony_ci	}
35662306a36Sopenharmony_ci
35762306a36Sopenharmony_ci	switch ((buf[1] >> 0) & 7) {
35862306a36Sopenharmony_ci	case 0:
35962306a36Sopenharmony_ci		c->code_rate_LP = FEC_1_2;
36062306a36Sopenharmony_ci		break;
36162306a36Sopenharmony_ci	case 1:
36262306a36Sopenharmony_ci		c->code_rate_LP = FEC_2_3;
36362306a36Sopenharmony_ci		break;
36462306a36Sopenharmony_ci	case 2:
36562306a36Sopenharmony_ci		c->code_rate_LP = FEC_3_4;
36662306a36Sopenharmony_ci		break;
36762306a36Sopenharmony_ci	case 3:
36862306a36Sopenharmony_ci		c->code_rate_LP = FEC_5_6;
36962306a36Sopenharmony_ci		break;
37062306a36Sopenharmony_ci	case 4:
37162306a36Sopenharmony_ci		c->code_rate_LP = FEC_7_8;
37262306a36Sopenharmony_ci		break;
37362306a36Sopenharmony_ci	}
37462306a36Sopenharmony_ci
37562306a36Sopenharmony_ci	return 0;
37662306a36Sopenharmony_cierr:
37762306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
37862306a36Sopenharmony_ci	return ret;
37962306a36Sopenharmony_ci}
38062306a36Sopenharmony_ci
38162306a36Sopenharmony_cistatic int rtl2830_read_status(struct dvb_frontend *fe, enum fe_status *status)
38262306a36Sopenharmony_ci{
38362306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
38462306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
38562306a36Sopenharmony_ci	struct dtv_frontend_properties *c = &dev->fe.dtv_property_cache;
38662306a36Sopenharmony_ci	int ret, stmp;
38762306a36Sopenharmony_ci	unsigned int utmp;
38862306a36Sopenharmony_ci	u8 u8tmp, buf[2];
38962306a36Sopenharmony_ci
39062306a36Sopenharmony_ci	*status = 0;
39162306a36Sopenharmony_ci
39262306a36Sopenharmony_ci	if (dev->sleeping)
39362306a36Sopenharmony_ci		return 0;
39462306a36Sopenharmony_ci
39562306a36Sopenharmony_ci	ret = rtl2830_bulk_read(client, 0x351, &u8tmp, 1);
39662306a36Sopenharmony_ci	if (ret)
39762306a36Sopenharmony_ci		goto err;
39862306a36Sopenharmony_ci
39962306a36Sopenharmony_ci	u8tmp = (u8tmp >> 3) & 0x0f; /* [6:3] */
40062306a36Sopenharmony_ci	if (u8tmp == 11) {
40162306a36Sopenharmony_ci		*status |= FE_HAS_SIGNAL | FE_HAS_CARRIER |
40262306a36Sopenharmony_ci			FE_HAS_VITERBI | FE_HAS_SYNC | FE_HAS_LOCK;
40362306a36Sopenharmony_ci	} else if (u8tmp == 10) {
40462306a36Sopenharmony_ci		*status |= FE_HAS_SIGNAL | FE_HAS_CARRIER |
40562306a36Sopenharmony_ci			FE_HAS_VITERBI;
40662306a36Sopenharmony_ci	}
40762306a36Sopenharmony_ci
40862306a36Sopenharmony_ci	dev->fe_status = *status;
40962306a36Sopenharmony_ci
41062306a36Sopenharmony_ci	/* Signal strength */
41162306a36Sopenharmony_ci	if (dev->fe_status & FE_HAS_SIGNAL) {
41262306a36Sopenharmony_ci		/* Read IF AGC */
41362306a36Sopenharmony_ci		ret = rtl2830_bulk_read(client, 0x359, buf, 2);
41462306a36Sopenharmony_ci		if (ret)
41562306a36Sopenharmony_ci			goto err;
41662306a36Sopenharmony_ci
41762306a36Sopenharmony_ci		stmp = buf[0] << 8 | buf[1] << 0;
41862306a36Sopenharmony_ci		stmp = sign_extend32(stmp, 13);
41962306a36Sopenharmony_ci		utmp = clamp_val(-4 * stmp + 32767, 0x0000, 0xffff);
42062306a36Sopenharmony_ci
42162306a36Sopenharmony_ci		dev_dbg(&client->dev, "IF AGC=%d\n", stmp);
42262306a36Sopenharmony_ci
42362306a36Sopenharmony_ci		c->strength.stat[0].scale = FE_SCALE_RELATIVE;
42462306a36Sopenharmony_ci		c->strength.stat[0].uvalue = utmp;
42562306a36Sopenharmony_ci	} else {
42662306a36Sopenharmony_ci		c->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
42762306a36Sopenharmony_ci	}
42862306a36Sopenharmony_ci
42962306a36Sopenharmony_ci	/* CNR */
43062306a36Sopenharmony_ci	if (dev->fe_status & FE_HAS_VITERBI) {
43162306a36Sopenharmony_ci		unsigned int hierarchy, constellation;
43262306a36Sopenharmony_ci		#define CONSTELLATION_NUM 3
43362306a36Sopenharmony_ci		#define HIERARCHY_NUM 4
43462306a36Sopenharmony_ci		static const u32 constant[CONSTELLATION_NUM][HIERARCHY_NUM] = {
43562306a36Sopenharmony_ci			{70705899, 70705899, 70705899, 70705899},
43662306a36Sopenharmony_ci			{82433173, 82433173, 87483115, 94445660},
43762306a36Sopenharmony_ci			{92888734, 92888734, 95487525, 99770748},
43862306a36Sopenharmony_ci		};
43962306a36Sopenharmony_ci
44062306a36Sopenharmony_ci		ret = rtl2830_bulk_read(client, 0x33c, &u8tmp, 1);
44162306a36Sopenharmony_ci		if (ret)
44262306a36Sopenharmony_ci			goto err;
44362306a36Sopenharmony_ci
44462306a36Sopenharmony_ci		constellation = (u8tmp >> 2) & 0x03; /* [3:2] */
44562306a36Sopenharmony_ci		if (constellation > CONSTELLATION_NUM - 1)
44662306a36Sopenharmony_ci			goto err;
44762306a36Sopenharmony_ci
44862306a36Sopenharmony_ci		hierarchy = (u8tmp >> 4) & 0x07; /* [6:4] */
44962306a36Sopenharmony_ci		if (hierarchy > HIERARCHY_NUM - 1)
45062306a36Sopenharmony_ci			goto err;
45162306a36Sopenharmony_ci
45262306a36Sopenharmony_ci		ret = rtl2830_bulk_read(client, 0x40c, buf, 2);
45362306a36Sopenharmony_ci		if (ret)
45462306a36Sopenharmony_ci			goto err;
45562306a36Sopenharmony_ci
45662306a36Sopenharmony_ci		utmp = buf[0] << 8 | buf[1] << 0;
45762306a36Sopenharmony_ci		if (utmp)
45862306a36Sopenharmony_ci			stmp = (constant[constellation][hierarchy] -
45962306a36Sopenharmony_ci			       intlog10(utmp)) / ((1 << 24) / 10000);
46062306a36Sopenharmony_ci		else
46162306a36Sopenharmony_ci			stmp = 0;
46262306a36Sopenharmony_ci
46362306a36Sopenharmony_ci		dev_dbg(&client->dev, "CNR raw=%u\n", utmp);
46462306a36Sopenharmony_ci
46562306a36Sopenharmony_ci		c->cnr.stat[0].scale = FE_SCALE_DECIBEL;
46662306a36Sopenharmony_ci		c->cnr.stat[0].svalue = stmp;
46762306a36Sopenharmony_ci	} else {
46862306a36Sopenharmony_ci		c->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
46962306a36Sopenharmony_ci	}
47062306a36Sopenharmony_ci
47162306a36Sopenharmony_ci	/* BER */
47262306a36Sopenharmony_ci	if (dev->fe_status & FE_HAS_LOCK) {
47362306a36Sopenharmony_ci		ret = rtl2830_bulk_read(client, 0x34e, buf, 2);
47462306a36Sopenharmony_ci		if (ret)
47562306a36Sopenharmony_ci			goto err;
47662306a36Sopenharmony_ci
47762306a36Sopenharmony_ci		utmp = buf[0] << 8 | buf[1] << 0;
47862306a36Sopenharmony_ci		dev->post_bit_error += utmp;
47962306a36Sopenharmony_ci		dev->post_bit_count += 1000000;
48062306a36Sopenharmony_ci
48162306a36Sopenharmony_ci		dev_dbg(&client->dev, "BER errors=%u total=1000000\n", utmp);
48262306a36Sopenharmony_ci
48362306a36Sopenharmony_ci		c->post_bit_error.stat[0].scale = FE_SCALE_COUNTER;
48462306a36Sopenharmony_ci		c->post_bit_error.stat[0].uvalue = dev->post_bit_error;
48562306a36Sopenharmony_ci		c->post_bit_count.stat[0].scale = FE_SCALE_COUNTER;
48662306a36Sopenharmony_ci		c->post_bit_count.stat[0].uvalue = dev->post_bit_count;
48762306a36Sopenharmony_ci	} else {
48862306a36Sopenharmony_ci		c->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
48962306a36Sopenharmony_ci		c->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
49062306a36Sopenharmony_ci	}
49162306a36Sopenharmony_ci
49262306a36Sopenharmony_ci
49362306a36Sopenharmony_ci	return ret;
49462306a36Sopenharmony_cierr:
49562306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
49662306a36Sopenharmony_ci	return ret;
49762306a36Sopenharmony_ci}
49862306a36Sopenharmony_ci
49962306a36Sopenharmony_cistatic int rtl2830_read_snr(struct dvb_frontend *fe, u16 *snr)
50062306a36Sopenharmony_ci{
50162306a36Sopenharmony_ci	struct dtv_frontend_properties *c = &fe->dtv_property_cache;
50262306a36Sopenharmony_ci
50362306a36Sopenharmony_ci	if (c->cnr.stat[0].scale == FE_SCALE_DECIBEL)
50462306a36Sopenharmony_ci		*snr = div_s64(c->cnr.stat[0].svalue, 100);
50562306a36Sopenharmony_ci	else
50662306a36Sopenharmony_ci		*snr = 0;
50762306a36Sopenharmony_ci
50862306a36Sopenharmony_ci	return 0;
50962306a36Sopenharmony_ci}
51062306a36Sopenharmony_ci
51162306a36Sopenharmony_cistatic int rtl2830_read_ber(struct dvb_frontend *fe, u32 *ber)
51262306a36Sopenharmony_ci{
51362306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
51462306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
51562306a36Sopenharmony_ci
51662306a36Sopenharmony_ci	*ber = (dev->post_bit_error - dev->post_bit_error_prev);
51762306a36Sopenharmony_ci	dev->post_bit_error_prev = dev->post_bit_error;
51862306a36Sopenharmony_ci
51962306a36Sopenharmony_ci	return 0;
52062306a36Sopenharmony_ci}
52162306a36Sopenharmony_ci
52262306a36Sopenharmony_cistatic int rtl2830_read_ucblocks(struct dvb_frontend *fe, u32 *ucblocks)
52362306a36Sopenharmony_ci{
52462306a36Sopenharmony_ci	*ucblocks = 0;
52562306a36Sopenharmony_ci
52662306a36Sopenharmony_ci	return 0;
52762306a36Sopenharmony_ci}
52862306a36Sopenharmony_ci
52962306a36Sopenharmony_cistatic int rtl2830_read_signal_strength(struct dvb_frontend *fe, u16 *strength)
53062306a36Sopenharmony_ci{
53162306a36Sopenharmony_ci	struct dtv_frontend_properties *c = &fe->dtv_property_cache;
53262306a36Sopenharmony_ci
53362306a36Sopenharmony_ci	if (c->strength.stat[0].scale == FE_SCALE_RELATIVE)
53462306a36Sopenharmony_ci		*strength = c->strength.stat[0].uvalue;
53562306a36Sopenharmony_ci	else
53662306a36Sopenharmony_ci		*strength = 0;
53762306a36Sopenharmony_ci
53862306a36Sopenharmony_ci	return 0;
53962306a36Sopenharmony_ci}
54062306a36Sopenharmony_ci
54162306a36Sopenharmony_cistatic const struct dvb_frontend_ops rtl2830_ops = {
54262306a36Sopenharmony_ci	.delsys = {SYS_DVBT},
54362306a36Sopenharmony_ci	.info = {
54462306a36Sopenharmony_ci		.name = "Realtek RTL2830 (DVB-T)",
54562306a36Sopenharmony_ci		.caps = FE_CAN_FEC_1_2 |
54662306a36Sopenharmony_ci			FE_CAN_FEC_2_3 |
54762306a36Sopenharmony_ci			FE_CAN_FEC_3_4 |
54862306a36Sopenharmony_ci			FE_CAN_FEC_5_6 |
54962306a36Sopenharmony_ci			FE_CAN_FEC_7_8 |
55062306a36Sopenharmony_ci			FE_CAN_FEC_AUTO |
55162306a36Sopenharmony_ci			FE_CAN_QPSK |
55262306a36Sopenharmony_ci			FE_CAN_QAM_16 |
55362306a36Sopenharmony_ci			FE_CAN_QAM_64 |
55462306a36Sopenharmony_ci			FE_CAN_QAM_AUTO |
55562306a36Sopenharmony_ci			FE_CAN_TRANSMISSION_MODE_AUTO |
55662306a36Sopenharmony_ci			FE_CAN_GUARD_INTERVAL_AUTO |
55762306a36Sopenharmony_ci			FE_CAN_HIERARCHY_AUTO |
55862306a36Sopenharmony_ci			FE_CAN_RECOVER |
55962306a36Sopenharmony_ci			FE_CAN_MUTE_TS
56062306a36Sopenharmony_ci	},
56162306a36Sopenharmony_ci
56262306a36Sopenharmony_ci	.init = rtl2830_init,
56362306a36Sopenharmony_ci	.sleep = rtl2830_sleep,
56462306a36Sopenharmony_ci
56562306a36Sopenharmony_ci	.get_tune_settings = rtl2830_get_tune_settings,
56662306a36Sopenharmony_ci
56762306a36Sopenharmony_ci	.set_frontend = rtl2830_set_frontend,
56862306a36Sopenharmony_ci	.get_frontend = rtl2830_get_frontend,
56962306a36Sopenharmony_ci
57062306a36Sopenharmony_ci	.read_status = rtl2830_read_status,
57162306a36Sopenharmony_ci	.read_snr = rtl2830_read_snr,
57262306a36Sopenharmony_ci	.read_ber = rtl2830_read_ber,
57362306a36Sopenharmony_ci	.read_ucblocks = rtl2830_read_ucblocks,
57462306a36Sopenharmony_ci	.read_signal_strength = rtl2830_read_signal_strength,
57562306a36Sopenharmony_ci};
57662306a36Sopenharmony_ci
57762306a36Sopenharmony_cistatic int rtl2830_pid_filter_ctrl(struct dvb_frontend *fe, int onoff)
57862306a36Sopenharmony_ci{
57962306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
58062306a36Sopenharmony_ci	int ret;
58162306a36Sopenharmony_ci	u8 u8tmp;
58262306a36Sopenharmony_ci
58362306a36Sopenharmony_ci	dev_dbg(&client->dev, "onoff=%d\n", onoff);
58462306a36Sopenharmony_ci
58562306a36Sopenharmony_ci	/* enable / disable PID filter */
58662306a36Sopenharmony_ci	if (onoff)
58762306a36Sopenharmony_ci		u8tmp = 0x80;
58862306a36Sopenharmony_ci	else
58962306a36Sopenharmony_ci		u8tmp = 0x00;
59062306a36Sopenharmony_ci
59162306a36Sopenharmony_ci	ret = rtl2830_update_bits(client, 0x061, 0x80, u8tmp);
59262306a36Sopenharmony_ci	if (ret)
59362306a36Sopenharmony_ci		goto err;
59462306a36Sopenharmony_ci
59562306a36Sopenharmony_ci	return 0;
59662306a36Sopenharmony_cierr:
59762306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
59862306a36Sopenharmony_ci	return ret;
59962306a36Sopenharmony_ci}
60062306a36Sopenharmony_ci
60162306a36Sopenharmony_cistatic int rtl2830_pid_filter(struct dvb_frontend *fe, u8 index, u16 pid, int onoff)
60262306a36Sopenharmony_ci{
60362306a36Sopenharmony_ci	struct i2c_client *client = fe->demodulator_priv;
60462306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
60562306a36Sopenharmony_ci	int ret;
60662306a36Sopenharmony_ci	u8 buf[4];
60762306a36Sopenharmony_ci
60862306a36Sopenharmony_ci	dev_dbg(&client->dev, "index=%d pid=%04x onoff=%d\n",
60962306a36Sopenharmony_ci		index, pid, onoff);
61062306a36Sopenharmony_ci
61162306a36Sopenharmony_ci	/* skip invalid PIDs (0x2000) */
61262306a36Sopenharmony_ci	if (pid > 0x1fff || index > 32)
61362306a36Sopenharmony_ci		return 0;
61462306a36Sopenharmony_ci
61562306a36Sopenharmony_ci	if (onoff)
61662306a36Sopenharmony_ci		set_bit(index, &dev->filters);
61762306a36Sopenharmony_ci	else
61862306a36Sopenharmony_ci		clear_bit(index, &dev->filters);
61962306a36Sopenharmony_ci
62062306a36Sopenharmony_ci	/* enable / disable PIDs */
62162306a36Sopenharmony_ci	buf[0] = (dev->filters >>  0) & 0xff;
62262306a36Sopenharmony_ci	buf[1] = (dev->filters >>  8) & 0xff;
62362306a36Sopenharmony_ci	buf[2] = (dev->filters >> 16) & 0xff;
62462306a36Sopenharmony_ci	buf[3] = (dev->filters >> 24) & 0xff;
62562306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x062, buf, 4);
62662306a36Sopenharmony_ci	if (ret)
62762306a36Sopenharmony_ci		goto err;
62862306a36Sopenharmony_ci
62962306a36Sopenharmony_ci	/* add PID */
63062306a36Sopenharmony_ci	buf[0] = (pid >> 8) & 0xff;
63162306a36Sopenharmony_ci	buf[1] = (pid >> 0) & 0xff;
63262306a36Sopenharmony_ci	ret = rtl2830_bulk_write(client, 0x066 + 2 * index, buf, 2);
63362306a36Sopenharmony_ci	if (ret)
63462306a36Sopenharmony_ci		goto err;
63562306a36Sopenharmony_ci
63662306a36Sopenharmony_ci	return 0;
63762306a36Sopenharmony_cierr:
63862306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
63962306a36Sopenharmony_ci	return ret;
64062306a36Sopenharmony_ci}
64162306a36Sopenharmony_ci
64262306a36Sopenharmony_ci/*
64362306a36Sopenharmony_ci * I2C gate/mux/repeater logic
64462306a36Sopenharmony_ci * We must use unlocked __i2c_transfer() here (through regmap) because of I2C
64562306a36Sopenharmony_ci * adapter lock is already taken by tuner driver.
64662306a36Sopenharmony_ci * Gate is closed automatically after single I2C transfer.
64762306a36Sopenharmony_ci */
64862306a36Sopenharmony_cistatic int rtl2830_select(struct i2c_mux_core *muxc, u32 chan_id)
64962306a36Sopenharmony_ci{
65062306a36Sopenharmony_ci	struct i2c_client *client = i2c_mux_priv(muxc);
65162306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
65262306a36Sopenharmony_ci	int ret;
65362306a36Sopenharmony_ci
65462306a36Sopenharmony_ci	dev_dbg(&client->dev, "\n");
65562306a36Sopenharmony_ci
65662306a36Sopenharmony_ci	/* open I2C repeater for 1 transfer, closes automatically */
65762306a36Sopenharmony_ci	/* XXX: regmap_update_bits() does not lock I2C adapter */
65862306a36Sopenharmony_ci	ret = regmap_update_bits(dev->regmap, 0x101, 0x08, 0x08);
65962306a36Sopenharmony_ci	if (ret)
66062306a36Sopenharmony_ci		goto err;
66162306a36Sopenharmony_ci
66262306a36Sopenharmony_ci	return 0;
66362306a36Sopenharmony_cierr:
66462306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
66562306a36Sopenharmony_ci	return ret;
66662306a36Sopenharmony_ci}
66762306a36Sopenharmony_ci
66862306a36Sopenharmony_cistatic struct dvb_frontend *rtl2830_get_dvb_frontend(struct i2c_client *client)
66962306a36Sopenharmony_ci{
67062306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
67162306a36Sopenharmony_ci
67262306a36Sopenharmony_ci	dev_dbg(&client->dev, "\n");
67362306a36Sopenharmony_ci
67462306a36Sopenharmony_ci	return &dev->fe;
67562306a36Sopenharmony_ci}
67662306a36Sopenharmony_ci
67762306a36Sopenharmony_cistatic struct i2c_adapter *rtl2830_get_i2c_adapter(struct i2c_client *client)
67862306a36Sopenharmony_ci{
67962306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
68062306a36Sopenharmony_ci
68162306a36Sopenharmony_ci	dev_dbg(&client->dev, "\n");
68262306a36Sopenharmony_ci
68362306a36Sopenharmony_ci	return dev->muxc->adapter[0];
68462306a36Sopenharmony_ci}
68562306a36Sopenharmony_ci
68662306a36Sopenharmony_ci/*
68762306a36Sopenharmony_ci * We implement own I2C access routines for regmap in order to get manual access
68862306a36Sopenharmony_ci * to I2C adapter lock, which is needed for I2C mux adapter.
68962306a36Sopenharmony_ci */
69062306a36Sopenharmony_cistatic int rtl2830_regmap_read(void *context, const void *reg_buf,
69162306a36Sopenharmony_ci			       size_t reg_size, void *val_buf, size_t val_size)
69262306a36Sopenharmony_ci{
69362306a36Sopenharmony_ci	struct i2c_client *client = context;
69462306a36Sopenharmony_ci	int ret;
69562306a36Sopenharmony_ci	struct i2c_msg msg[2] = {
69662306a36Sopenharmony_ci		{
69762306a36Sopenharmony_ci			.addr = client->addr,
69862306a36Sopenharmony_ci			.flags = 0,
69962306a36Sopenharmony_ci			.len = reg_size,
70062306a36Sopenharmony_ci			.buf = (u8 *)reg_buf,
70162306a36Sopenharmony_ci		}, {
70262306a36Sopenharmony_ci			.addr = client->addr,
70362306a36Sopenharmony_ci			.flags = I2C_M_RD,
70462306a36Sopenharmony_ci			.len = val_size,
70562306a36Sopenharmony_ci			.buf = val_buf,
70662306a36Sopenharmony_ci		}
70762306a36Sopenharmony_ci	};
70862306a36Sopenharmony_ci
70962306a36Sopenharmony_ci	ret = __i2c_transfer(client->adapter, msg, 2);
71062306a36Sopenharmony_ci	if (ret != 2) {
71162306a36Sopenharmony_ci		dev_warn(&client->dev, "i2c reg read failed %d\n", ret);
71262306a36Sopenharmony_ci		if (ret >= 0)
71362306a36Sopenharmony_ci			ret = -EREMOTEIO;
71462306a36Sopenharmony_ci		return ret;
71562306a36Sopenharmony_ci	}
71662306a36Sopenharmony_ci	return 0;
71762306a36Sopenharmony_ci}
71862306a36Sopenharmony_ci
71962306a36Sopenharmony_cistatic int rtl2830_regmap_write(void *context, const void *data, size_t count)
72062306a36Sopenharmony_ci{
72162306a36Sopenharmony_ci	struct i2c_client *client = context;
72262306a36Sopenharmony_ci	int ret;
72362306a36Sopenharmony_ci	struct i2c_msg msg[1] = {
72462306a36Sopenharmony_ci		{
72562306a36Sopenharmony_ci			.addr = client->addr,
72662306a36Sopenharmony_ci			.flags = 0,
72762306a36Sopenharmony_ci			.len = count,
72862306a36Sopenharmony_ci			.buf = (u8 *)data,
72962306a36Sopenharmony_ci		}
73062306a36Sopenharmony_ci	};
73162306a36Sopenharmony_ci
73262306a36Sopenharmony_ci	ret = __i2c_transfer(client->adapter, msg, 1);
73362306a36Sopenharmony_ci	if (ret != 1) {
73462306a36Sopenharmony_ci		dev_warn(&client->dev, "i2c reg write failed %d\n", ret);
73562306a36Sopenharmony_ci		if (ret >= 0)
73662306a36Sopenharmony_ci			ret = -EREMOTEIO;
73762306a36Sopenharmony_ci		return ret;
73862306a36Sopenharmony_ci	}
73962306a36Sopenharmony_ci	return 0;
74062306a36Sopenharmony_ci}
74162306a36Sopenharmony_ci
74262306a36Sopenharmony_cistatic int rtl2830_regmap_gather_write(void *context, const void *reg,
74362306a36Sopenharmony_ci				       size_t reg_len, const void *val,
74462306a36Sopenharmony_ci				       size_t val_len)
74562306a36Sopenharmony_ci{
74662306a36Sopenharmony_ci	struct i2c_client *client = context;
74762306a36Sopenharmony_ci	int ret;
74862306a36Sopenharmony_ci	u8 buf[256];
74962306a36Sopenharmony_ci	struct i2c_msg msg[1] = {
75062306a36Sopenharmony_ci		{
75162306a36Sopenharmony_ci			.addr = client->addr,
75262306a36Sopenharmony_ci			.flags = 0,
75362306a36Sopenharmony_ci			.len = 1 + val_len,
75462306a36Sopenharmony_ci			.buf = buf,
75562306a36Sopenharmony_ci		}
75662306a36Sopenharmony_ci	};
75762306a36Sopenharmony_ci
75862306a36Sopenharmony_ci	buf[0] = *(u8 const *)reg;
75962306a36Sopenharmony_ci	memcpy(&buf[1], val, val_len);
76062306a36Sopenharmony_ci
76162306a36Sopenharmony_ci	ret = __i2c_transfer(client->adapter, msg, 1);
76262306a36Sopenharmony_ci	if (ret != 1) {
76362306a36Sopenharmony_ci		dev_warn(&client->dev, "i2c reg write failed %d\n", ret);
76462306a36Sopenharmony_ci		if (ret >= 0)
76562306a36Sopenharmony_ci			ret = -EREMOTEIO;
76662306a36Sopenharmony_ci		return ret;
76762306a36Sopenharmony_ci	}
76862306a36Sopenharmony_ci	return 0;
76962306a36Sopenharmony_ci}
77062306a36Sopenharmony_ci
77162306a36Sopenharmony_cistatic int rtl2830_probe(struct i2c_client *client)
77262306a36Sopenharmony_ci{
77362306a36Sopenharmony_ci	struct rtl2830_platform_data *pdata = client->dev.platform_data;
77462306a36Sopenharmony_ci	struct rtl2830_dev *dev;
77562306a36Sopenharmony_ci	int ret;
77662306a36Sopenharmony_ci	u8 u8tmp;
77762306a36Sopenharmony_ci	static const struct regmap_bus regmap_bus = {
77862306a36Sopenharmony_ci		.read = rtl2830_regmap_read,
77962306a36Sopenharmony_ci		.write = rtl2830_regmap_write,
78062306a36Sopenharmony_ci		.gather_write = rtl2830_regmap_gather_write,
78162306a36Sopenharmony_ci		.val_format_endian_default = REGMAP_ENDIAN_NATIVE,
78262306a36Sopenharmony_ci	};
78362306a36Sopenharmony_ci	static const struct regmap_range_cfg regmap_range_cfg[] = {
78462306a36Sopenharmony_ci		{
78562306a36Sopenharmony_ci			.selector_reg     = 0x00,
78662306a36Sopenharmony_ci			.selector_mask    = 0xff,
78762306a36Sopenharmony_ci			.selector_shift   = 0,
78862306a36Sopenharmony_ci			.window_start     = 0,
78962306a36Sopenharmony_ci			.window_len       = 0x100,
79062306a36Sopenharmony_ci			.range_min        = 0 * 0x100,
79162306a36Sopenharmony_ci			.range_max        = 5 * 0x100,
79262306a36Sopenharmony_ci		},
79362306a36Sopenharmony_ci	};
79462306a36Sopenharmony_ci	static const struct regmap_config regmap_config = {
79562306a36Sopenharmony_ci		.reg_bits    =  8,
79662306a36Sopenharmony_ci		.val_bits    =  8,
79762306a36Sopenharmony_ci		.max_register = 5 * 0x100,
79862306a36Sopenharmony_ci		.ranges = regmap_range_cfg,
79962306a36Sopenharmony_ci		.num_ranges = ARRAY_SIZE(regmap_range_cfg),
80062306a36Sopenharmony_ci	};
80162306a36Sopenharmony_ci
80262306a36Sopenharmony_ci	dev_dbg(&client->dev, "\n");
80362306a36Sopenharmony_ci
80462306a36Sopenharmony_ci	if (pdata == NULL) {
80562306a36Sopenharmony_ci		ret = -EINVAL;
80662306a36Sopenharmony_ci		goto err;
80762306a36Sopenharmony_ci	}
80862306a36Sopenharmony_ci
80962306a36Sopenharmony_ci	/* allocate memory for the internal state */
81062306a36Sopenharmony_ci	dev = kzalloc(sizeof(*dev), GFP_KERNEL);
81162306a36Sopenharmony_ci	if (dev == NULL) {
81262306a36Sopenharmony_ci		ret = -ENOMEM;
81362306a36Sopenharmony_ci		goto err;
81462306a36Sopenharmony_ci	}
81562306a36Sopenharmony_ci
81662306a36Sopenharmony_ci	/* setup the state */
81762306a36Sopenharmony_ci	i2c_set_clientdata(client, dev);
81862306a36Sopenharmony_ci	dev->client = client;
81962306a36Sopenharmony_ci	dev->pdata = client->dev.platform_data;
82062306a36Sopenharmony_ci	dev->sleeping = true;
82162306a36Sopenharmony_ci	dev->regmap = regmap_init(&client->dev, &regmap_bus, client,
82262306a36Sopenharmony_ci				  &regmap_config);
82362306a36Sopenharmony_ci	if (IS_ERR(dev->regmap)) {
82462306a36Sopenharmony_ci		ret = PTR_ERR(dev->regmap);
82562306a36Sopenharmony_ci		goto err_kfree;
82662306a36Sopenharmony_ci	}
82762306a36Sopenharmony_ci
82862306a36Sopenharmony_ci	/* check if the demod is there */
82962306a36Sopenharmony_ci	ret = rtl2830_bulk_read(client, 0x000, &u8tmp, 1);
83062306a36Sopenharmony_ci	if (ret)
83162306a36Sopenharmony_ci		goto err_regmap_exit;
83262306a36Sopenharmony_ci
83362306a36Sopenharmony_ci	/* create muxed i2c adapter for tuner */
83462306a36Sopenharmony_ci	dev->muxc = i2c_mux_alloc(client->adapter, &client->dev, 1, 0, 0,
83562306a36Sopenharmony_ci				  rtl2830_select, NULL);
83662306a36Sopenharmony_ci	if (!dev->muxc) {
83762306a36Sopenharmony_ci		ret = -ENOMEM;
83862306a36Sopenharmony_ci		goto err_regmap_exit;
83962306a36Sopenharmony_ci	}
84062306a36Sopenharmony_ci	dev->muxc->priv = client;
84162306a36Sopenharmony_ci	ret = i2c_mux_add_adapter(dev->muxc, 0, 0, 0);
84262306a36Sopenharmony_ci	if (ret)
84362306a36Sopenharmony_ci		goto err_regmap_exit;
84462306a36Sopenharmony_ci
84562306a36Sopenharmony_ci	/* create dvb frontend */
84662306a36Sopenharmony_ci	memcpy(&dev->fe.ops, &rtl2830_ops, sizeof(dev->fe.ops));
84762306a36Sopenharmony_ci	dev->fe.demodulator_priv = client;
84862306a36Sopenharmony_ci
84962306a36Sopenharmony_ci	/* setup callbacks */
85062306a36Sopenharmony_ci	pdata->get_dvb_frontend = rtl2830_get_dvb_frontend;
85162306a36Sopenharmony_ci	pdata->get_i2c_adapter = rtl2830_get_i2c_adapter;
85262306a36Sopenharmony_ci	pdata->pid_filter = rtl2830_pid_filter;
85362306a36Sopenharmony_ci	pdata->pid_filter_ctrl = rtl2830_pid_filter_ctrl;
85462306a36Sopenharmony_ci
85562306a36Sopenharmony_ci	dev_info(&client->dev, "Realtek RTL2830 successfully attached\n");
85662306a36Sopenharmony_ci
85762306a36Sopenharmony_ci	return 0;
85862306a36Sopenharmony_cierr_regmap_exit:
85962306a36Sopenharmony_ci	regmap_exit(dev->regmap);
86062306a36Sopenharmony_cierr_kfree:
86162306a36Sopenharmony_ci	kfree(dev);
86262306a36Sopenharmony_cierr:
86362306a36Sopenharmony_ci	dev_dbg(&client->dev, "failed=%d\n", ret);
86462306a36Sopenharmony_ci	return ret;
86562306a36Sopenharmony_ci}
86662306a36Sopenharmony_ci
86762306a36Sopenharmony_cistatic void rtl2830_remove(struct i2c_client *client)
86862306a36Sopenharmony_ci{
86962306a36Sopenharmony_ci	struct rtl2830_dev *dev = i2c_get_clientdata(client);
87062306a36Sopenharmony_ci
87162306a36Sopenharmony_ci	dev_dbg(&client->dev, "\n");
87262306a36Sopenharmony_ci
87362306a36Sopenharmony_ci	i2c_mux_del_adapters(dev->muxc);
87462306a36Sopenharmony_ci	regmap_exit(dev->regmap);
87562306a36Sopenharmony_ci	kfree(dev);
87662306a36Sopenharmony_ci}
87762306a36Sopenharmony_ci
87862306a36Sopenharmony_cistatic const struct i2c_device_id rtl2830_id_table[] = {
87962306a36Sopenharmony_ci	{"rtl2830", 0},
88062306a36Sopenharmony_ci	{}
88162306a36Sopenharmony_ci};
88262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, rtl2830_id_table);
88362306a36Sopenharmony_ci
88462306a36Sopenharmony_cistatic struct i2c_driver rtl2830_driver = {
88562306a36Sopenharmony_ci	.driver = {
88662306a36Sopenharmony_ci		.name			= "rtl2830",
88762306a36Sopenharmony_ci		.suppress_bind_attrs	= true,
88862306a36Sopenharmony_ci	},
88962306a36Sopenharmony_ci	.probe		= rtl2830_probe,
89062306a36Sopenharmony_ci	.remove		= rtl2830_remove,
89162306a36Sopenharmony_ci	.id_table	= rtl2830_id_table,
89262306a36Sopenharmony_ci};
89362306a36Sopenharmony_ci
89462306a36Sopenharmony_cimodule_i2c_driver(rtl2830_driver);
89562306a36Sopenharmony_ci
89662306a36Sopenharmony_ciMODULE_AUTHOR("Antti Palosaari <crope@iki.fi>");
89762306a36Sopenharmony_ciMODULE_DESCRIPTION("Realtek RTL2830 DVB-T demodulator driver");
89862306a36Sopenharmony_ciMODULE_LICENSE("GPL");
899