Subversion Repositories shark

Rev

Rev 181 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
    bttv-cards.c

    this file has configuration informations - card-specific stuff
    like the big tvcards array for the most part

    Copyright (C) 1996,97,98 Ralph  Metzler (rjkm@thp.uni-koeln.de)
                           & Marcus Metzler (mocm@thp.uni-koeln.de)
    (c) 1999-2001 Gerd Knorr <kraxel@goldbach.in-berlin.de>

    This program is free software; you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation; either version 2 of the License, or
    (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with this program; if not, write to the Free Software
    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
   
*/


/* SHARK version by Giacomo Guidi <giacomo@gandalf.sssup.it> */

#define __NO_VERSION__ 1

#include <unistd.h>
#include <linux/pci.h>

#include "drivers/bttv.h"
#include "drivers/tuner.h"

/* fwd decl */
static void boot_msp34xx(struct bttv *btv, int pin);
static void hauppauge_eeprom(struct bttv *btv);
static void osprey_eeprom(struct bttv *btv);
static void avermedia_eeprom(struct bttv *btv);
static void init_PXC200(struct bttv *btv);
#if 0
static void init_tea5757(struct bttv *btv);
#endif

static void winview_audio(struct bttv *btv, struct video_audio *v, int set);
static void lt9415_audio(struct bttv *btv, struct video_audio *v, int set);
static void avermedia_tvphone_audio(struct bttv *btv, struct video_audio *v,
                                    int set);
static void terratv_audio(struct bttv *btv, struct video_audio *v, int set);
static void gvbctv3pci_audio(struct bttv *btv, struct video_audio *v, int set);
static void winfast2000_audio(struct bttv *btv, struct video_audio *v, int set);
static void pvbt878p9b_audio(struct bttv *btv, struct video_audio *v, int set);
static void fv2000s_audio(struct bttv *btv, struct video_audio *v, int set);
static void windvr_audio(struct bttv *btv, struct video_audio *v, int set);
static void rv605_muxsel(struct bttv *btv, unsigned int input);

/* config variables */
static int triton1=0;
static int vsfx=0;
int no_overlay=-1;
static unsigned int card[4]  = { -1, -1, -1, -1 };
static unsigned int pll[4]   = { -1, -1, -1, -1 };
static unsigned int tuner[4] = { -1, -1, -1, -1 };
static unsigned int gpiomask = -1;
static unsigned int audioall = -1;
static unsigned int audiomux[5] = { -1, -1, -1, -1, -1 };

/* ----------------------------------------------------------------------- */
/* list of card IDs for bt878+ cards                                       */

static struct CARD {
        unsigned id;
        int cardnr;
        char *name;
} cards[] = {
        { 0x13eb0070, BTTV_HAUPPAUGE878,  "Hauppauge WinTV" },
        { 0x39000070, BTTV_HAUPPAUGE878,  "Hauppauge WinTV-D" },
        { 0x45000070, BTTV_HAUPPAUGE878,  "Hauppauge WinTV/PVR" },
        { 0xff000070, BTTV_OSPREY1x0,     "Osprey-100" },
        { 0xff010070, BTTV_OSPREY2x0_SVID,"Osprey-200" },
        { 0xff020070, BTTV_OSPREY5x0,     "Osprey-500" },
        { 0xff030070, BTTV_OSPREY2000,    "Osprey-2000" },

        { 0x00011002, BTTV_ATI_TVWONDER,  "ATI TV Wonder" },
        { 0x00031002, BTTV_ATI_TVWONDERVE,"ATI TV Wonder/VE" },

        { 0x6606107d, BTTV_WINFAST2000,   "Leadtek WinFast TV 2000" },
        { 0x263610b4, BTTV_STB2,          "STB TV PCI FM, P/N 6000704" },
        { 0x402010fc, BTTV_GVBCTV3PCI,    "I-O Data Co. GV-BCTV3/PCI" },
        { 0x405010fc, BTTV_GVBCTV4PCI,    "I-O Data Co. GV-BCTV4/PCI" },

        { 0x1200bd11, BTTV_PINNACLE,      "Pinnacle PCTV" },
        { 0x001211bd, BTTV_PINNACLE,      "Pinnacle PCTV" },
        { 0x001c11bd, BTTV_PINNACLE,      "Pinnacle PCTV Sat" },

        { 0x3000121a, BTTV_VOODOOTV_FM,   "3Dfx VoodooTV FM/ VoodooTV 200" },
       
        { 0x3000144f, BTTV_MAGICTVIEW063, "(Askey Magic/others) TView99 CPH06x" },
        { 0x3002144f, BTTV_MAGICTVIEW061, "(Askey Magic/others) TView99 CPH05x" },
        { 0x3005144f, BTTV_MAGICTVIEW061, "(Askey Magic/others) TView99 CPH061/06L (T1/LC)" },
       
        { 0x00011461, BTTV_AVPHONE98,     "AVerMedia TVPhone98" },
        { 0x00021461, BTTV_AVERMEDIA98,   "AVermedia TVCapture 98" },
        { 0x00031461, BTTV_AVPHONE98,     "AVerMedia TVPhone98" },
        { 0x00041461, BTTV_AVERMEDIA98,   "AVerMedia TVCapture 98" },

        { 0x300014ff, BTTV_MAGICTVIEW061, "TView 99 (CPH061)" },
        { 0x300214ff, BTTV_PHOEBE_TVMAS,  "Phoebe TV Master (CPH060)" },

        { 0x1117153b, BTTV_TERRATVALUE,   "Terratec TValue" },
        { 0x1118153b, BTTV_TERRATVALUE,   "Terratec TValue" },
        { 0x1119153b, BTTV_TERRATVALUE,   "Terratec TValue" },
        { 0x111a153b, BTTV_TERRATVALUE,   "Terratec TValue" },
        { 0x1123153b, BTTV_TERRATVRADIO,  "Terratec TV Radio+" },
        { 0x1127153b, BTTV_TERRATV,       "Terratec TV+"    },
        // clashes with flyvideo
        //{ 0x18521852, BTTV_TERRATV,     "Terratec TV+"    },
        { 0x1134153b, BTTV_TERRATVALUE,   "Terratec TValue" },
        { 0x1135153b, BTTV_TERRATVALUER,  "Terratec TValue Radio" },
        { 0x5018153b, BTTV_TERRATVALUE,   "Terratec TValue" },

        { 0x400a15b0, BTTV_ZOLTRIX_GENIE, "Zoltrix Genie TV" },
        { 0x400d15b0, BTTV_ZOLTRIX_GENIE, "Zoltrix Genie TV / Radio" },
        { 0x401015b0, BTTV_ZOLTRIX_GENIE, "Zoltrix Genie TV / Radio" },
        { 0x401615b0, BTTV_ZOLTRIX_GENIE, "Zoltrix Genie TV / Radio" },

        { 0x010115cb, BTTV_GMV1,          "AG GMV1" },
        { 0x010114c7, BTTV_MODTEC_205,    "Modular Technology MM205 PCTV" },
        { 0x18501851, BTTV_CHRONOS_VS2,   "Flyvideo 98 (LR50)/ Chronos Video Shuttle II" },
        { 0x18511851, BTTV_FLYVIDEO98EZ,  "Flyvideo 98EZ (LR51)/ CyberMail AV" },
        { 0x18521852, BTTV_TYPHOON_TVIEW, "Flyvideo 98FM (LR50)/ Typhoon TView TV/FM Tuner" },
        { 0x10b42636, BTTV_HAUPPAUGE878,  "STB ???" },
        { 0x217d6606, BTTV_WINFAST2000,   "Leadtek WinFast TV 2000" },
        { 0x03116000, BTTV_SENSORAY311,   "Sensoray 311" },
        { 0x00790e11, BTTV_WINDVR,        "Canopus WinDVR PCI" },
        { 0xa0fca1a0, BTTV_ZOLTRIX,       "Face to Face Tvmax" },
        { 0x0350109e, BTTV_PXC200,        "Imagenation PXC200" },

        { 0, -1, NULL }
};

/* ----------------------------------------------------------------------- */
/* array with description for bt848 / bt878 tv/grabber cards               */

struct tvcard bttv_tvcards[] = {
{
/* ---- card 0x00 ---------------------------------- */
        name:           " *** UNKNOWN/GENERIC *** ",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        muxsel:         { 2, 3, 1, 0},
        tuner_type:     -1,
},{
        name:           "MIRO PCTV",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       15,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 2, 0, 0, 0, 10},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "Hauppauge (bt848)",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       7,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 1, 2, 3, 4},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "STB",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       7,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 4, 0, 2, 3, 1},
        no_msp34xx:     1,
        needs_tvaudio:  1,
        tuner_type:     -1,
},{

/* ---- card 0x04 ---------------------------------- */
        name:           "Intel",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           -1,
        gpiomask:       7,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 1, 2, 3, 4},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "Diamond DTV2000",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       3,
        muxsel:         { 2, 3, 1, 0},
        audiomux:       { 0, 1, 0, 1, 3},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "AVerMedia TVPhone",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           3,
        muxsel:         { 2, 3, 1, 1},
        gpiomask:       0x0f,
        audiomux:       { 0x0c, 0x04, 0x08, 0x04, 0},
        /*                0x04 for some cards ?? */
        needs_tvaudio:  1,
        tuner_type:     -1,
        audio_hook:     avermedia_tvphone_audio,
},{
        name:           "MATRIX-Vision MV-Delta",
        video_inputs:   5,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           3,
        gpiomask:       0,
        muxsel:         { 2, 3, 1, 0, 0},
        audiomux:       {0 },
        needs_tvaudio:  1,
        tuner_type:     -1,
},{

/* ---- card 0x08 ---------------------------------- */
        name:           "FlyVideo II (Bt848) LR26",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xc00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 0xc00, 0x800, 0x400, 0xc00, 0},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "IXMicro TurboTV",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       3,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 1, 1, 2, 3, 0},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "Hauppauge (bt878)",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x0f, /* old: 7 */
        muxsel:         { 2, 0, 1, 1},
        audiomux:       { 0, 1, 2, 3, 4},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "MIRO PCTV pro",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x3014f,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0x20001,0x10001, 0, 0,10},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{

/* ---- card 0x0c ---------------------------------- */
        name:           "ADS Technologies Channel Surfer TV",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       15,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 13, 14, 11, 7, 0, 0},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "AVerMedia TVCapture 98",
        video_inputs:   3,
        audio_inputs:   4,
        tuner:          0,
        svhs:           2,
        gpiomask:       15,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 13, 14, 11, 7, 0, 0},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "Aimslab Video Highway Xtreme (VHX)",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       7,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 2, 1, 3, 4}, /* old: { 0, 1, 2, 3, 4} */
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "Zoltrix TV-Max",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       15,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       {0 , 0, 1 , 0, 10},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{

/* ---- card 0x10 ---------------------------------- */
        name:           "Pixelview PlayTV (bt878)",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x01fe00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0x01c000, 0, 0x018000, 0x014000, 0x002000, 0 },
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "Leadtek WinView 601",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x8300f8,
        muxsel:         { 2, 3, 1, 1,0},
        audiomux:       { 0x4fa007,0xcfa007,0xcfa007,0xcfa007,0xcfa007,0xcfa007},
        needs_tvaudio:  1,
        tuner_type:     -1,
        audio_hook:     winview_audio,
        has_radio:      1,
},{
        name:           "AVEC Intercapture",
        video_inputs:   3,
        audio_inputs:   2,
        tuner:          0,
        svhs:           2,
        gpiomask:       0,
        muxsel:         {2, 3, 1, 1},
        audiomux:       {1, 0, 0, 0, 0},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "LifeView FlyKit w/o Tuner",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           -1,
        gpiomask:       0x8dff00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0 },
        no_msp34xx:     1,
        tuner_type:     -1,
},{

/* ---- card 0x14 ---------------------------------- */
        name:           "CEI Raffles Card",
        video_inputs:   3,
        audio_inputs:   3,
        tuner:          0,
        svhs:           2,
        muxsel:         {2, 3, 1, 1},
        tuner_type:     -1,
},{
        name:           "Lucky Star Image World ConferenceTV",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x00fffe07,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 131072, 1, 1638400, 3, 4},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     TUNER_PHILIPS_PAL_I,
},{
        name:           "Phoebe Tv Master + FM (CPH050)",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xc00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       {0, 1, 0x800, 0x400, 0xc00, 0},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "Modular Technology MM205 PCTV, bt878",
        video_inputs:   2,
        audio_inputs:   1,
        tuner:          0,
        svhs:           -1,
        gpiomask:       7,
        muxsel:         { 2, 3 },
        audiomux:       { 0, 0, 0, 0, 0 },
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     TUNER_ALPS_TSBB5_PAL_I,
},{

/* ---- card 0x18 ---------------------------------- */
        name:           "[many vendors] CPH05X/06X (bt878)",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xe00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       {0x400, 0x400, 0x400, 0x400, 0xc00},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "Terratec/Vobis TV-Boostar",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       16777215,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 131072, 1, 1638400, 3,4},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "Newer Hauppauge WinCam (bt878)",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           3,
        gpiomask:       7,
        muxsel:         { 2, 0, 1, 1},
        audiomux:       { 0, 1, 2, 3, 4},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "MAXI TV Video PCI2",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xffff,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 1, 2, 3, 0xc00},
        needs_tvaudio:  1,
        tuner_type:     TUNER_PHILIPS_SECAM,
},{

/* ---- card 0x1c ---------------------------------- */
        name:           "Terratec TerraTV+",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x70000,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0x20000, 0x30000, 0x10000, 0x00000, 0x40000},
        needs_tvaudio:  1,
        tuner_type:     TUNER_PHILIPS_PAL,
        audio_hook:     terratv_audio,
},{
        /* Jannik Fritsch <jannik@techfak.uni-bielefeld.de> */
        name:           "Imagenation PXC200",
        video_inputs:   5,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           1, /* was: 4 */
        gpiomask:       0,
        muxsel:         { 2, 3, 1, 0, 0},
        audiomux:       { 0 },
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "FlyVideo 98",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x1800,  //0x8dfe00
        muxsel:         {2, 3, 1, 1},
        audiomux:       { 0, 0x0800, 0x1000, 0x1000, 0x1800, 0 },
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "iProTV",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       1,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 1, 0, 0, 0, 0 },
        tuner_type:     -1,
},{

/* ---- card 0x20 ---------------------------------- */
        name:           "Intel Create and Share PCI",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           2,
        gpiomask:       7,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 4, 4, 4, 4, 4},
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "Terratec TerraTValue",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xffff00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0x500, 0, 0x300, 0x900, 0x900},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     TUNER_PHILIPS_PAL,
},{
        name:           "Leadtek WinFast 2000",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xc33000,
        muxsel:         { 2, 3, 1, 1,0},
        audiomux:       { 0x422000,0x001000,0x621100,0x620000,0x800000,0x620000},
        needs_tvaudio:  0,
        pll:            PLL_28,
        tuner_type:     -1,
        audio_hook:     winfast2000_audio,
},{
        name:           "Flyvideo 98 (LR50Q) / Chronos Video Shuttle II",
        video_inputs:   3,
        audio_inputs:   3,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x1800,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 0x800, 0x1000, 0x1000, 0x1800},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{

/* ---- card 0x24 ---------------------------------- */
        name:           "Flyvideo 98FM (LR50Q) / Typhoon TView TV/FM Tuner",
        video_inputs:   3,
        audio_inputs:   3,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x1800,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 0x800, 0x1000, 0x1000, 0x1800, 0 },
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
        has_radio:      1,
},{
        name:           "PixelView PlayTV pro",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xff,
        muxsel:         { 2, 3, 1, 1 },
        audiomux:       { 0x21, 0x20, 0x24, 0x2c, 0x29, 0x29 },
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "TView99 CPH06X",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x551e00,
        muxsel:         { 2, 3, 1, 0},
        audiomux:       { 0x551400, 0x551200, 0, 0, 0x551c00, 0x551200 },
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "Pinnacle PCTV Studio/Rave",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x03000F,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 2, 0, 0, 0, 1},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{

/* ---- card 0x28 ---------------------------------- */
        name:           "STB2",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       7,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 4, 0, 2, 3, 1},
        no_msp34xx:     1,
        needs_tvaudio:  1,
        tuner_type:     -1,
},{
        name:           "AVerMedia TVPhone 98",
        video_inputs:   3,
        audio_inputs:   4,
        tuner:          0,
        svhs:           2,
        gpiomask:       12,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 13, 4, 11, 7, 0, 0},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
        has_radio:      1,
},{
        name:           "ProVideo PV951", /* pic16c54 */
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 0, 0, 0, 0},
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     1,
},{
        name:           "Little OnAir TV",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xe00b,
        muxsel:         {2, 3, 1, 1},
        audiomux:       {0xff9ff6, 0xff9ff6, 0xff1ff7, 0, 0xff3ffc},
        no_msp34xx:     1,
        tuner_type:     -1,
},{

/* ---- card 0x2c ---------------------------------- */
        name:           "Sigma TVII-FM",
        video_inputs:   2,
        audio_inputs:   1,
        tuner:          0,
        svhs:           -1,
        gpiomask:       3,
        muxsel:         {2, 3, 1, 1},
        audiomux:       {1, 1, 0, 2, 3},
        no_msp34xx:     1,
        pll:            PLL_NONE,
        tuner_type:     -1,
},{
        name:           "MATRIX-Vision MV-Delta 2",
        video_inputs:   5,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           3,
        gpiomask:       0,
        muxsel:         { 2, 3, 1, 0, 0},
        audiomux:       {0 },
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "Zoltrix Genie TV/FM",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xbcf03f,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0xbc803f, 0xbc903f, 0xbcb03f, 0, 0xbcb03f},
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     21,
},{
        name:           "Terratec TV/Radio+",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x1f0000,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0xe2ffff, 0xebffff, 0, 0, 0xe0ffff, 0xe2ffff },
        needs_tvaudio:  1,
        no_msp34xx:     1,
        pll:            PLL_35,
        tuner_type:     1,
        has_radio:      1,
},{

/* ---- card 0x30 ---------------------------------- */
        name:           "Dynalink Magic TView ",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       15,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       {2,0,0,0,1},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        name:           "IODATA GV-BCTV3/PCI",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x010f00,
        muxsel:         {2, 3, 0, 0},
        audiomux:       {0x10000, 0, 0x10000, 0, 0, 0},
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     TUNER_ALPS_TSHC6_NTSC,
        audio_hook:     gvbctv3pci_audio,
},{
        name:           "Prolink PV-BT878P+4E / PixelView PlayTV PAK / Lenco MXTV-9578 CP",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           3,
        gpiomask:       0xAA0000,
        muxsel:         { 2,3,1,1 },
        audiomux:       { 0x20000, 0, 0x80000, 0x80000, 0xa8000, 0x46000  },
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     TUNER_PHILIPS_PAL_I,
},{
        name:           "Eagle Wireless Capricorn2 (bt878A)",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       7,
        muxsel:         { 2, 0, 1, 1},
        audiomux:       { 0, 1, 2, 3, 4},
        pll:            PLL_28,
        tuner_type:     -1 /* TUNER_ALPS_TMDH2_NTSC */,
},{

/* ---- card 0x34 ---------------------------------- */
        /* David Härdeman <david@2gen.com> */
        name:           "Pinnacle PCTV Studio Pro",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x03000F,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 1, 0x10001, 0, 0, 10},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        /* Claas Langbehn <claas@bigfoot.com>,
           Sven Grothklags <sven@upb.de> */

        name:           "Typhoon TView RDS + FM Stereo / KNC1 TV Station RDS",
        video_inputs:   3,
        audio_inputs:   3,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x1c,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 0, 0x10, 8, 4 },
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     TUNER_PHILIPS_PAL_I,
        has_radio:      1,
},{
        /* Tim Röstermundt <rosterm@uni-muenster.de>
           in de.comp.os.unix.linux.hardware:
                options bttv card=0 pll=1 radio=1 gpiomask=0x18e0
                audiomux=0x44c71f,0x44d71f,0,0x44d71f,0x44dfff
                options tuner type=5 */

        name:           "Lifetec LT 9415 TV (LR90 Rev.F)",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x18e0,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0x0000,0x0800,0x1000,0x1000,0x18e0 },
                       /* 0x0000: Tuner normal stereo
                          0x0080: Tuner A2 SAP (second audio program = Zweikanalton)
                          0x0880: Tuner A2 stereo */

        pll:            PLL_28,
        tuner_type:     TUNER_PHILIPS_PAL,
        audio_hook:     lt9415_audio,
        has_radio:      1,
},{
        /* Miguel Angel Alvarez <maacruz@navegalia.com>
           old Easy TV BT848 version (model CPH031) */

        name:           "BESTBUY Easy TV (CPH031)",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xF,
        muxsel:         { 2, 3, 1, 0},
        audiomux:       { 2, 0, 0, 0, 10},
        needs_tvaudio:  0,
        pll:            PLL_28,
        tuner_type:     TUNER_TEMIC_PAL,
},{

/* ---- card 0x38 ---------------------------------- */
        /* Gordon Heydon <gjheydon@bigfoot.com ('98) */
        name:           "FlyVideo '98/FM",
        video_inputs:   3,
        audio_inputs:   3,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x1800,
        muxsel:         { 2, 3, 0, 1},
        audiomux:       { 0, 0x800, 0x1000, 0x1000, 0x1800, 0 },
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     5,
},{
        /* This is the ultimate cheapo capture card
         * just a BT848A on a small PCB!
         * Steve Hosgood <steve@equiinet.com> */

        name:           "GrandTec 'Grand Video Capture' (Bt848)",
        video_inputs:   2,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           1,
        gpiomask:       0,
        muxsel:         { 3, 1 },
        audiomux:       { 0 },
        needs_tvaudio:  0,
        no_msp34xx:     1,
        pll:            PLL_35,
        tuner_type:     -1,
},{
        /* Daniel Herrington <daniel.herrington@home.com> */
        name:           "Phoebe TV Master Only (No FM) CPH060",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xe00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0x400, 0x400, 0x400, 0x400, 0x800, 0x400 },
        needs_tvaudio:  1,
        pll:            PLL_NONE,
        tuner_type:     TUNER_TEMIC_4036FY5_NTSC,
},{
        /* Matti Mottus <mottus@physic.ut.ee> */
        name:           "TV Capturer (CPH03X)",
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x03000F,
        muxsel:         { 2, 3, 1, 0},
        audiomux:       { 2,0,0,0,1 },
        pll:            PLL_28,
        tuner_type:     0,
},{

/* ---- card 0x3c ---------------------------------- */
        /* Philip Blundell <philb@gnu.org> */
        name:           "Modular Technology MM100PCTV",
        video_inputs:   2,
        audio_inputs:   2,
        tuner:          0,
        svhs:           -1,
        gpiomask:       11,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 2, 0, 0, 1, 8},
        pll:            PLL_35,
        tuner_type:     TUNER_TEMIC_PAL,

},{
        /* Adrian Cox <adrian@humboldt.co.uk */
        name:           "AG Electronics GMV1",
        video_inputs:   2,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           1,
        gpiomask:       0xF,
        muxsel:         { 2, 2},
        audiomux:       { },
        no_msp34xx:     1,
        needs_tvaudio:  0,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        /* Miguel Angel Alvarez <maacruz@navegalia.com>
           new Easy TV BT878 version (model CPH061)
           special thanks to Informatica Mieres for providing the card */

        name:           "BESTBUY Easy TV (bt878)",
        video_inputs:   3,
        audio_inputs:   2,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xFF,
        muxsel:         { 2, 3, 1, 0},
        audiomux:       { 1, 0, 4, 4, 9},
        needs_tvaudio:  0,
        pll:            PLL_28,
        tuner_type:     TUNER_PHILIPS_PAL,
},{
        /* Lukas Gebauer <geby@volny.cz> */
        name:           "ATI TV-Wonder",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xf03f,
        muxsel:         { 2, 3, 1, 0 },
        audiomux:       { 0xbffe, 0, 0xbfff, 0, 0xbffe},
        pll:            PLL_28,
        tuner_type:     TUNER_TEMIC_4006FN5_MULTI_PAL,
},{

/* ---- card 0x40 ---------------------------------- */
        /* Lukas Gebauer <geby@volny.cz> */
        name:           "ATI TV-Wonder VE",
        video_inputs:   2,
        audio_inputs:   1,
        tuner:          0,
        svhs:           -1,
        gpiomask:       1,
        muxsel:         { 2, 3, 0, 1},
        audiomux:       { 0, 0, 1, 0, 0},
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     TUNER_TEMIC_4006FN5_MULTI_PAL,
},{
        /* DeeJay <deejay@westel900.net (2000S) */
        name:           "FlyVideo 2000S",
        video_inputs:   3,
        audio_inputs:   3,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x18e0,
        muxsel:         { 2, 3, 0, 1},
                        /* Radio changed from 1e80 to 0x800 to make
                           Flyvideo2000S in .hu happy (gm)*/

                        /* -dk-???: set mute=0x1800 for tda9874h daughterboard */
        audiomux:       { 0x0000,0x0800,0x1000,0x1000,0x1800, 0x1080 },
        audio_hook:     fv2000s_audio,
        no_msp34xx:     1,
        no_tda9875:     1,
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     5,
},{
        name:           "Terratec TValueRadio",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0xffff00,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0x500, 0x500, 0x300, 0x900, 0x900},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     TUNER_PHILIPS_PAL,
        has_radio:      1,
},{
        /* TANAKA Kei <peg00625@nifty.com> */
        name:           "IODATA GV-BCTV4/PCI",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x010f00,
        muxsel:         {2, 3, 0, 0},
        audiomux:       {0x10000, 0, 0x10000, 0, 0, 0},
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     TUNER_SHARP_2U5JF5540_NTSC,
        audio_hook:     gvbctv3pci_audio,
},{

/* ---- card 0x44 ---------------------------------- */
        name:           "3Dfx VoodooTV FM (Euro), VoodooTV 200 (USA)",
        // try "insmod msp3400 simple=0" if you have
        // sound problems with this card.
        video_inputs:   4,
        audio_inputs:   1,
        tuner:          0,
        svhs:           -1,
        gpiomask:       0x4f8a00,
        // 0x100000: 1=MSP enabled (0=disable again)
        // 0x010000: Connected to "S0" on tda9880 (0=Pal/BG, 1=NTSC)
        audiomux:       {0x947fff, 0x987fff,0x947fff,0x947fff, 0x947fff},
        // tvtuner, radio,   external,internal, mute,  stereo
        /* tuner, Composit, SVid, Composit-on-Svid-adapter*/
        muxsel:         { 2, 3 ,0 ,1},
        tuner_type:     TUNER_MT2032,
        pll:            PLL_28,
        has_radio:      1,
},{
        /* Philip Blundell <pb@nexus.co.uk> */
        name:           "Active Imaging AIMMS",
        video_inputs:   1,
        audio_inputs:   0,
        tuner:          -1,
        tuner_type:     -1,
        pll:            PLL_28,
        muxsel:         { 2 },
        gpiomask:       0
},{
        /* Tomasz Pyra <hellfire@sedez.iq.pl> */
        name:           "Pixelview PV-BT878P+ (Rev.4C)",
        video_inputs:   3,
        audio_inputs:   4,
        tuner:          0,
        svhs:           2,
        gpiomask:       15,
        muxsel:         { 2, 3, 1, 1},
        audiomux:       { 0, 0, 11, 7, 13, 0},
        needs_tvaudio:  1,
        pll:            PLL_28,
        tuner_type:     25,
},{
        name:           "Flyvideo 98EZ (capture only)",
        video_inputs:   4,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           2,
        muxsel:         { 2, 3, 1, 1}, // AV1, AV2, SVHS, CVid adapter on SVHS
        pll:            PLL_28,
        no_msp34xx:     1,
},{

/* ---- card 0x48 ---------------------------------- */
        /* Dariusz Kowalewski <darekk@automex.pl> */
        name:           "Pixelview PV-BT878P+9B (PlayTV Pro rev.9B FM+NICAM)",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x3f,
        muxsel:         { 2, 3, 0, 1 },
        audiomux:       { 0x01, 0x00, 0x03, 0x03, 0x09, 0x02 },
        needs_tvaudio:  1,
        no_msp34xx:     1,
        no_tda9875:     1,
        pll:            PLL_28,
        tuner_type:     -1,
        audio_hook:     pvbt878p9b_audio,
        has_radio:      1,
},{
        /* Clay Kunz <ckunz@mail.arc.nasa.gov> */
        /* you must jumper JP5 for the card to work */
        name:           "Sensoray 311",
        video_inputs:   5,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           4,
        gpiomask:       0,
        muxsel:         { 2, 3, 1, 0, 0},
        audiomux:       { 0 },
        needs_tvaudio:  0,
        tuner_type:     -1,
},{
        /* Miguel Freitas <miguel@cetuc.puc-rio.br> */
        name:           "RemoteVision MX (RV605)",
        video_inputs:   16,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           -1,
        gpiomask:       0x00,
        gpiomask2:      0x07ff,
        muxsel:         { 0x33, 0x13, 0x23, 0x43, 0xf3, 0x73, 0xe3, 0x03,
                          0xd3, 0xb3, 0xc3, 0x63, 0x93, 0x53, 0x83, 0xa3 },
        no_msp34xx:     1,
        no_tda9875:     1,
        tuner_type:     -1,
        muxsel_hook:    rv605_muxsel,
},{
        name:           "Powercolor MTV878/ MTV878R/ MTV878F",
        video_inputs:   3,
        audio_inputs:   2,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x1C800F,  // Bit0-2: Audio select, 8-12:remote control 14:remote valid 15:remote reset
        muxsel:         { 2, 1, 1, },
        audiomux:       { 0, 1, 2, 2, 4 },
        needs_tvaudio:  0,
        tuner_type:     TUNER_PHILIPS_PAL,
        pll:            PLL_28,
        has_radio:      1,
},{

/* ---- card 0x4c ---------------------------------- */
        /* Masaki Suzuki <masaki@btree.org> */
        name:           "Canopus WinDVR PCI (COMPAQ Presario 3524JP, 5112JP)",
        video_inputs:   3,
        audio_inputs:   1,
        tuner:          0,
        svhs:           2,
        gpiomask:       0x140007,
        muxsel:         { 2, 3, 1, 1 },
        audiomux:       { 0, 1, 2, 3, 4, 0 },
        tuner_type:     TUNER_PHILIPS_NTSC,
        audio_hook:     windvr_audio,
},{
        name:           "GrandTec Multi Capture Card (Bt878)",
        video_inputs:   4,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           -1,
        gpiomask:       0,
        muxsel:         { 2, 3, 1, 0 },
        audiomux:       { 0 },
        needs_tvaudio:  0,
        no_msp34xx:     1,
        pll:            PLL_28,
        tuner_type:     -1,
},{
        /* http://www.aopen.com/products/video/va1000.htm */
        name:           "AOPEN VA1000",
        video_inputs:   3, /* coax, AV, s-vid */
        audio_inputs:   1,
        tuner:          0,
        tuner_type:     TUNER_LG_PAL, /* actually TP18PSB12P (PAL B/G) */
        audiomux:       { 2, 0, 0, 0 },
        muxsel:         { 2, 3, 1, 0 },
        pll:            PLL_28,
},{
        name:           "Osprey 100/150 (878)", /* 0x1(2|3)-45C6-C1 */
        video_inputs:   4,                  /* id-inputs-clock */
        audio_inputs:   0,
        tuner:          -1,
        svhs:           3,
        muxsel:         { 3, 2, 0, 1 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
/* ---- card 0x50 ---------------------------------- */
        name:           "Osprey 100/150 (848)", /* 0x04-54C0-C1 & older boards */
        video_inputs:   3,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           2,
        muxsel:         { 2, 3, 1 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
        name:           "Osprey 101 (848)", /* 0x05-40C0-C1 */
        video_inputs:   2,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           1,
        muxsel:         { 3, 1 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
        name:           "Osprey 101/151",       /* 0x1(4|5)-0004-C4 */
        video_inputs:   1,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           -1,
        muxsel:         { 0 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
        name:           "Osprey 101/151 w/ svid",  /* 0x(16|17|20)-00C4-C1 */
        video_inputs:   2,
        audio_inputs:   0,
        tuner:          -1,
        svhs:           1,
        muxsel:         { 0, 1 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
/* ---- card 0x54 ---------------------------------- */
        name:           "Osprey 200/201/250/251",  /* 0x1(8|9|E|F)-0004-C4 */
        video_inputs:   1,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           -1,
        muxsel:         { 0 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
        name:           "Osprey 200/250",   /* 0x1(A|B)-00C4-C1 */
        video_inputs:   2,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           1,
        muxsel:         { 0, 1 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
        name:           "Osprey 210/220",   /* 0x1(A|B)-04C0-C1 */
        video_inputs:   2,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           1,
        muxsel:         { 2, 3 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
        name:           "Osprey 5x0",   /* 500 and 540 */
        video_inputs:   2,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           1,
        muxsel:         { 0, 1 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,
},{
/* ---- card 0x58 ---------------------------------- */
        name:           "Osprey 2000",  /* 2000 */
        video_inputs:   2,
        audio_inputs:   1,
        tuner:          -1,
        svhs:           1,
        muxsel:         { 2, 3 },
        pll:            PLL_28,
        tuner_type:     -1,
        no_msp34xx:     1,
        no_tda9875:     1,
        no_tda7432:     1,      /* must avoid, conflicts with the bt860 */
}};

const int bttv_num_tvcards = (sizeof(bttv_tvcards)/sizeof(struct tvcard));

/* ----------------------------------------------------------------------- */

static unsigned char eeprom_data[256];

/*
 * identify card
 */

void bttv_idcard(struct bttv *btv)
{
        unsigned int gpiobits;
        int i,type;
        unsigned short tmp;

        /* read PCI subsystem ID */
        pci_read_config_word(btv->dev, 2, &tmp);
        btv->cardid = tmp << 16;
        pci_read_config_word(btv->dev, 0, &tmp);
        btv->cardid |= tmp;

        if (0 != btv->cardid && 0xffffffff != btv->cardid) {
                /* look for the card */
                for (type = -1, i = 0; cards[i].id != 0; i++)
                        if (cards[i].id  == btv->cardid)
                                type = i;
               
                if (type != -1) {
                        /* found it */
                        printk(KERN_INFO "bttv%d: %s [card=%d], "
                               "PCI ID is %04x:%04x\n",
                               btv->nr,cards[type].name,cards[type].cardnr,
                               btv->cardid & 0xffff, btv->cardid >> 16);
                        btv->type = cards[type].cardnr;
                } else {
                        /* 404 */
                        printk(KERN_INFO "bttv%d: subsystem: %04x:%04x (UNKNOWN)\n",
                               btv->nr, btv->cardid&0xffff, btv->cardid>>16);
                }
        }

        /* let the user override the autodetected type */
        if (card[btv->nr] >= 0 && card[btv->nr] < bttv_num_tvcards)
                btv->type=card[btv->nr];
       
        /* print which card config we are using */
        sprintf(btv->video_dev.name,"BT%d%s(%.23s)",
                btv->id,
                (btv->id==848 && btv->revision==0x12) ? "A" : "",
                bttv_tvcards[btv->type].name);
        printk(KERN_INFO "bttv%d: using: %s [card=%d,%s]\n",btv->nr,
               btv->video_dev.name,btv->type,
               (card[btv->nr] >= 0 && card[btv->nr] < bttv_num_tvcards) ?
               "insmod option" : "autodetected");

        /* overwrite gpio stuff ?? */
        if (-1 == audioall && -1 == audiomux[0])
                return;

        if (-1 != audiomux[0]) {
                gpiobits = 0;
                for (i = 0; i < 5; i++) {
                        bttv_tvcards[btv->type].audiomux[i] = audiomux[i];
                        gpiobits |= audiomux[i];
                }
        } else {
                gpiobits = audioall;
                for (i = 0; i < 5; i++) {
                        bttv_tvcards[btv->type].audiomux[i] = audioall;
                }
        }
        bttv_tvcards[btv->type].gpiomask = (-1 != gpiomask) ? gpiomask : gpiobits;
        cprintf("[info  ] bttv%d: gpio config override: mask=0x%lx, mux=",
               btv->nr,bttv_tvcards[btv->type].gpiomask);
        for (i = 0; i < 5; i++) {
                cprintf("%s0x%lx", i ? "," : "", bttv_tvcards[btv->type].audiomux[i]);
        }
        cprintf("\n");
}

/*
 * (most) board specific initialisations goes here
 */


static void flyvideo_gpio(struct bttv *btv)
{
        int gpio,outbits;
        int tuner=-1,ttype;
       
        outbits = btread(BT848_GPIO_OUT_EN);
        btwrite(0x00, BT848_GPIO_OUT_EN);
        udelay(8);  // without this we would see the 0x1800 mask
        gpio=gpioread();
        btwrite(outbits, BT848_GPIO_OUT_EN);
        // all cards provide GPIO info, some have an additional eeprom
       
        // lowest 3 bytes are remote control codes (no handshake needed)
        ttype=(gpio&0x0f0000)>>16;
        switch(ttype) {
        case 0: tuner=4; // None
                break;
        case 4: tuner=5; // Philips PAL
                break;
        case 6: tuner=37; // LG PAL (newer TAPC series)
                break;
        case 0xC: tuner=3; // Philips SECAM(+PAL)
                break;
        default:
                printk(KERN_INFO "bttv%d: flyvideo_gpio: unknown tuner type.\n", btv->nr);
        }
       
        printk(KERN_INFO "bttv%d: Flyvideo Radio=%s RemoteControl=%s Tuner=%d gpio=0x%06x\n",
               btv->nr,
               gpio&0x400000? "yes":"no",
               gpio&0x800000? "yes":"no", tuner, gpio);
       
        btv->tuner_type = tuner;
        btv->has_radio = gpio&0x400000? 1:0;
}

int miro_tunermap[] = { 0,6,2,3,   4,5,6,0,  3,0,4,5,  5,2,16,1,
                        14,2,17,1, 4,1,4,3,  1,2,16,1, 4,4,4,4 };
int miro_fmtuner[]  = { 0,0,0,0,   0,0,0,0,  0,0,0,0,  0,0,0,1,
                        1,1,1,1,   1,1,1,0,  0,0,0,0,  0,0,0,0 };

/* initialization part one -- before registering i2c bus */
void bttv_init_card1(struct bttv *btv)
{
        if (btv->type == BTTV_HAUPPAUGE || btv->type == BTTV_HAUPPAUGE878)
                boot_msp34xx(btv,5);
        if (btv->type == BTTV_VOODOOTV_FM)
                boot_msp34xx(btv,20);
       
        /*printk("bttv: end of init -- pre i2c\n");*/
}

/* initialization part one -- after registering i2c bus */
void bttv_init_card2(struct bttv *btv)
{
        /* miro/pinnacle */
        if (btv->type == BTTV_MIRO      ||
            btv->type == BTTV_MIROPRO   ||
            btv->type == BTTV_PINNACLE  ||
            btv->type == BTTV_PINNACLEPRO) {
                int id,msp;
                id  = ((gpioread()>>10) & 31) -1;
                msp = bttv_I2CRead(btv, I2C_MSP3400, "MSP34xx");
                btv->tuner_type = miro_tunermap[id];
                if (0 == (gpioread() & 0x20)) {
                        btv->has_radio = 1;
                        if (!miro_fmtuner[id]) {
                                btv->has_matchbox = 1;
                                btv->mbox_we    = (1<<6);
                                btv->mbox_most  = (1<<7);
                                btv->mbox_clk   = (1<<8);
                                btv->mbox_data  = (1<<9);
                                btv->mbox_mask  = (1<<6)|(1<<7)|(1<<8)|(1<<9);
                        }
                } else {
                        btv->has_radio = 0;
                }
                if (-1 != msp) {
                        if (btv->type == BTTV_MIRO)
                                btv->type = BTTV_MIROPRO;
                        if (btv->type == BTTV_PINNACLE)
                                btv->type = BTTV_PINNACLEPRO;
                }
                if (bttv_verbose)
                        printk(KERN_INFO "bttv%d: miro: id=%d tuner=%d "
                               "radio=%s stereo=%s\n",
                               btv->nr, id+1, btv->tuner_type,
                               !btv->has_radio ? "no" :
                                   (btv->has_matchbox ? "matchbox" : "fmtuner"),
                               (-1 == msp) ? "no" : "yes");
#if 0
                if (btv->has_matchbox) {
                        if (bttv_verbose)
                                printk(KERN_INFO "Initializing TEA5757...\n");
                        init_tea5757(btv);
                }
#endif
        }

        if (btv->type == BTTV_FLYVIDEO_98    ||
            btv->type == BTTV_FLYVIDEO       ||
            btv->type == BTTV_TYPHOON_TVIEW  ||
            btv->type == BTTV_CHRONOS_VS2    ||
            btv->type == BTTV_FLYVIDEO_98FM  ||
            btv->type == BTTV_FLYVIDEO2000   ||
            btv->type == BTTV_FLYVIDEO98EZ)
                flyvideo_gpio(btv);

        if (btv->type == BTTV_HAUPPAUGE || btv->type == BTTV_HAUPPAUGE878) {
                /* pick up some config infos from the eeprom */
                bttv_readee(btv,eeprom_data,0xa0);
                hauppauge_eeprom(btv);
        }

        if ((btv->type >= BTTV_OSPREY_BEGIN) && (btv->type <= BTTV_OSPREY_END)) {
                /* check for config info in the eeprom */
                bttv_readee(btv,eeprom_data,0xa0);
                osprey_eeprom(btv);
        }

        if (btv->type == BTTV_AVERMEDIA98 || btv->type == BTTV_AVPHONE98) {
                bttv_readee(btv,eeprom_data,0xa0);
                avermedia_eeprom(btv);
        }

        if (btv->type == BTTV_PXC200)
                init_PXC200(btv);

        if (btv->type == BTTV_VHX) {
                btv->has_radio    = 1;
                btv->has_matchbox = 1;
                btv->mbox_we      = 0x20;
                btv->mbox_most    = 0;
                btv->mbox_clk     = 0x08;
                btv->mbox_data    = 0x10;
                btv->mbox_mask    = 0x38;
        }

        if (btv->type == BTTV_LIFETEC_9415) {
                if (gpioread() & 0x4000)
                        cprintf("bttv%d: lifetec: tv mono/fm stereo card\n", btv->nr);
                else
                        cprintf("bttv%d: lifetec: stereo(TDA9821) card\n",btv->nr);
        }

        if (btv->type == BTTV_MAGICTVIEW061) {
                if(btv->cardid == 0x4002144f) {
                        btv->has_radio=1;
                        cprintf("bttv%d: radio detected by subsystem id (CPH05x)\n",btv->nr);
                }
        }

        /* pll configuration */
        if (!(btv->id==848 && btv->revision==0x11)) {
                /* defaults from card list */
                if (PLL_28 == bttv_tvcards[btv->type].pll) {
                        btv->pll.pll_ifreq=28636363;
                        btv->pll.pll_crystal=BT848_IFORM_XT0;
                }
                if (PLL_35 == bttv_tvcards[btv->type].pll) {
                        btv->pll.pll_ifreq=35468950;
                        btv->pll.pll_crystal=BT848_IFORM_XT1;
                }
                /* insmod options can override */
                switch (pll[btv->nr]) {
                case 0: /* none */
                        btv->pll.pll_crystal = 0;
                        btv->pll.pll_ifreq   = 0;
                        btv->pll.pll_ofreq   = 0;
                        break;
                case 1: /* 28 MHz */
                case 28:
                        btv->pll.pll_ifreq   = 28636363;
                        btv->pll.pll_ofreq   = 0;
                        btv->pll.pll_crystal = BT848_IFORM_XT0;
                        break;
                case 2: /* 35 MHz */
                case 35:
                        btv->pll.pll_ifreq   = 35468950;
                        btv->pll.pll_ofreq   = 0;
                        btv->pll.pll_crystal = BT848_IFORM_XT1;
                        break;
                }
        }

        /* tuner configuration (from card list / insmod option) */
        if (-1 != bttv_tvcards[btv->type].tuner_type)
                btv->tuner_type = bttv_tvcards[btv->type].tuner_type;
        if (-1 != tuner[btv->nr])
                btv->tuner_type = tuner[btv->nr];

        if (bttv_tvcards[btv->type].has_radio)
                btv->has_radio=1;

        /* try to detect audio/fader chips */
        if (!bttv_tvcards[btv->type].no_msp34xx &&
            bttv_I2CRead(btv, I2C_MSP3400, "MSP34xx") >=0) {
        }

        if (!bttv_tvcards[btv->type].no_tda9875 &&
            bttv_I2CRead(btv, I2C_TDA9875, "TDA9875") >=0) {
        }

        if (!bttv_tvcards[btv->type].no_tda7432 &&
            bttv_I2CRead(btv, I2C_TDA7432, "TDA7432") >=0) {
        }

        if (bttv_tvcards[btv->type].needs_tvaudio) {
        }

        if (bttv_tvcards[btv->type].tuner != -1) {
        }
}


/* ----------------------------------------------------------------------- */
/* some hauppauge specific stuff                                           */

static struct HAUPPAUGE_TUNER
{
        int  id;
        char *name;
}
hauppauge_tuner[] =
{
        { TUNER_ABSENT,        "" },
        { TUNER_ABSENT,        "External" },
        { TUNER_ABSENT,        "Unspecified" },
        { TUNER_PHILIPS_PAL,   "Philips FI1216" },
        { TUNER_PHILIPS_SECAM, "Philips FI1216MF" },
        { TUNER_PHILIPS_NTSC,  "Philips FI1236" },
        { TUNER_PHILIPS_PAL_I, "Philips FI1246" },
        { TUNER_PHILIPS_PAL_DK,"Philips FI1256" },
        { TUNER_PHILIPS_PAL,   "Philips FI1216 MK2" },
        { TUNER_PHILIPS_SECAM, "Philips FI1216MF MK2" },
        { TUNER_PHILIPS_NTSC,  "Philips FI1236 MK2" },
        { TUNER_PHILIPS_PAL_I, "Philips FI1246 MK2" },
        { TUNER_PHILIPS_PAL_DK,"Philips FI1256 MK2" },
        { TUNER_TEMIC_NTSC,    "Temic 4032FY5" },
        { TUNER_TEMIC_PAL,     "Temic 4002FH5" },
        { TUNER_TEMIC_PAL_I,   "Temic 4062FY5" },
        { TUNER_PHILIPS_PAL,   "Philips FR1216 MK2" },
        { TUNER_PHILIPS_SECAM, "Philips FR1216MF MK2" },
        { TUNER_PHILIPS_NTSC,  "Philips FR1236 MK2" },
        { TUNER_PHILIPS_PAL_I, "Philips FR1246 MK2" },
        { TUNER_PHILIPS_PAL_DK,"Philips FR1256 MK2" },
        { TUNER_PHILIPS_PAL,   "Philips FM1216" },
        { TUNER_PHILIPS_SECAM, "Philips FM1216MF" },
        { TUNER_PHILIPS_NTSC,  "Philips FM1236" },
        { TUNER_PHILIPS_PAL_I, "Philips FM1246" },
        { TUNER_PHILIPS_PAL_DK,"Philips FM1256" },
        { TUNER_TEMIC_4036FY5_NTSC, "Temic 4036FY5" },
        { TUNER_ABSENT,        "Samsung TCPN9082D" },
        { TUNER_ABSENT,        "Samsung TCPM9092P" },
        { TUNER_TEMIC_4006FH5_PAL, "Temic 4006FH5" },
        { TUNER_ABSENT,        "Samsung TCPN9085D" },
        { TUNER_ABSENT,        "Samsung TCPB9085P" },
        { TUNER_ABSENT,        "Samsung TCPL9091P" },
        { TUNER_TEMIC_4039FR5_NTSC, "Temic 4039FR5" },
        { TUNER_PHILIPS_FQ1216ME,   "Philips FQ1216 ME" },
        { TUNER_TEMIC_4066FY5_PAL_I, "Temic 4066FY5" },
        { TUNER_ABSENT,        "Philips TD1536" },
        { TUNER_ABSENT,        "Philips TD1536D" },
        { TUNER_PHILIPS_NTSC,  "Philips FMR1236" }, /* mono radio */
        { TUNER_ABSENT,        "Philips FI1256MP" },
        { TUNER_ABSENT,        "Samsung TCPQ9091P" },
        { TUNER_TEMIC_4006FN5_MULTI_PAL, "Temic 4006FN5" },
        { TUNER_TEMIC_4009FR5_PAL, "Temic 4009FR5" },
        { TUNER_TEMIC_4046FM5,     "Temic 4046FM5" },
        { TUNER_TEMIC_4009FN5_MULTI_PAL_FM, "Temic 4009FN5" },
        { TUNER_ABSENT,        "Philips TD1536D_FH_44"},
        { TUNER_LG_NTSC_FM,    "LG TP18NSR01F"},
        { TUNER_LG_PAL_FM,     "LG TP18PSB01D"},
        { TUNER_LG_PAL,        "LG TP18PSB11D"},       
        { TUNER_LG_PAL_I_FM,   "LG TAPC-I001D"},
        { TUNER_LG_PAL_I,      "LG TAPC-I701D"}
};

static void hauppauge_eeprom(struct bttv *btv)
{
        int blk2,tuner,radio,model;

        if (eeprom_data[0] != 0x84 || eeprom_data[2] != 0)
                printk(KERN_WARNING "bttv%d: Hauppauge eeprom: invalid\n",
                       btv->nr);

        /* Block 2 starts after len+3 bytes header */
        blk2 = eeprom_data[1] + 3;

        /* decode + use some config infos */
        model = eeprom_data[12] << 8 | eeprom_data[11];
        tuner = eeprom_data[9];
        radio = eeprom_data[blk2-1] & 0x01;
       
        if (tuner < sizeof(hauppauge_tuner)/sizeof(struct HAUPPAUGE_TUNER))
                btv->tuner_type = hauppauge_tuner[tuner].id;
        if (radio)
                btv->has_radio = 1;
       
        if (bttv_verbose)
                printk(KERN_INFO "bttv%d: Hauppauge eeprom: model=%d, "
                       "tuner=%s (%d), radio=%s\n",
                       btv->nr, model, hauppauge_tuner[tuner].name,
                       btv->tuner_type, radio ? "yes" : "no");
}

static void osprey_eeprom(struct bttv *btv)
{
    int i = 0;
    int detected_type = btv->type;
    unsigned char *ee = eeprom_data;
    unsigned long serial = 0;
   
    if (bttv_debug) {
        printk(KERN_INFO "Osprey: EEPROM dump\n");
        for(;i<256; i+= 16) {
            cprintf("[info  ] Osprey 100:(%02X) %02X %02X %02X %02X %02X %02X %02X %02X -- %02X %02X %02X %02X %02X %02X %02X %02X\n",
                    i,ee[i+ 0],ee[i+ 1],ee[i+ 2],ee[i+ 3],
                      ee[i+ 4],ee[i+ 5],ee[i+ 6],ee[i+ 7],
                      ee[i+ 8],ee[i+ 9],ee[i+10],ee[i+11],
                      ee[i+12],ee[i+13],ee[i+14],ee[i+15] );
        }
    }
   
    if (btv->type == 0) {
        /* this might be an antique...check for MMAC label in eeprom */
        if ((ee[0]=='M') && (ee[1]=='M') && (ee[2]=='A') && (ee[3]=='C')) {
            unsigned char checksum = 0;
            /* that looks like an old card...verify checksum */
            for (i =0; i<21; i++) checksum += ee[i];
            if (checksum == ee[21]) {
                btv->type = BTTV_OSPREY1x0_848;
                for (i = 12; i < 21; i++) serial *= 10,serial += ee[i] - '0';
            }
        }
    } else {
        /* we know this is an osprey card, find a valid eeprom descriptor */
        int offset = 4*16;
       
        for(; offset < 8*16; offset += 16) {
            unsigned short checksum = 0;
            /* verify the checksum */
            for(i = 0; i<14; i++) checksum += ee[i+offset];
            checksum = ~checksum;  /* no idea why */
            if ((((checksum>>8) & 0x0FF) == ee[offset+14]) &&
                ((checksum & 0x0FF) == ee[offset+15])) {
                break;
            }
        }
       
        if (offset < 8*16) {
            /* found a valid descriptor */
            unsigned short type = (ee[offset+4]<<8) | (ee[offset+5]);
           
            switch(type) {
                /* 848 based */
                case 0x0004: {
                    btv->type = BTTV_OSPREY1x0_848;
                    break;
                }
                case 0x0005: {
                    btv->type = BTTV_OSPREY101_848;
                    break;
                }
                /* 878 based */
                case 0x0012:
                case 0x0013: {
                    btv->type = BTTV_OSPREY1x0;
                    break;
                }
                case 0x0014:
                case 0x0015: {
                    btv->type = BTTV_OSPREY1x1;
                    break;
                }
                case 0x0016:
                case 0x0017:
                case 0x0020: {
                    btv->type = BTTV_OSPREY1x1_SVID;
                    break;
                }
                case 0x0018:
                case 0x0019:
                case 0x001E:
                case 0x001F: {
                    btv->type = BTTV_OSPREY2xx;
                    break;
                }
                case 0x001A:
                case 0x001B: {
                    btv->type = BTTV_OSPREY2x0_SVID;
                    break;
                }
                case 0x0060:
                case 0x0070: {
                    btv->type = BTTV_OSPREY2x0;
                    //enable output on select control lines
                    btwrite(0x000303, BT848_GPIO_OUT_EN);
                    break;
                }
                default: {
                    /* unknown...leave as generic, but get a serial # */
                    break;
                }
            }
           
            serial =   (ee[offset+6] << 24)
                     | (ee[offset+7] << 16)
                     | (ee[offset+8] <<  8)
                     | (ee[offset+9]);
        }
    }
   
    if ((btv->type != detected_type) || (serial != 0)){
        /* reprint which card config we are using */
        sprintf(btv->video_dev.name,"BT%d%s(%.23s)",
                btv->id,
                (btv->id==848 && btv->revision==0x12) ? "A" : "",
                bttv_tvcards[btv->type].name);
        printk(KERN_INFO "bttv%d: is now:%s [card=%d,%s] Serial#: %ld\n",
               btv->nr,
               btv->video_dev.name,
               btv->type,
               (card[btv->nr] >= 0 && card[btv->nr] < bttv_num_tvcards) ?
               "insmod option" : "autodetected",
               serial);
    }
}


/* ----------------------------------------------------------------------- */
/* AVermedia specific stuff, from  bktr_card.c                             */

int tuner_0_table[] = {
        TUNER_PHILIPS_NTSC,  TUNER_PHILIPS_PAL,
        TUNER_PHILIPS_PAL,   TUNER_PHILIPS_PAL,
        TUNER_PHILIPS_PAL,   TUNER_PHILIPS_PAL,
        TUNER_PHILIPS_SECAM, TUNER_PHILIPS_SECAM,
        TUNER_PHILIPS_SECAM, TUNER_PHILIPS_PAL};
#if 0
int tuner_0_fm_table[] = {
        PHILIPS_FR1236_NTSC,  PHILIPS_FR1216_PAL,
        PHILIPS_FR1216_PAL,   PHILIPS_FR1216_PAL,
        PHILIPS_FR1216_PAL,   PHILIPS_FR1216_PAL,
        PHILIPS_FR1236_SECAM, PHILIPS_FR1236_SECAM,
        PHILIPS_FR1236_SECAM, PHILIPS_FR1216_PAL};
#endif

int tuner_1_table[] = {
        TUNER_TEMIC_NTSC,  TUNER_TEMIC_PAL,
        TUNER_TEMIC_PAL,   TUNER_TEMIC_PAL,
        TUNER_TEMIC_PAL,   TUNER_TEMIC_PAL,
        TUNER_TEMIC_4012FY5, TUNER_TEMIC_4012FY5, //TUNER_TEMIC_SECAM
        TUNER_TEMIC_4012FY5, TUNER_TEMIC_PAL};

static void avermedia_eeprom(struct bttv *btv)
{
        int tuner_make,tuner_tv_fm,tuner_format,tuner=0,remote;

        tuner_make   = (eeprom_data[0x41] & 0x7);
        tuner_tv_fm  = (eeprom_data[0x41] & 0x18) >> 3;
        tuner_format = (eeprom_data[0x42] & 0xf0) >> 4;
        remote       = (eeprom_data[0x42] & 0x01);

        if (tuner_make == 0 || tuner_make == 2)
                if(tuner_format <=9)
                        tuner = tuner_0_table[tuner_format];
        if (tuner_make == 1)
                if(tuner_format <=9)
                        tuner = tuner_1_table[tuner_format];
       
        cprintf("[info  ] bttv%d: Avermedia eeprom[0x%02x%02x]: tuner=",
                btv->nr,eeprom_data[0x41],eeprom_data[0x42]);
        if(tuner) {
                btv->tuner_type=tuner;
                cprintf("%d",tuner);
        } else
                cprintf("Unknown type");
        cprintf(" radio:%s remote control:%s\n",
                tuner_tv_fm?"yes":"no",
                remote?"yes":"no");
}

/* used on Voodoo TV/FM (Voodoo 200), S0 wired to 0x10000 */
void bttv_tda9880_setnorm(struct bttv *btv, int norm)
{
        // fix up our card entry
        if(norm==VIDEO_MODE_NTSC) {
                bttv_tvcards[BTTV_VOODOOTV_FM].audiomux[0]=0x957fff;
                bttv_tvcards[BTTV_VOODOOTV_FM].audiomux[4]=0x957fff;
                dprintk("bttv_tda9880_setnorm to NTSC\n");
        }
        else {
                bttv_tvcards[BTTV_VOODOOTV_FM].audiomux[0]=0x947fff;
                bttv_tvcards[BTTV_VOODOOTV_FM].audiomux[4]=0x947fff;
                dprintk("bttv_tda9880_setnorm to PAL\n");
        }
        // set GPIO according
        gpioaor(bttv_tvcards[btv->type].audiomux[btv->audio],
              ~bttv_tvcards[btv->type].gpiomask);
}


/*
 * reset/enable the MSP on some Hauppauge cards
 * Thanks to Kyösti Mälkki (kmalkki@cc.hut.fi)!
 *
 * Hauppauge:  pin  5
 * Voodoo:     pin 20
 */

static void boot_msp34xx(struct bttv *btv, int pin)
{
        int mask = (1 << pin);

        btor(mask, BT848_GPIO_OUT_EN);
        gpioand(~mask);
        udelay(2500);
        gpioor(mask);
        if (bttv_gpio)
                bttv_gpio_tracking(btv,"msp34xx");

        if (bttv_verbose)
                printk(KERN_INFO "bttv%d: Hauppauge/Voodoo msp34xx: reset line "
                       "init [%d]\n", btv->nr, pin);
}


/* ----------------------------------------------------------------------- */
/*  Imagenation L-Model PXC200 Framegrabber */
/*  This is basically the same procedure as
 *  used by Alessandro Rubini in his pxc200
 *  driver, but using BTTV functions */


static void init_PXC200(struct bttv *btv)
{
        /* Initialise GPIO-connevted stuff */
        btwrite(1<<13,BT848_GPIO_OUT_EN); /* Reset pin only */
        gpiowrite(0);
        udelay(3);
        gpiowrite(1<<13);
        /* GPIO inputs are pulled up, so no need to drive
         * reset pin any longer */

        btwrite(0,BT848_GPIO_OUT_EN);
        if (bttv_gpio)
                bttv_gpio_tracking(btv,"pxc200");

        /*  we could/should try and reset/control the AD pots? but
            right now  we simply  turned off the crushing.  Without
            this the AGC drifts drifts
            remember the EN is reverse logic -->
            setting BT848_ADC_AGC_EN disable the AGC
            tboult@eecs.lehigh.edu
        */

        btwrite(BT848_ADC_RESERVED|BT848_ADC_AGC_EN, BT848_ADC);
       
        /*      Initialise MAX517 DAC */
        printk(KERN_INFO "Setting DAC reference voltage level ...\n");
        bttv_I2CWrite(btv,0x5E,0,0x80,1);
       
        /*      Initialise 12C508 PIC */
        /*      The I2CWrite and I2CRead commmands are actually to the
         *      same chips - but the R/W bit is included in the address
         *      argument so the numbers are different */

       
        printk(KERN_INFO "PXC200 Initialised.\n");
}

/* ----------------------------------------------------------------------- */
/* Miro Pro radio stuff -- the tea5757 is connected to some GPIO ports     */
/*
 * Copyright (c) 1999 Csaba Halasz <qgehali@uni-miskolc.hu>
 * This code is placed under the terms of the GNU General Public License
 *
 * Brutally hacked by Dan Sheridan <dan.sheridan@contact.org.uk> djs52 8/3/00
 */


#if 0
/* bus bits on the GPIO port */
#define TEA_WE                  6
#define TEA_DATA                9
#define TEA_CLK                 8
#define TEA_MOST                7
#endif

#define BUS_LOW(bit)    gpioand(~(bit))
#define BUS_HIGH(bit)   gpioor((bit))
#define BUS_IN(bit)     (gpioread() & (bit))

/* TEA5757 register bits */
#define TEA_FREQ                0:14
#define TEA_BUFFER              15:15

#define TEA_SIGNAL_STRENGTH     16:17

#define TEA_PORT1               18:18
#define TEA_PORT0               19:19

#define TEA_BAND                20:21
#define TEA_BAND_FM             0
#define TEA_BAND_MW             1
#define TEA_BAND_LW             2
#define TEA_BAND_SW             3

#define TEA_MONO                22:22
#define TEA_ALLOW_STEREO        0
#define TEA_FORCE_MONO          1

#define TEA_SEARCH_DIRECTION    23:23
#define TEA_SEARCH_DOWN         0
#define TEA_SEARCH_UP           1

#define TEA_STATUS              24:24
#define TEA_STATUS_TUNED        0
#define TEA_STATUS_SEARCHING    1

/* ----------------------------------------------------------------------- */
/* winview                                                                 */

void winview_audio(struct bttv *btv, struct video_audio *v, int set)
{
        /* PT2254A programming Jon Tombs, jon@gte.esi.us.es */
        int bits_out, loops, vol, data;

        if (!set) {
                /* Fixed by Leandro Lucarella <luca@linuxmendoza.org.ar (07/31/01) */
                v->flags |= VIDEO_AUDIO_VOLUME;
                return;
        }
       
        /* 32 levels logarithmic */
        vol = 32 - ((v->volume>>11));
        /* units */
        bits_out = (PT2254_DBS_IN_2>>(vol%5));
        /* tens */
        bits_out |= (PT2254_DBS_IN_10>>(vol/5));
        bits_out |= PT2254_L_CHANNEL | PT2254_R_CHANNEL;
        data = gpioread();
        data &= ~(WINVIEW_PT2254_CLK| WINVIEW_PT2254_DATA|
                  WINVIEW_PT2254_STROBE);
        for (loops = 17; loops >= 0 ; loops--) {
                if (bits_out & (1<<loops))
                        data |=  WINVIEW_PT2254_DATA;
                else
                        data &= ~WINVIEW_PT2254_DATA;
                gpiowrite(data);
                udelay(5);
                data |= WINVIEW_PT2254_CLK;
                gpiowrite(data);
                udelay(5);
                data &= ~WINVIEW_PT2254_CLK;
                gpiowrite(data);
        }
        data |=  WINVIEW_PT2254_STROBE;
        data &= ~WINVIEW_PT2254_DATA;
        gpiowrite(data);
        udelay(10);                    
        data &= ~WINVIEW_PT2254_STROBE;
        gpiowrite(data);
}

/* ----------------------------------------------------------------------- */
/* mono/stereo control for various cards (which don't use i2c chips but    */
/* connect something to the GPIO pins                                      */

static void
gvbctv3pci_audio(struct bttv *btv, struct video_audio *v, int set)
{
        unsigned int con = 0;

        if (set) {
                btor(0x300, BT848_GPIO_OUT_EN);
                if (v->mode & VIDEO_SOUND_LANG1)
                        con = 0x000;
                if (v->mode & VIDEO_SOUND_LANG2)
                        con = 0x300;
                if (v->mode & VIDEO_SOUND_STEREO)
                        con = 0x200;
//              if (v->mode & VIDEO_SOUND_MONO)
//                      con = 0x100;
                gpioaor(con, ~0x300);
        } else {
                v->mode = VIDEO_SOUND_STEREO |
                          VIDEO_SOUND_LANG1  | VIDEO_SOUND_LANG2;
        }
}

/*
 * Mario Medina Nussbaum <medisoft@alohabbs.org.mx>
 *  I discover that on BT848_GPIO_DATA address a byte 0xcce enable stereo,
 *  0xdde enables mono and 0xccd enables sap
 *
 * Petr Vandrovec <VANDROVE@vc.cvut.cz>
 *  P.S.: At least mask in line above is wrong - GPIO pins 3,2 select
 *  input/output sound connection, so both must be set for output mode.
 *
 * Looks like it's needed only for the "tvphone", the "tvphone 98"
 * handles this with a tda9840
 *
 */

static void
avermedia_tvphone_audio(struct bttv *btv, struct video_audio *v, int set)
{
        int val = 0;

        if (set) {
                if (v->mode & VIDEO_SOUND_LANG1)   /* SAP */
                        val = 0x02;
                if (v->mode & VIDEO_SOUND_STEREO)
                        val = 0x01;
                if (val) {
                        gpioaor(val, ~0x03);
                        if (bttv_gpio)
                                bttv_gpio_tracking(btv,"avermedia");
                }
        } else {
                v->mode = VIDEO_SOUND_MONO | VIDEO_SOUND_STEREO |
                        VIDEO_SOUND_LANG1;
                return;
        }
}

/* Lifetec 9415 handling */
static void
lt9415_audio(struct bttv *btv, struct video_audio *v, int set)
{
        int val = 0;

        if (gpioread() & 0x4000) {
                v->mode = VIDEO_SOUND_MONO;
                return;
        }

        if (set) {
                if (v->mode & VIDEO_SOUND_LANG2)  /* A2 SAP */
                        val = 0x0080;
                if (v->mode & VIDEO_SOUND_STEREO) /* A2 stereo */
                        val = 0x0880;
                if ((v->mode & VIDEO_SOUND_LANG1) ||
                    (v->mode & VIDEO_SOUND_MONO))
                        val = 0;
                gpioaor(val, ~0x0880);
                if (bttv_gpio)
                        bttv_gpio_tracking(btv,"lt9415");
        } else {
                /* autodetect doesn't work with this card :-( */
                v->mode = VIDEO_SOUND_MONO | VIDEO_SOUND_STEREO |
                        VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2;
                return;
        }
}


static void
terratv_audio(struct bttv *btv, struct video_audio *v, int set)
{
        unsigned int con = 0;

        if (set) {
                btor(0x180000, BT848_GPIO_OUT_EN);
                if (v->mode & VIDEO_SOUND_LANG2)
                        con = 0x080000;
                if (v->mode & VIDEO_SOUND_STEREO)
                        con = 0x180000;
                gpioaor(con, ~0x180000);
                if (bttv_gpio)
                        bttv_gpio_tracking(btv,"terratv");
        } else {
                v->mode = VIDEO_SOUND_MONO | VIDEO_SOUND_STEREO |
                        VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2;
        }
}

static void
winfast2000_audio(struct bttv *btv, struct video_audio *v, int set)
{
        unsigned long val = 0;

        if (set) {
                /*btor (0xc32000, BT848_GPIO_OUT_EN);*/
                if (v->mode & VIDEO_SOUND_MONO)         /* Mono */
                        val = 0x420000;
                if (v->mode & VIDEO_SOUND_LANG1)        /* Mono */
                        val = 0x420000;
                if (v->mode & VIDEO_SOUND_LANG2)        /* SAP */
                        val = 0x410000;
                if (v->mode & VIDEO_SOUND_STEREO)       /* Stereo */
                        val = 0x020000;
                if (val) {
                        gpioaor(val, ~0x430000);
                        if (bttv_gpio)
                                bttv_gpio_tracking(btv,"winfast2000");
                }
        } else {
                v->mode = VIDEO_SOUND_MONO | VIDEO_SOUND_STEREO |
                          VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2;
        }
}

/*
 * Dariusz Kowalewski <darekk@automex.pl>
 * sound control for Prolink PV-BT878P+9B (PixelView PlayTV Pro FM+NICAM
 * revision 9B has on-board TDA9874A sound decoder).
 */

static void
pvbt878p9b_audio(struct bttv *btv, struct video_audio *v, int set)
{
        unsigned int val = 0;

        if (btv->radio)
                return;

        if (set) {
                if (v->mode & VIDEO_SOUND_MONO) {
                        val = 0x01;
                }
                if ((v->mode & (VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2))
                    || (v->mode & VIDEO_SOUND_STEREO)) {
                        val = 0x02;
                }
                if (val) {
                        gpioaor(val, ~0x03);
                        if (bttv_gpio)
                                bttv_gpio_tracking(btv,"pvbt878p9b");
                }
        } else {
                v->mode = VIDEO_SOUND_MONO | VIDEO_SOUND_STEREO |
                        VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2;
        }
}

/*
 * Dariusz Kowalewski <darekk@automex.pl>
 * sound control for FlyVideo 2000S (with tda9874 decoder)
 * based on pvbt878p9b_audio() - this is not tested, please fix!!!
 */

static void
fv2000s_audio(struct bttv *btv, struct video_audio *v, int set)
{
        unsigned int val = 0xffff;

        if (btv->radio)
                return;
       
        if (set) {
                if (v->mode & VIDEO_SOUND_MONO) {
                        val = 0x0000;
                }
                if ((v->mode & (VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2))
                    || (v->mode & VIDEO_SOUND_STEREO)) {
                        val = 0x1080; //-dk-???: 0x0880, 0x0080, 0x1800 ...
                }
                if (val != 0xffff) {
                        gpioaor(val, ~0x1800);
                        if (bttv_gpio)
                                bttv_gpio_tracking(btv,"fv2000s");
                }
        } else {
                v->mode = VIDEO_SOUND_MONO | VIDEO_SOUND_STEREO |
                        VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2;
        }
}

/*
 * sound control for Canopus WinDVR PCI
 * Masaki Suzuki <masaki@btree.org>
 */

static void
windvr_audio(struct bttv *btv, struct video_audio *v, int set)
{
        unsigned long val = 0;

        if (set) {
                if (v->mode & VIDEO_SOUND_MONO)
                        val = 0x040000;
                if (v->mode & VIDEO_SOUND_LANG1)
                        val = 0;
                if (v->mode & VIDEO_SOUND_LANG2)
                        val = 0x100000;
                if (v->mode & VIDEO_SOUND_STEREO)
                        val = 0;
                if (val) {
                        gpioaor(val, ~0x140000);
                        if (bttv_gpio)
                                bttv_gpio_tracking(btv,"windvr");
                }
        } else {
                v->mode = VIDEO_SOUND_MONO | VIDEO_SOUND_STEREO |
                          VIDEO_SOUND_LANG1 | VIDEO_SOUND_LANG2;
        }
}

/* RemoteVision MX (rv605) muxsel helper [Miguel Freitas]
 *
 * This is needed because rv605 don't use a normal multiplex, but a crosspoint
 * switch instead (CD22M3494E). This IC can have multiple active connections
 * between Xn (input) and Yn (output) pins. We need to clear any existing
 * connection prior to establish a new one, pulsing the STROBE pin.
 *
 * The board hardwire Y0 (xpoint) to MUX1 and MUXOUT to Yin.
 * GPIO pins are wired as:
 *  GPIO[0:3] - AX[0:3] (xpoint) - P1[0:3] (microcontroler)
 *  GPIO[4:6] - AY[0:2] (xpoint) - P1[4:6] (microcontroler)
 *  GPIO[7]   - DATA (xpoint)    - P1[7] (microcontroler)
 *  GPIO[8]   -                  - P3[5] (microcontroler)
 *  GPIO[9]   - RESET (xpoint)   - P3[6] (microcontroler)
 *  GPIO[10]  - STROBE (xpoint)  - P3[7] (microcontroler)
 *  GPINTR    -                  - P3[4] (microcontroler)
 *
 * The microcontroler is a 80C32 like. It should be possible to change xpoint
 * configuration either directly (as we are doing) or using the microcontroler
 * which is also wired to I2C interface. I have no further info on the
 * microcontroler features, one would need to disassembly the firmware.
 * note: the vendor refused to give any information on this product, all
 *       that stuff was found using a multimeter! :)
 */

static void rv605_muxsel(struct bttv *btv, unsigned int input)
{
        /* reset all conections */
        gpioaor(0x200,~0x200);
        udelay(1000);
        gpioaor(0x000,~0x200);
        udelay(1000);

        /* create a new conection */
        gpioaor(0x080,~0x480);
        gpioaor(0x480,~0x480);
        udelay(1000);
        gpioaor(0x080,~0x480);
        udelay(1000);
}


/* ----------------------------------------------------------------------- */
/* motherboard chipset specific stuff                                      */

void bttv_check_chipset(void)
{
        int pcipci_fail = 0;
        struct pci_dev dev;
        BYTE bus, dv;

        triton1 = 1;
       
        if (pci_device(PCI_VENDOR_ID_SI, PCI_DEVICE_ID_SI_496, 0, &bus, &dv)) pcipci_fail = 1;
        if (pci_device(PCI_VENDOR_ID_SI, PCI_DEVICE_ID_SI_5597, 0, &bus, &dv)) pcipci_fail = 1;
        if (pci_device(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82437, 0, &bus, &dv)) triton1 = 1;
        if (pci_device(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82443BX_0, 0, &bus, &dv)) triton1 = 1;
        if (pci_device(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C597_0, 0, &bus, &dv)) triton1 = 1;
        if (pci_device(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C576, 0, &bus, &dv)) vsfx = 1;
       
        /* print warnings about any quirks found */
        if (triton1)
                printk(KERN_INFO "bttv: Host bridge needs ETBF enabled.\n");
        if (vsfx)
                printk(KERN_INFO "bttv: Host bridge needs VSFX enabled.\n");

        if (pcipci_fail) {
                printk(KERN_WARNING "bttv: BT848 and your chipset may not work together.\n");
                if (-1 == no_overlay) {
                        printk(KERN_WARNING "bttv: going to disable overlay.\n");
                        no_overlay = 1;
                }
        }

        if(pci_device(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82441, 0, &bus, &dv)) {
                unsigned char b;
                dev.bus->number = bus;
                dev.devfn = dv;
                pci_read_config_byte(&dev, 0x53, &b);
                if (bttv_debug)
                        printk(KERN_INFO "bttv: Host bridge: 82441FX Natoma, "
                               "bufcon=0x%02x\n",b);
        }

}

int bttv_handle_chipset(struct bttv *btv)
{
        unsigned char command;

        if (!triton1 && !vsfx)
                return 0;

        if (bttv_verbose) {
                if (triton1)
                        printk(KERN_INFO "bttv%d: enabling ETBF (430FX/VP3 compatibilty)\n",btv->nr);
                if (vsfx && btv->id >= 878)
                        printk(KERN_INFO "bttv%d: enabling VSFX\n",btv->nr);
        }

        if (btv->id < 878) {
                /* bt848 (mis)uses a bit in the irq mask for etbf */
                if (triton1)
                        btv->triton1 = BT848_INT_ETBF;
        } else {
                /* bt878 has a bit in the pci config space for it */
                pci_read_config_byte(btv->dev, BT878_DEVCTRL, &command);
                if (triton1)
                        command |= BT878_EN_TBFX;
                if (vsfx)
                        command |= BT878_EN_VSFX;
                pci_write_config_byte(btv->dev, BT878_DEVCTRL, command);
        }
        return 0;
}