int fc0012Tuner::fc0012_init( void )
{
	int ret = 0;
	unsigned int i;
	uint8_t reg[] =
	{
		0x00,	/* dummy reg. 0 */
		0x05,	/* reg. 0x01 */
		0x10,	/* reg. 0x02 */
		0x00,	/* reg. 0x03 */
		0x00,	/* reg. 0x04 */
		0x0f,	/* reg. 0x05: may also be 0x0a */
		0x00,	/* reg. 0x06: divider 2, VCO slow */
		0x00,	/* reg. 0x07: may also be 0x0f */
		0xff,	/* reg. 0x08: AGC Clock divide by 256, AGC gain 1/256,
			   Loop Bw 1/8 */
		0x6e,	/* reg. 0x09: Disable LoopThrough, Enable LoopThrough: 0x6f */
		0xb8,	/* reg. 0x0a: Disable LO Test Buffer */
		0x82,	/* reg. 0x0b: Output Clock is same as clock frequency,
			   may also be 0x83 */
		0xfc,	/* reg. 0x0c: depending on AGC Up-Down mode, may need 0xf8 */
		0x02,	/* reg. 0x0d: AGC Not Forcing & LNA Forcing, 0x02 for DVB-T */
		0x00,	/* reg. 0x0e */
		0x00,	/* reg. 0x0f */
		0x00,	/* reg. 0x10: may also be 0x0d */
		0x00,	/* reg. 0x11 */
		0x1f,	/* reg. 0x12: Set to maximum gain */
		0x08,	/* reg. 0x13: Set to Middle Gain: 0x08,
			   Low Gain: 0x00, High Gain: 0x10, enable IX2: 0x80 */
		0x00,	/* reg. 0x14 */
		0x04,	/* reg. 0x15: Enable LNA COMPS */
	};

#if 0
	switch ( rtlsdr_get_tuner_clock())
	{
	case FC_XTAL_27_MHZ:
	case FC_XTAL_28_8_MHZ:
		reg[ 0x07 ] |= 0x20;
		break;
	case FC_XTAL_36_MHZ:
	default:
		break;
	}
#endif
	reg[ 0x07 ] |= 0x20;

//	if (priv->dual_master)
	reg[ 0x0c ] |= 0x02;

	for ( i = 1; i < sizeof( reg ); i++ )
	{
		ret = fc0012_writereg( i, reg[ i ]);
		if ( ret )
			break;
	}

	return ret;
}
Exemplo n.º 2
0
int fc0012_set_params(void *dev, uint32_t freq, uint32_t bandwidth)
{
    int i, ret = 0;
    uint8_t reg[7], am, pm, multi, tmp;
    uint64_t f_vco;
    uint32_t xtal_freq_div_2;
    uint16_t xin, xdiv;
    int vco_select = 0;

    xtal_freq_div_2 = rtlsdr_get_tuner_clock(dev) / 2;

    /* select frequency divider and the frequency of VCO */
    if (freq < 37084000) {		/* freq * 96 < 3560000000 */
        multi = 96;
        reg[5] = 0x82;
        reg[6] = 0x00;
    } else if (freq < 55625000) {	/* freq * 64 < 3560000000 */
        multi = 64;
        reg[5] = 0x82;
        reg[6] = 0x02;
    } else if (freq < 74167000) {	/* freq * 48 < 3560000000 */
        multi = 48;
        reg[5] = 0x42;
        reg[6] = 0x00;
    } else if (freq < 111250000) {	/* freq * 32 < 3560000000 */
        multi = 32;
        reg[5] = 0x42;
        reg[6] = 0x02;
    } else if (freq < 148334000) {	/* freq * 24 < 3560000000 */
        multi = 24;
        reg[5] = 0x22;
        reg[6] = 0x00;
    } else if (freq < 222500000) {	/* freq * 16 < 3560000000 */
        multi = 16;
        reg[5] = 0x22;
        reg[6] = 0x02;
    } else if (freq < 296667000) {	/* freq * 12 < 3560000000 */
        multi = 12;
        reg[5] = 0x12;
        reg[6] = 0x00;
    } else if (freq < 445000000) {	/* freq * 8 < 3560000000 */
        multi = 8;
        reg[5] = 0x12;
        reg[6] = 0x02;
    } else if (freq < 593334000) {	/* freq * 6 < 3560000000 */
        multi = 6;
        reg[5] = 0x0a;
        reg[6] = 0x00;
    } else {
        multi = 4;
        reg[5] = 0x0a;
        reg[6] = 0x02;
    }

    f_vco = freq * multi;

    if (f_vco >= 3060000000U) {
        reg[6] |= 0x08;
        vco_select = 1;
    }

    /* From divided value (XDIV) determined the FA and FP value */
    xdiv = (uint16_t)(f_vco / xtal_freq_div_2);
    if ((f_vco - xdiv * xtal_freq_div_2) >= (xtal_freq_div_2 / 2))
        xdiv++;

    pm = (uint8_t)(xdiv / 8);
    am = (uint8_t)(xdiv - (8 * pm));

    if (am < 2) {
        am += 8;
        pm--;
    }

    if (pm > 31) {
        reg[1] = am + (8 * (pm - 31));
        reg[2] = 31;
    } else {
        reg[1] = am;
        reg[2] = pm;
    }

    if ((reg[1] > 15) || (reg[2] < 0x0b)) {
        fprintf(stderr, "[FC0012] no valid PLL combination "
                "found for %u Hz!\n", freq);
        return -1;
    }

    /* fix clock out */
    reg[6] |= 0x20;

    /* From VCO frequency determines the XIN ( fractional part of Delta
       Sigma PLL) and divided value (XDIV) */
    xin = (uint16_t)((f_vco - (f_vco / xtal_freq_div_2) * xtal_freq_div_2) / 1000);
    xin = (xin << 15) / (xtal_freq_div_2 / 1000);
    if (xin >= 16384)
        xin += 32768;

    reg[3] = xin >> 8;	/* xin with 9 bit resolution */
    reg[4] = xin & 0xff;

    reg[6] &= 0x3f; /* bits 6 and 7 describe the bandwidth */
    switch (bandwidth) {
    case 6000000:
        reg[6] |= 0x80;
        break;
    case 7000000:
        reg[6] |= 0x40;
        break;
    case 8000000:
    default:
        break;
    }

    /* modified for Realtek demod */
    reg[5] |= 0x07;

    for (i = 1; i <= 6; i++) {
        ret = fc0012_writereg(dev, i, reg[i]);
        if (ret)
            goto exit;
    }

    /* VCO Calibration */
    ret = fc0012_writereg(dev, 0x0e, 0x80);
    if (!ret)
        ret = fc0012_writereg(dev, 0x0e, 0x00);

    /* VCO Re-Calibration if needed */
    if (!ret)
        ret = fc0012_writereg(dev, 0x0e, 0x00);

    if (!ret) {
//		msleep(10);
        ret = fc0012_readreg(dev, 0x0e, &tmp);
    }
    if (ret)
        goto exit;

    /* vco selection */
    tmp &= 0x3f;

    if (vco_select) {
        if (tmp > 0x3c) {
            reg[6] &= ~0x08;
            ret = fc0012_writereg(dev, 0x06, reg[6]);
            if (!ret)
                ret = fc0012_writereg(dev, 0x0e, 0x80);
            if (!ret)
                ret = fc0012_writereg(dev, 0x0e, 0x00);
        }
    } else {
        if (tmp < 0x02) {
            reg[6] |= 0x08;
            ret = fc0012_writereg(dev, 0x06, reg[6]);
            if (!ret)
                ret = fc0012_writereg(dev, 0x0e, 0x80);
            if (!ret)
                ret = fc0012_writereg(dev, 0x0e, 0x00);
        }
    }

exit:
    if (!ret)
        return freq;
    else
        return -1;
}