blob: 21260aad1e54684a46c82248dc0bee8681327c21 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * TTUSB DEC Frontend Driver
3 *
4 * Copyright (C) 2003-2004 Alex Woods <linux-dvb@giblets.org>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19 *
20 */
21
22#include "dvb_frontend.h"
23#include "ttusbdecfe.h"
24
25
26#define LOF_HI 10600000
27#define LOF_LO 9750000
28
29struct ttusbdecfe_state {
30
Linus Torvalds1da177e2005-04-16 15:20:36 -070031 /* configuration settings */
32 const struct ttusbdecfe_config* config;
33
34 struct dvb_frontend frontend;
35
36 u8 hi_band;
37 u8 voltage;
38};
39
40
Peter Beutner4d2858c2008-09-06 13:54:06 -030041static int ttusbdecfe_dvbs_read_status(struct dvb_frontend *fe,
42 fe_status_t *status)
43{
44 *status = FE_HAS_SIGNAL | FE_HAS_VITERBI |
45 FE_HAS_SYNC | FE_HAS_CARRIER | FE_HAS_LOCK;
46 return 0;
47}
48
49
50static int ttusbdecfe_dvbt_read_status(struct dvb_frontend *fe,
51 fe_status_t *status)
Linus Torvalds1da177e2005-04-16 15:20:36 -070052{
Alex Woodsf961e712006-01-09 15:25:24 -020053 struct ttusbdecfe_state* state = fe->demodulator_priv;
54 u8 b[] = { 0x00, 0x00, 0x00, 0x00,
55 0x00, 0x00, 0x00, 0x00 };
56 u8 result[4];
57 int len, ret;
58
59 *status=0;
60
61 ret=state->config->send_command(fe, 0x73, sizeof(b), b, &len, result);
62 if(ret)
63 return ret;
64
65 if(len != 4) {
Harvey Harrisone9815ce2008-04-08 23:20:00 -030066 printk(KERN_ERR "%s: unexpected reply\n", __func__);
Alex Woodsf961e712006-01-09 15:25:24 -020067 return -EIO;
68 }
69
70 switch(result[3]) {
71 case 1: /* not tuned yet */
72 case 2: /* no signal/no lock*/
73 break;
74 case 3: /* signal found and locked*/
75 *status = FE_HAS_SIGNAL | FE_HAS_VITERBI |
76 FE_HAS_SYNC | FE_HAS_CARRIER | FE_HAS_LOCK;
77 break;
78 case 4:
79 *status = FE_TIMEDOUT;
80 break;
81 default:
82 pr_info("%s: returned unknown value: %d\n",
Harvey Harrisone9815ce2008-04-08 23:20:00 -030083 __func__, result[3]);
Alex Woodsf961e712006-01-09 15:25:24 -020084 return -EIO;
85 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070086
87 return 0;
88}
89
90static int ttusbdecfe_dvbt_set_frontend(struct dvb_frontend* fe, struct dvb_frontend_parameters *p)
91{
92 struct ttusbdecfe_state* state = (struct ttusbdecfe_state*) fe->demodulator_priv;
93 u8 b[] = { 0x00, 0x00, 0x00, 0x03,
94 0x00, 0x00, 0x00, 0x00,
95 0x00, 0x00, 0x00, 0x01,
96 0x00, 0x00, 0x00, 0xff,
97 0x00, 0x00, 0x00, 0xff };
98
Al Virod4f979a2008-05-21 00:31:31 -030099 __be32 freq = htonl(p->frequency / 1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100 memcpy(&b[4], &freq, sizeof (u32));
101 state->config->send_command(fe, 0x71, sizeof(b), b, NULL, NULL);
102
103 return 0;
104}
105
Alex Woodsf961e712006-01-09 15:25:24 -0200106static int ttusbdecfe_dvbt_get_tune_settings(struct dvb_frontend* fe,
107 struct dvb_frontend_tune_settings* fesettings)
108{
109 fesettings->min_delay_ms = 1500;
110 /* Drift compensation makes no sense for DVB-T */
111 fesettings->step_size = 0;
112 fesettings->max_drift = 0;
113 return 0;
114}
115
Linus Torvalds1da177e2005-04-16 15:20:36 -0700116static int ttusbdecfe_dvbs_set_frontend(struct dvb_frontend* fe, struct dvb_frontend_parameters *p)
117{
118 struct ttusbdecfe_state* state = (struct ttusbdecfe_state*) fe->demodulator_priv;
119
120 u8 b[] = { 0x00, 0x00, 0x00, 0x01,
121 0x00, 0x00, 0x00, 0x00,
122 0x00, 0x00, 0x00, 0x01,
123 0x00, 0x00, 0x00, 0x00,
124 0x00, 0x00, 0x00, 0x00,
125 0x00, 0x00, 0x00, 0x00,
126 0x00, 0x00, 0x00, 0x00,
127 0x00, 0x00, 0x00, 0x00,
128 0x00, 0x00, 0x00, 0x00,
129 0x00, 0x00, 0x00, 0x00 };
Al Virod4f979a2008-05-21 00:31:31 -0300130 __be32 freq;
131 __be32 sym_rate;
132 __be32 band;
133 __be32 lnb_voltage;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700134
135 freq = htonl(p->frequency +
136 (state->hi_band ? LOF_HI : LOF_LO));
137 memcpy(&b[4], &freq, sizeof(u32));
138 sym_rate = htonl(p->u.qam.symbol_rate);
139 memcpy(&b[12], &sym_rate, sizeof(u32));
140 band = htonl(state->hi_band ? LOF_HI : LOF_LO);
141 memcpy(&b[24], &band, sizeof(u32));
142 lnb_voltage = htonl(state->voltage);
143 memcpy(&b[28], &lnb_voltage, sizeof(u32));
144
145 state->config->send_command(fe, 0x71, sizeof(b), b, NULL, NULL);
146
147 return 0;
148}
149
150static int ttusbdecfe_dvbs_diseqc_send_master_cmd(struct dvb_frontend* fe, struct dvb_diseqc_master_cmd *cmd)
151{
152 struct ttusbdecfe_state* state = (struct ttusbdecfe_state*) fe->demodulator_priv;
153 u8 b[] = { 0x00, 0xff, 0x00, 0x00,
154 0x00, 0x00, 0x00, 0x00,
155 0x00, 0x00 };
156
157 memcpy(&b[4], cmd->msg, cmd->msg_len);
158
159 state->config->send_command(fe, 0x72,
160 sizeof(b) - (6 - cmd->msg_len), b,
161 NULL, NULL);
162
163 return 0;
164}
165
166
167static int ttusbdecfe_dvbs_set_tone(struct dvb_frontend* fe, fe_sec_tone_mode_t tone)
168{
169 struct ttusbdecfe_state* state = (struct ttusbdecfe_state*) fe->demodulator_priv;
170
171 state->hi_band = (SEC_TONE_ON == tone);
172
173 return 0;
174}
175
176
177static int ttusbdecfe_dvbs_set_voltage(struct dvb_frontend* fe, fe_sec_voltage_t voltage)
178{
179 struct ttusbdecfe_state* state = (struct ttusbdecfe_state*) fe->demodulator_priv;
180
181 switch (voltage) {
182 case SEC_VOLTAGE_13:
183 state->voltage = 13;
184 break;
185 case SEC_VOLTAGE_18:
186 state->voltage = 18;
187 break;
188 default:
189 return -EINVAL;
190 }
191
192 return 0;
193}
194
195static void ttusbdecfe_release(struct dvb_frontend* fe)
196{
197 struct ttusbdecfe_state* state = (struct ttusbdecfe_state*) fe->demodulator_priv;
198 kfree(state);
199}
200
201static struct dvb_frontend_ops ttusbdecfe_dvbt_ops;
202
203struct dvb_frontend* ttusbdecfe_dvbt_attach(const struct ttusbdecfe_config* config)
204{
205 struct ttusbdecfe_state* state = NULL;
206
207 /* allocate memory for the internal state */
Robert P. J. Day5cbded52006-12-13 00:35:56 -0800208 state = kmalloc(sizeof(struct ttusbdecfe_state), GFP_KERNEL);
Adrian Bunk771e7152005-07-07 17:57:52 -0700209 if (state == NULL)
210 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700211
212 /* setup the state */
213 state->config = config;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700214
215 /* create dvb_frontend */
Patrick Boettcherdea74862006-05-14 05:01:31 -0300216 memcpy(&state->frontend.ops, &ttusbdecfe_dvbt_ops, sizeof(struct dvb_frontend_ops));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700217 state->frontend.demodulator_priv = state;
218 return &state->frontend;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700219}
220
221static struct dvb_frontend_ops ttusbdecfe_dvbs_ops;
222
223struct dvb_frontend* ttusbdecfe_dvbs_attach(const struct ttusbdecfe_config* config)
224{
225 struct ttusbdecfe_state* state = NULL;
226
227 /* allocate memory for the internal state */
Robert P. J. Day5cbded52006-12-13 00:35:56 -0800228 state = kmalloc(sizeof(struct ttusbdecfe_state), GFP_KERNEL);
Adrian Bunk771e7152005-07-07 17:57:52 -0700229 if (state == NULL)
230 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700231
232 /* setup the state */
233 state->config = config;
234 state->voltage = 0;
235 state->hi_band = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236
237 /* create dvb_frontend */
Patrick Boettcherdea74862006-05-14 05:01:31 -0300238 memcpy(&state->frontend.ops, &ttusbdecfe_dvbs_ops, sizeof(struct dvb_frontend_ops));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700239 state->frontend.demodulator_priv = state;
240 return &state->frontend;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700241}
242
243static struct dvb_frontend_ops ttusbdecfe_dvbt_ops = {
244
245 .info = {
246 .name = "TechnoTrend/Hauppauge DEC2000-t Frontend",
247 .type = FE_OFDM,
248 .frequency_min = 51000000,
249 .frequency_max = 858000000,
250 .frequency_stepsize = 62500,
251 .caps = FE_CAN_FEC_1_2 | FE_CAN_FEC_2_3 | FE_CAN_FEC_3_4 |
252 FE_CAN_FEC_5_6 | FE_CAN_FEC_7_8 | FE_CAN_FEC_AUTO |
253 FE_CAN_QAM_16 | FE_CAN_QAM_64 | FE_CAN_QAM_AUTO |
254 FE_CAN_TRANSMISSION_MODE_AUTO | FE_CAN_GUARD_INTERVAL_AUTO |
255 FE_CAN_HIERARCHY_AUTO,
256 },
257
258 .release = ttusbdecfe_release,
259
260 .set_frontend = ttusbdecfe_dvbt_set_frontend,
261
Alex Woodsf961e712006-01-09 15:25:24 -0200262 .get_tune_settings = ttusbdecfe_dvbt_get_tune_settings,
263
Peter Beutner4d2858c2008-09-06 13:54:06 -0300264 .read_status = ttusbdecfe_dvbt_read_status,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700265};
266
267static struct dvb_frontend_ops ttusbdecfe_dvbs_ops = {
268
269 .info = {
270 .name = "TechnoTrend/Hauppauge DEC3000-s Frontend",
271 .type = FE_QPSK,
272 .frequency_min = 950000,
273 .frequency_max = 2150000,
274 .frequency_stepsize = 125,
Alex Woodsf961e712006-01-09 15:25:24 -0200275 .symbol_rate_min = 1000000, /* guessed */
276 .symbol_rate_max = 45000000, /* guessed */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700277 .caps = FE_CAN_FEC_1_2 | FE_CAN_FEC_2_3 | FE_CAN_FEC_3_4 |
278 FE_CAN_FEC_5_6 | FE_CAN_FEC_7_8 | FE_CAN_FEC_AUTO |
Alex Woodsf961e712006-01-09 15:25:24 -0200279 FE_CAN_QPSK
Linus Torvalds1da177e2005-04-16 15:20:36 -0700280 },
281
282 .release = ttusbdecfe_release,
283
284 .set_frontend = ttusbdecfe_dvbs_set_frontend,
285
Peter Beutner4d2858c2008-09-06 13:54:06 -0300286 .read_status = ttusbdecfe_dvbs_read_status,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700287
288 .diseqc_send_master_cmd = ttusbdecfe_dvbs_diseqc_send_master_cmd,
289 .set_voltage = ttusbdecfe_dvbs_set_voltage,
290 .set_tone = ttusbdecfe_dvbs_set_tone,
291};
292
293MODULE_DESCRIPTION("TTUSB DEC DVB-T/S Demodulator driver");
294MODULE_AUTHOR("Alex Woods/Andrew de Quincey");
295MODULE_LICENSE("GPL");
296
297EXPORT_SYMBOL(ttusbdecfe_dvbt_attach);
298EXPORT_SYMBOL(ttusbdecfe_dvbs_attach);