Path: blob/master/drivers/media/dvb/ngene/ngene-core.c
15111 views
/*1* ngene.c: nGene PCIe bridge driver2*3* Copyright (C) 2005-2007 Micronas4*5* Copyright (C) 2008-2009 Ralph Metzler <[email protected]>6* Modifications for new nGene firmware,7* support for EEPROM-copying,8* support for new dual DVB-S2 card prototype9*10*11* This program is free software; you can redistribute it and/or12* modify it under the terms of the GNU General Public License13* version 2 only, as published by the Free Software Foundation.14*15*16* This program is distributed in the hope that it will be useful,17* but WITHOUT ANY WARRANTY; without even the implied warranty of18* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the19* GNU General Public License for more details.20*21*22* You should have received a copy of the GNU General Public License23* along with this program; if not, write to the Free Software24* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA25* 02110-1301, USA26* Or, point your browser to http://www.gnu.org/copyleft/gpl.html27*/2829#include <linux/module.h>30#include <linux/init.h>31#include <linux/delay.h>32#include <linux/poll.h>33#include <linux/io.h>34#include <asm/div64.h>35#include <linux/pci.h>36#include <linux/timer.h>37#include <linux/byteorder/generic.h>38#include <linux/firmware.h>39#include <linux/vmalloc.h>4041#include "ngene.h"4243static int one_adapter = 1;44module_param(one_adapter, int, 0444);45MODULE_PARM_DESC(one_adapter, "Use only one adapter.");4647static int shutdown_workaround;48module_param(shutdown_workaround, int, 0644);49MODULE_PARM_DESC(shutdown_workaround, "Activate workaround for shutdown problem with some chipsets.");5051static int debug;52module_param(debug, int, 0444);53MODULE_PARM_DESC(debug, "Print debugging information.");5455DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);5657#define dprintk if (debug) printk5859#define ngwriteb(dat, adr) writeb((dat), (char *)(dev->iomem + (adr)))60#define ngwritel(dat, adr) writel((dat), (char *)(dev->iomem + (adr)))61#define ngwriteb(dat, adr) writeb((dat), (char *)(dev->iomem + (adr)))62#define ngreadl(adr) readl(dev->iomem + (adr))63#define ngreadb(adr) readb(dev->iomem + (adr))64#define ngcpyto(adr, src, count) memcpy_toio((char *) \65(dev->iomem + (adr)), (src), (count))66#define ngcpyfrom(dst, adr, count) memcpy_fromio((dst), (char *) \67(dev->iomem + (adr)), (count))6869/****************************************************************************/70/* nGene interrupt handler **************************************************/71/****************************************************************************/7273static void event_tasklet(unsigned long data)74{75struct ngene *dev = (struct ngene *)data;7677while (dev->EventQueueReadIndex != dev->EventQueueWriteIndex) {78struct EVENT_BUFFER Event =79dev->EventQueue[dev->EventQueueReadIndex];80dev->EventQueueReadIndex =81(dev->EventQueueReadIndex + 1) & (EVENT_QUEUE_SIZE - 1);8283if ((Event.UARTStatus & 0x01) && (dev->TxEventNotify))84dev->TxEventNotify(dev, Event.TimeStamp);85if ((Event.UARTStatus & 0x02) && (dev->RxEventNotify))86dev->RxEventNotify(dev, Event.TimeStamp,87Event.RXCharacter);88}89}9091static void demux_tasklet(unsigned long data)92{93struct ngene_channel *chan = (struct ngene_channel *)data;94struct SBufferHeader *Cur = chan->nextBuffer;9596spin_lock_irq(&chan->state_lock);9798while (Cur->ngeneBuffer.SR.Flags & 0x80) {99if (chan->mode & NGENE_IO_TSOUT) {100u32 Flags = chan->DataFormatFlags;101if (Cur->ngeneBuffer.SR.Flags & 0x20)102Flags |= BEF_OVERFLOW;103if (chan->pBufferExchange) {104if (!chan->pBufferExchange(chan,105Cur->Buffer1,106chan->Capture1Length,107Cur->ngeneBuffer.SR.108Clock, Flags)) {109/*110We didn't get data111Clear in service flag to make sure we112get called on next interrupt again.113leave fill/empty (0x80) flag alone114to avoid hardware running out of115buffers during startup, we hold only116in run state ( the source may be late117delivering data )118*/119120if (chan->HWState == HWSTATE_RUN) {121Cur->ngeneBuffer.SR.Flags &=122~0x40;123break;124/* Stop processing stream */125}126} else {127/* We got a valid buffer,128so switch to run state */129chan->HWState = HWSTATE_RUN;130}131} else {132printk(KERN_ERR DEVICE_NAME ": OOPS\n");133if (chan->HWState == HWSTATE_RUN) {134Cur->ngeneBuffer.SR.Flags &= ~0x40;135break; /* Stop processing stream */136}137}138if (chan->AudioDTOUpdated) {139printk(KERN_INFO DEVICE_NAME140": Update AudioDTO = %d\n",141chan->AudioDTOValue);142Cur->ngeneBuffer.SR.DTOUpdate =143chan->AudioDTOValue;144chan->AudioDTOUpdated = 0;145}146} else {147if (chan->HWState == HWSTATE_RUN) {148u32 Flags = chan->DataFormatFlags;149IBufferExchange *exch1 = chan->pBufferExchange;150IBufferExchange *exch2 = chan->pBufferExchange2;151if (Cur->ngeneBuffer.SR.Flags & 0x01)152Flags |= BEF_EVEN_FIELD;153if (Cur->ngeneBuffer.SR.Flags & 0x20)154Flags |= BEF_OVERFLOW;155spin_unlock_irq(&chan->state_lock);156if (exch1)157exch1(chan, Cur->Buffer1,158chan->Capture1Length,159Cur->ngeneBuffer.SR.Clock,160Flags);161if (exch2)162exch2(chan, Cur->Buffer2,163chan->Capture2Length,164Cur->ngeneBuffer.SR.Clock,165Flags);166spin_lock_irq(&chan->state_lock);167} else if (chan->HWState != HWSTATE_STOP)168chan->HWState = HWSTATE_RUN;169}170Cur->ngeneBuffer.SR.Flags = 0x00;171Cur = Cur->Next;172}173chan->nextBuffer = Cur;174175spin_unlock_irq(&chan->state_lock);176}177178static irqreturn_t irq_handler(int irq, void *dev_id)179{180struct ngene *dev = (struct ngene *)dev_id;181u32 icounts = 0;182irqreturn_t rc = IRQ_NONE;183u32 i = MAX_STREAM;184u8 *tmpCmdDoneByte;185186if (dev->BootFirmware) {187icounts = ngreadl(NGENE_INT_COUNTS);188if (icounts != dev->icounts) {189ngwritel(0, FORCE_NMI);190dev->cmd_done = 1;191wake_up(&dev->cmd_wq);192dev->icounts = icounts;193rc = IRQ_HANDLED;194}195return rc;196}197198ngwritel(0, FORCE_NMI);199200spin_lock(&dev->cmd_lock);201tmpCmdDoneByte = dev->CmdDoneByte;202if (tmpCmdDoneByte &&203(*tmpCmdDoneByte ||204(dev->ngenetohost[0] == 1 && dev->ngenetohost[1] != 0))) {205dev->CmdDoneByte = NULL;206dev->cmd_done = 1;207wake_up(&dev->cmd_wq);208rc = IRQ_HANDLED;209}210spin_unlock(&dev->cmd_lock);211212if (dev->EventBuffer->EventStatus & 0x80) {213u8 nextWriteIndex =214(dev->EventQueueWriteIndex + 1) &215(EVENT_QUEUE_SIZE - 1);216if (nextWriteIndex != dev->EventQueueReadIndex) {217dev->EventQueue[dev->EventQueueWriteIndex] =218*(dev->EventBuffer);219dev->EventQueueWriteIndex = nextWriteIndex;220} else {221printk(KERN_ERR DEVICE_NAME ": event overflow\n");222dev->EventQueueOverflowCount += 1;223dev->EventQueueOverflowFlag = 1;224}225dev->EventBuffer->EventStatus &= ~0x80;226tasklet_schedule(&dev->event_tasklet);227rc = IRQ_HANDLED;228}229230while (i > 0) {231i--;232spin_lock(&dev->channel[i].state_lock);233/* if (dev->channel[i].State>=KSSTATE_RUN) { */234if (dev->channel[i].nextBuffer) {235if ((dev->channel[i].nextBuffer->236ngeneBuffer.SR.Flags & 0xC0) == 0x80) {237dev->channel[i].nextBuffer->238ngeneBuffer.SR.Flags |= 0x40;239tasklet_schedule(240&dev->channel[i].demux_tasklet);241rc = IRQ_HANDLED;242}243}244spin_unlock(&dev->channel[i].state_lock);245}246247/* Request might have been processed by a previous call. */248return IRQ_HANDLED;249}250251/****************************************************************************/252/* nGene command interface **************************************************/253/****************************************************************************/254255static void dump_command_io(struct ngene *dev)256{257u8 buf[8], *b;258259ngcpyfrom(buf, HOST_TO_NGENE, 8);260printk(KERN_ERR "host_to_ngene (%04x): %02x %02x %02x %02x %02x %02x %02x %02x\n",261HOST_TO_NGENE, buf[0], buf[1], buf[2], buf[3],262buf[4], buf[5], buf[6], buf[7]);263264ngcpyfrom(buf, NGENE_TO_HOST, 8);265printk(KERN_ERR "ngene_to_host (%04x): %02x %02x %02x %02x %02x %02x %02x %02x\n",266NGENE_TO_HOST, buf[0], buf[1], buf[2], buf[3],267buf[4], buf[5], buf[6], buf[7]);268269b = dev->hosttongene;270printk(KERN_ERR "dev->hosttongene (%p): %02x %02x %02x %02x %02x %02x %02x %02x\n",271b, b[0], b[1], b[2], b[3], b[4], b[5], b[6], b[7]);272273b = dev->ngenetohost;274printk(KERN_ERR "dev->ngenetohost (%p): %02x %02x %02x %02x %02x %02x %02x %02x\n",275b, b[0], b[1], b[2], b[3], b[4], b[5], b[6], b[7]);276}277278static int ngene_command_mutex(struct ngene *dev, struct ngene_command *com)279{280int ret;281u8 *tmpCmdDoneByte;282283dev->cmd_done = 0;284285if (com->cmd.hdr.Opcode == CMD_FWLOAD_PREPARE) {286dev->BootFirmware = 1;287dev->icounts = ngreadl(NGENE_INT_COUNTS);288ngwritel(0, NGENE_COMMAND);289ngwritel(0, NGENE_COMMAND_HI);290ngwritel(0, NGENE_STATUS);291ngwritel(0, NGENE_STATUS_HI);292ngwritel(0, NGENE_EVENT);293ngwritel(0, NGENE_EVENT_HI);294} else if (com->cmd.hdr.Opcode == CMD_FWLOAD_FINISH) {295u64 fwio = dev->PAFWInterfaceBuffer;296297ngwritel(fwio & 0xffffffff, NGENE_COMMAND);298ngwritel(fwio >> 32, NGENE_COMMAND_HI);299ngwritel((fwio + 256) & 0xffffffff, NGENE_STATUS);300ngwritel((fwio + 256) >> 32, NGENE_STATUS_HI);301ngwritel((fwio + 512) & 0xffffffff, NGENE_EVENT);302ngwritel((fwio + 512) >> 32, NGENE_EVENT_HI);303}304305memcpy(dev->FWInterfaceBuffer, com->cmd.raw8, com->in_len + 2);306307if (dev->BootFirmware)308ngcpyto(HOST_TO_NGENE, com->cmd.raw8, com->in_len + 2);309310spin_lock_irq(&dev->cmd_lock);311tmpCmdDoneByte = dev->ngenetohost + com->out_len;312if (!com->out_len)313tmpCmdDoneByte++;314*tmpCmdDoneByte = 0;315dev->ngenetohost[0] = 0;316dev->ngenetohost[1] = 0;317dev->CmdDoneByte = tmpCmdDoneByte;318spin_unlock_irq(&dev->cmd_lock);319320/* Notify 8051. */321ngwritel(1, FORCE_INT);322323ret = wait_event_timeout(dev->cmd_wq, dev->cmd_done == 1, 2 * HZ);324if (!ret) {325/*ngwritel(0, FORCE_NMI);*/326327printk(KERN_ERR DEVICE_NAME328": Command timeout cmd=%02x prev=%02x\n",329com->cmd.hdr.Opcode, dev->prev_cmd);330dump_command_io(dev);331return -1;332}333if (com->cmd.hdr.Opcode == CMD_FWLOAD_FINISH)334dev->BootFirmware = 0;335336dev->prev_cmd = com->cmd.hdr.Opcode;337338if (!com->out_len)339return 0;340341memcpy(com->cmd.raw8, dev->ngenetohost, com->out_len);342343return 0;344}345346int ngene_command(struct ngene *dev, struct ngene_command *com)347{348int result;349350down(&dev->cmd_mutex);351result = ngene_command_mutex(dev, com);352up(&dev->cmd_mutex);353return result;354}355356357static int ngene_command_load_firmware(struct ngene *dev,358u8 *ngene_fw, u32 size)359{360#define FIRSTCHUNK (1024)361u32 cleft;362struct ngene_command com;363364com.cmd.hdr.Opcode = CMD_FWLOAD_PREPARE;365com.cmd.hdr.Length = 0;366com.in_len = 0;367com.out_len = 0;368369ngene_command(dev, &com);370371cleft = (size + 3) & ~3;372if (cleft > FIRSTCHUNK) {373ngcpyto(PROGRAM_SRAM + FIRSTCHUNK, ngene_fw + FIRSTCHUNK,374cleft - FIRSTCHUNK);375cleft = FIRSTCHUNK;376}377ngcpyto(DATA_FIFO_AREA, ngene_fw, cleft);378379memset(&com, 0, sizeof(struct ngene_command));380com.cmd.hdr.Opcode = CMD_FWLOAD_FINISH;381com.cmd.hdr.Length = 4;382com.cmd.FWLoadFinish.Address = DATA_FIFO_AREA;383com.cmd.FWLoadFinish.Length = (unsigned short)cleft;384com.in_len = 4;385com.out_len = 0;386387return ngene_command(dev, &com);388}389390391static int ngene_command_config_buf(struct ngene *dev, u8 config)392{393struct ngene_command com;394395com.cmd.hdr.Opcode = CMD_CONFIGURE_BUFFER;396com.cmd.hdr.Length = 1;397com.cmd.ConfigureBuffers.config = config;398com.in_len = 1;399com.out_len = 0;400401if (ngene_command(dev, &com) < 0)402return -EIO;403return 0;404}405406static int ngene_command_config_free_buf(struct ngene *dev, u8 *config)407{408struct ngene_command com;409410com.cmd.hdr.Opcode = CMD_CONFIGURE_FREE_BUFFER;411com.cmd.hdr.Length = 6;412memcpy(&com.cmd.ConfigureBuffers.config, config, 6);413com.in_len = 6;414com.out_len = 0;415416if (ngene_command(dev, &com) < 0)417return -EIO;418419return 0;420}421422int ngene_command_gpio_set(struct ngene *dev, u8 select, u8 level)423{424struct ngene_command com;425426com.cmd.hdr.Opcode = CMD_SET_GPIO_PIN;427com.cmd.hdr.Length = 1;428com.cmd.SetGpioPin.select = select | (level << 7);429com.in_len = 1;430com.out_len = 0;431432return ngene_command(dev, &com);433}434435436/*43702000640 is sample on rising edge.43802000740 is sample on falling edge.43902000040 is ignore "valid" signal4404410: FD_CTL1 Bit 7,6 must be 0,14427 disable(fw controlled)4436 0-AUX,1-TS4445 0-par,1-ser4454 0-lsb/1-msb4463,2 reserved4471,0 0-no sync, 1-use ext. start, 2-use 0x47, 3-both4481: FD_CTL2 has 3-valid must be hi, 2-use valid, 1-edge4492: FD_STA is read-only. 0-sync4503: FD_INSYNC is number of 47s to trigger "in sync".4514: FD_OUTSYNC is number of 47s to trigger "out of sync".4525: FD_MAXBYTE1 is low-order of bytes per packet.4536: FD_MAXBYTE2 is high-order of bytes per packet.4547: Top byte is unused.455*/456457/****************************************************************************/458459static u8 TSFeatureDecoderSetup[8 * 5] = {4600x42, 0x00, 0x00, 0x02, 0x02, 0xbc, 0x00, 0x00,4610x40, 0x06, 0x00, 0x02, 0x02, 0xbc, 0x00, 0x00, /* DRXH */4620x71, 0x07, 0x00, 0x02, 0x02, 0xbc, 0x00, 0x00, /* DRXHser */4630x72, 0x06, 0x00, 0x02, 0x02, 0xbc, 0x00, 0x00, /* S2ser */4640x40, 0x07, 0x00, 0x02, 0x02, 0xbc, 0x00, 0x00, /* LGDT3303 */465};466467/* Set NGENE I2S Config to 16 bit packed */468static u8 I2SConfiguration[] = {4690x00, 0x10, 0x00, 0x00,4700x80, 0x10, 0x00, 0x00,471};472473static u8 SPDIFConfiguration[10] = {4740x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00475};476477/* Set NGENE I2S Config to transport stream compatible mode */478479static u8 TS_I2SConfiguration[4] = { 0x3E, 0x18, 0x00, 0x00 };480481static u8 TS_I2SOutConfiguration[4] = { 0x80, 0x04, 0x00, 0x00 };482483static u8 ITUDecoderSetup[4][16] = {484{0x1c, 0x13, 0x01, 0x68, 0x3d, 0x90, 0x14, 0x20, /* SDTV */4850x00, 0x00, 0x01, 0xb0, 0x9c, 0x00, 0x00, 0x00},486{0x9c, 0x03, 0x23, 0xC0, 0x60, 0x0E, 0x13, 0x00,4870x00, 0x00, 0x00, 0x01, 0xB0, 0x00, 0x00, 0x00},488{0x9f, 0x00, 0x23, 0xC0, 0x60, 0x0F, 0x13, 0x00, /* HDTV 1080i50 */4890x00, 0x00, 0x00, 0x01, 0xB0, 0x00, 0x00, 0x00},490{0x9c, 0x01, 0x23, 0xC0, 0x60, 0x0E, 0x13, 0x00, /* HDTV 1080i60 */4910x00, 0x00, 0x00, 0x01, 0xB0, 0x00, 0x00, 0x00},492};493494/*495* 50 48 60 gleich496* 27p50 9f 00 22 80 42 69 18 ...497* 27p60 93 00 22 80 82 69 1c ...498*/499500/* Maxbyte to 1144 (for raw data) */501static u8 ITUFeatureDecoderSetup[8] = {5020x00, 0x00, 0x00, 0x00, 0x00, 0x78, 0x04, 0x00503};504505void FillTSBuffer(void *Buffer, int Length, u32 Flags)506{507u32 *ptr = Buffer;508509memset(Buffer, 0xff, Length);510while (Length > 0) {511if (Flags & DF_SWAP32)512*ptr = 0x471FFF10;513else514*ptr = 0x10FF1F47;515ptr += (188 / 4);516Length -= 188;517}518}519520521static void flush_buffers(struct ngene_channel *chan)522{523u8 val;524525do {526msleep(1);527spin_lock_irq(&chan->state_lock);528val = chan->nextBuffer->ngeneBuffer.SR.Flags & 0x80;529spin_unlock_irq(&chan->state_lock);530} while (val);531}532533static void clear_buffers(struct ngene_channel *chan)534{535struct SBufferHeader *Cur = chan->nextBuffer;536537do {538memset(&Cur->ngeneBuffer.SR, 0, sizeof(Cur->ngeneBuffer.SR));539if (chan->mode & NGENE_IO_TSOUT)540FillTSBuffer(Cur->Buffer1,541chan->Capture1Length,542chan->DataFormatFlags);543Cur = Cur->Next;544} while (Cur != chan->nextBuffer);545546if (chan->mode & NGENE_IO_TSOUT) {547chan->nextBuffer->ngeneBuffer.SR.DTOUpdate =548chan->AudioDTOValue;549chan->AudioDTOUpdated = 0;550551Cur = chan->TSIdleBuffer.Head;552553do {554memset(&Cur->ngeneBuffer.SR, 0,555sizeof(Cur->ngeneBuffer.SR));556FillTSBuffer(Cur->Buffer1,557chan->Capture1Length,558chan->DataFormatFlags);559Cur = Cur->Next;560} while (Cur != chan->TSIdleBuffer.Head);561}562}563564static int ngene_command_stream_control(struct ngene *dev, u8 stream,565u8 control, u8 mode, u8 flags)566{567struct ngene_channel *chan = &dev->channel[stream];568struct ngene_command com;569u16 BsUVI = ((stream & 1) ? 0x9400 : 0x9300);570u16 BsSDI = ((stream & 1) ? 0x9600 : 0x9500);571u16 BsSPI = ((stream & 1) ? 0x9800 : 0x9700);572u16 BsSDO = 0x9B00;573574down(&dev->stream_mutex);575memset(&com, 0, sizeof(com));576com.cmd.hdr.Opcode = CMD_CONTROL;577com.cmd.hdr.Length = sizeof(struct FW_STREAM_CONTROL) - 2;578com.cmd.StreamControl.Stream = stream | (control ? 8 : 0);579if (chan->mode & NGENE_IO_TSOUT)580com.cmd.StreamControl.Stream |= 0x07;581com.cmd.StreamControl.Control = control |582(flags & SFLAG_ORDER_LUMA_CHROMA);583com.cmd.StreamControl.Mode = mode;584com.in_len = sizeof(struct FW_STREAM_CONTROL);585com.out_len = 0;586587dprintk(KERN_INFO DEVICE_NAME588": Stream=%02x, Control=%02x, Mode=%02x\n",589com.cmd.StreamControl.Stream, com.cmd.StreamControl.Control,590com.cmd.StreamControl.Mode);591592chan->Mode = mode;593594if (!(control & 0x80)) {595spin_lock_irq(&chan->state_lock);596if (chan->State == KSSTATE_RUN) {597chan->State = KSSTATE_ACQUIRE;598chan->HWState = HWSTATE_STOP;599spin_unlock_irq(&chan->state_lock);600if (ngene_command(dev, &com) < 0) {601up(&dev->stream_mutex);602return -1;603}604/* clear_buffers(chan); */605flush_buffers(chan);606up(&dev->stream_mutex);607return 0;608}609spin_unlock_irq(&chan->state_lock);610up(&dev->stream_mutex);611return 0;612}613614if (mode & SMODE_AUDIO_CAPTURE) {615com.cmd.StreamControl.CaptureBlockCount =616chan->Capture1Length / AUDIO_BLOCK_SIZE;617com.cmd.StreamControl.Buffer_Address = chan->RingBuffer.PAHead;618} else if (mode & SMODE_TRANSPORT_STREAM) {619com.cmd.StreamControl.CaptureBlockCount =620chan->Capture1Length / TS_BLOCK_SIZE;621com.cmd.StreamControl.MaxLinesPerField =622chan->Capture1Length / TS_BLOCK_SIZE;623com.cmd.StreamControl.Buffer_Address =624chan->TSRingBuffer.PAHead;625if (chan->mode & NGENE_IO_TSOUT) {626com.cmd.StreamControl.BytesPerVBILine =627chan->Capture1Length / TS_BLOCK_SIZE;628com.cmd.StreamControl.Stream |= 0x07;629}630} else {631com.cmd.StreamControl.BytesPerVideoLine = chan->nBytesPerLine;632com.cmd.StreamControl.MaxLinesPerField = chan->nLines;633com.cmd.StreamControl.MinLinesPerField = 100;634com.cmd.StreamControl.Buffer_Address = chan->RingBuffer.PAHead;635636if (mode & SMODE_VBI_CAPTURE) {637com.cmd.StreamControl.MaxVBILinesPerField =638chan->nVBILines;639com.cmd.StreamControl.MinVBILinesPerField = 0;640com.cmd.StreamControl.BytesPerVBILine =641chan->nBytesPerVBILine;642}643if (flags & SFLAG_COLORBAR)644com.cmd.StreamControl.Stream |= 0x04;645}646647spin_lock_irq(&chan->state_lock);648if (mode & SMODE_AUDIO_CAPTURE) {649chan->nextBuffer = chan->RingBuffer.Head;650if (mode & SMODE_AUDIO_SPDIF) {651com.cmd.StreamControl.SetupDataLen =652sizeof(SPDIFConfiguration);653com.cmd.StreamControl.SetupDataAddr = BsSPI;654memcpy(com.cmd.StreamControl.SetupData,655SPDIFConfiguration, sizeof(SPDIFConfiguration));656} else {657com.cmd.StreamControl.SetupDataLen = 4;658com.cmd.StreamControl.SetupDataAddr = BsSDI;659memcpy(com.cmd.StreamControl.SetupData,660I2SConfiguration +6614 * dev->card_info->i2s[stream], 4);662}663} else if (mode & SMODE_TRANSPORT_STREAM) {664chan->nextBuffer = chan->TSRingBuffer.Head;665if (stream >= STREAM_AUDIOIN1) {666if (chan->mode & NGENE_IO_TSOUT) {667com.cmd.StreamControl.SetupDataLen =668sizeof(TS_I2SOutConfiguration);669com.cmd.StreamControl.SetupDataAddr = BsSDO;670memcpy(com.cmd.StreamControl.SetupData,671TS_I2SOutConfiguration,672sizeof(TS_I2SOutConfiguration));673} else {674com.cmd.StreamControl.SetupDataLen =675sizeof(TS_I2SConfiguration);676com.cmd.StreamControl.SetupDataAddr = BsSDI;677memcpy(com.cmd.StreamControl.SetupData,678TS_I2SConfiguration,679sizeof(TS_I2SConfiguration));680}681} else {682com.cmd.StreamControl.SetupDataLen = 8;683com.cmd.StreamControl.SetupDataAddr = BsUVI + 0x10;684memcpy(com.cmd.StreamControl.SetupData,685TSFeatureDecoderSetup +6868 * dev->card_info->tsf[stream], 8);687}688} else {689chan->nextBuffer = chan->RingBuffer.Head;690com.cmd.StreamControl.SetupDataLen =69116 + sizeof(ITUFeatureDecoderSetup);692com.cmd.StreamControl.SetupDataAddr = BsUVI;693memcpy(com.cmd.StreamControl.SetupData,694ITUDecoderSetup[chan->itumode], 16);695memcpy(com.cmd.StreamControl.SetupData + 16,696ITUFeatureDecoderSetup, sizeof(ITUFeatureDecoderSetup));697}698clear_buffers(chan);699chan->State = KSSTATE_RUN;700if (mode & SMODE_TRANSPORT_STREAM)701chan->HWState = HWSTATE_RUN;702else703chan->HWState = HWSTATE_STARTUP;704spin_unlock_irq(&chan->state_lock);705706if (ngene_command(dev, &com) < 0) {707up(&dev->stream_mutex);708return -1;709}710up(&dev->stream_mutex);711return 0;712}713714void set_transfer(struct ngene_channel *chan, int state)715{716u8 control = 0, mode = 0, flags = 0;717struct ngene *dev = chan->dev;718int ret;719720/*721printk(KERN_INFO DEVICE_NAME ": st %d\n", state);722msleep(100);723*/724725if (state) {726if (chan->running) {727printk(KERN_INFO DEVICE_NAME ": already running\n");728return;729}730} else {731if (!chan->running) {732printk(KERN_INFO DEVICE_NAME ": already stopped\n");733return;734}735}736737if (dev->card_info->switch_ctrl)738dev->card_info->switch_ctrl(chan, 1, state ^ 1);739740if (state) {741spin_lock_irq(&chan->state_lock);742743/* printk(KERN_INFO DEVICE_NAME ": lock=%08x\n",744ngreadl(0x9310)); */745dvb_ringbuffer_flush(&dev->tsout_rbuf);746control = 0x80;747if (chan->mode & (NGENE_IO_TSIN | NGENE_IO_TSOUT)) {748chan->Capture1Length = 512 * 188;749mode = SMODE_TRANSPORT_STREAM;750}751if (chan->mode & NGENE_IO_TSOUT) {752chan->pBufferExchange = tsout_exchange;753/* 0x66666666 = 50MHz *2^33 /250MHz */754chan->AudioDTOValue = 0x80000000;755chan->AudioDTOUpdated = 1;756}757if (chan->mode & NGENE_IO_TSIN)758chan->pBufferExchange = tsin_exchange;759spin_unlock_irq(&chan->state_lock);760} else761;/* printk(KERN_INFO DEVICE_NAME ": lock=%08x\n",762ngreadl(0x9310)); */763764ret = ngene_command_stream_control(dev, chan->number,765control, mode, flags);766if (!ret)767chan->running = state;768else769printk(KERN_ERR DEVICE_NAME ": set_transfer %d failed\n",770state);771if (!state) {772spin_lock_irq(&chan->state_lock);773chan->pBufferExchange = NULL;774dvb_ringbuffer_flush(&dev->tsout_rbuf);775spin_unlock_irq(&chan->state_lock);776}777}778779780/****************************************************************************/781/* nGene hardware init and release functions ********************************/782/****************************************************************************/783784static void free_ringbuffer(struct ngene *dev, struct SRingBufferDescriptor *rb)785{786struct SBufferHeader *Cur = rb->Head;787u32 j;788789if (!Cur)790return;791792for (j = 0; j < rb->NumBuffers; j++, Cur = Cur->Next) {793if (Cur->Buffer1)794pci_free_consistent(dev->pci_dev,795rb->Buffer1Length,796Cur->Buffer1,797Cur->scList1->Address);798799if (Cur->Buffer2)800pci_free_consistent(dev->pci_dev,801rb->Buffer2Length,802Cur->Buffer2,803Cur->scList2->Address);804}805806if (rb->SCListMem)807pci_free_consistent(dev->pci_dev, rb->SCListMemSize,808rb->SCListMem, rb->PASCListMem);809810pci_free_consistent(dev->pci_dev, rb->MemSize, rb->Head, rb->PAHead);811}812813static void free_idlebuffer(struct ngene *dev,814struct SRingBufferDescriptor *rb,815struct SRingBufferDescriptor *tb)816{817int j;818struct SBufferHeader *Cur = tb->Head;819820if (!rb->Head)821return;822free_ringbuffer(dev, rb);823for (j = 0; j < tb->NumBuffers; j++, Cur = Cur->Next) {824Cur->Buffer2 = NULL;825Cur->scList2 = NULL;826Cur->ngeneBuffer.Address_of_first_entry_2 = 0;827Cur->ngeneBuffer.Number_of_entries_2 = 0;828}829}830831static void free_common_buffers(struct ngene *dev)832{833u32 i;834struct ngene_channel *chan;835836for (i = STREAM_VIDEOIN1; i < MAX_STREAM; i++) {837chan = &dev->channel[i];838free_idlebuffer(dev, &chan->TSIdleBuffer, &chan->TSRingBuffer);839free_ringbuffer(dev, &chan->RingBuffer);840free_ringbuffer(dev, &chan->TSRingBuffer);841}842843if (dev->OverflowBuffer)844pci_free_consistent(dev->pci_dev,845OVERFLOW_BUFFER_SIZE,846dev->OverflowBuffer, dev->PAOverflowBuffer);847848if (dev->FWInterfaceBuffer)849pci_free_consistent(dev->pci_dev,8504096,851dev->FWInterfaceBuffer,852dev->PAFWInterfaceBuffer);853}854855/****************************************************************************/856/* Ring buffer handling *****************************************************/857/****************************************************************************/858859static int create_ring_buffer(struct pci_dev *pci_dev,860struct SRingBufferDescriptor *descr, u32 NumBuffers)861{862dma_addr_t tmp;863struct SBufferHeader *Head;864u32 i;865u32 MemSize = SIZEOF_SBufferHeader * NumBuffers;866u64 PARingBufferHead;867u64 PARingBufferCur;868u64 PARingBufferNext;869struct SBufferHeader *Cur, *Next;870871descr->Head = NULL;872descr->MemSize = 0;873descr->PAHead = 0;874descr->NumBuffers = 0;875876if (MemSize < 4096)877MemSize = 4096;878879Head = pci_alloc_consistent(pci_dev, MemSize, &tmp);880PARingBufferHead = tmp;881882if (!Head)883return -ENOMEM;884885memset(Head, 0, MemSize);886887PARingBufferCur = PARingBufferHead;888Cur = Head;889890for (i = 0; i < NumBuffers - 1; i++) {891Next = (struct SBufferHeader *)892(((u8 *) Cur) + SIZEOF_SBufferHeader);893PARingBufferNext = PARingBufferCur + SIZEOF_SBufferHeader;894Cur->Next = Next;895Cur->ngeneBuffer.Next = PARingBufferNext;896Cur = Next;897PARingBufferCur = PARingBufferNext;898}899/* Last Buffer points back to first one */900Cur->Next = Head;901Cur->ngeneBuffer.Next = PARingBufferHead;902903descr->Head = Head;904descr->MemSize = MemSize;905descr->PAHead = PARingBufferHead;906descr->NumBuffers = NumBuffers;907908return 0;909}910911static int AllocateRingBuffers(struct pci_dev *pci_dev,912dma_addr_t of,913struct SRingBufferDescriptor *pRingBuffer,914u32 Buffer1Length, u32 Buffer2Length)915{916dma_addr_t tmp;917u32 i, j;918int status = 0;919u32 SCListMemSize = pRingBuffer->NumBuffers920* ((Buffer2Length != 0) ? (NUM_SCATTER_GATHER_ENTRIES * 2) :921NUM_SCATTER_GATHER_ENTRIES)922* sizeof(struct HW_SCATTER_GATHER_ELEMENT);923924u64 PASCListMem;925struct HW_SCATTER_GATHER_ELEMENT *SCListEntry;926u64 PASCListEntry;927struct SBufferHeader *Cur;928void *SCListMem;929930if (SCListMemSize < 4096)931SCListMemSize = 4096;932933SCListMem = pci_alloc_consistent(pci_dev, SCListMemSize, &tmp);934935PASCListMem = tmp;936if (SCListMem == NULL)937return -ENOMEM;938939memset(SCListMem, 0, SCListMemSize);940941pRingBuffer->SCListMem = SCListMem;942pRingBuffer->PASCListMem = PASCListMem;943pRingBuffer->SCListMemSize = SCListMemSize;944pRingBuffer->Buffer1Length = Buffer1Length;945pRingBuffer->Buffer2Length = Buffer2Length;946947SCListEntry = SCListMem;948PASCListEntry = PASCListMem;949Cur = pRingBuffer->Head;950951for (i = 0; i < pRingBuffer->NumBuffers; i += 1, Cur = Cur->Next) {952u64 PABuffer;953954void *Buffer = pci_alloc_consistent(pci_dev, Buffer1Length,955&tmp);956PABuffer = tmp;957958if (Buffer == NULL)959return -ENOMEM;960961Cur->Buffer1 = Buffer;962963SCListEntry->Address = PABuffer;964SCListEntry->Length = Buffer1Length;965966Cur->scList1 = SCListEntry;967Cur->ngeneBuffer.Address_of_first_entry_1 = PASCListEntry;968Cur->ngeneBuffer.Number_of_entries_1 =969NUM_SCATTER_GATHER_ENTRIES;970971SCListEntry += 1;972PASCListEntry += sizeof(struct HW_SCATTER_GATHER_ELEMENT);973974#if NUM_SCATTER_GATHER_ENTRIES > 1975for (j = 0; j < NUM_SCATTER_GATHER_ENTRIES - 1; j += 1) {976SCListEntry->Address = of;977SCListEntry->Length = OVERFLOW_BUFFER_SIZE;978SCListEntry += 1;979PASCListEntry +=980sizeof(struct HW_SCATTER_GATHER_ELEMENT);981}982#endif983984if (!Buffer2Length)985continue;986987Buffer = pci_alloc_consistent(pci_dev, Buffer2Length, &tmp);988PABuffer = tmp;989990if (Buffer == NULL)991return -ENOMEM;992993Cur->Buffer2 = Buffer;994995SCListEntry->Address = PABuffer;996SCListEntry->Length = Buffer2Length;997998Cur->scList2 = SCListEntry;999Cur->ngeneBuffer.Address_of_first_entry_2 = PASCListEntry;1000Cur->ngeneBuffer.Number_of_entries_2 =1001NUM_SCATTER_GATHER_ENTRIES;10021003SCListEntry += 1;1004PASCListEntry += sizeof(struct HW_SCATTER_GATHER_ELEMENT);10051006#if NUM_SCATTER_GATHER_ENTRIES > 11007for (j = 0; j < NUM_SCATTER_GATHER_ENTRIES - 1; j++) {1008SCListEntry->Address = of;1009SCListEntry->Length = OVERFLOW_BUFFER_SIZE;1010SCListEntry += 1;1011PASCListEntry +=1012sizeof(struct HW_SCATTER_GATHER_ELEMENT);1013}1014#endif10151016}10171018return status;1019}10201021static int FillTSIdleBuffer(struct SRingBufferDescriptor *pIdleBuffer,1022struct SRingBufferDescriptor *pRingBuffer)1023{1024int status = 0;10251026/* Copy pointer to scatter gather list in TSRingbuffer1027structure for buffer 21028Load number of buffer1029*/1030u32 n = pRingBuffer->NumBuffers;10311032/* Point to first buffer entry */1033struct SBufferHeader *Cur = pRingBuffer->Head;1034int i;1035/* Loop thru all buffer and set Buffer 2 pointers to TSIdlebuffer */1036for (i = 0; i < n; i++) {1037Cur->Buffer2 = pIdleBuffer->Head->Buffer1;1038Cur->scList2 = pIdleBuffer->Head->scList1;1039Cur->ngeneBuffer.Address_of_first_entry_2 =1040pIdleBuffer->Head->ngeneBuffer.1041Address_of_first_entry_1;1042Cur->ngeneBuffer.Number_of_entries_2 =1043pIdleBuffer->Head->ngeneBuffer.Number_of_entries_1;1044Cur = Cur->Next;1045}1046return status;1047}10481049static u32 RingBufferSizes[MAX_STREAM] = {1050RING_SIZE_VIDEO,1051RING_SIZE_VIDEO,1052RING_SIZE_AUDIO,1053RING_SIZE_AUDIO,1054RING_SIZE_AUDIO,1055};10561057static u32 Buffer1Sizes[MAX_STREAM] = {1058MAX_VIDEO_BUFFER_SIZE,1059MAX_VIDEO_BUFFER_SIZE,1060MAX_AUDIO_BUFFER_SIZE,1061MAX_AUDIO_BUFFER_SIZE,1062MAX_AUDIO_BUFFER_SIZE1063};10641065static u32 Buffer2Sizes[MAX_STREAM] = {1066MAX_VBI_BUFFER_SIZE,1067MAX_VBI_BUFFER_SIZE,10680,10690,107001071};107210731074static int AllocCommonBuffers(struct ngene *dev)1075{1076int status = 0, i;10771078dev->FWInterfaceBuffer = pci_alloc_consistent(dev->pci_dev, 4096,1079&dev->PAFWInterfaceBuffer);1080if (!dev->FWInterfaceBuffer)1081return -ENOMEM;1082dev->hosttongene = dev->FWInterfaceBuffer;1083dev->ngenetohost = dev->FWInterfaceBuffer + 256;1084dev->EventBuffer = dev->FWInterfaceBuffer + 512;10851086dev->OverflowBuffer = pci_alloc_consistent(dev->pci_dev,1087OVERFLOW_BUFFER_SIZE,1088&dev->PAOverflowBuffer);1089if (!dev->OverflowBuffer)1090return -ENOMEM;1091memset(dev->OverflowBuffer, 0, OVERFLOW_BUFFER_SIZE);10921093for (i = STREAM_VIDEOIN1; i < MAX_STREAM; i++) {1094int type = dev->card_info->io_type[i];10951096dev->channel[i].State = KSSTATE_STOP;10971098if (type & (NGENE_IO_TV | NGENE_IO_HDTV | NGENE_IO_AIN)) {1099status = create_ring_buffer(dev->pci_dev,1100&dev->channel[i].RingBuffer,1101RingBufferSizes[i]);1102if (status < 0)1103break;11041105if (type & (NGENE_IO_TV | NGENE_IO_AIN)) {1106status = AllocateRingBuffers(dev->pci_dev,1107dev->1108PAOverflowBuffer,1109&dev->channel[i].1110RingBuffer,1111Buffer1Sizes[i],1112Buffer2Sizes[i]);1113if (status < 0)1114break;1115} else if (type & NGENE_IO_HDTV) {1116status = AllocateRingBuffers(dev->pci_dev,1117dev->1118PAOverflowBuffer,1119&dev->channel[i].1120RingBuffer,1121MAX_HDTV_BUFFER_SIZE,11220);1123if (status < 0)1124break;1125}1126}11271128if (type & (NGENE_IO_TSIN | NGENE_IO_TSOUT)) {11291130status = create_ring_buffer(dev->pci_dev,1131&dev->channel[i].1132TSRingBuffer, RING_SIZE_TS);1133if (status < 0)1134break;11351136status = AllocateRingBuffers(dev->pci_dev,1137dev->PAOverflowBuffer,1138&dev->channel[i].1139TSRingBuffer,1140MAX_TS_BUFFER_SIZE, 0);1141if (status)1142break;1143}11441145if (type & NGENE_IO_TSOUT) {1146status = create_ring_buffer(dev->pci_dev,1147&dev->channel[i].1148TSIdleBuffer, 1);1149if (status < 0)1150break;1151status = AllocateRingBuffers(dev->pci_dev,1152dev->PAOverflowBuffer,1153&dev->channel[i].1154TSIdleBuffer,1155MAX_TS_BUFFER_SIZE, 0);1156if (status)1157break;1158FillTSIdleBuffer(&dev->channel[i].TSIdleBuffer,1159&dev->channel[i].TSRingBuffer);1160}1161}1162return status;1163}11641165static void ngene_release_buffers(struct ngene *dev)1166{1167if (dev->iomem)1168iounmap(dev->iomem);1169free_common_buffers(dev);1170vfree(dev->tsout_buf);1171vfree(dev->tsin_buf);1172vfree(dev->ain_buf);1173vfree(dev->vin_buf);1174vfree(dev);1175}11761177static int ngene_get_buffers(struct ngene *dev)1178{1179if (AllocCommonBuffers(dev))1180return -ENOMEM;1181if (dev->card_info->io_type[4] & NGENE_IO_TSOUT) {1182dev->tsout_buf = vmalloc(TSOUT_BUF_SIZE);1183if (!dev->tsout_buf)1184return -ENOMEM;1185dvb_ringbuffer_init(&dev->tsout_rbuf,1186dev->tsout_buf, TSOUT_BUF_SIZE);1187}1188if (dev->card_info->io_type[2]&NGENE_IO_TSIN) {1189dev->tsin_buf = vmalloc(TSIN_BUF_SIZE);1190if (!dev->tsin_buf)1191return -ENOMEM;1192dvb_ringbuffer_init(&dev->tsin_rbuf,1193dev->tsin_buf, TSIN_BUF_SIZE);1194}1195if (dev->card_info->io_type[2] & NGENE_IO_AIN) {1196dev->ain_buf = vmalloc(AIN_BUF_SIZE);1197if (!dev->ain_buf)1198return -ENOMEM;1199dvb_ringbuffer_init(&dev->ain_rbuf, dev->ain_buf, AIN_BUF_SIZE);1200}1201if (dev->card_info->io_type[0] & NGENE_IO_HDTV) {1202dev->vin_buf = vmalloc(VIN_BUF_SIZE);1203if (!dev->vin_buf)1204return -ENOMEM;1205dvb_ringbuffer_init(&dev->vin_rbuf, dev->vin_buf, VIN_BUF_SIZE);1206}1207dev->iomem = ioremap(pci_resource_start(dev->pci_dev, 0),1208pci_resource_len(dev->pci_dev, 0));1209if (!dev->iomem)1210return -ENOMEM;12111212return 0;1213}12141215static void ngene_init(struct ngene *dev)1216{1217int i;12181219tasklet_init(&dev->event_tasklet, event_tasklet, (unsigned long)dev);12201221memset_io(dev->iomem + 0xc000, 0x00, 0x220);1222memset_io(dev->iomem + 0xc400, 0x00, 0x100);12231224for (i = 0; i < MAX_STREAM; i++) {1225dev->channel[i].dev = dev;1226dev->channel[i].number = i;1227}12281229dev->fw_interface_version = 0;12301231ngwritel(0, NGENE_INT_ENABLE);12321233dev->icounts = ngreadl(NGENE_INT_COUNTS);12341235dev->device_version = ngreadl(DEV_VER) & 0x0f;1236printk(KERN_INFO DEVICE_NAME ": Device version %d\n",1237dev->device_version);1238}12391240static int ngene_load_firm(struct ngene *dev)1241{1242u32 size;1243const struct firmware *fw = NULL;1244u8 *ngene_fw;1245char *fw_name;1246int err, version;12471248version = dev->card_info->fw_version;12491250switch (version) {1251default:1252case 15:1253version = 15;1254size = 23466;1255fw_name = "ngene_15.fw";1256dev->cmd_timeout_workaround = true;1257break;1258case 16:1259size = 23498;1260fw_name = "ngene_16.fw";1261dev->cmd_timeout_workaround = true;1262break;1263case 17:1264size = 24446;1265fw_name = "ngene_17.fw";1266dev->cmd_timeout_workaround = true;1267break;1268case 18:1269size = 0;1270fw_name = "ngene_18.fw";1271break;1272}12731274if (request_firmware(&fw, fw_name, &dev->pci_dev->dev) < 0) {1275printk(KERN_ERR DEVICE_NAME1276": Could not load firmware file %s.\n", fw_name);1277printk(KERN_INFO DEVICE_NAME1278": Copy %s to your hotplug directory!\n", fw_name);1279return -1;1280}1281if (size == 0)1282size = fw->size;1283if (size != fw->size) {1284printk(KERN_ERR DEVICE_NAME1285": Firmware %s has invalid size!", fw_name);1286err = -1;1287} else {1288printk(KERN_INFO DEVICE_NAME1289": Loading firmware file %s.\n", fw_name);1290ngene_fw = (u8 *) fw->data;1291err = ngene_command_load_firmware(dev, ngene_fw, size);1292}12931294release_firmware(fw);12951296return err;1297}12981299static void ngene_stop(struct ngene *dev)1300{1301down(&dev->cmd_mutex);1302i2c_del_adapter(&(dev->channel[0].i2c_adapter));1303i2c_del_adapter(&(dev->channel[1].i2c_adapter));1304ngwritel(0, NGENE_INT_ENABLE);1305ngwritel(0, NGENE_COMMAND);1306ngwritel(0, NGENE_COMMAND_HI);1307ngwritel(0, NGENE_STATUS);1308ngwritel(0, NGENE_STATUS_HI);1309ngwritel(0, NGENE_EVENT);1310ngwritel(0, NGENE_EVENT_HI);1311free_irq(dev->pci_dev->irq, dev);1312#ifdef CONFIG_PCI_MSI1313if (dev->msi_enabled)1314pci_disable_msi(dev->pci_dev);1315#endif1316}13171318static int ngene_buffer_config(struct ngene *dev)1319{1320int stat;13211322if (dev->card_info->fw_version >= 17) {1323u8 tsin12_config[6] = { 0x60, 0x60, 0x00, 0x00, 0x00, 0x00 };1324u8 tsin1234_config[6] = { 0x30, 0x30, 0x00, 0x30, 0x30, 0x00 };1325u8 tsio1235_config[6] = { 0x30, 0x30, 0x00, 0x28, 0x00, 0x38 };1326u8 *bconf = tsin12_config;13271328if (dev->card_info->io_type[2]&NGENE_IO_TSIN &&1329dev->card_info->io_type[3]&NGENE_IO_TSIN) {1330bconf = tsin1234_config;1331if (dev->card_info->io_type[4]&NGENE_IO_TSOUT &&1332dev->ci.en)1333bconf = tsio1235_config;1334}1335stat = ngene_command_config_free_buf(dev, bconf);1336} else {1337int bconf = BUFFER_CONFIG_4422;13381339if (dev->card_info->io_type[3] == NGENE_IO_TSIN)1340bconf = BUFFER_CONFIG_3333;1341stat = ngene_command_config_buf(dev, bconf);1342}1343return stat;1344}134513461347static int ngene_start(struct ngene *dev)1348{1349int stat;1350int i;13511352pci_set_master(dev->pci_dev);1353ngene_init(dev);13541355stat = request_irq(dev->pci_dev->irq, irq_handler,1356IRQF_SHARED, "nGene",1357(void *)dev);1358if (stat < 0)1359return stat;13601361init_waitqueue_head(&dev->cmd_wq);1362init_waitqueue_head(&dev->tx_wq);1363init_waitqueue_head(&dev->rx_wq);1364sema_init(&dev->cmd_mutex, 1);1365sema_init(&dev->stream_mutex, 1);1366sema_init(&dev->pll_mutex, 1);1367sema_init(&dev->i2c_switch_mutex, 1);1368spin_lock_init(&dev->cmd_lock);1369for (i = 0; i < MAX_STREAM; i++)1370spin_lock_init(&dev->channel[i].state_lock);1371ngwritel(1, TIMESTAMPS);13721373ngwritel(1, NGENE_INT_ENABLE);13741375stat = ngene_load_firm(dev);1376if (stat < 0)1377goto fail;13781379#ifdef CONFIG_PCI_MSI1380/* enable MSI if kernel and card support it */1381if (pci_msi_enabled() && dev->card_info->msi_supported) {1382unsigned long flags;13831384ngwritel(0, NGENE_INT_ENABLE);1385free_irq(dev->pci_dev->irq, dev);1386stat = pci_enable_msi(dev->pci_dev);1387if (stat) {1388printk(KERN_INFO DEVICE_NAME1389": MSI not available\n");1390flags = IRQF_SHARED;1391} else {1392flags = 0;1393dev->msi_enabled = true;1394}1395stat = request_irq(dev->pci_dev->irq, irq_handler,1396flags, "nGene", dev);1397if (stat < 0)1398goto fail2;1399ngwritel(1, NGENE_INT_ENABLE);1400}1401#endif14021403stat = ngene_i2c_init(dev, 0);1404if (stat < 0)1405goto fail;14061407stat = ngene_i2c_init(dev, 1);1408if (stat < 0)1409goto fail;14101411if (!stat)1412return stat;14131414/* otherwise error: fall through */1415fail:1416ngwritel(0, NGENE_INT_ENABLE);1417free_irq(dev->pci_dev->irq, dev);1418#ifdef CONFIG_PCI_MSI1419fail2:1420if (dev->msi_enabled)1421pci_disable_msi(dev->pci_dev);1422#endif1423return stat;1424}14251426/****************************************************************************/1427/****************************************************************************/1428/****************************************************************************/14291430static void release_channel(struct ngene_channel *chan)1431{1432struct dvb_demux *dvbdemux = &chan->demux;1433struct ngene *dev = chan->dev;14341435if (chan->running)1436set_transfer(chan, 0);14371438tasklet_kill(&chan->demux_tasklet);14391440if (chan->ci_dev) {1441dvb_unregister_device(chan->ci_dev);1442chan->ci_dev = NULL;1443}14441445if (chan->fe) {1446dvb_unregister_frontend(chan->fe);1447dvb_frontend_detach(chan->fe);1448chan->fe = NULL;1449}14501451if (chan->has_demux) {1452dvb_net_release(&chan->dvbnet);1453dvbdemux->dmx.close(&dvbdemux->dmx);1454dvbdemux->dmx.remove_frontend(&dvbdemux->dmx,1455&chan->hw_frontend);1456dvbdemux->dmx.remove_frontend(&dvbdemux->dmx,1457&chan->mem_frontend);1458dvb_dmxdev_release(&chan->dmxdev);1459dvb_dmx_release(&chan->demux);1460chan->has_demux = false;1461}14621463if (chan->has_adapter) {1464dvb_unregister_adapter(&dev->adapter[chan->number]);1465chan->has_adapter = false;1466}1467}14681469static int init_channel(struct ngene_channel *chan)1470{1471int ret = 0, nr = chan->number;1472struct dvb_adapter *adapter = NULL;1473struct dvb_demux *dvbdemux = &chan->demux;1474struct ngene *dev = chan->dev;1475struct ngene_info *ni = dev->card_info;1476int io = ni->io_type[nr];14771478tasklet_init(&chan->demux_tasklet, demux_tasklet, (unsigned long)chan);1479chan->users = 0;1480chan->type = io;1481chan->mode = chan->type; /* for now only one mode */14821483if (io & NGENE_IO_TSIN) {1484chan->fe = NULL;1485if (ni->demod_attach[nr]) {1486ret = ni->demod_attach[nr](chan);1487if (ret < 0)1488goto err;1489}1490if (chan->fe && ni->tuner_attach[nr]) {1491ret = ni->tuner_attach[nr](chan);1492if (ret < 0)1493goto err;1494}1495}14961497if (!dev->ci.en && (io & NGENE_IO_TSOUT))1498return 0;14991500if (io & (NGENE_IO_TSIN | NGENE_IO_TSOUT)) {1501if (nr >= STREAM_AUDIOIN1)1502chan->DataFormatFlags = DF_SWAP32;15031504if (nr == 0 || !one_adapter || dev->first_adapter == NULL) {1505adapter = &dev->adapter[nr];1506ret = dvb_register_adapter(adapter, "nGene",1507THIS_MODULE,1508&chan->dev->pci_dev->dev,1509adapter_nr);1510if (ret < 0)1511goto err;1512if (dev->first_adapter == NULL)1513dev->first_adapter = adapter;1514chan->has_adapter = true;1515} else1516adapter = dev->first_adapter;1517}15181519if (dev->ci.en && (io & NGENE_IO_TSOUT)) {1520dvb_ca_en50221_init(adapter, dev->ci.en, 0, 1);1521set_transfer(chan, 1);1522chan->dev->channel[2].DataFormatFlags = DF_SWAP32;1523set_transfer(&chan->dev->channel[2], 1);1524dvb_register_device(adapter, &chan->ci_dev,1525&ngene_dvbdev_ci, (void *) chan,1526DVB_DEVICE_SEC);1527if (!chan->ci_dev)1528goto err;1529}15301531if (chan->fe) {1532if (dvb_register_frontend(adapter, chan->fe) < 0)1533goto err;1534chan->has_demux = true;1535}15361537if (chan->has_demux) {1538ret = my_dvb_dmx_ts_card_init(dvbdemux, "SW demux",1539ngene_start_feed,1540ngene_stop_feed, chan);1541ret = my_dvb_dmxdev_ts_card_init(&chan->dmxdev, &chan->demux,1542&chan->hw_frontend,1543&chan->mem_frontend, adapter);1544ret = dvb_net_init(adapter, &chan->dvbnet, &chan->demux.dmx);1545}15461547return ret;15481549err:1550if (chan->fe) {1551dvb_frontend_detach(chan->fe);1552chan->fe = NULL;1553}1554release_channel(chan);1555return 0;1556}15571558static int init_channels(struct ngene *dev)1559{1560int i, j;15611562for (i = 0; i < MAX_STREAM; i++) {1563dev->channel[i].number = i;1564if (init_channel(&dev->channel[i]) < 0) {1565for (j = i - 1; j >= 0; j--)1566release_channel(&dev->channel[j]);1567return -1;1568}1569}1570return 0;1571}15721573static void cxd_attach(struct ngene *dev)1574{1575struct ngene_ci *ci = &dev->ci;15761577ci->en = cxd2099_attach(0x40, dev, &dev->channel[0].i2c_adapter);1578ci->dev = dev;1579return;1580}15811582static void cxd_detach(struct ngene *dev)1583{1584struct ngene_ci *ci = &dev->ci;15851586dvb_ca_en50221_release(ci->en);1587kfree(ci->en);1588ci->en = 0;1589}15901591/***********************************/1592/* workaround for shutdown failure */1593/***********************************/15941595static void ngene_unlink(struct ngene *dev)1596{1597struct ngene_command com;15981599com.cmd.hdr.Opcode = CMD_MEM_WRITE;1600com.cmd.hdr.Length = 3;1601com.cmd.MemoryWrite.address = 0x910c;1602com.cmd.MemoryWrite.data = 0xff;1603com.in_len = 3;1604com.out_len = 1;16051606down(&dev->cmd_mutex);1607ngwritel(0, NGENE_INT_ENABLE);1608ngene_command_mutex(dev, &com);1609up(&dev->cmd_mutex);1610}16111612void ngene_shutdown(struct pci_dev *pdev)1613{1614struct ngene *dev = (struct ngene *)pci_get_drvdata(pdev);16151616if (!dev || !shutdown_workaround)1617return;16181619printk(KERN_INFO DEVICE_NAME ": shutdown workaround...\n");1620ngene_unlink(dev);1621pci_disable_device(pdev);1622}16231624/****************************************************************************/1625/* device probe/remove calls ************************************************/1626/****************************************************************************/16271628void __devexit ngene_remove(struct pci_dev *pdev)1629{1630struct ngene *dev = pci_get_drvdata(pdev);1631int i;16321633tasklet_kill(&dev->event_tasklet);1634for (i = MAX_STREAM - 1; i >= 0; i--)1635release_channel(&dev->channel[i]);1636if (dev->ci.en)1637cxd_detach(dev);1638ngene_stop(dev);1639ngene_release_buffers(dev);1640pci_set_drvdata(pdev, NULL);1641pci_disable_device(pdev);1642}16431644int __devinit ngene_probe(struct pci_dev *pci_dev,1645const struct pci_device_id *id)1646{1647struct ngene *dev;1648int stat = 0;16491650if (pci_enable_device(pci_dev) < 0)1651return -ENODEV;16521653dev = vzalloc(sizeof(struct ngene));1654if (dev == NULL) {1655stat = -ENOMEM;1656goto fail0;1657}16581659dev->pci_dev = pci_dev;1660dev->card_info = (struct ngene_info *)id->driver_data;1661printk(KERN_INFO DEVICE_NAME ": Found %s\n", dev->card_info->name);16621663pci_set_drvdata(pci_dev, dev);16641665/* Alloc buffers and start nGene */1666stat = ngene_get_buffers(dev);1667if (stat < 0)1668goto fail1;1669stat = ngene_start(dev);1670if (stat < 0)1671goto fail1;16721673cxd_attach(dev);16741675stat = ngene_buffer_config(dev);1676if (stat < 0)1677goto fail1;167816791680dev->i2c_current_bus = -1;16811682/* Register DVB adapters and devices for both channels */1683if (init_channels(dev) < 0)1684goto fail2;16851686return 0;16871688fail2:1689ngene_stop(dev);1690fail1:1691ngene_release_buffers(dev);1692fail0:1693pci_disable_device(pci_dev);1694pci_set_drvdata(pci_dev, NULL);1695return stat;1696}169716981699