2     Samsung S5H1411 VSB/QAM demodulator driver
 
   4     Copyright (C) 2008 Steven Toth <stoth@hauppauge.com>
 
   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.
 
  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.
 
  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.
 
  22 #include <linux/kernel.h>
 
  23 #include <linux/init.h>
 
  24 #include <linux/module.h>
 
  25 #include <linux/string.h>
 
  26 #include <linux/slab.h>
 
  27 #include <linux/delay.h>
 
  28 #include "dvb_frontend.h"
 
  32 struct s5h1411_state {
 
  34         struct i2c_adapter *i2c;
 
  36         /* configuration settings */
 
  37         const struct s5h1411_config *config;
 
  39         struct dvb_frontend frontend;
 
  41         fe_modulation_t current_modulation;
 
  43         u32 current_frequency;
 
  51 #define dprintk(arg...) do {    \
 
  56 /* Register values to initialise the demod, defaults to VSB */
 
  57 static struct init_tab {
 
  62         { S5H1411_I2C_TOP_ADDR, 0x00, 0x0071, },
 
  63         { S5H1411_I2C_TOP_ADDR, 0x08, 0x0047, },
 
  64         { S5H1411_I2C_TOP_ADDR, 0x1c, 0x0400, },
 
  65         { S5H1411_I2C_TOP_ADDR, 0x1e, 0x0370, },
 
  66         { S5H1411_I2C_TOP_ADDR, 0x1f, 0x342a, },
 
  67         { S5H1411_I2C_TOP_ADDR, 0x24, 0x0231, },
 
  68         { S5H1411_I2C_TOP_ADDR, 0x25, 0x1011, },
 
  69         { S5H1411_I2C_TOP_ADDR, 0x26, 0x0f07, },
 
  70         { S5H1411_I2C_TOP_ADDR, 0x27, 0x0f04, },
 
  71         { S5H1411_I2C_TOP_ADDR, 0x28, 0x070f, },
 
  72         { S5H1411_I2C_TOP_ADDR, 0x29, 0x2820, },
 
  73         { S5H1411_I2C_TOP_ADDR, 0x2a, 0x102e, },
 
  74         { S5H1411_I2C_TOP_ADDR, 0x2b, 0x0220, },
 
  75         { S5H1411_I2C_TOP_ADDR, 0x2e, 0x0d0e, },
 
  76         { S5H1411_I2C_TOP_ADDR, 0x2f, 0x1013, },
 
  77         { S5H1411_I2C_TOP_ADDR, 0x31, 0x171b, },
 
  78         { S5H1411_I2C_TOP_ADDR, 0x32, 0x0e0f, },
 
  79         { S5H1411_I2C_TOP_ADDR, 0x33, 0x0f10, },
 
  80         { S5H1411_I2C_TOP_ADDR, 0x34, 0x170e, },
 
  81         { S5H1411_I2C_TOP_ADDR, 0x35, 0x4b10, },
 
  82         { S5H1411_I2C_TOP_ADDR, 0x36, 0x0f17, },
 
  83         { S5H1411_I2C_TOP_ADDR, 0x3c, 0x1577, },
 
  84         { S5H1411_I2C_TOP_ADDR, 0x3d, 0x081a, },
 
  85         { S5H1411_I2C_TOP_ADDR, 0x3e, 0x77ee, },
 
  86         { S5H1411_I2C_TOP_ADDR, 0x40, 0x1e09, },
 
  87         { S5H1411_I2C_TOP_ADDR, 0x41, 0x0f0c, },
 
  88         { S5H1411_I2C_TOP_ADDR, 0x42, 0x1f10, },
 
  89         { S5H1411_I2C_TOP_ADDR, 0x4d, 0x0509, },
 
  90         { S5H1411_I2C_TOP_ADDR, 0x4e, 0x0a00, },
 
  91         { S5H1411_I2C_TOP_ADDR, 0x50, 0x0000, },
 
  92         { S5H1411_I2C_TOP_ADDR, 0x5b, 0x0000, },
 
  93         { S5H1411_I2C_TOP_ADDR, 0x5c, 0x0008, },
 
  94         { S5H1411_I2C_TOP_ADDR, 0x57, 0x1101, },
 
  95         { S5H1411_I2C_TOP_ADDR, 0x65, 0x007c, },
 
  96         { S5H1411_I2C_TOP_ADDR, 0x68, 0x0512, },
 
  97         { S5H1411_I2C_TOP_ADDR, 0x69, 0x0258, },
 
  98         { S5H1411_I2C_TOP_ADDR, 0x70, 0x0004, },
 
  99         { S5H1411_I2C_TOP_ADDR, 0x71, 0x0007, },
 
 100         { S5H1411_I2C_TOP_ADDR, 0x76, 0x00a9, },
 
 101         { S5H1411_I2C_TOP_ADDR, 0x78, 0x3141, },
 
 102         { S5H1411_I2C_TOP_ADDR, 0x7a, 0x3141, },
 
 103         { S5H1411_I2C_TOP_ADDR, 0xb3, 0x8003, },
 
 104         { S5H1411_I2C_TOP_ADDR, 0xb5, 0xafbb, },
 
 105         { S5H1411_I2C_TOP_ADDR, 0xb5, 0xa6bb, },
 
 106         { S5H1411_I2C_TOP_ADDR, 0xb6, 0x0609, },
 
 107         { S5H1411_I2C_TOP_ADDR, 0xb7, 0x2f06, },
 
 108         { S5H1411_I2C_TOP_ADDR, 0xb8, 0x003f, },
 
 109         { S5H1411_I2C_TOP_ADDR, 0xb9, 0x2700, },
 
 110         { S5H1411_I2C_TOP_ADDR, 0xba, 0xfac8, },
 
 111         { S5H1411_I2C_TOP_ADDR, 0xbe, 0x1003, },
 
 112         { S5H1411_I2C_TOP_ADDR, 0xbf, 0x103f, },
 
 113         { S5H1411_I2C_TOP_ADDR, 0xce, 0x2000, },
 
 114         { S5H1411_I2C_TOP_ADDR, 0xcf, 0x0800, },
 
 115         { S5H1411_I2C_TOP_ADDR, 0xd0, 0x0800, },
 
 116         { S5H1411_I2C_TOP_ADDR, 0xd1, 0x0400, },
 
 117         { S5H1411_I2C_TOP_ADDR, 0xd2, 0x0800, },
 
 118         { S5H1411_I2C_TOP_ADDR, 0xd3, 0x2000, },
 
 119         { S5H1411_I2C_TOP_ADDR, 0xd4, 0x3000, },
 
 120         { S5H1411_I2C_TOP_ADDR, 0xdb, 0x4a9b, },
 
 121         { S5H1411_I2C_TOP_ADDR, 0xdc, 0x1000, },
 
 122         { S5H1411_I2C_TOP_ADDR, 0xde, 0x0001, },
 
 123         { S5H1411_I2C_TOP_ADDR, 0xdf, 0x0000, },
 
 124         { S5H1411_I2C_TOP_ADDR, 0xe3, 0x0301, },
 
 125         { S5H1411_I2C_QAM_ADDR, 0xf3, 0x0000, },
 
 126         { S5H1411_I2C_QAM_ADDR, 0xf3, 0x0001, },
 
 127         { S5H1411_I2C_QAM_ADDR, 0x08, 0x0600, },
 
 128         { S5H1411_I2C_QAM_ADDR, 0x18, 0x4201, },
 
 129         { S5H1411_I2C_QAM_ADDR, 0x1e, 0x6476, },
 
 130         { S5H1411_I2C_QAM_ADDR, 0x21, 0x0830, },
 
 131         { S5H1411_I2C_QAM_ADDR, 0x0c, 0x5679, },
 
 132         { S5H1411_I2C_QAM_ADDR, 0x0d, 0x579b, },
 
 133         { S5H1411_I2C_QAM_ADDR, 0x24, 0x0102, },
 
 134         { S5H1411_I2C_QAM_ADDR, 0x31, 0x7488, },
 
 135         { S5H1411_I2C_QAM_ADDR, 0x32, 0x0a08, },
 
 136         { S5H1411_I2C_QAM_ADDR, 0x3d, 0x8689, },
 
 137         { S5H1411_I2C_QAM_ADDR, 0x49, 0x0048, },
 
 138         { S5H1411_I2C_QAM_ADDR, 0x57, 0x2012, },
 
 139         { S5H1411_I2C_QAM_ADDR, 0x5d, 0x7676, },
 
 140         { S5H1411_I2C_QAM_ADDR, 0x04, 0x0400, },
 
 141         { S5H1411_I2C_QAM_ADDR, 0x58, 0x00c0, },
 
 142         { S5H1411_I2C_QAM_ADDR, 0x5b, 0x0100, },
 
 145 /* VSB SNR lookup table */
 
 146 static struct vsb_snr_tab {
 
 190 /* QAM64 SNR lookup table */
 
 191 static struct qam64_snr_tab {
 
 194 } qam64_snr_tab[] = {
 
 263 /* QAM256 SNR lookup table */
 
 264 static struct qam256_snr_tab {
 
 267 } qam256_snr_tab[] = {
 
 342 /* 8 bit registers, 16 bit values */
 
 343 static int s5h1411_writereg(struct s5h1411_state *state,
 
 344         u8 addr, u8 reg, u16 data)
 
 347         u8 buf [] = { reg, data >> 8,  data & 0xff };
 
 349         struct i2c_msg msg = { .addr = addr, .flags = 0, .buf = buf, .len = 3 };
 
 351         ret = i2c_transfer(state->i2c, &msg, 1);
 
 354                 printk(KERN_ERR "%s: writereg error 0x%02x 0x%02x 0x%04x, "
 
 355                        "ret == %i)\n", __func__, addr, reg, data, ret);
 
 357         return (ret != 1) ? -1 : 0;
 
 360 static u16 s5h1411_readreg(struct s5h1411_state *state, u8 addr, u8 reg)
 
 366         struct i2c_msg msg [] = {
 
 367                 { .addr = addr, .flags = 0, .buf = b0, .len = 1 },
 
 368                 { .addr = addr, .flags = I2C_M_RD, .buf = b1, .len = 2 } };
 
 370         ret = i2c_transfer(state->i2c, msg, 2);
 
 373                 printk(KERN_ERR "%s: readreg error (ret == %i)\n",
 
 375         return (b1[0] << 8) | b1[1];
 
 378 static int s5h1411_softreset(struct dvb_frontend *fe)
 
 380         struct s5h1411_state *state = fe->demodulator_priv;
 
 382         dprintk("%s()\n", __func__);
 
 384         s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf7, 0);
 
 385         s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf7, 1);
 
 389 static int s5h1411_set_if_freq(struct dvb_frontend *fe, int KHz)
 
 391         struct s5h1411_state *state = fe->demodulator_priv;
 
 393         dprintk("%s(%d KHz)\n", __func__, KHz);
 
 397                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x38, 0x10d9);
 
 398                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x39, 0x5342);
 
 399                 s5h1411_writereg(state, S5H1411_I2C_QAM_ADDR, 0x2c, 0x10d9);
 
 402                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x38, 0x1225);
 
 403                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x39, 0x1e96);
 
 404                 s5h1411_writereg(state, S5H1411_I2C_QAM_ADDR, 0x2c, 0x1225);
 
 407                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x38, 0x14bc);
 
 408                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x39, 0xb53e);
 
 409                 s5h1411_writereg(state, S5H1411_I2C_QAM_ADDR, 0x2c, 0x14bd);
 
 412                 dprintk("%s(%d KHz) Invalid, defaulting to 5380\n",
 
 414                 /* no break, need to continue */
 
 417                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x38, 0x1be4);
 
 418                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x39, 0x3655);
 
 419                 s5h1411_writereg(state, S5H1411_I2C_QAM_ADDR, 0x2c, 0x1be4);
 
 423         state->if_freq = KHz;
 
 428 static int s5h1411_set_mpeg_timing(struct dvb_frontend *fe, int mode)
 
 430         struct s5h1411_state *state = fe->demodulator_priv;
 
 433         dprintk("%s(%d)\n", __func__, mode);
 
 435         val = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xbe) & 0xcfff;
 
 437         case S5H1411_MPEGTIMING_CONTINOUS_INVERTING_CLOCK:
 
 440         case S5H1411_MPEGTIMING_CONTINOUS_NONINVERTING_CLOCK:
 
 441                 dprintk("%s(%d) Mode1 or Defaulting\n", __func__, mode);
 
 444         case S5H1411_MPEGTIMING_NONCONTINOUS_INVERTING_CLOCK:
 
 447         case S5H1411_MPEGTIMING_NONCONTINOUS_NONINVERTING_CLOCK:
 
 454         /* Configure MPEG Signal Timing charactistics */
 
 455         return s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xbe, val);
 
 458 static int s5h1411_set_spectralinversion(struct dvb_frontend *fe, int inversion)
 
 460         struct s5h1411_state *state = fe->demodulator_priv;
 
 463         dprintk("%s(%d)\n", __func__, inversion);
 
 464         val = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0x24) & ~0x1000;
 
 467                 val |= 0x1000; /* Inverted */
 
 471         state->inversion = inversion;
 
 472         return s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x24, val);
 
 475 static int s5h1411_enable_modulation(struct dvb_frontend *fe,
 
 478         struct s5h1411_state *state = fe->demodulator_priv;
 
 480         dprintk("%s(0x%08x)\n", __func__, m);
 
 484                 dprintk("%s() VSB_8\n", __func__);
 
 485                 s5h1411_set_if_freq(fe, state->config->vsb_if);
 
 486                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x00, 0x71);
 
 487                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf6, 0x00);
 
 488                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xcd, 0xf1);
 
 492                 dprintk("%s() QAM_AUTO (64/256)\n", __func__);
 
 493                 s5h1411_set_if_freq(fe, state->config->qam_if);
 
 494                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0x00, 0x0171);
 
 495                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf6, 0x0001);
 
 496                 s5h1411_writereg(state, S5H1411_I2C_QAM_ADDR, 0x16, 0x1101);
 
 497                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xcd, 0x00f0);
 
 500                 dprintk("%s() Invalid modulation\n", __func__);
 
 504         state->current_modulation = m;
 
 505         s5h1411_softreset(fe);
 
 510 static int s5h1411_i2c_gate_ctrl(struct dvb_frontend *fe, int enable)
 
 512         struct s5h1411_state *state = fe->demodulator_priv;
 
 514         dprintk("%s(%d)\n", __func__, enable);
 
 517                 return s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf5, 1);
 
 519                 return s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf5, 0);
 
 522 static int s5h1411_set_gpio(struct dvb_frontend *fe, int enable)
 
 524         struct s5h1411_state *state = fe->demodulator_priv;
 
 527         dprintk("%s(%d)\n", __func__, enable);
 
 529         val = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xe0) & ~0x02;
 
 532                 return s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xe0,
 
 535                 return s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xe0, val);
 
 538 static int s5h1411_sleep(struct dvb_frontend *fe, int enable)
 
 540         struct s5h1411_state *state = fe->demodulator_priv;
 
 542         dprintk("%s(%d)\n", __func__, enable);
 
 545                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf4, 1);
 
 547                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf4, 0);
 
 548                 s5h1411_softreset(fe);
 
 554 static int s5h1411_register_reset(struct dvb_frontend *fe)
 
 556         struct s5h1411_state *state = fe->demodulator_priv;
 
 558         dprintk("%s()\n", __func__);
 
 560         return s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf3, 0);
 
 563 /* Talk to the demod, set the FEC, GUARD, QAM settings etc */
 
 564 static int s5h1411_set_frontend(struct dvb_frontend *fe,
 
 565         struct dvb_frontend_parameters *p)
 
 567         struct s5h1411_state *state = fe->demodulator_priv;
 
 569         dprintk("%s(frequency=%d)\n", __func__, p->frequency);
 
 571         s5h1411_softreset(fe);
 
 573         state->current_frequency = p->frequency;
 
 575         s5h1411_enable_modulation(fe, p->u.vsb.modulation);
 
 577         /* Allow the demod to settle */
 
 580         if (fe->ops.tuner_ops.set_params) {
 
 581                 if (fe->ops.i2c_gate_ctrl)
 
 582                         fe->ops.i2c_gate_ctrl(fe, 1);
 
 584                 fe->ops.tuner_ops.set_params(fe, p);
 
 586                 if (fe->ops.i2c_gate_ctrl)
 
 587                         fe->ops.i2c_gate_ctrl(fe, 0);
 
 593 /* Reset the demod hardware and reset all of the configuration registers
 
 594    to a default state. */
 
 595 static int s5h1411_init(struct dvb_frontend *fe)
 
 597         struct s5h1411_state *state = fe->demodulator_priv;
 
 600         dprintk("%s()\n", __func__);
 
 602         s5h1411_sleep(fe, 0);
 
 603         s5h1411_register_reset(fe);
 
 605         for (i = 0; i < ARRAY_SIZE(init_tab); i++)
 
 606                 s5h1411_writereg(state, init_tab[i].addr,
 
 610         /* The datasheet says that after initialisation, VSB is default */
 
 611         state->current_modulation = VSB_8;
 
 613         if (state->config->output_mode == S5H1411_SERIAL_OUTPUT)
 
 615                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xbd, 0x1101);
 
 618                 s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xbd, 0x1001);
 
 620         s5h1411_set_spectralinversion(fe, state->config->inversion);
 
 621         s5h1411_set_if_freq(fe, state->config->vsb_if);
 
 622         s5h1411_set_gpio(fe, state->config->gpio);
 
 623         s5h1411_set_mpeg_timing(fe, state->config->mpeg_timing);
 
 624         s5h1411_softreset(fe);
 
 626         /* Note: Leaving the I2C gate closed. */
 
 627         s5h1411_i2c_gate_ctrl(fe, 0);
 
 632 static int s5h1411_read_status(struct dvb_frontend *fe, fe_status_t *status)
 
 634         struct s5h1411_state *state = fe->demodulator_priv;
 
 636         u32 tuner_status = 0;
 
 640         /* Get the demodulator status */
 
 641         reg = (s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xf2) >> 15)
 
 644                 *status |= FE_HAS_LOCK | FE_HAS_CARRIER | FE_HAS_SIGNAL;
 
 646         switch (state->current_modulation) {
 
 649                 reg = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xf0);
 
 651                         *status |= FE_HAS_VITERBI;
 
 653                         *status |= FE_HAS_SYNC;
 
 656                 reg = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0x5e);
 
 658                         *status |= FE_HAS_SYNC;
 
 659                 reg = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xf2);
 
 661                         *status |= FE_HAS_VITERBI;
 
 667         switch (state->config->status_mode) {
 
 668         case S5H1411_DEMODLOCKING:
 
 669                 if (*status & FE_HAS_VITERBI)
 
 670                         *status |= FE_HAS_CARRIER | FE_HAS_SIGNAL;
 
 672         case S5H1411_TUNERLOCKING:
 
 673                 /* Get the tuner status */
 
 674                 if (fe->ops.tuner_ops.get_status) {
 
 675                         if (fe->ops.i2c_gate_ctrl)
 
 676                                 fe->ops.i2c_gate_ctrl(fe, 1);
 
 678                         fe->ops.tuner_ops.get_status(fe, &tuner_status);
 
 680                         if (fe->ops.i2c_gate_ctrl)
 
 681                                 fe->ops.i2c_gate_ctrl(fe, 0);
 
 684                         *status |= FE_HAS_CARRIER | FE_HAS_SIGNAL;
 
 688         dprintk("%s() status 0x%08x\n", __func__, *status);
 
 693 static int s5h1411_qam256_lookup_snr(struct dvb_frontend *fe, u16 *snr, u16 v)
 
 695         int i, ret = -EINVAL;
 
 696         dprintk("%s()\n", __func__);
 
 698         for (i = 0; i < ARRAY_SIZE(qam256_snr_tab); i++) {
 
 699                 if (v < qam256_snr_tab[i].val) {
 
 700                         *snr = qam256_snr_tab[i].data;
 
 708 static int s5h1411_qam64_lookup_snr(struct dvb_frontend *fe, u16 *snr, u16 v)
 
 710         int i, ret = -EINVAL;
 
 711         dprintk("%s()\n", __func__);
 
 713         for (i = 0; i < ARRAY_SIZE(qam64_snr_tab); i++) {
 
 714                 if (v < qam64_snr_tab[i].val) {
 
 715                         *snr = qam64_snr_tab[i].data;
 
 723 static int s5h1411_vsb_lookup_snr(struct dvb_frontend *fe, u16 *snr, u16 v)
 
 725         int i, ret = -EINVAL;
 
 726         dprintk("%s()\n", __func__);
 
 728         for (i = 0; i < ARRAY_SIZE(vsb_snr_tab); i++) {
 
 729                 if (v > vsb_snr_tab[i].val) {
 
 730                         *snr = vsb_snr_tab[i].data;
 
 735         dprintk("%s() snr=%d\n", __func__, *snr);
 
 739 static int s5h1411_read_snr(struct dvb_frontend *fe, u16 *snr)
 
 741         struct s5h1411_state *state = fe->demodulator_priv;
 
 743         dprintk("%s()\n", __func__);
 
 745         switch (state->current_modulation) {
 
 747                 reg = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xf1);
 
 748                 return s5h1411_qam64_lookup_snr(fe, snr, reg);
 
 750                 reg = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xf1);
 
 751                 return s5h1411_qam256_lookup_snr(fe, snr, reg);
 
 753                 reg = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR,
 
 755                 return s5h1411_vsb_lookup_snr(fe, snr, reg);
 
 763 static int s5h1411_read_signal_strength(struct dvb_frontend *fe,
 
 764         u16 *signal_strength)
 
 766         return s5h1411_read_snr(fe, signal_strength);
 
 769 static int s5h1411_read_ucblocks(struct dvb_frontend *fe, u32 *ucblocks)
 
 771         struct s5h1411_state *state = fe->demodulator_priv;
 
 773         *ucblocks = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0xc9);
 
 778 static int s5h1411_read_ber(struct dvb_frontend *fe, u32 *ber)
 
 780         return s5h1411_read_ucblocks(fe, ber);
 
 783 static int s5h1411_get_frontend(struct dvb_frontend *fe,
 
 784                                 struct dvb_frontend_parameters *p)
 
 786         struct s5h1411_state *state = fe->demodulator_priv;
 
 788         p->frequency = state->current_frequency;
 
 789         p->u.vsb.modulation = state->current_modulation;
 
 794 static int s5h1411_get_tune_settings(struct dvb_frontend *fe,
 
 795                                      struct dvb_frontend_tune_settings *tune)
 
 797         tune->min_delay_ms = 1000;
 
 801 static void s5h1411_release(struct dvb_frontend *fe)
 
 803         struct s5h1411_state *state = fe->demodulator_priv;
 
 807 static struct dvb_frontend_ops s5h1411_ops;
 
 809 struct dvb_frontend *s5h1411_attach(const struct s5h1411_config *config,
 
 810                                     struct i2c_adapter *i2c)
 
 812         struct s5h1411_state *state = NULL;
 
 815         /* allocate memory for the internal state */
 
 816         state = kmalloc(sizeof(struct s5h1411_state), GFP_KERNEL);
 
 820         /* setup the state */
 
 821         state->config = config;
 
 823         state->current_modulation = VSB_8;
 
 824         state->inversion = state->config->inversion;
 
 826         /* check if the demod exists */
 
 827         reg = s5h1411_readreg(state, S5H1411_I2C_TOP_ADDR, 0x05);
 
 831         /* create dvb_frontend */
 
 832         memcpy(&state->frontend.ops, &s5h1411_ops,
 
 833                sizeof(struct dvb_frontend_ops));
 
 835         state->frontend.demodulator_priv = state;
 
 837         if (s5h1411_init(&state->frontend) != 0) {
 
 838                 printk(KERN_ERR "%s: Failed to initialize correctly\n",
 
 843         /* Note: Leaving the I2C gate open here. */
 
 844         s5h1411_writereg(state, S5H1411_I2C_TOP_ADDR, 0xf5, 1);
 
 846         return &state->frontend;
 
 852 EXPORT_SYMBOL(s5h1411_attach);
 
 854 static struct dvb_frontend_ops s5h1411_ops = {
 
 857                 .name                   = "Samsung S5H1411 QAM/8VSB Frontend",
 
 859                 .frequency_min          = 54000000,
 
 860                 .frequency_max          = 858000000,
 
 861                 .frequency_stepsize     = 62500,
 
 862                 .caps = FE_CAN_QAM_64 | FE_CAN_QAM_256 | FE_CAN_8VSB
 
 865         .init                 = s5h1411_init,
 
 866         .i2c_gate_ctrl        = s5h1411_i2c_gate_ctrl,
 
 867         .set_frontend         = s5h1411_set_frontend,
 
 868         .get_frontend         = s5h1411_get_frontend,
 
 869         .get_tune_settings    = s5h1411_get_tune_settings,
 
 870         .read_status          = s5h1411_read_status,
 
 871         .read_ber             = s5h1411_read_ber,
 
 872         .read_signal_strength = s5h1411_read_signal_strength,
 
 873         .read_snr             = s5h1411_read_snr,
 
 874         .read_ucblocks        = s5h1411_read_ucblocks,
 
 875         .release              = s5h1411_release,
 
 878 module_param(debug, int, 0644);
 
 879 MODULE_PARM_DESC(debug, "Enable verbose debug messages");
 
 881 MODULE_DESCRIPTION("Samsung S5H1411 QAM-B/ATSC Demodulator driver");
 
 882 MODULE_AUTHOR("Steven Toth");
 
 883 MODULE_LICENSE("GPL");