dtt200u-fe.c 6.2 KB
Newer Older
1 2
/* Frontend part of the Linux driver for the WideView/ Yakumo/ Hama/
 * Typhoon/ Yuan DVB-T USB2.0 receiver.
3
 *
4
 * Copyright (C) 2005 Patrick Boettcher <patrick.boettcher@posteo.de>
5 6 7 8 9 10 11 12 13 14 15 16
 *
 *	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, version 2.
 *
 * see Documentation/dvb/README.dvb-usb for more information
 */
#include "dtt200u.h"

struct dtt200u_fe_state {
	struct dvb_usb_device *d;

17
	enum fe_status stat;
18

19
	struct dtv_frontend_properties fep;
20
	struct dvb_frontend frontend;
21 22 23

	unsigned char data[80];
	struct mutex data_mutex;
24 25
};

26 27
static int dtt200u_fe_read_status(struct dvb_frontend *fe,
				  enum fe_status *stat)
28 29
{
	struct dtt200u_fe_state *state = fe->demodulator_priv;
30
	int ret;
31

32 33 34
	mutex_lock(&state->data_mutex);
	state->data[0] = GET_TUNE_STATUS;

35 36 37 38 39 40
	ret = dvb_usb_generic_rw(state->d, state->data, 1, state->data, 3, 0);
	if (ret < 0) {
		*stat = 0;
		mutex_unlock(&state->data_mutex);
		return ret;
	}
41

42
	switch (state->data[0]) {
43
		case 0x01:
44 45
			*stat = FE_HAS_SIGNAL | FE_HAS_CARRIER |
				FE_HAS_VITERBI | FE_HAS_SYNC | FE_HAS_LOCK;
46
			break;
47 48
		case 0x00: /* pending */
			*stat = FE_TIMEDOUT; /* during set_frontend */
49 50
			break;
		default:
51 52
		case 0x02: /* failed */
			*stat = 0;
53 54
			break;
	}
55
	mutex_unlock(&state->data_mutex);
56 57
	return 0;
}
58

59 60 61
static int dtt200u_fe_read_ber(struct dvb_frontend* fe, u32 *ber)
{
	struct dtt200u_fe_state *state = fe->demodulator_priv;
62
	int ret;
63 64 65 66

	mutex_lock(&state->data_mutex);
	state->data[0] = GET_VIT_ERR_CNT;

67 68 69
	ret = dvb_usb_generic_rw(state->d, state->data, 1, state->data, 3, 0);
	if (ret >= 0)
		*ber = (state->data[0] << 16) | (state->data[1] << 8) | state->data[2];
70 71

	mutex_unlock(&state->data_mutex);
72
	return ret;
73 74 75 76 77
}

static int dtt200u_fe_read_unc_blocks(struct dvb_frontend* fe, u32 *unc)
{
	struct dtt200u_fe_state *state = fe->demodulator_priv;
78
	int ret;
79

80 81 82
	mutex_lock(&state->data_mutex);
	state->data[0] = GET_RS_UNCOR_BLK_CNT;

83 84 85
	ret = dvb_usb_generic_rw(state->d, state->data, 1, state->data, 2, 0);
	if (ret >= 0)
		*unc = (state->data[0] << 8) | state->data[1];
86 87 88

	mutex_unlock(&state->data_mutex);
	return ret;
89 90 91 92 93
}

static int dtt200u_fe_read_signal_strength(struct dvb_frontend* fe, u16 *strength)
{
	struct dtt200u_fe_state *state = fe->demodulator_priv;
94
	int ret;
95 96 97 98

	mutex_lock(&state->data_mutex);
	state->data[0] = GET_AGC;

99 100 101
	ret = dvb_usb_generic_rw(state->d, state->data, 1, state->data, 1, 0);
	if (ret >= 0)
		*strength = (state->data[0] << 8) | state->data[0];
102 103 104

	mutex_unlock(&state->data_mutex);
	return ret;
105 106 107 108 109
}

static int dtt200u_fe_read_snr(struct dvb_frontend* fe, u16 *snr)
{
	struct dtt200u_fe_state *state = fe->demodulator_priv;
110
	int ret;
111 112 113 114

	mutex_lock(&state->data_mutex);
	state->data[0] = GET_SNR;

115 116 117
	ret = dvb_usb_generic_rw(state->d, state->data, 1, state->data, 1, 0);
	if (ret >= 0)
		*snr = ~((state->data[0] << 8) | state->data[0]);
118 119 120

	mutex_unlock(&state->data_mutex);
	return ret;
121 122 123 124 125
}

static int dtt200u_fe_init(struct dvb_frontend* fe)
{
	struct dtt200u_fe_state *state = fe->demodulator_priv;
126 127 128 129 130 131 132 133 134
	int ret;

	mutex_lock(&state->data_mutex);
	state->data[0] = SET_INIT;

	ret = dvb_usb_generic_write(state->d, state->data, 1);
	mutex_unlock(&state->data_mutex);

	return ret;
135 136 137 138 139 140 141 142 143 144
}

static int dtt200u_fe_sleep(struct dvb_frontend* fe)
{
	return dtt200u_fe_init(fe);
}

static int dtt200u_fe_get_tune_settings(struct dvb_frontend* fe, struct dvb_frontend_tune_settings *tune)
{
	tune->min_delay_ms = 1500;
145 146
	tune->step_size = 0;
	tune->max_drift = 0;
147 148 149
	return 0;
}

150
static int dtt200u_fe_set_frontend(struct dvb_frontend *fe)
151
{
152
	struct dtv_frontend_properties *fep = &fe->dtv_property_cache;
153
	struct dtt200u_fe_state *state = fe->demodulator_priv;
154
	int ret;
155 156
	u16 freq = fep->frequency / 250000;

157 158
	mutex_lock(&state->data_mutex);
	state->data[0] = SET_BANDWIDTH;
159 160
	switch (fep->bandwidth_hz) {
	case 8000000:
161
		state->data[1] = 8;
162 163
		break;
	case 7000000:
164
		state->data[1] = 7;
165 166
		break;
	case 6000000:
167
		state->data[1] = 6;
168 169
		break;
	default:
170 171
		ret = -EINVAL;
		goto ret;
172 173
	}

174 175 176
	ret = dvb_usb_generic_write(state->d, state->data, 2);
	if (ret < 0)
		goto ret;
177

178 179 180
	state->data[0] = SET_RF_FREQ;
	state->data[1] = freq & 0xff;
	state->data[2] = (freq >> 8) & 0xff;
181 182 183
	ret = dvb_usb_generic_write(state->d, state->data, 3);
	if (ret < 0)
		goto ret;
184

185 186 187
ret:
	mutex_unlock(&state->data_mutex);
	return ret;
188 189
}

190 191
static int dtt200u_fe_get_frontend(struct dvb_frontend* fe,
				   struct dtv_frontend_properties *fep)
192 193
{
	struct dtt200u_fe_state *state = fe->demodulator_priv;
194

195
	memcpy(fep, &state->fep, sizeof(struct dtv_frontend_properties));
196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211
	return 0;
}

static void dtt200u_fe_release(struct dvb_frontend* fe)
{
	struct dtt200u_fe_state *state = (struct dtt200u_fe_state*) fe->demodulator_priv;
	kfree(state);
}

static struct dvb_frontend_ops dtt200u_fe_ops;

struct dvb_frontend* dtt200u_fe_attach(struct dvb_usb_device *d)
{
	struct dtt200u_fe_state* state = NULL;

	/* allocate memory for the internal state */
P
 
Panagiotis Issaris 已提交
212
	state = kzalloc(sizeof(struct dtt200u_fe_state), GFP_KERNEL);
213 214 215 216 217 218
	if (state == NULL)
		goto error;

	deb_info("attaching frontend dtt200u\n");

	state->d = d;
219
	mutex_init(&state->data_mutex);
220

221
	memcpy(&state->frontend.ops,&dtt200u_fe_ops,sizeof(struct dvb_frontend_ops));
222 223
	state->frontend.demodulator_priv = state;

224
	return &state->frontend;
225 226 227 228 229
error:
	return NULL;
}

static struct dvb_frontend_ops dtt200u_fe_ops = {
230
	.delsys = { SYS_DVBT },
231
	.info = {
232
		.name			= "WideView USB DVB-T",
233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250
		.frequency_min		= 44250000,
		.frequency_max		= 867250000,
		.frequency_stepsize	= 250000,
		.caps = FE_CAN_INVERSION_AUTO |
				FE_CAN_FEC_1_2 | FE_CAN_FEC_2_3 | FE_CAN_FEC_3_4 |
				FE_CAN_FEC_5_6 | FE_CAN_FEC_7_8 | FE_CAN_FEC_AUTO |
				FE_CAN_QPSK | FE_CAN_QAM_16 | FE_CAN_QAM_64 | FE_CAN_QAM_AUTO |
				FE_CAN_TRANSMISSION_MODE_AUTO |
				FE_CAN_GUARD_INTERVAL_AUTO |
				FE_CAN_RECOVER |
				FE_CAN_HIERARCHY_AUTO,
	},

	.release = dtt200u_fe_release,

	.init = dtt200u_fe_init,
	.sleep = dtt200u_fe_sleep,

251 252
	.set_frontend = dtt200u_fe_set_frontend,
	.get_frontend = dtt200u_fe_get_frontend,
253 254 255 256 257 258 259 260
	.get_tune_settings = dtt200u_fe_get_tune_settings,

	.read_status = dtt200u_fe_read_status,
	.read_ber = dtt200u_fe_read_ber,
	.read_signal_strength = dtt200u_fe_read_signal_strength,
	.read_snr = dtt200u_fe_read_snr,
	.read_ucblocks = dtt200u_fe_read_unc_blocks,
};