📄 cx88-tvaudio.c
字号:
{AUD_DN2_AFC, 0x00000002}, {AUD_DCOC_1_SRC, 0x0000001b}, {AUD_DCOC1_SHIFT, 0x00000000}, {AUD_DCOC_1_SHIFT_IN0, 0x0000000a}, {AUD_DCOC_1_SHIFT_IN1, 0x00000008}, {AUD_IIR3_1_SEL, 0x00000023}, {AUD_DN0_FREQ, 0x000035a3}, {AUD_DN2_FREQ, 0x000029c7}, {AUD_CRDC0_SRC_SEL, 0x00000511}, {AUD_IIR1_0_SEL, 0x00000001}, {AUD_IIR1_1_SEL, 0x00000000}, {AUD_IIR3_2_SEL, 0x00000003}, {AUD_IIR3_2_SHIFT, 0x00000000}, {AUD_IIR3_0_SEL, 0x00000002}, {AUD_IIR2_0_SEL, 0x00000021}, {AUD_IIR2_0_SHIFT, 0x00000002}, {AUD_DEEMPH0_SRC_SEL, 0x0000000b}, {AUD_DEEMPH1_SRC_SEL, 0x0000000b}, {AUD_POLYPH80SCALEFAC, 0x00000001}, {AUD_START_TIMER, 0x00000000}, { /* end of list */ }, }; static const struct rlist am_l[] = { {AUD_ERRLOGPERIOD_R, 0x00000064}, {AUD_ERRINTRPTTHSHLD1_R, 0x00000FFF}, {AUD_ERRINTRPTTHSHLD2_R, 0x0000001F}, {AUD_ERRINTRPTTHSHLD3_R, 0x0000000F}, {AUD_PDF_DDS_CNST_BYTE2, 0x48}, {AUD_PDF_DDS_CNST_BYTE1, 0x3D}, {AUD_QAM_MODE, 0x00}, {AUD_PDF_DDS_CNST_BYTE0, 0xf5}, {AUD_PHACC_FREQ_8MSB, 0x3a}, {AUD_PHACC_FREQ_8LSB, 0x4a}, {AUD_DEEMPHGAIN_R, 0x00006680}, {AUD_DEEMPHNUMER1_R, 0x000353DE}, {AUD_DEEMPHNUMER2_R, 0x000001B1}, {AUD_DEEMPHDENOM1_R, 0x0000F3D0}, {AUD_DEEMPHDENOM2_R, 0x00000000}, {AUD_FM_MODE_ENABLE, 0x00000007}, {AUD_POLYPH80SCALEFAC, 0x00000003}, {AUD_AFE_12DB_EN, 0x00000001}, {AAGC_GAIN, 0x00000000}, {AAGC_HYST, 0x00000018}, {AAGC_DEF, 0x00000020}, {AUD_DN0_FREQ, 0x00000000}, {AUD_POLY0_DDS_CONSTANT, 0x000E4DB2}, {AUD_DCOC_0_SRC, 0x00000021}, {AUD_IIR1_0_SEL, 0x00000000}, {AUD_IIR1_0_SHIFT, 0x00000007}, {AUD_IIR1_1_SEL, 0x00000002}, {AUD_IIR1_1_SHIFT, 0x00000000}, {AUD_DCOC_1_SRC, 0x00000003}, {AUD_DCOC1_SHIFT, 0x00000000}, {AUD_DCOC_PASS_IN, 0x00000000}, {AUD_IIR1_2_SEL, 0x00000023}, {AUD_IIR1_2_SHIFT, 0x00000000}, {AUD_IIR1_3_SEL, 0x00000004}, {AUD_IIR1_3_SHIFT, 0x00000007}, {AUD_IIR1_4_SEL, 0x00000005}, {AUD_IIR1_4_SHIFT, 0x00000007}, {AUD_IIR3_0_SEL, 0x00000007}, {AUD_IIR3_0_SHIFT, 0x00000000}, {AUD_DEEMPH0_SRC_SEL, 0x00000011}, {AUD_DEEMPH0_SHIFT, 0x00000000}, {AUD_DEEMPH0_G0, 0x00007000}, {AUD_DEEMPH0_A0, 0x00000000}, {AUD_DEEMPH0_B0, 0x00000000}, {AUD_DEEMPH0_A1, 0x00000000}, {AUD_DEEMPH0_B1, 0x00000000}, {AUD_DEEMPH1_SRC_SEL, 0x00000011}, {AUD_DEEMPH1_SHIFT, 0x00000000}, {AUD_DEEMPH1_G0, 0x00007000}, {AUD_DEEMPH1_A0, 0x00000000}, {AUD_DEEMPH1_B0, 0x00000000}, {AUD_DEEMPH1_A1, 0x00000000}, {AUD_DEEMPH1_B1, 0x00000000}, {AUD_OUT0_SEL, 0x0000003F}, {AUD_OUT1_SEL, 0x0000003F}, {AUD_DMD_RA_DDS, 0x00F5C285}, {AUD_PLL_INT, 0x0000001E}, {AUD_PLL_DDS, 0x00000000}, {AUD_PLL_FRAC, 0x0000E542}, {AUD_RATE_ADJ1, 0x00000100}, {AUD_RATE_ADJ2, 0x00000200}, {AUD_RATE_ADJ3, 0x00000300}, {AUD_RATE_ADJ4, 0x00000400}, {AUD_RATE_ADJ5, 0x00000500}, {AUD_RATE_THRES_DMD, 0x000000C0}, { /* end of list */ }, }; static const struct rlist a2_deemph50[] = { {AUD_DEEMPH0_G0, 0x00000380}, {AUD_DEEMPH1_G0, 0x00000380}, {AUD_DEEMPHGAIN_R, 0x000011e1}, {AUD_DEEMPHNUMER1_R, 0x0002a7bc}, {AUD_DEEMPHNUMER2_R, 0x0003023c}, { /* end of list */ }, }; set_audio_start(core, SEL_A2); switch (core->tvaudio) { case WW_BG: dprintk("%s PAL-BG A1/2 (status: known-good)\n", __FUNCTION__); set_audio_registers(core, a2_bgdk_common); set_audio_registers(core, a2_bg); set_audio_registers(core, a2_deemph50); break; case WW_DK: dprintk("%s PAL-DK A1/2 (status: known-good)\n", __FUNCTION__); set_audio_registers(core, a2_bgdk_common); set_audio_registers(core, a2_dk); set_audio_registers(core, a2_deemph50); break; case WW_I: dprintk("%s PAL-I A1 (status: known-good)\n", __FUNCTION__); set_audio_registers(core, a1_i); set_audio_registers(core, a2_deemph50); break; case WW_L: dprintk("%s AM-L (status: devel)\n", __FUNCTION__); set_audio_registers(core, am_l); break; default: dprintk("%s Warning: wrong value\n", __FUNCTION__); return; break; }; mode |= EN_FMRADIO_EN_RDS | EN_DMTRX_SUMDIFF; set_audio_finish(core, mode);}static void set_audio_standard_EIAJ(struct cx88_core *core){ static const struct rlist eiaj[] = { /* TODO: eiaj register settings are not there yet ... */ { /* end of list */ }, }; dprintk("%s (status: unknown)\n", __FUNCTION__); set_audio_start(core, SEL_EIAJ); set_audio_registers(core, eiaj); set_audio_finish(core, EN_EIAJ_AUTO_STEREO);}static void set_audio_standard_FM(struct cx88_core *core, enum cx88_deemph_type deemph){ static const struct rlist fm_deemph_50[] = { {AUD_DEEMPH0_G0, 0x0C45}, {AUD_DEEMPH0_A0, 0x6262}, {AUD_DEEMPH0_B0, 0x1C29}, {AUD_DEEMPH0_A1, 0x3FC66}, {AUD_DEEMPH0_B1, 0x399A}, {AUD_DEEMPH1_G0, 0x0D80}, {AUD_DEEMPH1_A0, 0x6262}, {AUD_DEEMPH1_B0, 0x1C29}, {AUD_DEEMPH1_A1, 0x3FC66}, {AUD_DEEMPH1_B1, 0x399A}, {AUD_POLYPH80SCALEFAC, 0x0003}, { /* end of list */ }, }; static const struct rlist fm_deemph_75[] = { {AUD_DEEMPH0_G0, 0x091B}, {AUD_DEEMPH0_A0, 0x6B68}, {AUD_DEEMPH0_B0, 0x11EC}, {AUD_DEEMPH0_A1, 0x3FC66}, {AUD_DEEMPH0_B1, 0x399A}, {AUD_DEEMPH1_G0, 0x0AA0}, {AUD_DEEMPH1_A0, 0x6B68}, {AUD_DEEMPH1_B0, 0x11EC}, {AUD_DEEMPH1_A1, 0x3FC66}, {AUD_DEEMPH1_B1, 0x399A}, {AUD_POLYPH80SCALEFAC, 0x0003}, { /* end of list */ }, }; /* It is enough to leave default values? */ static const struct rlist fm_no_deemph[] = { {AUD_POLYPH80SCALEFAC, 0x0003}, { /* end of list */ }, }; dprintk("%s (status: unknown)\n", __FUNCTION__); set_audio_start(core, SEL_FMRADIO); switch (deemph) { case FM_NO_DEEMPH: set_audio_registers(core, fm_no_deemph); break; case FM_DEEMPH_50: set_audio_registers(core, fm_deemph_50); break; case FM_DEEMPH_75: set_audio_registers(core, fm_deemph_75); break; } set_audio_finish(core, EN_FMRADIO_AUTO_STEREO);}/* ----------------------------------------------------------- */int cx88_detect_nicam(struct cx88_core *core){ int i, j = 0; dprintk("start nicam autodetect.\n"); for (i = 0; i < 6; i++) { /* if bit1=1 then nicam is detected */ j += ((cx_read(AUD_NICAM_STATUS2) & 0x02) >> 1); if (j == 1) { dprintk("nicam is detected.\n"); return 1; } /* wait a little bit for next reading status */ msleep(10); } dprintk("nicam is not detected.\n"); return 0;}void cx88_set_tvaudio(struct cx88_core *core){ switch (core->tvaudio) { case WW_BTSC: set_audio_standard_BTSC(core, 0, EN_BTSC_AUTO_STEREO); break; case WW_BG: case WW_DK: case WW_I: case WW_L: /* prepare all dsp registers */ set_audio_standard_A2(core, EN_A2_FORCE_MONO1); /* set nicam mode - otherwise AUD_NICAM_STATUS2 contains wrong values */ set_audio_standard_NICAM(core, EN_NICAM_AUTO_STEREO); if (0 == cx88_detect_nicam(core)) { /* fall back to fm / am mono */ set_audio_standard_A2(core, EN_A2_FORCE_MONO1); core->use_nicam = 0; } else { core->use_nicam = 1; } break; case WW_EIAJ: set_audio_standard_EIAJ(core); break; case WW_FM: set_audio_standard_FM(core, FM_NO_DEEMPH); break; case WW_NONE: default: printk("%s/0: unknown tv audio mode [%d]\n", core->name, core->tvaudio); break; } return;}void cx88_newstation(struct cx88_core *core){ core->audiomode_manual = UNSET;}void cx88_get_stereo(struct cx88_core *core, struct v4l2_tuner *t){ static char *m[] = { "stereo", "dual mono", "mono", "sap" }; static char *p[] = { "no pilot", "pilot c1", "pilot c2", "?" }; u32 reg, mode, pilot; reg = cx_read(AUD_STATUS); mode = reg & 0x03; pilot = (reg >> 2) & 0x03; if (core->astat != reg) dprintk("AUD_STATUS: 0x%x [%s/%s] ctl=%s\n", reg, m[mode], p[pilot], aud_ctl_names[cx_read(AUD_CTL) & 63]); core->astat = reg;/* TODO Reading from AUD_STATUS is not enough for auto-detecting sap/dual-fm/nicam. Add some code here later.*/# if 0 t->capability = V4L2_TUNER_CAP_STEREO | V4L2_TUNER_CAP_SAP | V4L2_TUNER_CAP_LANG1 | V4L2_TUNER_CAP_LANG2; t->rxsubchans = V4L2_TUNER_SUB_MONO; t->audmode = V4L2_TUNER_MODE_MONO; switch (core->tvaudio) { case WW_BTSC: t->capability = V4L2_TUNER_CAP_STEREO | V4L2_TUNER_CAP_SAP; t->rxsubchans = V4L2_TUNER_SUB_STEREO; if (1 == pilot) { /* SAP */ t->rxsubchans |= V4L2_TUNER_SUB_SAP; } break; case WW_A2_BG: case WW_A2_DK: case WW_A2_M: if (1 == pilot) { /* stereo */ t->rxsubchans = V4L2_TUNER_SUB_MONO | V4L2_TUNER_SUB_STEREO; if (0 == mode) t->audmode = V4L2_TUNER_MODE_STEREO; } if (2 == pilot) { /* dual language -- FIXME */ t->rxsubchans = V4L2_TUNER_SUB_LANG1 | V4L2_TUNER_SUB_LANG2; t->audmode = V4L2_TUNER_MODE_LANG1; } break; case WW_NICAM_BGDKL: if (0 == mode) { t->audmode = V4L2_TUNER_MODE_STEREO; t->rxsubchans |= V4L2_TUNER_SUB_STEREO; } break; case WW_SYSTEM_L_AM: if (0x0 == mode && !(cx_read(AUD_INIT) & 0x04)) { t->audmode = V4L2_TUNER_MODE_STEREO; t->rxsubchans |= V4L2_TUNER_SUB_STEREO; } break; default: /* nothing */ break; }# endif return;}void cx88_set_stereo(struct cx88_core *core, u32 mode, int manual){ u32 ctl = UNSET; u32 mask = UNSET; if (manual) { core->audiomode_manual = mode; } else { if (UNSET != core->audiomode_manual) return; } core->audiomode_current = mode; switch (core->tvaudio) { case WW_BTSC: switch (mode) { case V4L2_TUNER_MODE_MONO: set_audio_standard_BTSC(core, 0, EN_BTSC_FORCE_MONO); break; case V4L2_TUNER_MODE_LANG1: set_audio_standard_BTSC(core, 0, EN_BTSC_AUTO_STEREO); break; case V4L2_TUNER_MODE_LANG2: set_audio_standard_BTSC(core, 1, EN_BTSC_FORCE_SAP); break; case V4L2_TUNER_MODE_STEREO: case V4L2_TUNER_MODE_LANG1_LANG2: set_audio_standard_BTSC(core, 0, EN_BTSC_FORCE_STEREO); break; } break; case WW_BG: case WW_DK: case WW_I: case WW_L: if (1 == core->use_nicam) { switch (mode) { case V4L2_TUNER_MODE_MONO: case V4L2_TUNER_MODE_LANG1: set_audio_standard_NICAM(core, EN_NICAM_FORCE_MONO1); break; case V4L2_TUNER_MODE_LANG2: set_audio_standard_NICAM(core, EN_NICAM_FORCE_MONO2); break; case V4L2_TUNER_MODE_STEREO: case V4L2_TUNER_MODE_LANG1_LANG2: set_audio_standard_NICAM(core, EN_NICAM_FORCE_STEREO); break; } } else { if ((core->tvaudio == WW_I) || (core->tvaudio == WW_L)) { /* fall back to fm / am mono */ set_audio_standard_A2(core, EN_A2_FORCE_MONO1); } else { /* TODO: Add A2 autodection */ switch (mode) { case V4L2_TUNER_MODE_MONO: case V4L2_TUNER_MODE_LANG1: set_audio_standard_A2(core, EN_A2_FORCE_MONO1); break; case V4L2_TUNER_MODE_LANG2: set_audio_standard_A2(core, EN_A2_FORCE_MONO2); break; case V4L2_TUNER_MODE_STEREO: case V4L2_TUNER_MODE_LANG1_LANG2: set_audio_standard_A2(core, EN_A2_FORCE_STEREO); break; } } } break; case WW_FM: switch (mode) { case V4L2_TUNER_MODE_MONO: ctl = EN_FMRADIO_FORCE_MONO; mask = 0x3f; break; case V4L2_TUNER_MODE_STEREO: ctl = EN_FMRADIO_AUTO_STEREO; mask = 0x3f; break; } break; } if (UNSET != ctl) { dprintk("cx88_set_stereo: mask 0x%x, ctl 0x%x " "[status=0x%x,ctl=0x%x,vol=0x%x]\n", mask, ctl, cx_read(AUD_STATUS), cx_read(AUD_CTL), cx_sread(SHADOW_AUD_VOL_CTL)); cx_andor(AUD_CTL, mask, ctl); } return;}int cx88_audio_thread(void *data){ struct cx88_core *core = data; struct v4l2_tuner t; u32 mode = 0; dprintk("cx88: tvaudio thread started\n"); for (;;) { msleep_interruptible(1000); if (kthread_should_stop()) break; /* just monitor the audio status for now ... */ memset(&t, 0, sizeof(t)); cx88_get_stereo(core, &t); if (UNSET != core->audiomode_manual) /* manually set, don't do anything. */ continue; /* monitor signal */ if (t.rxsubchans & V4L2_TUNER_SUB_STEREO) mode = V4L2_TUNER_MODE_STEREO; else mode = V4L2_TUNER_MODE_MONO; if (mode == core->audiomode_current) continue; /* automatically switch to best available mode */ cx88_set_stereo(core, mode, 0); } dprintk("cx88: tvaudio thread exiting\n"); return 0;}/* ----------------------------------------------------------- */EXPORT_SYMBOL(cx88_set_tvaudio);EXPORT_SYMBOL(cx88_newstation);EXPORT_SYMBOL(cx88_set_stereo);EXPORT_SYMBOL(cx88_get_stereo);EXPORT_SYMBOL(cx88_audio_thread);/* * Local variables: * c-basic-offset: 8 * End: * kate: eol "unix"; indent-width 3; remove-trailing-space on; replace-trailing-space-save on; tab-width 8; replace-tabs off; space-indent off; mixed-indent off */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -