aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/media/video/cx88/cx88-cards.c168
-rw-r--r--drivers/media/video/cx88/cx88-core.c163
-rw-r--r--drivers/media/video/cx88/cx88.h12
3 files changed, 172 insertions, 171 deletions
diff --git a/drivers/media/video/cx88/cx88-cards.c b/drivers/media/video/cx88/cx88-cards.c
index 8be90ad4e910..6204a453c11e 100644
--- a/drivers/media/video/cx88/cx88-cards.c
+++ b/drivers/media/video/cx88/cx88-cards.c
@@ -27,10 +27,26 @@
27 27
28#include "cx88.h" 28#include "cx88.h"
29 29
30static unsigned int tuner[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
31static unsigned int radio[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
32static unsigned int card[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
33
34module_param_array(tuner, int, NULL, 0444);
35module_param_array(radio, int, NULL, 0444);
36module_param_array(card, int, NULL, 0444);
37
38MODULE_PARM_DESC(tuner,"tuner type");
39MODULE_PARM_DESC(radio,"radio tuner type");
40MODULE_PARM_DESC(card,"card type");
41
42static unsigned int latency = UNSET;
43module_param(latency,int,0444);
44MODULE_PARM_DESC(latency,"pci latency timer");
45
30/* ------------------------------------------------------------------ */ 46/* ------------------------------------------------------------------ */
31/* board config info */ 47/* board config info */
32 48
33const struct cx88_board cx88_boards[] = { 49static const struct cx88_board cx88_boards[] = {
34 [CX88_BOARD_UNKNOWN] = { 50 [CX88_BOARD_UNKNOWN] = {
35 .name = "UNKNOWN/GENERIC", 51 .name = "UNKNOWN/GENERIC",
36 .tuner_type = UNSET, 52 .tuner_type = UNSET,
@@ -1355,12 +1371,11 @@ const struct cx88_board cx88_boards[] = {
1355 }}, 1371 }},
1356 }, 1372 },
1357}; 1373};
1358const unsigned int cx88_bcount = ARRAY_SIZE(cx88_boards);
1359 1374
1360/* ------------------------------------------------------------------ */ 1375/* ------------------------------------------------------------------ */
1361/* PCI subsystem IDs */ 1376/* PCI subsystem IDs */
1362 1377
1363struct cx88_subid cx88_subids[] = { 1378static const struct cx88_subid cx88_subids[] = {
1364 { 1379 {
1365 .subvendor = 0x0070, 1380 .subvendor = 0x0070,
1366 .subdevice = 0x3400, 1381 .subdevice = 0x3400,
@@ -1666,7 +1681,6 @@ struct cx88_subid cx88_subids[] = {
1666 .card = CX88_BOARD_ADSTECH_PTV_390, 1681 .card = CX88_BOARD_ADSTECH_PTV_390,
1667 }, 1682 },
1668}; 1683};
1669const unsigned int cx88_idcount = ARRAY_SIZE(cx88_subids);
1670 1684
1671/* ----------------------------------------------------------------------- */ 1685/* ----------------------------------------------------------------------- */
1672/* some leadtek specific stuff */ 1686/* some leadtek specific stuff */
@@ -1833,7 +1847,7 @@ static void dvico_fusionhdtv_hybrid_init(struct cx88_core *core)
1833 1847
1834/* ----------------------------------------------------------------------- */ 1848/* ----------------------------------------------------------------------- */
1835 1849
1836void cx88_card_list(struct cx88_core *core, struct pci_dev *pci) 1850static void cx88_card_list(struct cx88_core *core, struct pci_dev *pci)
1837{ 1851{
1838 int i; 1852 int i;
1839 1853
@@ -1854,12 +1868,12 @@ void cx88_card_list(struct cx88_core *core, struct pci_dev *pci)
1854 } 1868 }
1855 printk("%s: Here is a list of valid choices for the card=<n> insmod option:\n", 1869 printk("%s: Here is a list of valid choices for the card=<n> insmod option:\n",
1856 core->name); 1870 core->name);
1857 for (i = 0; i < cx88_bcount; i++) 1871 for (i = 0; i < ARRAY_SIZE(cx88_boards); i++)
1858 printk("%s: card=%d -> %s\n", 1872 printk("%s: card=%d -> %s\n",
1859 core->name, i, cx88_boards[i].name); 1873 core->name, i, cx88_boards[i].name);
1860} 1874}
1861 1875
1862void cx88_card_setup_pre_i2c(struct cx88_core *core) 1876static void cx88_card_setup_pre_i2c(struct cx88_core *core)
1863{ 1877{
1864 switch (core->boardnr) { 1878 switch (core->boardnr) {
1865 case CX88_BOARD_HAUPPAUGE_HVR1300: 1879 case CX88_BOARD_HAUPPAUGE_HVR1300:
@@ -1875,7 +1889,7 @@ void cx88_card_setup_pre_i2c(struct cx88_core *core)
1875 } 1889 }
1876} 1890}
1877 1891
1878void cx88_card_setup(struct cx88_core *core) 1892static void cx88_card_setup(struct cx88_core *core)
1879{ 1893{
1880 static u8 eeprom[256]; 1894 static u8 eeprom[256];
1881 1895
@@ -1970,6 +1984,144 @@ void cx88_card_setup(struct cx88_core *core)
1970 1984
1971/* ------------------------------------------------------------------ */ 1985/* ------------------------------------------------------------------ */
1972 1986
1987static int cx88_pci_quirks(const char *name, struct pci_dev *pci)
1988{
1989 unsigned int lat = UNSET;
1990 u8 ctrl = 0;
1991 u8 value;
1992
1993 /* check pci quirks */
1994 if (pci_pci_problems & PCIPCI_TRITON) {
1995 printk(KERN_INFO "%s: quirk: PCIPCI_TRITON -- set TBFX\n",
1996 name);
1997 ctrl |= CX88X_EN_TBFX;
1998 }
1999 if (pci_pci_problems & PCIPCI_NATOMA) {
2000 printk(KERN_INFO "%s: quirk: PCIPCI_NATOMA -- set TBFX\n",
2001 name);
2002 ctrl |= CX88X_EN_TBFX;
2003 }
2004 if (pci_pci_problems & PCIPCI_VIAETBF) {
2005 printk(KERN_INFO "%s: quirk: PCIPCI_VIAETBF -- set TBFX\n",
2006 name);
2007 ctrl |= CX88X_EN_TBFX;
2008 }
2009 if (pci_pci_problems & PCIPCI_VSFX) {
2010 printk(KERN_INFO "%s: quirk: PCIPCI_VSFX -- set VSFX\n",
2011 name);
2012 ctrl |= CX88X_EN_VSFX;
2013 }
2014#ifdef PCIPCI_ALIMAGIK
2015 if (pci_pci_problems & PCIPCI_ALIMAGIK) {
2016 printk(KERN_INFO "%s: quirk: PCIPCI_ALIMAGIK -- latency fixup\n",
2017 name);
2018 lat = 0x0A;
2019 }
2020#endif
2021
2022 /* check insmod options */
2023 if (UNSET != latency)
2024 lat = latency;
2025
2026 /* apply stuff */
2027 if (ctrl) {
2028 pci_read_config_byte(pci, CX88X_DEVCTRL, &value);
2029 value |= ctrl;
2030 pci_write_config_byte(pci, CX88X_DEVCTRL, value);
2031 }
2032 if (UNSET != lat) {
2033 printk(KERN_INFO "%s: setting pci latency timer to %d\n",
2034 name, latency);
2035 pci_write_config_byte(pci, PCI_LATENCY_TIMER, latency);
2036 }
2037 return 0;
2038}
2039
2040int cx88_get_resources(const struct cx88_core *core, struct pci_dev *pci)
2041{
2042 if (request_mem_region(pci_resource_start(pci,0),
2043 pci_resource_len(pci,0),
2044 core->name))
2045 return 0;
2046 printk(KERN_ERR
2047 "%s/%d: Can't get MMIO memory @ 0x%llx, subsystem: %04x:%04x\n",
2048 core->name, PCI_FUNC(pci->devfn),
2049 (unsigned long long)pci_resource_start(pci, 0),
2050 pci->subsystem_vendor, pci->subsystem_device);
2051 return -EBUSY;
2052}
2053
2054/* Allocate and initialize the cx88 core struct. One should hold the
2055 * devlist mutex before calling this. */
2056struct cx88_core *cx88_core_create(struct pci_dev *pci, int nr)
2057{
2058 struct cx88_core *core;
2059 int i;
2060
2061 core = kzalloc(sizeof(*core), GFP_KERNEL);
2062
2063 atomic_inc(&core->refcount);
2064 core->pci_bus = pci->bus->number;
2065 core->pci_slot = PCI_SLOT(pci->devfn);
2066 core->pci_irqmask = 0x00fc00;
2067 mutex_init(&core->lock);
2068
2069 core->nr = nr;
2070 sprintf(core->name, "cx88[%d]", core->nr);
2071 if (0 != cx88_get_resources(core, pci)) {
2072 kfree(core);
2073 return NULL;
2074 }
2075
2076 /* PCI stuff */
2077 cx88_pci_quirks(core->name, pci);
2078 core->lmmio = ioremap(pci_resource_start(pci, 0),
2079 pci_resource_len(pci, 0));
2080 core->bmmio = (u8 __iomem *)core->lmmio;
2081
2082 /* board config */
2083 core->boardnr = UNSET;
2084 if (card[core->nr] < ARRAY_SIZE(cx88_boards))
2085 core->boardnr = card[core->nr];
2086 for (i = 0; UNSET == core->boardnr && i < ARRAY_SIZE(cx88_subids); i++)
2087 if (pci->subsystem_vendor == cx88_subids[i].subvendor &&
2088 pci->subsystem_device == cx88_subids[i].subdevice)
2089 core->boardnr = cx88_subids[i].card;
2090 if (UNSET == core->boardnr) {
2091 core->boardnr = CX88_BOARD_UNKNOWN;
2092 cx88_card_list(core, pci);
2093 }
2094
2095 memcpy(&core->board, &cx88_boards[core->boardnr], sizeof(core->board));
2096
2097 printk(KERN_INFO "CORE %s: subsystem: %04x:%04x, board: %s [card=%d,%s]\n",
2098 core->name,pci->subsystem_vendor,
2099 pci->subsystem_device, core->board.name,
2100 core->boardnr, card[core->nr] == core->boardnr ?
2101 "insmod option" : "autodetected");
2102
2103 if (tuner[core->nr] != UNSET)
2104 core->board.tuner_type = tuner[core->nr];
2105 if (radio[core->nr] != UNSET)
2106 core->board.radio_type = radio[core->nr];
2107
2108 printk(KERN_INFO "TV tuner %d at 0x%02x, Radio tuner %d at 0x%02x\n",
2109 core->board.tuner_type, core->board.tuner_addr<<1,
2110 core->board.radio_type, core->board.radio_addr<<1);
2111
2112 /* init hardware */
2113 cx88_reset(core);
2114 cx88_card_setup_pre_i2c(core);
2115 cx88_i2c_init(core, pci);
2116 cx88_call_i2c_clients (core, TUNER_SET_STANDBY, NULL);
2117 cx88_card_setup(core);
2118 cx88_ir_init(core, pci);
2119
2120 return core;
2121}
2122
2123/* ------------------------------------------------------------------ */
2124
1973/* 2125/*
1974 * Local variables: 2126 * Local variables:
1975 * c-basic-offset: 8 2127 * c-basic-offset: 8
diff --git a/drivers/media/video/cx88/cx88-core.c b/drivers/media/video/cx88/cx88-core.c
index ece788bd3d55..0765e9db5339 100644
--- a/drivers/media/video/cx88/cx88-core.c
+++ b/drivers/media/video/cx88/cx88-core.c
@@ -52,22 +52,6 @@ static unsigned int core_debug = 0;
52module_param(core_debug,int,0644); 52module_param(core_debug,int,0644);
53MODULE_PARM_DESC(core_debug,"enable debug messages [core]"); 53MODULE_PARM_DESC(core_debug,"enable debug messages [core]");
54 54
55static unsigned int latency = UNSET;
56module_param(latency,int,0444);
57MODULE_PARM_DESC(latency,"pci latency timer");
58
59static unsigned int tuner[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
60static unsigned int radio[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
61static unsigned int card[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
62
63module_param_array(tuner, int, NULL, 0444);
64module_param_array(radio, int, NULL, 0444);
65module_param_array(card, int, NULL, 0444);
66
67MODULE_PARM_DESC(tuner,"tuner type");
68MODULE_PARM_DESC(radio,"radio tuner type");
69MODULE_PARM_DESC(card,"card type");
70
71static unsigned int nicam = 0; 55static unsigned int nicam = 0;
72module_param(nicam,int,0644); 56module_param(nicam,int,0644);
73MODULE_PARM_DESC(nicam,"tv audio is nicam"); 57MODULE_PARM_DESC(nicam,"tv audio is nicam");
@@ -997,61 +981,6 @@ int cx88_set_tvnorm(struct cx88_core *core, v4l2_std_id norm)
997 981
998/* ------------------------------------------------------------------ */ 982/* ------------------------------------------------------------------ */
999 983
1000static int cx88_pci_quirks(char *name, struct pci_dev *pci)
1001{
1002 unsigned int lat = UNSET;
1003 u8 ctrl = 0;
1004 u8 value;
1005
1006 /* check pci quirks */
1007 if (pci_pci_problems & PCIPCI_TRITON) {
1008 printk(KERN_INFO "%s: quirk: PCIPCI_TRITON -- set TBFX\n",
1009 name);
1010 ctrl |= CX88X_EN_TBFX;
1011 }
1012 if (pci_pci_problems & PCIPCI_NATOMA) {
1013 printk(KERN_INFO "%s: quirk: PCIPCI_NATOMA -- set TBFX\n",
1014 name);
1015 ctrl |= CX88X_EN_TBFX;
1016 }
1017 if (pci_pci_problems & PCIPCI_VIAETBF) {
1018 printk(KERN_INFO "%s: quirk: PCIPCI_VIAETBF -- set TBFX\n",
1019 name);
1020 ctrl |= CX88X_EN_TBFX;
1021 }
1022 if (pci_pci_problems & PCIPCI_VSFX) {
1023 printk(KERN_INFO "%s: quirk: PCIPCI_VSFX -- set VSFX\n",
1024 name);
1025 ctrl |= CX88X_EN_VSFX;
1026 }
1027#ifdef PCIPCI_ALIMAGIK
1028 if (pci_pci_problems & PCIPCI_ALIMAGIK) {
1029 printk(KERN_INFO "%s: quirk: PCIPCI_ALIMAGIK -- latency fixup\n",
1030 name);
1031 lat = 0x0A;
1032 }
1033#endif
1034
1035 /* check insmod options */
1036 if (UNSET != latency)
1037 lat = latency;
1038
1039 /* apply stuff */
1040 if (ctrl) {
1041 pci_read_config_byte(pci, CX88X_DEVCTRL, &value);
1042 value |= ctrl;
1043 pci_write_config_byte(pci, CX88X_DEVCTRL, value);
1044 }
1045 if (UNSET != lat) {
1046 printk(KERN_INFO "%s: setting pci latency timer to %d\n",
1047 name, latency);
1048 pci_write_config_byte(pci, PCI_LATENCY_TIMER, latency);
1049 }
1050 return 0;
1051}
1052
1053/* ------------------------------------------------------------------ */
1054
1055struct video_device *cx88_vdev_init(struct cx88_core *core, 984struct video_device *cx88_vdev_init(struct cx88_core *core,
1056 struct pci_dev *pci, 985 struct pci_dev *pci,
1057 struct video_device *template, 986 struct video_device *template,
@@ -1071,25 +1000,10 @@ struct video_device *cx88_vdev_init(struct cx88_core *core,
1071 return vfd; 1000 return vfd;
1072} 1001}
1073 1002
1074static int get_ressources(struct cx88_core *core, struct pci_dev *pci)
1075{
1076 if (request_mem_region(pci_resource_start(pci,0),
1077 pci_resource_len(pci,0),
1078 core->name))
1079 return 0;
1080 printk(KERN_ERR
1081 "%s/%d: Can't get MMIO memory @ 0x%llx, subsystem: %04x:%04x\n",
1082 core->name, PCI_FUNC(pci->devfn),
1083 (unsigned long long)pci_resource_start(pci, 0),
1084 pci->subsystem_vendor, pci->subsystem_device);
1085 return -EBUSY;
1086}
1087
1088struct cx88_core* cx88_core_get(struct pci_dev *pci) 1003struct cx88_core* cx88_core_get(struct pci_dev *pci)
1089{ 1004{
1090 struct cx88_core *core; 1005 struct cx88_core *core;
1091 struct list_head *item; 1006 struct list_head *item;
1092 int i;
1093 1007
1094 mutex_lock(&devlist); 1008 mutex_lock(&devlist);
1095 list_for_each(item,&cx88_devlist) { 1009 list_for_each(item,&cx88_devlist) {
@@ -1099,82 +1013,23 @@ struct cx88_core* cx88_core_get(struct pci_dev *pci)
1099 if (PCI_SLOT(pci->devfn) != core->pci_slot) 1013 if (PCI_SLOT(pci->devfn) != core->pci_slot)
1100 continue; 1014 continue;
1101 1015
1102 if (0 != get_ressources(core,pci)) 1016 if (0 != cx88_get_resources(core, pci)) {
1103 goto fail_unlock; 1017 mutex_unlock(&devlist);
1018 return NULL;
1019 }
1104 atomic_inc(&core->refcount); 1020 atomic_inc(&core->refcount);
1105 mutex_unlock(&devlist); 1021 mutex_unlock(&devlist);
1106 return core; 1022 return core;
1107 } 1023 }
1108 core = kzalloc(sizeof(*core),GFP_KERNEL);
1109 if (NULL == core)
1110 goto fail_unlock;
1111
1112 atomic_inc(&core->refcount);
1113 core->pci_bus = pci->bus->number;
1114 core->pci_slot = PCI_SLOT(pci->devfn);
1115 core->pci_irqmask = 0x00fc00;
1116 mutex_init(&core->lock);
1117
1118 core->nr = cx88_devcount++;
1119 sprintf(core->name,"cx88[%d]",core->nr);
1120 if (0 != get_ressources(core,pci)) {
1121 cx88_devcount--;
1122 goto fail_free;
1123 }
1124 list_add_tail(&core->devlist,&cx88_devlist);
1125
1126 /* PCI stuff */
1127 cx88_pci_quirks(core->name, pci);
1128 core->lmmio = ioremap(pci_resource_start(pci,0),
1129 pci_resource_len(pci,0));
1130 core->bmmio = (u8 __iomem *)core->lmmio;
1131
1132 /* board config */
1133 core->boardnr = UNSET;
1134 if (card[core->nr] < cx88_bcount)
1135 core->boardnr = card[core->nr];
1136 for (i = 0; UNSET == core->boardnr && i < cx88_idcount; i++)
1137 if (pci->subsystem_vendor == cx88_subids[i].subvendor &&
1138 pci->subsystem_device == cx88_subids[i].subdevice)
1139 core->boardnr = cx88_subids[i].card;
1140 if (UNSET == core->boardnr) {
1141 core->boardnr = CX88_BOARD_UNKNOWN;
1142 cx88_card_list(core,pci);
1143 }
1144 1024
1145 memcpy(&core->board, &cx88_boards[core->boardnr], sizeof(core->board)); 1025 core = cx88_core_create(pci, cx88_devcount);
1146 1026 if (NULL != core) {
1147 printk(KERN_INFO "CORE %s: subsystem: %04x:%04x, board: %s [card=%d,%s]\n", 1027 cx88_devcount++;
1148 core->name,pci->subsystem_vendor, 1028 list_add_tail(&core->devlist, &cx88_devlist);
1149 pci->subsystem_device, core->board.name, 1029 }
1150 core->boardnr, card[core->nr] == core->boardnr ?
1151 "insmod option" : "autodetected");
1152
1153 if (tuner[core->nr] != UNSET)
1154 core->board.tuner_type = tuner[core->nr];
1155 if (radio[core->nr] != UNSET)
1156 core->board.radio_type = radio[core->nr];
1157
1158 printk(KERN_INFO "TV tuner %d at 0x%02x, Radio tuner %d at 0x%02x\n",
1159 core->board.tuner_type, core->board.tuner_addr<<1,
1160 core->board.radio_type, core->board.radio_addr<<1);
1161
1162 /* init hardware */
1163 cx88_reset(core);
1164 cx88_card_setup_pre_i2c(core);
1165 cx88_i2c_init(core,pci);
1166 cx88_call_i2c_clients (core, TUNER_SET_STANDBY, NULL);
1167 cx88_card_setup(core);
1168 cx88_ir_init(core,pci);
1169 1030
1170 mutex_unlock(&devlist); 1031 mutex_unlock(&devlist);
1171 return core; 1032 return core;
1172
1173fail_free:
1174 kfree(core);
1175fail_unlock:
1176 mutex_unlock(&devlist);
1177 return NULL;
1178} 1033}
1179 1034
1180void cx88_core_put(struct cx88_core *core, struct pci_dev *pci) 1035void cx88_core_put(struct cx88_core *core, struct pci_dev *pci)
diff --git a/drivers/media/video/cx88/cx88.h b/drivers/media/video/cx88/cx88.h
index 3636ec26de59..80e49f986fbb 100644
--- a/drivers/media/video/cx88/cx88.h
+++ b/drivers/media/video/cx88/cx88.h
@@ -580,15 +580,9 @@ extern void cx88_call_i2c_clients(struct cx88_core *core,
580/* ----------------------------------------------------------- */ 580/* ----------------------------------------------------------- */
581/* cx88-cards.c */ 581/* cx88-cards.c */
582 582
583extern const struct cx88_board cx88_boards[]; 583extern int cx88_get_resources(const struct cx88_core *core,
584extern const unsigned int cx88_bcount; 584 struct pci_dev *pci);
585 585extern struct cx88_core *cx88_core_create(struct pci_dev *pci, int nr);
586extern struct cx88_subid cx88_subids[];
587extern const unsigned int cx88_idcount;
588
589extern void cx88_card_list(struct cx88_core *core, struct pci_dev *pci);
590extern void cx88_card_setup(struct cx88_core *core);
591extern void cx88_card_setup_pre_i2c(struct cx88_core *core);
592 586
593/* ----------------------------------------------------------- */ 587/* ----------------------------------------------------------- */
594/* cx88-tvaudio.c */ 588/* cx88-tvaudio.c */