Path: blob/master/drivers/media/dvb/ttpci/av7110_v4l.c
15111 views
/*1* av7110_v4l.c: av7110 video4linux interface for DVB and Siemens DVB-C analog module2*3* Copyright (C) 1999-2002 Ralph Metzler4* & Marcus Metzler for convergence integrated media GmbH5*6* originally based on code by:7* Copyright (C) 1998,1999 Christian Theiss <[email protected]>8*9* This program is free software; you can redistribute it and/or10* modify it under the terms of the GNU General Public License11* as published by the Free Software Foundation; either version 212* of the License, or (at your option) any later version.13*14* This program is distributed in the hope that it will be useful,15* but WITHOUT ANY WARRANTY; without even the implied warranty of16* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the17* GNU General Public License for more details.18*19* You should have received a copy of the GNU General Public License20* along with this program; if not, write to the Free Software21* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.22* Or, point your browser to http://www.gnu.org/copyleft/gpl.html23*24* the project's page is at http://www.linuxtv.org/25*/2627#include <linux/kernel.h>28#include <linux/types.h>29#include <linux/delay.h>30#include <linux/fs.h>31#include <linux/timer.h>32#include <linux/poll.h>3334#include "av7110.h"35#include "av7110_hw.h"36#include "av7110_av.h"3738int msp_writereg(struct av7110 *av7110, u8 dev, u16 reg, u16 val)39{40u8 msg[5] = { dev, reg >> 8, reg & 0xff, val >> 8 , val & 0xff };41struct i2c_msg msgs = { .flags = 0, .len = 5, .buf = msg };4243switch (av7110->adac_type) {44case DVB_ADAC_MSP34x0:45msgs.addr = 0x40;46break;47case DVB_ADAC_MSP34x5:48msgs.addr = 0x42;49break;50default:51return 0;52}5354if (i2c_transfer(&av7110->i2c_adap, &msgs, 1) != 1) {55dprintk(1, "dvb-ttpci: failed @ card %d, %u = %u\n",56av7110->dvb_adapter.num, reg, val);57return -EIO;58}59return 0;60}6162static int msp_readreg(struct av7110 *av7110, u8 dev, u16 reg, u16 *val)63{64u8 msg1[3] = { dev, reg >> 8, reg & 0xff };65u8 msg2[2];66struct i2c_msg msgs[2] = {67{ .flags = 0 , .len = 3, .buf = msg1 },68{ .flags = I2C_M_RD, .len = 2, .buf = msg2 }69};7071switch (av7110->adac_type) {72case DVB_ADAC_MSP34x0:73msgs[0].addr = 0x40;74msgs[1].addr = 0x40;75break;76case DVB_ADAC_MSP34x5:77msgs[0].addr = 0x42;78msgs[1].addr = 0x42;79break;80default:81return 0;82}8384if (i2c_transfer(&av7110->i2c_adap, &msgs[0], 2) != 2) {85dprintk(1, "dvb-ttpci: failed @ card %d, %u\n",86av7110->dvb_adapter.num, reg);87return -EIO;88}89*val = (msg2[0] << 8) | msg2[1];90return 0;91}9293static struct v4l2_input inputs[4] = {94{95.index = 0,96.name = "DVB",97.type = V4L2_INPUT_TYPE_CAMERA,98.audioset = 1,99.tuner = 0, /* ignored */100.std = V4L2_STD_PAL_BG|V4L2_STD_NTSC_M,101.status = 0,102.capabilities = V4L2_IN_CAP_STD,103}, {104.index = 1,105.name = "Television",106.type = V4L2_INPUT_TYPE_TUNER,107.audioset = 2,108.tuner = 0,109.std = V4L2_STD_PAL_BG|V4L2_STD_NTSC_M,110.status = 0,111.capabilities = V4L2_IN_CAP_STD,112}, {113.index = 2,114.name = "Video",115.type = V4L2_INPUT_TYPE_CAMERA,116.audioset = 0,117.tuner = 0,118.std = V4L2_STD_PAL_BG|V4L2_STD_NTSC_M,119.status = 0,120.capabilities = V4L2_IN_CAP_STD,121}, {122.index = 3,123.name = "Y/C",124.type = V4L2_INPUT_TYPE_CAMERA,125.audioset = 0,126.tuner = 0,127.std = V4L2_STD_PAL_BG|V4L2_STD_NTSC_M,128.status = 0,129.capabilities = V4L2_IN_CAP_STD,130}131};132133static int ves1820_writereg(struct saa7146_dev *dev, u8 addr, u8 reg, u8 data)134{135struct av7110 *av7110 = dev->ext_priv;136u8 buf[] = { 0x00, reg, data };137struct i2c_msg msg = { .addr = addr, .flags = 0, .buf = buf, .len = 3 };138139dprintk(4, "dev: %p\n", dev);140141if (1 != i2c_transfer(&av7110->i2c_adap, &msg, 1))142return -1;143return 0;144}145146static int tuner_write(struct saa7146_dev *dev, u8 addr, u8 data [4])147{148struct av7110 *av7110 = dev->ext_priv;149struct i2c_msg msg = { .addr = addr, .flags = 0, .buf = data, .len = 4 };150151dprintk(4, "dev: %p\n", dev);152153if (1 != i2c_transfer(&av7110->i2c_adap, &msg, 1))154return -1;155return 0;156}157158static int ves1820_set_tv_freq(struct saa7146_dev *dev, u32 freq)159{160u32 div;161u8 config;162u8 buf[4];163164dprintk(4, "freq: 0x%08x\n", freq);165166/* magic number: 614. tuning with the frequency given by v4l2167is always off by 614*62.5 = 38375 kHz...*/168div = freq + 614;169170buf[0] = (div >> 8) & 0x7f;171buf[1] = div & 0xff;172buf[2] = 0x8e;173174if (freq < (u32) (16 * 168.25))175config = 0xa0;176else if (freq < (u32) (16 * 447.25))177config = 0x90;178else179config = 0x30;180config &= ~0x02;181182buf[3] = config;183184return tuner_write(dev, 0x61, buf);185}186187static int stv0297_set_tv_freq(struct saa7146_dev *dev, u32 freq)188{189struct av7110 *av7110 = (struct av7110*)dev->ext_priv;190u32 div;191u8 data[4];192193div = (freq + 38900000 + 31250) / 62500;194195data[0] = (div >> 8) & 0x7f;196data[1] = div & 0xff;197data[2] = 0xce;198199if (freq < 45000000)200return -EINVAL;201else if (freq < 137000000)202data[3] = 0x01;203else if (freq < 403000000)204data[3] = 0x02;205else if (freq < 860000000)206data[3] = 0x04;207else208return -EINVAL;209210if (av7110->fe->ops.i2c_gate_ctrl)211av7110->fe->ops.i2c_gate_ctrl(av7110->fe, 1);212return tuner_write(dev, 0x63, data);213}214215216217static struct saa7146_standard analog_standard[];218static struct saa7146_standard dvb_standard[];219static struct saa7146_standard standard[];220221static struct v4l2_audio msp3400_v4l2_audio = {222.index = 0,223.name = "Television",224.capability = V4L2_AUDCAP_STEREO225};226227static int av7110_dvb_c_switch(struct saa7146_fh *fh)228{229struct saa7146_dev *dev = fh->dev;230struct saa7146_vv *vv = dev->vv_data;231struct av7110 *av7110 = (struct av7110*)dev->ext_priv;232u16 adswitch;233int source, sync, err;234235dprintk(4, "%p\n", av7110);236237if ((vv->video_status & STATUS_OVERLAY) != 0) {238vv->ov_suspend = vv->video_fh;239err = saa7146_stop_preview(vv->video_fh); /* side effect: video_status is now 0, video_fh is NULL */240if (err != 0) {241dprintk(2, "suspending video failed\n");242vv->ov_suspend = NULL;243}244}245246if (0 != av7110->current_input) {247dprintk(1, "switching to analog TV:\n");248adswitch = 1;249source = SAA7146_HPS_SOURCE_PORT_B;250sync = SAA7146_HPS_SYNC_PORT_B;251memcpy(standard, analog_standard, sizeof(struct saa7146_standard) * 2);252253switch (av7110->current_input) {254case 1:255dprintk(1, "switching SAA7113 to Analog Tuner Input.\n");256msp_writereg(av7110, MSP_WR_DSP, 0x0008, 0x0000); // loudspeaker source257msp_writereg(av7110, MSP_WR_DSP, 0x0009, 0x0000); // headphone source258msp_writereg(av7110, MSP_WR_DSP, 0x000a, 0x0000); // SCART 1 source259msp_writereg(av7110, MSP_WR_DSP, 0x000e, 0x3000); // FM matrix, mono260msp_writereg(av7110, MSP_WR_DSP, 0x0000, 0x4f00); // loudspeaker + headphone261msp_writereg(av7110, MSP_WR_DSP, 0x0007, 0x4f00); // SCART 1 volume262263if (av7110->analog_tuner_flags & ANALOG_TUNER_VES1820) {264if (ves1820_writereg(dev, 0x09, 0x0f, 0x60))265dprintk(1, "setting band in demodulator failed.\n");266} else if (av7110->analog_tuner_flags & ANALOG_TUNER_STV0297) {267saa7146_setgpio(dev, 1, SAA7146_GPIO_OUTHI); // TDA9819 pin9(STD)268saa7146_setgpio(dev, 3, SAA7146_GPIO_OUTHI); // TDA9819 pin30(VIF)269}270if (i2c_writereg(av7110, 0x48, 0x02, 0xd0) != 1)271dprintk(1, "saa7113 write failed @ card %d", av7110->dvb_adapter.num);272break;273case 2:274dprintk(1, "switching SAA7113 to Video AV CVBS Input.\n");275if (i2c_writereg(av7110, 0x48, 0x02, 0xd2) != 1)276dprintk(1, "saa7113 write failed @ card %d", av7110->dvb_adapter.num);277break;278case 3:279dprintk(1, "switching SAA7113 to Video AV Y/C Input.\n");280if (i2c_writereg(av7110, 0x48, 0x02, 0xd9) != 1)281dprintk(1, "saa7113 write failed @ card %d", av7110->dvb_adapter.num);282break;283default:284dprintk(1, "switching SAA7113 to Input: AV7110: SAA7113: invalid input.\n");285}286} else {287adswitch = 0;288source = SAA7146_HPS_SOURCE_PORT_A;289sync = SAA7146_HPS_SYNC_PORT_A;290memcpy(standard, dvb_standard, sizeof(struct saa7146_standard) * 2);291dprintk(1, "switching DVB mode\n");292msp_writereg(av7110, MSP_WR_DSP, 0x0008, 0x0220); // loudspeaker source293msp_writereg(av7110, MSP_WR_DSP, 0x0009, 0x0220); // headphone source294msp_writereg(av7110, MSP_WR_DSP, 0x000a, 0x0220); // SCART 1 source295msp_writereg(av7110, MSP_WR_DSP, 0x000e, 0x3000); // FM matrix, mono296msp_writereg(av7110, MSP_WR_DSP, 0x0000, 0x7f00); // loudspeaker + headphone297msp_writereg(av7110, MSP_WR_DSP, 0x0007, 0x7f00); // SCART 1 volume298299if (av7110->analog_tuner_flags & ANALOG_TUNER_VES1820) {300if (ves1820_writereg(dev, 0x09, 0x0f, 0x20))301dprintk(1, "setting band in demodulator failed.\n");302} else if (av7110->analog_tuner_flags & ANALOG_TUNER_STV0297) {303saa7146_setgpio(dev, 1, SAA7146_GPIO_OUTLO); // TDA9819 pin9(STD)304saa7146_setgpio(dev, 3, SAA7146_GPIO_OUTLO); // TDA9819 pin30(VIF)305}306}307308/* hmm, this does not do anything!? */309if (av7110_fw_cmd(av7110, COMTYPE_AUDIODAC, ADSwitch, 1, adswitch))310dprintk(1, "ADSwitch error\n");311312saa7146_set_hps_source_and_sync(dev, source, sync);313314if (vv->ov_suspend != NULL) {315saa7146_start_preview(vv->ov_suspend);316vv->ov_suspend = NULL;317}318319return 0;320}321322static int vidioc_g_tuner(struct file *file, void *fh, struct v4l2_tuner *t)323{324struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;325struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;326u16 stereo_det;327s8 stereo;328329dprintk(2, "VIDIOC_G_TUNER: %d\n", t->index);330331if (!av7110->analog_tuner_flags || t->index != 0)332return -EINVAL;333334memset(t, 0, sizeof(*t));335strcpy((char *)t->name, "Television");336337t->type = V4L2_TUNER_ANALOG_TV;338t->capability = V4L2_TUNER_CAP_NORM | V4L2_TUNER_CAP_STEREO |339V4L2_TUNER_CAP_LANG1 | V4L2_TUNER_CAP_LANG2 | V4L2_TUNER_CAP_SAP;340t->rangelow = 772; /* 48.25 MHZ / 62.5 kHz = 772, see fi1216mk2-specs, page 2 */341t->rangehigh = 13684; /* 855.25 MHz / 62.5 kHz = 13684 */342/* FIXME: add the real signal strength here */343t->signal = 0xffff;344t->afc = 0;345346/* FIXME: standard / stereo detection is still broken */347msp_readreg(av7110, MSP_RD_DEM, 0x007e, &stereo_det);348dprintk(1, "VIDIOC_G_TUNER: msp3400 TV standard detection: 0x%04x\n", stereo_det);349msp_readreg(av7110, MSP_RD_DSP, 0x0018, &stereo_det);350dprintk(1, "VIDIOC_G_TUNER: msp3400 stereo detection: 0x%04x\n", stereo_det);351stereo = (s8)(stereo_det >> 8);352if (stereo > 0x10) {353/* stereo */354t->rxsubchans = V4L2_TUNER_SUB_STEREO | V4L2_TUNER_SUB_MONO;355t->audmode = V4L2_TUNER_MODE_STEREO;356} else if (stereo < -0x10) {357/* bilingual */358t->rxsubchans = V4L2_TUNER_SUB_LANG1 | V4L2_TUNER_SUB_LANG2;359t->audmode = V4L2_TUNER_MODE_LANG1;360} else /* mono */361t->rxsubchans = V4L2_TUNER_SUB_MONO;362363return 0;364}365366static int vidioc_s_tuner(struct file *file, void *fh, struct v4l2_tuner *t)367{368struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;369struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;370u16 fm_matrix, src;371dprintk(2, "VIDIOC_S_TUNER: %d\n", t->index);372373if (!av7110->analog_tuner_flags || av7110->current_input != 1)374return -EINVAL;375376switch (t->audmode) {377case V4L2_TUNER_MODE_STEREO:378dprintk(2, "VIDIOC_S_TUNER: V4L2_TUNER_MODE_STEREO\n");379fm_matrix = 0x3001; /* stereo */380src = 0x0020;381break;382case V4L2_TUNER_MODE_LANG1_LANG2:383dprintk(2, "VIDIOC_S_TUNER: V4L2_TUNER_MODE_LANG1_LANG2\n");384fm_matrix = 0x3000; /* bilingual */385src = 0x0020;386break;387case V4L2_TUNER_MODE_LANG1:388dprintk(2, "VIDIOC_S_TUNER: V4L2_TUNER_MODE_LANG1\n");389fm_matrix = 0x3000; /* mono */390src = 0x0000;391break;392case V4L2_TUNER_MODE_LANG2:393dprintk(2, "VIDIOC_S_TUNER: V4L2_TUNER_MODE_LANG2\n");394fm_matrix = 0x3000; /* mono */395src = 0x0010;396break;397default: /* case V4L2_TUNER_MODE_MONO: */398dprintk(2, "VIDIOC_S_TUNER: TDA9840_SET_MONO\n");399fm_matrix = 0x3000; /* mono */400src = 0x0030;401break;402}403msp_writereg(av7110, MSP_WR_DSP, 0x000e, fm_matrix);404msp_writereg(av7110, MSP_WR_DSP, 0x0008, src);405msp_writereg(av7110, MSP_WR_DSP, 0x0009, src);406msp_writereg(av7110, MSP_WR_DSP, 0x000a, src);407return 0;408}409410static int vidioc_g_frequency(struct file *file, void *fh, struct v4l2_frequency *f)411{412struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;413struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;414415dprintk(2, "VIDIOC_G_FREQ: freq:0x%08x.\n", f->frequency);416417if (!av7110->analog_tuner_flags || av7110->current_input != 1)418return -EINVAL;419420memset(f, 0, sizeof(*f));421f->type = V4L2_TUNER_ANALOG_TV;422f->frequency = av7110->current_freq;423return 0;424}425426static int vidioc_s_frequency(struct file *file, void *fh, struct v4l2_frequency *f)427{428struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;429struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;430431dprintk(2, "VIDIOC_S_FREQUENCY: freq:0x%08x.\n", f->frequency);432433if (!av7110->analog_tuner_flags || av7110->current_input != 1)434return -EINVAL;435436if (V4L2_TUNER_ANALOG_TV != f->type)437return -EINVAL;438439msp_writereg(av7110, MSP_WR_DSP, 0x0000, 0xffe0); /* fast mute */440msp_writereg(av7110, MSP_WR_DSP, 0x0007, 0xffe0);441442/* tune in desired frequency */443if (av7110->analog_tuner_flags & ANALOG_TUNER_VES1820)444ves1820_set_tv_freq(dev, f->frequency);445else if (av7110->analog_tuner_flags & ANALOG_TUNER_STV0297)446stv0297_set_tv_freq(dev, f->frequency);447av7110->current_freq = f->frequency;448449msp_writereg(av7110, MSP_WR_DSP, 0x0015, 0x003f); /* start stereo detection */450msp_writereg(av7110, MSP_WR_DSP, 0x0015, 0x0000);451msp_writereg(av7110, MSP_WR_DSP, 0x0000, 0x4f00); /* loudspeaker + headphone */452msp_writereg(av7110, MSP_WR_DSP, 0x0007, 0x4f00); /* SCART 1 volume */453return 0;454}455456static int vidioc_enum_input(struct file *file, void *fh, struct v4l2_input *i)457{458struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;459struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;460461dprintk(2, "VIDIOC_ENUMINPUT: %d\n", i->index);462463if (av7110->analog_tuner_flags) {464if (i->index >= 4)465return -EINVAL;466} else {467if (i->index != 0)468return -EINVAL;469}470471memcpy(i, &inputs[i->index], sizeof(struct v4l2_input));472473return 0;474}475476static int vidioc_g_input(struct file *file, void *fh, unsigned int *input)477{478struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;479struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;480481*input = av7110->current_input;482dprintk(2, "VIDIOC_G_INPUT: %d\n", *input);483return 0;484}485486static int vidioc_s_input(struct file *file, void *fh, unsigned int input)487{488struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;489struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;490491dprintk(2, "VIDIOC_S_INPUT: %d\n", input);492493if (!av7110->analog_tuner_flags)494return 0;495496if (input >= 4)497return -EINVAL;498499av7110->current_input = input;500return av7110_dvb_c_switch(fh);501}502503static int vidioc_g_audio(struct file *file, void *fh, struct v4l2_audio *a)504{505dprintk(2, "VIDIOC_G_AUDIO: %d\n", a->index);506if (a->index != 0)507return -EINVAL;508memcpy(a, &msp3400_v4l2_audio, sizeof(struct v4l2_audio));509return 0;510}511512static int vidioc_s_audio(struct file *file, void *fh, struct v4l2_audio *a)513{514dprintk(2, "VIDIOC_S_AUDIO: %d\n", a->index);515return 0;516}517518static int vidioc_g_sliced_vbi_cap(struct file *file, void *fh,519struct v4l2_sliced_vbi_cap *cap)520{521struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;522struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;523524dprintk(2, "VIDIOC_G_SLICED_VBI_CAP\n");525if (cap->type != V4L2_BUF_TYPE_SLICED_VBI_OUTPUT)526return -EINVAL;527if (FW_VERSION(av7110->arm_app) >= 0x2623) {528cap->service_set = V4L2_SLICED_WSS_625;529cap->service_lines[0][23] = V4L2_SLICED_WSS_625;530}531return 0;532}533534static int vidioc_g_fmt_sliced_vbi_out(struct file *file, void *fh,535struct v4l2_format *f)536{537struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;538struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;539540dprintk(2, "VIDIOC_G_FMT:\n");541if (FW_VERSION(av7110->arm_app) < 0x2623)542return -EINVAL;543memset(&f->fmt.sliced, 0, sizeof f->fmt.sliced);544if (av7110->wssMode) {545f->fmt.sliced.service_set = V4L2_SLICED_WSS_625;546f->fmt.sliced.service_lines[0][23] = V4L2_SLICED_WSS_625;547f->fmt.sliced.io_size = sizeof(struct v4l2_sliced_vbi_data);548}549return 0;550}551552static int vidioc_s_fmt_sliced_vbi_out(struct file *file, void *fh,553struct v4l2_format *f)554{555struct saa7146_dev *dev = ((struct saa7146_fh *)fh)->dev;556struct av7110 *av7110 = (struct av7110 *)dev->ext_priv;557558dprintk(2, "VIDIOC_S_FMT\n");559if (FW_VERSION(av7110->arm_app) < 0x2623)560return -EINVAL;561if (f->fmt.sliced.service_set != V4L2_SLICED_WSS_625 &&562f->fmt.sliced.service_lines[0][23] != V4L2_SLICED_WSS_625) {563memset(&f->fmt.sliced, 0, sizeof(f->fmt.sliced));564/* WSS controlled by firmware */565av7110->wssMode = 0;566av7110->wssData = 0;567return av7110_fw_cmd(av7110, COMTYPE_ENCODER,568SetWSSConfig, 1, 0);569} else {570memset(&f->fmt.sliced, 0, sizeof(f->fmt.sliced));571f->fmt.sliced.service_set = V4L2_SLICED_WSS_625;572f->fmt.sliced.service_lines[0][23] = V4L2_SLICED_WSS_625;573f->fmt.sliced.io_size = sizeof(struct v4l2_sliced_vbi_data);574/* WSS controlled by userspace */575av7110->wssMode = 1;576av7110->wssData = 0;577}578return 0;579}580581static int av7110_vbi_reset(struct file *file)582{583struct saa7146_fh *fh = file->private_data;584struct saa7146_dev *dev = fh->dev;585struct av7110 *av7110 = (struct av7110*) dev->ext_priv;586587dprintk(2, "%s\n", __func__);588av7110->wssMode = 0;589av7110->wssData = 0;590if (FW_VERSION(av7110->arm_app) < 0x2623)591return 0;592else593return av7110_fw_cmd(av7110, COMTYPE_ENCODER, SetWSSConfig, 1, 0);594}595596static ssize_t av7110_vbi_write(struct file *file, const char __user *data, size_t count, loff_t *ppos)597{598struct saa7146_fh *fh = file->private_data;599struct saa7146_dev *dev = fh->dev;600struct av7110 *av7110 = (struct av7110*) dev->ext_priv;601struct v4l2_sliced_vbi_data d;602int rc;603604dprintk(2, "%s\n", __func__);605if (FW_VERSION(av7110->arm_app) < 0x2623 || !av7110->wssMode || count != sizeof d)606return -EINVAL;607if (copy_from_user(&d, data, count))608return -EFAULT;609if ((d.id != 0 && d.id != V4L2_SLICED_WSS_625) || d.field != 0 || d.line != 23)610return -EINVAL;611if (d.id)612av7110->wssData = ((d.data[1] << 8) & 0x3f00) | d.data[0];613else614av7110->wssData = 0x8000;615rc = av7110_fw_cmd(av7110, COMTYPE_ENCODER, SetWSSConfig, 2, 1, av7110->wssData);616return (rc < 0) ? rc : count;617}618619/****************************************************************************620* INITIALIZATION621****************************************************************************/622623static u8 saa7113_init_regs[] = {6240x02, 0xd0,6250x03, 0x23,6260x04, 0x00,6270x05, 0x00,6280x06, 0xe9,6290x07, 0x0d,6300x08, 0x98,6310x09, 0x02,6320x0a, 0x80,6330x0b, 0x40,6340x0c, 0x40,6350x0d, 0x00,6360x0e, 0x01,6370x0f, 0x7c,6380x10, 0x48,6390x11, 0x0c,6400x12, 0x8b,6410x13, 0x1a,6420x14, 0x00,6430x15, 0x00,6440x16, 0x00,6450x17, 0x00,6460x18, 0x00,6470x19, 0x00,6480x1a, 0x00,6490x1b, 0x00,6500x1c, 0x00,6510x1d, 0x00,6520x1e, 0x00,6536540x41, 0x77,6550x42, 0x77,6560x43, 0x77,6570x44, 0x77,6580x45, 0x77,6590x46, 0x77,6600x47, 0x77,6610x48, 0x77,6620x49, 0x77,6630x4a, 0x77,6640x4b, 0x77,6650x4c, 0x77,6660x4d, 0x77,6670x4e, 0x77,6680x4f, 0x77,6690x50, 0x77,6700x51, 0x77,6710x52, 0x77,6720x53, 0x77,6730x54, 0x77,6740x55, 0x77,6750x56, 0x77,6760x57, 0xff,6776780xff679};680681682static struct saa7146_ext_vv av7110_vv_data_st;683static struct saa7146_ext_vv av7110_vv_data_c;684685int av7110_init_analog_module(struct av7110 *av7110)686{687u16 version1, version2;688689if (i2c_writereg(av7110, 0x80, 0x0, 0x80) == 1 &&690i2c_writereg(av7110, 0x80, 0x0, 0) == 1) {691printk("dvb-ttpci: DVB-C analog module @ card %d detected, initializing MSP3400\n",692av7110->dvb_adapter.num);693av7110->adac_type = DVB_ADAC_MSP34x0;694} else if (i2c_writereg(av7110, 0x84, 0x0, 0x80) == 1 &&695i2c_writereg(av7110, 0x84, 0x0, 0) == 1) {696printk("dvb-ttpci: DVB-C analog module @ card %d detected, initializing MSP3415\n",697av7110->dvb_adapter.num);698av7110->adac_type = DVB_ADAC_MSP34x5;699} else700return -ENODEV;701702msleep(100); // the probing above resets the msp...703msp_readreg(av7110, MSP_RD_DSP, 0x001e, &version1);704msp_readreg(av7110, MSP_RD_DSP, 0x001f, &version2);705dprintk(1, "dvb-ttpci: @ card %d MSP34xx version 0x%04x 0x%04x\n",706av7110->dvb_adapter.num, version1, version2);707msp_writereg(av7110, MSP_WR_DSP, 0x0013, 0x0c00);708msp_writereg(av7110, MSP_WR_DSP, 0x0000, 0x7f00); // loudspeaker + headphone709msp_writereg(av7110, MSP_WR_DSP, 0x0008, 0x0220); // loudspeaker source710msp_writereg(av7110, MSP_WR_DSP, 0x0009, 0x0220); // headphone source711msp_writereg(av7110, MSP_WR_DSP, 0x0004, 0x7f00); // loudspeaker volume712msp_writereg(av7110, MSP_WR_DSP, 0x000a, 0x0220); // SCART 1 source713msp_writereg(av7110, MSP_WR_DSP, 0x0007, 0x7f00); // SCART 1 volume714msp_writereg(av7110, MSP_WR_DSP, 0x000d, 0x1900); // prescale SCART715716if (i2c_writereg(av7110, 0x48, 0x01, 0x00)!=1) {717INFO(("saa7113 not accessible.\n"));718} else {719u8 *i = saa7113_init_regs;720721if ((av7110->dev->pci->subsystem_vendor == 0x110a) && (av7110->dev->pci->subsystem_device == 0x0000)) {722/* Fujitsu/Siemens DVB-Cable */723av7110->analog_tuner_flags |= ANALOG_TUNER_VES1820;724} else if ((av7110->dev->pci->subsystem_vendor == 0x13c2) && (av7110->dev->pci->subsystem_device == 0x0002)) {725/* Hauppauge/TT DVB-C premium */726av7110->analog_tuner_flags |= ANALOG_TUNER_VES1820;727} else if ((av7110->dev->pci->subsystem_vendor == 0x13c2) && (av7110->dev->pci->subsystem_device == 0x000A)) {728/* Hauppauge/TT DVB-C premium */729av7110->analog_tuner_flags |= ANALOG_TUNER_STV0297;730}731732/* setup for DVB by default */733if (av7110->analog_tuner_flags & ANALOG_TUNER_VES1820) {734if (ves1820_writereg(av7110->dev, 0x09, 0x0f, 0x20))735dprintk(1, "setting band in demodulator failed.\n");736} else if (av7110->analog_tuner_flags & ANALOG_TUNER_STV0297) {737saa7146_setgpio(av7110->dev, 1, SAA7146_GPIO_OUTLO); // TDA9819 pin9(STD)738saa7146_setgpio(av7110->dev, 3, SAA7146_GPIO_OUTLO); // TDA9819 pin30(VIF)739}740741/* init the saa7113 */742while (*i != 0xff) {743if (i2c_writereg(av7110, 0x48, i[0], i[1]) != 1) {744dprintk(1, "saa7113 initialization failed @ card %d", av7110->dvb_adapter.num);745break;746}747i += 2;748}749/* setup msp for analog sound: B/G Dual-FM */750msp_writereg(av7110, MSP_WR_DEM, 0x00bb, 0x02d0); // AD_CV751msp_writereg(av7110, MSP_WR_DEM, 0x0001, 3); // FIR1752msp_writereg(av7110, MSP_WR_DEM, 0x0001, 18); // FIR1753msp_writereg(av7110, MSP_WR_DEM, 0x0001, 27); // FIR1754msp_writereg(av7110, MSP_WR_DEM, 0x0001, 48); // FIR1755msp_writereg(av7110, MSP_WR_DEM, 0x0001, 66); // FIR1756msp_writereg(av7110, MSP_WR_DEM, 0x0001, 72); // FIR1757msp_writereg(av7110, MSP_WR_DEM, 0x0005, 4); // FIR2758msp_writereg(av7110, MSP_WR_DEM, 0x0005, 64); // FIR2759msp_writereg(av7110, MSP_WR_DEM, 0x0005, 0); // FIR2760msp_writereg(av7110, MSP_WR_DEM, 0x0005, 3); // FIR2761msp_writereg(av7110, MSP_WR_DEM, 0x0005, 18); // FIR2762msp_writereg(av7110, MSP_WR_DEM, 0x0005, 27); // FIR2763msp_writereg(av7110, MSP_WR_DEM, 0x0005, 48); // FIR2764msp_writereg(av7110, MSP_WR_DEM, 0x0005, 66); // FIR2765msp_writereg(av7110, MSP_WR_DEM, 0x0005, 72); // FIR2766msp_writereg(av7110, MSP_WR_DEM, 0x0083, 0xa000); // MODE_REG767msp_writereg(av7110, MSP_WR_DEM, 0x0093, 0x00aa); // DCO1_LO 5.74MHz768msp_writereg(av7110, MSP_WR_DEM, 0x009b, 0x04fc); // DCO1_HI769msp_writereg(av7110, MSP_WR_DEM, 0x00a3, 0x038e); // DCO2_LO 5.5MHz770msp_writereg(av7110, MSP_WR_DEM, 0x00ab, 0x04c6); // DCO2_HI771msp_writereg(av7110, MSP_WR_DEM, 0x0056, 0); // LOAD_REG 1/2772}773774memcpy(standard, dvb_standard, sizeof(struct saa7146_standard) * 2);775/* set dd1 stream a & b */776saa7146_write(av7110->dev, DD1_STREAM_B, 0x00000000);777saa7146_write(av7110->dev, DD1_INIT, 0x03000700);778saa7146_write(av7110->dev, MC2, (MASK_09 | MASK_25 | MASK_10 | MASK_26));779780return 0;781}782783int av7110_init_v4l(struct av7110 *av7110)784{785struct saa7146_dev* dev = av7110->dev;786struct saa7146_ext_vv *vv_data;787int ret;788789/* special case DVB-C: these cards have an analog tuner790plus need some special handling, so we have separate791saa7146_ext_vv data for these... */792if (av7110->analog_tuner_flags)793vv_data = &av7110_vv_data_c;794else795vv_data = &av7110_vv_data_st;796ret = saa7146_vv_init(dev, vv_data);797798if (ret) {799ERR(("cannot init capture device. skipping.\n"));800return -ENODEV;801}802vv_data->ops.vidioc_enum_input = vidioc_enum_input;803vv_data->ops.vidioc_g_input = vidioc_g_input;804vv_data->ops.vidioc_s_input = vidioc_s_input;805vv_data->ops.vidioc_g_tuner = vidioc_g_tuner;806vv_data->ops.vidioc_s_tuner = vidioc_s_tuner;807vv_data->ops.vidioc_g_frequency = vidioc_g_frequency;808vv_data->ops.vidioc_s_frequency = vidioc_s_frequency;809vv_data->ops.vidioc_g_audio = vidioc_g_audio;810vv_data->ops.vidioc_s_audio = vidioc_s_audio;811vv_data->ops.vidioc_g_sliced_vbi_cap = vidioc_g_sliced_vbi_cap;812vv_data->ops.vidioc_g_fmt_sliced_vbi_out = vidioc_g_fmt_sliced_vbi_out;813vv_data->ops.vidioc_s_fmt_sliced_vbi_out = vidioc_s_fmt_sliced_vbi_out;814815if (saa7146_register_device(&av7110->v4l_dev, dev, "av7110", VFL_TYPE_GRABBER)) {816ERR(("cannot register capture device. skipping.\n"));817saa7146_vv_release(dev);818return -ENODEV;819}820if (saa7146_register_device(&av7110->vbi_dev, dev, "av7110", VFL_TYPE_VBI))821ERR(("cannot register vbi v4l2 device. skipping.\n"));822return 0;823}824825int av7110_exit_v4l(struct av7110 *av7110)826{827struct saa7146_dev* dev = av7110->dev;828829saa7146_unregister_device(&av7110->v4l_dev, av7110->dev);830saa7146_unregister_device(&av7110->vbi_dev, av7110->dev);831832saa7146_vv_release(dev);833834return 0;835}836837838839/* FIXME: these values are experimental values that look better than the840values from the latest "official" driver -- at least for me... (MiHu) */841static struct saa7146_standard standard[] = {842{843.name = "PAL", .id = V4L2_STD_PAL_BG,844.v_offset = 0x15, .v_field = 288,845.h_offset = 0x48, .h_pixels = 708,846.v_max_out = 576, .h_max_out = 768,847}, {848.name = "NTSC", .id = V4L2_STD_NTSC,849.v_offset = 0x10, .v_field = 244,850.h_offset = 0x40, .h_pixels = 708,851.v_max_out = 480, .h_max_out = 640,852}853};854855static struct saa7146_standard analog_standard[] = {856{857.name = "PAL", .id = V4L2_STD_PAL_BG,858.v_offset = 0x1b, .v_field = 288,859.h_offset = 0x08, .h_pixels = 708,860.v_max_out = 576, .h_max_out = 768,861}, {862.name = "NTSC", .id = V4L2_STD_NTSC,863.v_offset = 0x10, .v_field = 244,864.h_offset = 0x40, .h_pixels = 708,865.v_max_out = 480, .h_max_out = 640,866}867};868869static struct saa7146_standard dvb_standard[] = {870{871.name = "PAL", .id = V4L2_STD_PAL_BG,872.v_offset = 0x14, .v_field = 288,873.h_offset = 0x48, .h_pixels = 708,874.v_max_out = 576, .h_max_out = 768,875}, {876.name = "NTSC", .id = V4L2_STD_NTSC,877.v_offset = 0x10, .v_field = 244,878.h_offset = 0x40, .h_pixels = 708,879.v_max_out = 480, .h_max_out = 640,880}881};882883static int std_callback(struct saa7146_dev* dev, struct saa7146_standard *std)884{885struct av7110 *av7110 = (struct av7110*) dev->ext_priv;886887if (std->id & V4L2_STD_PAL) {888av7110->vidmode = AV7110_VIDEO_MODE_PAL;889av7110_set_vidmode(av7110, av7110->vidmode);890}891else if (std->id & V4L2_STD_NTSC) {892av7110->vidmode = AV7110_VIDEO_MODE_NTSC;893av7110_set_vidmode(av7110, av7110->vidmode);894}895else896return -1;897898return 0;899}900901902static struct saa7146_ext_vv av7110_vv_data_st = {903.inputs = 1,904.audios = 1,905.capabilities = V4L2_CAP_SLICED_VBI_OUTPUT,906.flags = 0,907908.stds = &standard[0],909.num_stds = ARRAY_SIZE(standard),910.std_callback = &std_callback,911912.vbi_fops.open = av7110_vbi_reset,913.vbi_fops.release = av7110_vbi_reset,914.vbi_fops.write = av7110_vbi_write,915};916917static struct saa7146_ext_vv av7110_vv_data_c = {918.inputs = 1,919.audios = 1,920.capabilities = V4L2_CAP_TUNER | V4L2_CAP_SLICED_VBI_OUTPUT,921.flags = SAA7146_USE_PORT_B_FOR_VBI,922923.stds = &standard[0],924.num_stds = ARRAY_SIZE(standard),925.std_callback = &std_callback,926927.vbi_fops.open = av7110_vbi_reset,928.vbi_fops.release = av7110_vbi_reset,929.vbi_fops.write = av7110_vbi_write,930};931932933934