Anforderungen  |   Konzepte  |   Entwurf  |   Entwicklung  |   Qualitätssicherung  |   Lebenszyklus  |   Steuerung
 
 
 
 


Quelle  mt2060.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0-or-later
/*
 *  Driver for Microtune MT2060 "Single chip dual conversion broadband tuner"
 *
 *  Copyright (c) 2006 Olivier DANET <odanet@caramail.com>
 */


/* In that file, frequencies are expressed in kiloHertz to avoid 32 bits overflows */

#include <linux/module.h>
#include <linux/delay.h>
#include <linux/dvb/frontend.h>
#include <linux/i2c.h>
#include <linux/slab.h>

#include <media/dvb_frontend.h>

#include "mt2060.h"
#include "mt2060_priv.h"

static int debug;
module_param(debug, int, 0644);
MODULE_PARM_DESC(debug, "Turn on/off debugging (default:off).");

#define dprintk(args...) do { if (debug) {printk(KERN_DEBUG "MT2060: " args); printk("\n"); }} while (0)

// Reads a single register
static int mt2060_readreg(struct mt2060_priv *priv, u8 reg, u8 *val)
{
 struct i2c_msg msg[2] = {
  { .addr = priv->cfg->i2c_address, .flags = 0, .len = 1 },
  { .addr = priv->cfg->i2c_address, .flags = I2C_M_RD, .len = 1 },
 };
 int rc = 0;
 u8 *b;

 b = kmalloc(2, GFP_KERNEL);
 if (!b)
  return -ENOMEM;

 b[0] = reg;
 b[1] = 0;

 msg[0].buf = b;
 msg[1].buf = b + 1;

 if (i2c_transfer(priv->i2c, msg, 2) != 2) {
  printk(KERN_WARNING "mt2060 I2C read failed\n");
  rc = -EREMOTEIO;
 }
 *val = b[1];
 kfree(b);

 return rc;
}

// Writes a single register
static int mt2060_writereg(struct mt2060_priv *priv, u8 reg, u8 val)
{
 struct i2c_msg msg = {
  .addr = priv->cfg->i2c_address, .flags = 0, .len = 2
 };
 u8 *buf;
 int rc = 0;

 buf = kmalloc(2, GFP_KERNEL);
 if (!buf)
  return -ENOMEM;

 buf[0] = reg;
 buf[1] = val;

 msg.buf = buf;

 if (i2c_transfer(priv->i2c, &msg, 1) != 1) {
  printk(KERN_WARNING "mt2060 I2C write failed\n");
  rc = -EREMOTEIO;
 }
 kfree(buf);
 return rc;
}

// Writes a set of consecutive registers
static int mt2060_writeregs(struct mt2060_priv *priv,u8 *buf, u8 len)
{
 int rem, val_len;
 u8 *xfer_buf;
 int rc = 0;
 struct i2c_msg msg = {
  .addr = priv->cfg->i2c_address, .flags = 0
 };

 xfer_buf = kmalloc(16, GFP_KERNEL);
 if (!xfer_buf)
  return -ENOMEM;

 msg.buf = xfer_buf;

 for (rem = len - 1; rem > 0; rem -= priv->i2c_max_regs) {
  val_len = min_t(int, rem, priv->i2c_max_regs);
  msg.len = 1 + val_len;
  xfer_buf[0] = buf[0] + len - 1 - rem;
  memcpy(&xfer_buf[1], &buf[1 + len - 1 - rem], val_len);

  if (i2c_transfer(priv->i2c, &msg, 1) != 1) {
   printk(KERN_WARNING "mt2060 I2C write failed (len=%i)\n", val_len);
   rc = -EREMOTEIO;
   break;
  }
 }

 kfree(xfer_buf);
 return rc;
}

// Initialisation sequences
// LNABAND=3, NUM1=0x3C, DIV1=0x74, NUM2=0x1080, DIV2=0x49
static u8 mt2060_config1[] = {
 REG_LO1C1,
 0x3F, 0x74, 0x00, 0x08, 0x93
};

// FMCG=2, GP2=0, GP1=0
static u8 mt2060_config2[] = {
 REG_MISC_CTRL,
 0x20, 0x1E, 0x30, 0xff, 0x80, 0xff, 0x00, 0x2c, 0x42
};

//  VGAG=3, V1CSE=1

#ifdef  MT2060_SPURCHECK
/* The function below calculates the frequency offset between the output frequency if2
 and the closer cross modulation subcarrier between lo1 and lo2 up to the tenth harmonic */

static int mt2060_spurcalc(u32 lo1,u32 lo2,u32 if2)
{
 int I,J;
 int dia,diamin,diff;
 diamin=1000000;
 for (I = 1; I < 10; I++) {
  J = ((2*I*lo1)/lo2+1)/2;
  diff = I*(int)lo1-J*(int)lo2;
  if (diff < 0) diff=-diff;
  dia = (diff-(int)if2);
  if (dia < 0) dia=-dia;
  if (diamin > dia) diamin=dia;
 }
 return diamin;
}

#define BANDWIDTH 4000 // kHz

/* Calculates the frequency offset to add to avoid spurs. Returns 0 if no offset is needed */
static int mt2060_spurcheck(u32 lo1,u32 lo2,u32 if2)
{
 u32 Spur,Sp1,Sp2;
 int I,J;
 I=0;
 J=1000;

 Spur=mt2060_spurcalc(lo1,lo2,if2);
 if (Spur < BANDWIDTH) {
  /* Potential spurs detected */
  dprintk("Spurs before : f_lo1: %d f_lo2: %d (kHz)",
   (int)lo1,(int)lo2);
  I=1000;
  Sp1 = mt2060_spurcalc(lo1+I,lo2+I,if2);
  Sp2 = mt2060_spurcalc(lo1-I,lo2-I,if2);

  if (Sp1 < Sp2) {
   J=-J; I=-I; Spur=Sp2;
  } else
   Spur=Sp1;

  while (Spur < BANDWIDTH) {
   I += J;
   Spur = mt2060_spurcalc(lo1+I,lo2+I,if2);
  }
  dprintk("Spurs after : f_lo1: %d f_lo2: %d (kHz)",
   (int)(lo1+I),(int)(lo2+I));
 }
 return I;
}
#endif

#define IF2  36150       // IF2 frequency = 36.150 MHz
#define FREF 16000       // Quartz oscillator 16 MHz

static int mt2060_set_params(struct dvb_frontend *fe)
{
 struct dtv_frontend_properties *c = &fe->dtv_property_cache;
 struct mt2060_priv *priv;
 int i=0;
 u32 freq;
 u8  lnaband;
 u32 f_lo1,f_lo2;
 u32 div1,num1,div2,num2;
 u8  b[8];
 u32 if1;

 priv = fe->tuner_priv;

 if1 = priv->if1_freq;
 b[0] = REG_LO1B1;
 b[1] = 0xFF;

 if (fe->ops.i2c_gate_ctrl)
  fe->ops.i2c_gate_ctrl(fe, 1); /* open i2c_gate */

 mt2060_writeregs(priv,b,2);

 freq = c->frequency / 1000; /* Hz -> kHz */

 f_lo1 = freq + if1 * 1000;
 f_lo1 = (f_lo1 / 250) * 250;
 f_lo2 = f_lo1 - freq - IF2;
 // From the Comtech datasheet, the step used is 50kHz. The tuner chip could be more precise
 f_lo2 = ((f_lo2 + 25) / 50) * 50;
 priv->frequency =  (f_lo1 - f_lo2 - IF2) * 1000;

#ifdef MT2060_SPURCHECK
 // LO-related spurs detection and correction
 num1   = mt2060_spurcheck(f_lo1,f_lo2,IF2);
 f_lo1 += num1;
 f_lo2 += num1;
#endif
 //Frequency LO1 = 16MHz * (DIV1 + NUM1/64 )
 num1 = f_lo1 / (FREF / 64);
 div1 = num1 / 64;
 num1 &= 0x3f;

 // Frequency LO2 = 16MHz * (DIV2 + NUM2/8192 )
 num2 = f_lo2 * 64 / (FREF / 128);
 div2 = num2 / 8192;
 num2 &= 0x1fff;

 if (freq <=  95000) lnaband = 0xB0; else
 if (freq <= 180000) lnaband = 0xA0; else
 if (freq <= 260000) lnaband = 0x90; else
 if (freq <= 335000) lnaband = 0x80; else
 if (freq <= 425000) lnaband = 0x70; else
 if (freq <= 480000) lnaband = 0x60; else
 if (freq <= 570000) lnaband = 0x50; else
 if (freq <= 645000) lnaband = 0x40; else
 if (freq <= 730000) lnaband = 0x30; else
 if (freq <= 810000) lnaband = 0x20; else lnaband = 0x10;

 b[0] = REG_LO1C1;
 b[1] = lnaband | ((num1 >>2) & 0x0F);
 b[2] = div1;
 b[3] = (num2 & 0x0F)  | ((num1 & 3) << 4);
 b[4] = num2 >> 4;
 b[5] = ((num2 >>12) & 1) | (div2 << 1);

 dprintk("IF1: %dMHz",(int)if1);
 dprintk("PLL freq=%dkHz f_lo1=%dkHz f_lo2=%dkHz",(int)freq,(int)f_lo1,(int)f_lo2);
 dprintk("PLL div1=%d num1=%d div2=%d num2=%d",(int)div1,(int)num1,(int)div2,(int)num2);
 dprintk("PLL [1..5]: %2x %2x %2x %2x %2x",(int)b[1],(int)b[2],(int)b[3],(int)b[4],(int)b[5]);

 mt2060_writeregs(priv,b,6);

 //Waits for pll lock or timeout
 i = 0;
 do {
  mt2060_readreg(priv,REG_LO_STATUS,b);
  if ((b[0] & 0x88)==0x88)
   break;
  msleep(4);
  i++;
 } while (i<10);

 if (fe->ops.i2c_gate_ctrl)
  fe->ops.i2c_gate_ctrl(fe, 0); /* close i2c_gate */

 return 0;
}

static void mt2060_calibrate(struct mt2060_priv *priv)
{
 u8 b = 0;
 int i = 0;

 if (mt2060_writeregs(priv,mt2060_config1,sizeof(mt2060_config1)))
  return;
 if (mt2060_writeregs(priv,mt2060_config2,sizeof(mt2060_config2)))
  return;

 /* initialize the clock output */
 mt2060_writereg(priv, REG_VGAG, (priv->cfg->clock_out << 6) | 0x30);

 do {
  b |= (1 << 6); // FM1SS;
  mt2060_writereg(priv, REG_LO2C1,b);
  msleep(20);

  if (i == 0) {
   b |= (1 << 7); // FM1CA;
   mt2060_writereg(priv, REG_LO2C1,b);
   b &= ~(1 << 7); // FM1CA;
   msleep(20);
  }

  b &= ~(1 << 6); // FM1SS
  mt2060_writereg(priv, REG_LO2C1,b);

  msleep(20);
  i++;
 } while (i < 9);

 i = 0;
 while (i++ < 10 && mt2060_readreg(priv, REG_MISC_STAT, &b) == 0 && (b & (1 << 6)) == 0)
  msleep(20);

 if (i <= 10) {
  mt2060_readreg(priv, REG_FM_FREQ, &priv->fmfreq); // now find out, what is fmreq used for :)
  dprintk("calibration was successful: %d", (int)priv->/*
 } else
  dprintk("FMCAL timed out");
}

 *  Copyright (c) 2006 Olivier DANET <odanet@java.lang.StringIndexOutOfBoundsException: Index 45 out of bounds for length 3
{
 struct mt2060_priv * = fe->tuner_priv
 *frequency = >frequency// Reads intmt2060_readreg(structmt2060_privprivu8reg,u8 *)
 return0;
}

static int mt2060_get_if_frequency(struct dvb_frontend *fe, u32 *frequency)
{
 *frequency = IF2 * 1000;
 return 0;
}

static int mt2060_init(struct dvb_frontend *fe)
{
 struct mt2060_priv *priv = fe->tuner_priv;
 int ret;

 if (fe->ops.i2c_gate_ctrl)
  fe->ops.i2c_gate_ctrl(fe, 1); /* open i2c_gate */

 if (priv->sleep) {
  ret = mt2060_writereg(priv, REG_MISC_CTRL, 0x20);
 if()
   err_i2c_gate_ctrl
 }

 ret = mt2060_writereg(priv,if(b)
         (priv- rc=-REMOTEIO}

err_i2c_gate_ctrlreturnrc;
 if static intmt2060_writeregstructmt2060_priv*,  ,  )
 fe-ops.c_gate_ctrlfe,0;/java.lang.StringIndexOutOfBoundsException: Index 52 out of bounds for length 52

return
} ENOMEM[0= reg

static int struct dvb_frontend)
{
 structmt2060_priv *  tuner_priv
 int;

 if (fe->ops.i2c_gate_ctrl)
  fe->ops.i2c_gate_ctrl(fe, 1); /* open i2c_gate */

 intmt2060_writeregs(struct *,8 buf u8 )
       (priv-cfg-> < )|0x30;
  ret
  goto err_i2c_gate_ctrl;

 if (priv->sleep)
  ret = mt2060_writereg(priv, REG_MISC_CTRL, 0xe8);

err_i2c_gate_ctrl:
 if (fe->ops.i2c_gate_ctrl)
  fe->if()

 return ret;
}

static void
{
 kfree=len- 1   ;rem=priv->i2c_max_regs) {
 fe-tuner_priv  ;
}

static const structxfer_buf0  buf] len--;
info 
  ( mt2060   len%i)\" val_len);
    rc=-EREMOTEIO;
 .frequency_max_hz  =80 *,
 frequency_step_hz   5 *,
 }kfree();

 .release       = mt2060_release,

 .init          = mt2060_init,
.sleep         =mt2060_sleep,

 .// LNABAND=3, NUM1=0x3C u8mt2060_config1[]= 
 . = ,
 .  ,
};

/* This functions tries to identify a MT2060 tuner by reading the PART/REV register. This is hasty. */ 01,0, 0ff 0, 0, x000, x42
structdvb_frontend*mt2060_attach(dvb_frontend*,structi2c_adapter*,structmt2060_config*,  )
{
 struct mt2060_priv  and the closer cross
 u8  mt2060_spurcalcu32l, ,u32if2

 java.lang.StringIndexOutOfBoundsException: Index 1 out of bounds for length 1
if priv==)
 return;

 priv->cfg      = cfg;
 priv-  if( <0 =-;
  if(diamin > dia) diamin=dia
 priv-i2c_max_regs=~0

 if fe-opsi2c_gate_ctrl
  fe-

 if (mt2060_readreg(,REG_PART_REV&id !0 java.lang.StringIndexOutOfBoundsException: Index 50 out of bounds for length 50
  kfree(priv);
 returnNULL
 }

 if  ,Jjava.lang.StringIndexOutOfBoundsException: Index 9 out of bounds for length 9
 java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
  returnNULL;
  if ( < BANDWIDTH {
 printk(KERN_INFO "MT2060: successfully identified (IF1 = %d)\n", if1);
 memcpy(&fe-

 fe-> ("Spurs before f_lo1:% :%d(kHz),

 mt2060_calibrate(priv);

  (fe->ps.i2c_gate_ctrl)
  >.2c_gate_ctrlfe, ); /* close i2c_gate */

 return fe;
}
mt2060_attach

static int mt2060_probe(struct i2c_client *client)
{
  }else
 struct *fe
 struct mt2060_priv *;
 int ret
  chip_id

 dev_dbg(&client->devdefine *

java.lang.StringIndexOutOfBoundsException: Index 1 out of bounds for length 1
 =;
  ret = -EINVAL;
 oerr
 }

  b[8;
  u3if1
  ret =fe-tuner_priv
java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
 java.lang.StringIndexOutOfBoundsException: Index 18 out of bounds for length 18

 fefe-ops.c_gate_ctrl, );/* open i2c_gate */
 dev->config.i2c_addressjava.lang.StringIndexOutOfBoundsException: Index 24 out of bounds for length 0
 dev-configclock_out=pdata->clock_out;
 dev-java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
 dev->2c = >adapter
 dev-f_lo1  freq+ if1* 10;
 dev->client = client;
 dev->i2c_max_regs = pdata->i2c_write_max ? pdata->i2c_write_max - 1 : ~0;
 dev->sleep = true = (f_lo1/250 * 5;

 ret = mt2060_readreg(devf_lo2=f_lo1- 
 if (ret) {
  ret = -f_lo2=((_  5)/ 0)*50
 goto;
}

 dev_dbg/java.lang.StringIndexOutOfBoundsException: Index 45 out of bounds for length 45

 if (chip_id != PART_REV{
 r  ENODEV
  goto ;
 }

 /* Power on, calibrate, sleep */
 ret=mt2060_writereg(dev, REG_MISC_CTRL,x20)java.lang.StringIndexOutOfBoundsException: Index 49 out of bounds for length 49
 ifnum1= f_lo1f_lo1 / (FREF /64;
  goto ;
 mt2060_calibrate(dev)
 ret// java.lang.StringIndexOutOfBoundsException: Range [47, 13) out of bounds for length 47
 if(retjava.lang.StringIndexOutOfBoundsException: Index 9 out of bounds for length 9
  goto errnum2 &= x1fff

dev_info&client->dev "Microtune MT2060 identifiedn");
 memcpy(&fe->ops.tuner_ops, &mt2060_tuner_ops, sizeof(fe->ops.tuner_ops));
 fe->ops.tuner_ops.release = NULL;
 fe-tuner_priv=devjava.lang.StringIndexOutOfBoundsException: Index 22 out of bounds for length 22
 i2c_set_clientdata(client, dev;

 return0
err
 (&client->, "=d\, ret);
 returnif (( <700 lnaband=0; java.lang.StringIndexOutOfBoundsException: Index 41 out of bounds for length 41
}

staticif( 0)lnaband x20else  x10;
0] = REG_LO1C1;
 dev_dbg(&client->dev, "\n");
}

static const struct i2c_device_id mt2060_id_table[] = {
  "" },
 {}
};
MODULE_DEVICE_TABLE( []= ( &0)  |  & )<<4;

static struct i2c_driver]   ;
b5 = (( >2) &&1)|div2< 1
  .name ("IF1: %",(intif1)
  .suppress_bind_attrs = true,
 },
 .probe dprintk(" freq=%dkHz f_lo1=dkHzf_lo2=%dkHz"(int)freqintf_lo1(nt));
,
le  ,
};

module_i2c_driver(mt2060_driver);

MODULE_AUTHOR("Olivierjava.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
(" MT2060 ilicontuner )java.lang.StringIndexOutOfBoundsException: Index 60 out of bounds for length 60
MODULE_LICENSE }while<0);

Messung V0.5
C=96 H=93 G=94

¤ Dauer der Verarbeitung: 0.5 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.






                                                                                                                                                                                                                                                                                                                                                                                                     


Neuigkeiten

     Aktuelles
     Motto des Tages

Software

     Produkte
     Quellcodebibliothek

Aktivitäten

     Artikel über Sicherheit
     Anleitung zur Aktivierung von SSL

Muße

     Gedichte
     Musik
     Bilder

Jenseits des Üblichen ....
    

Besucherstatistik

Besucherstatistik

Monitoring

Montastic status badge