1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19#include <linux/module.h>
20#include <linux/init.h>
21#include <linux/interrupt.h>
22#include <linux/delay.h>
23#include <linux/slab.h>
24#include <linux/poll.h>
25#include <linux/io.h>
26#include <linux/pci.h>
27#include <linux/pci_ids.h>
28#include <linux/timer.h>
29#include <linux/i2c.h>
30#include <linux/swab.h>
31#include <linux/vmalloc.h>
32
33#include "ddbridge.h"
34#include "ddbridge-regs.h"
35#include "ddbridge-io.h"
36#include "ddbridge-mci.h"
37
38#include "ddbridge-max.h"
39#include "mxl5xx.h"
40
41
42
43
44static int fmode;
45module_param(fmode, int, 0444);
46MODULE_PARM_DESC(fmode, "frontend emulation mode");
47
48static int fmode_sat = -1;
49module_param(fmode_sat, int, 0444);
50MODULE_PARM_DESC(fmode_sat, "set frontend emulation mode sat");
51
52static int old_quattro;
53module_param(old_quattro, int, 0444);
54MODULE_PARM_DESC(old_quattro, "old quattro LNB input order ");
55
56
57
58static int lnb_command(struct ddb *dev, u32 link, u32 lnb, u32 cmd)
59{
60 u32 c, v = 0, tag = DDB_LINK_TAG(link);
61
62 v = LNB_TONE & (dev->link[link].lnb.tone << (15 - lnb));
63 ddbwritel(dev, cmd | v, tag | LNB_CONTROL(lnb));
64 for (c = 0; c < 10; c++) {
65 v = ddbreadl(dev, tag | LNB_CONTROL(lnb));
66 if ((v & LNB_BUSY) == 0)
67 break;
68 msleep(20);
69 }
70 if (c == 10)
71 dev_info(dev->dev, "%s lnb = %08x cmd = %08x\n",
72 __func__, lnb, cmd);
73 return 0;
74}
75
76static int max_send_master_cmd(struct dvb_frontend *fe,
77 struct dvb_diseqc_master_cmd *cmd)
78{
79 struct ddb_input *input = fe->sec_priv;
80 struct ddb_port *port = input->port;
81 struct ddb *dev = port->dev;
82 struct ddb_dvb *dvb = &port->dvb[input->nr & 1];
83 u32 tag = DDB_LINK_TAG(port->lnr);
84 int i;
85 u32 fmode = dev->link[port->lnr].lnb.fmode;
86
87 if (fmode == 2 || fmode == 1)
88 return 0;
89 if (dvb->diseqc_send_master_cmd)
90 dvb->diseqc_send_master_cmd(fe, cmd);
91
92 mutex_lock(&dev->link[port->lnr].lnb.lock);
93 ddbwritel(dev, 0, tag | LNB_BUF_LEVEL(dvb->input));
94 for (i = 0; i < cmd->msg_len; i++)
95 ddbwritel(dev, cmd->msg[i], tag | LNB_BUF_WRITE(dvb->input));
96 lnb_command(dev, port->lnr, dvb->input, LNB_CMD_DISEQC);
97 mutex_unlock(&dev->link[port->lnr].lnb.lock);
98 return 0;
99}
100
101static int lnb_send_diseqc(struct ddb *dev, u32 link, u32 input,
102 struct dvb_diseqc_master_cmd *cmd)
103{
104 u32 tag = DDB_LINK_TAG(link);
105 int i;
106
107 ddbwritel(dev, 0, tag | LNB_BUF_LEVEL(input));
108 for (i = 0; i < cmd->msg_len; i++)
109 ddbwritel(dev, cmd->msg[i], tag | LNB_BUF_WRITE(input));
110 lnb_command(dev, link, input, LNB_CMD_DISEQC);
111 return 0;
112}
113
114static int lnb_set_sat(struct ddb *dev, u32 link, u32 input, u32 sat, u32 band,
115 u32 hor)
116{
117 struct dvb_diseqc_master_cmd cmd = {
118 .msg = {0xe0, 0x10, 0x38, 0xf0, 0x00, 0x00},
119 .msg_len = 4
120 };
121 cmd.msg[3] = 0xf0 | (((sat << 2) & 0x0c) | (band ? 1 : 0) |
122 (hor ? 2 : 0));
123 return lnb_send_diseqc(dev, link, input, &cmd);
124}
125
126static int lnb_set_tone(struct ddb *dev, u32 link, u32 input,
127 enum fe_sec_tone_mode tone)
128{
129 int s = 0;
130 u32 mask = (1ULL << input);
131
132 switch (tone) {
133 case SEC_TONE_OFF:
134 if (!(dev->link[link].lnb.tone & mask))
135 return 0;
136 dev->link[link].lnb.tone &= ~(1ULL << input);
137 break;
138 case SEC_TONE_ON:
139 if (dev->link[link].lnb.tone & mask)
140 return 0;
141 dev->link[link].lnb.tone |= (1ULL << input);
142 break;
143 default:
144 s = -EINVAL;
145 break;
146 }
147 if (!s)
148 s = lnb_command(dev, link, input, LNB_CMD_NOP);
149 return s;
150}
151
152static int lnb_set_voltage(struct ddb *dev, u32 link, u32 input,
153 enum fe_sec_voltage voltage)
154{
155 int s = 0;
156
157 if (dev->link[link].lnb.oldvoltage[input] == voltage)
158 return 0;
159 switch (voltage) {
160 case SEC_VOLTAGE_OFF:
161 if (dev->link[link].lnb.voltage[input])
162 return 0;
163 lnb_command(dev, link, input, LNB_CMD_OFF);
164 break;
165 case SEC_VOLTAGE_13:
166 lnb_command(dev, link, input, LNB_CMD_LOW);
167 break;
168 case SEC_VOLTAGE_18:
169 lnb_command(dev, link, input, LNB_CMD_HIGH);
170 break;
171 default:
172 s = -EINVAL;
173 break;
174 }
175 dev->link[link].lnb.oldvoltage[input] = voltage;
176 return s;
177}
178
179static int max_set_input_unlocked(struct dvb_frontend *fe, int in)
180{
181 struct ddb_input *input = fe->sec_priv;
182 struct ddb_port *port = input->port;
183 struct ddb *dev = port->dev;
184 struct ddb_dvb *dvb = &port->dvb[input->nr & 1];
185 int res = 0;
186
187 if (in > 3)
188 return -EINVAL;
189 if (dvb->input != in) {
190 u32 bit = (1ULL << input->nr);
191 u32 obit =
192 dev->link[port->lnr].lnb.voltage[dvb->input & 3] & bit;
193
194 dev->link[port->lnr].lnb.voltage[dvb->input & 3] &= ~bit;
195 dvb->input = in;
196 dev->link[port->lnr].lnb.voltage[dvb->input & 3] |= obit;
197 }
198 res = dvb->set_input(fe, in);
199 return res;
200}
201
202static int max_set_tone(struct dvb_frontend *fe, enum fe_sec_tone_mode tone)
203{
204 struct ddb_input *input = fe->sec_priv;
205 struct ddb_port *port = input->port;
206 struct ddb *dev = port->dev;
207 struct ddb_dvb *dvb = &port->dvb[input->nr & 1];
208 int tuner = 0;
209 int res = 0;
210 u32 fmode = dev->link[port->lnr].lnb.fmode;
211
212 mutex_lock(&dev->link[port->lnr].lnb.lock);
213 dvb->tone = tone;
214 switch (fmode) {
215 default:
216 case 0:
217 case 3:
218 res = lnb_set_tone(dev, port->lnr, dvb->input, tone);
219 break;
220 case 1:
221 case 2:
222 if (old_quattro) {
223 if (dvb->tone == SEC_TONE_ON)
224 tuner |= 2;
225 if (dvb->voltage == SEC_VOLTAGE_18)
226 tuner |= 1;
227 } else {
228 if (dvb->tone == SEC_TONE_ON)
229 tuner |= 1;
230 if (dvb->voltage == SEC_VOLTAGE_18)
231 tuner |= 2;
232 }
233 res = max_set_input_unlocked(fe, tuner);
234 break;
235 }
236 mutex_unlock(&dev->link[port->lnr].lnb.lock);
237 return res;
238}
239
240static int max_set_voltage(struct dvb_frontend *fe, enum fe_sec_voltage voltage)
241{
242 struct ddb_input *input = fe->sec_priv;
243 struct ddb_port *port = input->port;
244 struct ddb *dev = port->dev;
245 struct ddb_dvb *dvb = &port->dvb[input->nr & 1];
246 int tuner = 0;
247 u32 nv, ov = dev->link[port->lnr].lnb.voltages;
248 int res = 0;
249 u32 fmode = dev->link[port->lnr].lnb.fmode;
250
251 mutex_lock(&dev->link[port->lnr].lnb.lock);
252 dvb->voltage = voltage;
253
254 switch (fmode) {
255 case 3:
256 default:
257 case 0:
258 if (fmode == 3)
259 max_set_input_unlocked(fe, 0);
260 if (voltage == SEC_VOLTAGE_OFF)
261 dev->link[port->lnr].lnb.voltage[dvb->input] &=
262 ~(1ULL << input->nr);
263 else
264 dev->link[port->lnr].lnb.voltage[dvb->input] |=
265 (1ULL << input->nr);
266
267 res = lnb_set_voltage(dev, port->lnr, dvb->input, voltage);
268 break;
269 case 1:
270 case 2:
271 if (voltage == SEC_VOLTAGE_OFF)
272 dev->link[port->lnr].lnb.voltages &=
273 ~(1ULL << input->nr);
274 else
275 dev->link[port->lnr].lnb.voltages |=
276 (1ULL << input->nr);
277
278 nv = dev->link[port->lnr].lnb.voltages;
279
280 if (old_quattro) {
281 if (dvb->tone == SEC_TONE_ON)
282 tuner |= 2;
283 if (dvb->voltage == SEC_VOLTAGE_18)
284 tuner |= 1;
285 } else {
286 if (dvb->tone == SEC_TONE_ON)
287 tuner |= 1;
288 if (dvb->voltage == SEC_VOLTAGE_18)
289 tuner |= 2;
290 }
291 res = max_set_input_unlocked(fe, tuner);
292
293 if (nv != ov) {
294 if (nv) {
295 lnb_set_voltage(
296 dev, port->lnr,
297 0, SEC_VOLTAGE_13);
298 if (fmode == 1) {
299 lnb_set_voltage(
300 dev, port->lnr,
301 0, SEC_VOLTAGE_13);
302 if (old_quattro) {
303 lnb_set_voltage(
304 dev, port->lnr,
305 1, SEC_VOLTAGE_18);
306 lnb_set_voltage(
307 dev, port->lnr,
308 2, SEC_VOLTAGE_13);
309 } else {
310 lnb_set_voltage(
311 dev, port->lnr,
312 1, SEC_VOLTAGE_13);
313 lnb_set_voltage(
314 dev, port->lnr,
315 2, SEC_VOLTAGE_18);
316 }
317 lnb_set_voltage(
318 dev, port->lnr,
319 3, SEC_VOLTAGE_18);
320 }
321 } else {
322 lnb_set_voltage(
323 dev, port->lnr,
324 0, SEC_VOLTAGE_OFF);
325 if (fmode == 1) {
326 lnb_set_voltage(
327 dev, port->lnr,
328 1, SEC_VOLTAGE_OFF);
329 lnb_set_voltage(
330 dev, port->lnr,
331 2, SEC_VOLTAGE_OFF);
332 lnb_set_voltage(
333 dev, port->lnr,
334 3, SEC_VOLTAGE_OFF);
335 }
336 }
337 }
338 break;
339 }
340 mutex_unlock(&dev->link[port->lnr].lnb.lock);
341 return res;
342}
343
344static int max_enable_high_lnb_voltage(struct dvb_frontend *fe, long arg)
345{
346 return 0;
347}
348
349static int max_send_burst(struct dvb_frontend *fe, enum fe_sec_mini_cmd burst)
350{
351 return 0;
352}
353
354static int mxl_fw_read(void *priv, u8 *buf, u32 len)
355{
356 struct ddb_link *link = priv;
357 struct ddb *dev = link->dev;
358
359 dev_info(dev->dev, "Read mxl_fw from link %u\n", link->nr);
360
361 return ddbridge_flashread(dev, link->nr, buf, 0xc0000, len);
362}
363
364int ddb_lnb_init_fmode(struct ddb *dev, struct ddb_link *link, u32 fm)
365{
366 u32 l = link->nr;
367
368 if (link->lnb.fmode == fm)
369 return 0;
370 dev_info(dev->dev, "Set fmode link %u = %u\n", l, fm);
371 mutex_lock(&link->lnb.lock);
372 if (fm == 2 || fm == 1) {
373 if (fmode_sat >= 0) {
374 lnb_set_sat(dev, l, 0, fmode_sat, 0, 0);
375 if (old_quattro) {
376 lnb_set_sat(dev, l, 1, fmode_sat, 0, 1);
377 lnb_set_sat(dev, l, 2, fmode_sat, 1, 0);
378 } else {
379 lnb_set_sat(dev, l, 1, fmode_sat, 1, 0);
380 lnb_set_sat(dev, l, 2, fmode_sat, 0, 1);
381 }
382 lnb_set_sat(dev, l, 3, fmode_sat, 1, 1);
383 }
384 lnb_set_tone(dev, l, 0, SEC_TONE_OFF);
385 if (old_quattro) {
386 lnb_set_tone(dev, l, 1, SEC_TONE_OFF);
387 lnb_set_tone(dev, l, 2, SEC_TONE_ON);
388 } else {
389 lnb_set_tone(dev, l, 1, SEC_TONE_ON);
390 lnb_set_tone(dev, l, 2, SEC_TONE_OFF);
391 }
392 lnb_set_tone(dev, l, 3, SEC_TONE_ON);
393 }
394 link->lnb.fmode = fm;
395 mutex_unlock(&link->lnb.lock);
396 return 0;
397}
398
399static struct mxl5xx_cfg mxl5xx = {
400 .adr = 0x60,
401 .type = 0x01,
402 .clk = 27000000,
403 .ts_clk = 139,
404 .cap = 12,
405 .fw_read = mxl_fw_read,
406};
407
408int ddb_fe_attach_mxl5xx(struct ddb_input *input)
409{
410 struct ddb *dev = input->port->dev;
411 struct i2c_adapter *i2c = &input->port->i2c->adap;
412 struct ddb_dvb *dvb = &input->port->dvb[input->nr & 1];
413 struct ddb_port *port = input->port;
414 struct ddb_link *link = &dev->link[port->lnr];
415 struct mxl5xx_cfg cfg;
416 int demod, tuner;
417
418 cfg = mxl5xx;
419 cfg.fw_priv = link;
420 dvb->set_input = NULL;
421
422 demod = input->nr;
423 tuner = demod & 3;
424 if (fmode == 3)
425 tuner = 0;
426
427 dvb->fe = dvb_attach(mxl5xx_attach, i2c, &cfg,
428 demod, tuner, &dvb->set_input);
429
430 if (!dvb->fe) {
431 dev_err(dev->dev, "No MXL5XX found!\n");
432 return -ENODEV;
433 }
434
435 if (!dvb->set_input) {
436 dev_err(dev->dev, "No mxl5xx_set_input function pointer!\n");
437 return -ENODEV;
438 }
439
440 if (input->nr < 4) {
441 lnb_command(dev, port->lnr, input->nr, LNB_CMD_INIT);
442 lnb_set_voltage(dev, port->lnr, input->nr, SEC_VOLTAGE_OFF);
443 }
444 ddb_lnb_init_fmode(dev, link, fmode);
445
446 dvb->fe->ops.set_voltage = max_set_voltage;
447 dvb->fe->ops.enable_high_lnb_voltage = max_enable_high_lnb_voltage;
448 dvb->fe->ops.set_tone = max_set_tone;
449 dvb->diseqc_send_master_cmd = dvb->fe->ops.diseqc_send_master_cmd;
450 dvb->fe->ops.diseqc_send_master_cmd = max_send_master_cmd;
451 dvb->fe->ops.diseqc_send_burst = max_send_burst;
452 dvb->fe->sec_priv = input;
453 dvb->input = tuner;
454 return 0;
455}
456
457
458
459
460int ddb_fe_attach_mci(struct ddb_input *input)
461{
462 struct ddb *dev = input->port->dev;
463 struct ddb_dvb *dvb = &input->port->dvb[input->nr & 1];
464 struct ddb_port *port = input->port;
465 struct ddb_link *link = &dev->link[port->lnr];
466 int demod, tuner;
467
468 demod = input->nr;
469 tuner = demod & 3;
470 if (fmode == 3)
471 tuner = 0;
472 dvb->fe = ddb_mci_attach(input, 0, demod, &dvb->set_input);
473 if (!dvb->fe) {
474 dev_err(dev->dev, "No MAXSX8 found!\n");
475 return -ENODEV;
476 }
477 if (!dvb->set_input) {
478 dev_err(dev->dev, "No MCI set_input function pointer!\n");
479 return -ENODEV;
480 }
481 if (input->nr < 4) {
482 lnb_command(dev, port->lnr, input->nr, LNB_CMD_INIT);
483 lnb_set_voltage(dev, port->lnr, input->nr, SEC_VOLTAGE_OFF);
484 }
485 ddb_lnb_init_fmode(dev, link, fmode);
486
487 dvb->fe->ops.set_voltage = max_set_voltage;
488 dvb->fe->ops.enable_high_lnb_voltage = max_enable_high_lnb_voltage;
489 dvb->fe->ops.set_tone = max_set_tone;
490 dvb->diseqc_send_master_cmd = dvb->fe->ops.diseqc_send_master_cmd;
491 dvb->fe->ops.diseqc_send_master_cmd = max_send_master_cmd;
492 dvb->fe->ops.diseqc_send_burst = max_send_burst;
493 dvb->fe->sec_priv = input;
494 dvb->input = tuner;
495 return 0;
496}
497