blob: 8982d6fdf6bef0c3f72f1dfa82990963d2f107dd [file] [log] [blame]
Manu Abraham41e840b2009-12-02 21:57:10 -03001/*
2 Mantis PCI bridge driver
3 Copyright (C) 2005, 2006 Manu Abraham (abraham.manu@gmail.com)
4
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
9
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18*/
19
Manu Abrahamb3b96142009-12-04 05:41:11 -030020#include <linux/kernel.h>
Manu Abraham41e840b2009-12-02 21:57:10 -030021#include <linux/bitops.h>
Manu Abrahamb3b96142009-12-04 05:41:11 -030022
Manu Abrahamb3b96142009-12-04 05:41:11 -030023#include <linux/signal.h>
24#include <linux/sched.h>
25#include <linux/interrupt.h>
26#include <linux/pci.h>
27#include <linux/i2c.h>
Manu Abraham41e840b2009-12-02 21:57:10 -030028
29#include "dmxdev.h"
30#include "dvbdev.h"
31#include "dvb_demux.h"
32#include "dvb_frontend.h"
Manu Abrahamb3b96142009-12-04 05:41:11 -030033#include "dvb_net.h"
34
35#include "mantis_common.h"
36#include "mantis_dma.h"
37#include "mantis_ca.h"
38#include "mantis_ioc.h"
39#include "mantis_dvb.h"
Manu Abraham41e840b2009-12-02 21:57:10 -030040
Manu Abraham616f75e2009-12-04 05:33:56 -030041DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);
42
Manu Abrahamb3b96142009-12-04 05:41:11 -030043int mantis_frontend_power(struct mantis_pci *mantis, enum mantis_power power)
Manu Abraham41e840b2009-12-02 21:57:10 -030044{
Manu Abrahamb3b96142009-12-04 05:41:11 -030045 struct mantis_hwconfig *config = mantis->hwconfig;
Manu Abraham41e840b2009-12-02 21:57:10 -030046
Manu Abrahamb3b96142009-12-04 05:41:11 -030047 switch (power) {
48 case POWER_ON:
49 dprintk(MANTIS_DEBUG, 1, "Power ON");
50 gpio_set_bits(mantis, config->power, POWER_ON);
51 msleep(100);
52 gpio_set_bits(mantis, config->power, POWER_ON);
53 msleep(100);
54 break;
Manu Abraham41e840b2009-12-02 21:57:10 -030055
Manu Abrahamb3b96142009-12-04 05:41:11 -030056 case POWER_OFF:
57 dprintk(MANTIS_DEBUG, 1, "Power OFF");
58 gpio_set_bits(mantis, config->power, POWER_OFF);
59 msleep(100);
60 break;
Manu Abraham41e840b2009-12-02 21:57:10 -030061
Manu Abrahamb3b96142009-12-04 05:41:11 -030062 default:
63 dprintk(MANTIS_DEBUG, 1, "Unknown state <%02x>", power);
64 return -1;
65 }
Manu Abraham41e840b2009-12-02 21:57:10 -030066
67 return 0;
68}
Manu Abrahamb3b96142009-12-04 05:41:11 -030069EXPORT_SYMBOL_GPL(mantis_frontend_power);
Manu Abraham41e840b2009-12-02 21:57:10 -030070
Manu Abrahamb3b96142009-12-04 05:41:11 -030071void mantis_frontend_soft_reset(struct mantis_pci *mantis)
Manu Abraham41e840b2009-12-02 21:57:10 -030072{
Manu Abrahamb3b96142009-12-04 05:41:11 -030073 struct mantis_hwconfig *config = mantis->hwconfig;
74
75 dprintk(MANTIS_DEBUG, 1, "Frontend RESET");
76 gpio_set_bits(mantis, config->reset, 0);
77 msleep(100);
78 gpio_set_bits(mantis, config->reset, 0);
79 msleep(100);
80 gpio_set_bits(mantis, config->reset, 1);
81 msleep(100);
82 gpio_set_bits(mantis, config->reset, 1);
83 msleep(100);
84
85 return;
86}
87EXPORT_SYMBOL_GPL(mantis_frontend_soft_reset);
88
89static int mantis_frontend_shutdown(struct mantis_pci *mantis)
90{
91 int err;
92
93 mantis_frontend_soft_reset(mantis);
94 err = mantis_frontend_power(mantis, POWER_OFF);
95 if (err != 0) {
96 dprintk(MANTIS_ERROR, 1, "Frontend POWER OFF failed! <%d>", err);
97 return 1;
98 }
Manu Abraham41e840b2009-12-02 21:57:10 -030099
100 return 0;
101}
102
103static int mantis_dvb_start_feed(struct dvb_demux_feed *dvbdmxfeed)
104{
105 struct dvb_demux *dvbdmx = dvbdmxfeed->demux;
106 struct mantis_pci *mantis = dvbdmx->priv;
107
Manu Abrahamb3b96142009-12-04 05:41:11 -0300108 dprintk(MANTIS_DEBUG, 1, "Mantis DVB Start feed");
Manu Abraham41e840b2009-12-02 21:57:10 -0300109 if (!dvbdmx->dmx.frontend) {
Manu Abrahamb3b96142009-12-04 05:41:11 -0300110 dprintk(MANTIS_DEBUG, 1, "no frontend ?");
Manu Abraham41e840b2009-12-02 21:57:10 -0300111 return -EINVAL;
112 }
Manu Abrahamb3b96142009-12-04 05:41:11 -0300113
Manu Abraham41e840b2009-12-02 21:57:10 -0300114 mantis->feeds++;
Manu Abrahamb3b96142009-12-04 05:41:11 -0300115 dprintk(MANTIS_DEBUG, 1, "mantis start feed, feeds=%d", mantis->feeds);
Manu Abraham41e840b2009-12-02 21:57:10 -0300116
117 if (mantis->feeds == 1) {
Manu Abrahamb3b96142009-12-04 05:41:11 -0300118 dprintk(MANTIS_DEBUG, 1, "mantis start feed & dma");
Manu Abraham41e840b2009-12-02 21:57:10 -0300119 mantis_dma_start(mantis);
120 }
121
122 return mantis->feeds;
123}
124
125static int mantis_dvb_stop_feed(struct dvb_demux_feed *dvbdmxfeed)
126{
127 struct dvb_demux *dvbdmx = dvbdmxfeed->demux;
128 struct mantis_pci *mantis = dvbdmx->priv;
129
Manu Abrahamb3b96142009-12-04 05:41:11 -0300130 dprintk(MANTIS_DEBUG, 1, "Mantis DVB Stop feed");
Manu Abraham41e840b2009-12-02 21:57:10 -0300131 if (!dvbdmx->dmx.frontend) {
Manu Abrahamb3b96142009-12-04 05:41:11 -0300132 dprintk(MANTIS_DEBUG, 1, "no frontend ?");
Manu Abraham41e840b2009-12-02 21:57:10 -0300133 return -EINVAL;
134 }
Manu Abrahamb3b96142009-12-04 05:41:11 -0300135
Manu Abraham41e840b2009-12-02 21:57:10 -0300136 mantis->feeds--;
137 if (mantis->feeds == 0) {
Manu Abrahamb3b96142009-12-04 05:41:11 -0300138 dprintk(MANTIS_DEBUG, 1, "mantis stop feed and dma");
Manu Abraham41e840b2009-12-02 21:57:10 -0300139 mantis_dma_stop(mantis);
140 }
Manu Abrahamb3b96142009-12-04 05:41:11 -0300141
Manu Abraham41e840b2009-12-02 21:57:10 -0300142 return 0;
143}
144
145int __devinit mantis_dvb_init(struct mantis_pci *mantis)
146{
Manu Abrahamb3b96142009-12-04 05:41:11 -0300147 struct mantis_hwconfig *config = mantis->hwconfig;
148 int result = -1;
Manu Abraham41e840b2009-12-02 21:57:10 -0300149
Manu Abrahamb3b96142009-12-04 05:41:11 -0300150 dprintk(MANTIS_DEBUG, 1, "dvb_register_adapter");
Manu Abraham41e840b2009-12-02 21:57:10 -0300151
Manu Abrahamb3b96142009-12-04 05:41:11 -0300152 result = dvb_register_adapter(&mantis->dvb_adapter,
153 "Mantis DVB adapter",
154 THIS_MODULE,
155 &mantis->pdev->dev,
156 adapter_nr);
157
158 if (result < 0) {
159
160 dprintk(MANTIS_ERROR, 1, "Error registering adapter");
Manu Abraham41e840b2009-12-02 21:57:10 -0300161 return -ENODEV;
162 }
Manu Abrahamb3b96142009-12-04 05:41:11 -0300163
164 mantis->dvb_adapter.priv = mantis;
165 mantis->demux.dmx.capabilities = DMX_TS_FILTERING |
Manu Abraham41e840b2009-12-02 21:57:10 -0300166 DMX_SECTION_FILTERING |
167 DMX_MEMORY_BASED_FILTERING;
168
Manu Abrahamb3b96142009-12-04 05:41:11 -0300169 mantis->demux.priv = mantis;
170 mantis->demux.filternum = 256;
171 mantis->demux.feednum = 256;
172 mantis->demux.start_feed = mantis_dvb_start_feed;
173 mantis->demux.stop_feed = mantis_dvb_stop_feed;
174 mantis->demux.write_to_decoder = NULL;
175
176 dprintk(MANTIS_DEBUG, 1, "dvb_dmx_init");
177 result = dvb_dmx_init(&mantis->demux);
178 if (result < 0) {
179 dprintk(MANTIS_ERROR, 1, "dvb_dmx_init failed, ERROR=%d", result);
Manu Abraham41e840b2009-12-02 21:57:10 -0300180
181 goto err0;
182 }
Manu Abraham41e840b2009-12-02 21:57:10 -0300183
Manu Abrahamb3b96142009-12-04 05:41:11 -0300184 mantis->dmxdev.filternum = 256;
185 mantis->dmxdev.demux = &mantis->demux.dmx;
186 mantis->dmxdev.capabilities = 0;
187 dprintk(MANTIS_DEBUG, 1, "dvb_dmxdev_init");
188
189 result = dvb_dmxdev_init(&mantis->dmxdev, &mantis->dvb_adapter);
190 if (result < 0) {
191
192 dprintk(MANTIS_ERROR, 1, "dvb_dmxdev_init failed, ERROR=%d", result);
Manu Abraham41e840b2009-12-02 21:57:10 -0300193 goto err1;
194 }
Manu Abraham41e840b2009-12-02 21:57:10 -0300195
Manu Abrahamb3b96142009-12-04 05:41:11 -0300196 mantis->fe_hw.source = DMX_FRONTEND_0;
197 result = mantis->demux.dmx.add_frontend(&mantis->demux.dmx, &mantis->fe_hw);
198 if (result < 0) {
Manu Abraham41e840b2009-12-02 21:57:10 -0300199
Manu Abrahamb3b96142009-12-04 05:41:11 -0300200 dprintk(MANTIS_ERROR, 1, "dvb_dmx_init failed, ERROR=%d", result);
Manu Abraham41e840b2009-12-02 21:57:10 -0300201 goto err2;
202 }
Manu Abraham41e840b2009-12-02 21:57:10 -0300203
Manu Abrahamb3b96142009-12-04 05:41:11 -0300204 mantis->fe_mem.source = DMX_MEMORY_FE;
Manu Abrahamf5ae4f62009-12-15 08:47:21 -0300205 result = mantis->demux.dmx.add_frontend(&mantis->demux.dmx, &mantis->fe_mem);
Manu Abrahamb3b96142009-12-04 05:41:11 -0300206 if (result < 0) {
Manu Abrahamf5ae4f62009-12-15 08:47:21 -0300207 dprintk(MANTIS_ERROR, 1, "dvb_dmx_init failed, ERROR=%d", result);
Manu Abraham41e840b2009-12-02 21:57:10 -0300208 goto err3;
209 }
Manu Abraham41e840b2009-12-02 21:57:10 -0300210
Manu Abrahamb3b96142009-12-04 05:41:11 -0300211 result = mantis->demux.dmx.connect_frontend(&mantis->demux.dmx, &mantis->fe_hw);
212 if (result < 0) {
213 dprintk(MANTIS_ERROR, 1, "dvb_dmx_init failed, ERROR=%d", result);
Manu Abraham41e840b2009-12-02 21:57:10 -0300214 goto err4;
215 }
Manu Abrahamb3b96142009-12-04 05:41:11 -0300216
Manu Abraham41e840b2009-12-02 21:57:10 -0300217 dvb_net_init(&mantis->dvb_adapter, &mantis->dvbnet, &mantis->demux.dmx);
218 tasklet_init(&mantis->tasklet, mantis_dma_xfer, (unsigned long) mantis);
Manu Abrahamb3b96142009-12-04 05:41:11 -0300219 if (mantis->hwconfig) {
220 result = config->frontend_init(mantis, mantis->fe);
221 if (result < 0) {
222 dprintk(MANTIS_ERROR, 1, "!!! NO Frontends found !!!");
223 goto err5;
224 } else {
Manu Abrahamb3b96142009-12-04 05:41:11 -0300225 if (mantis->fe == NULL) {
226 dprintk(MANTIS_ERROR, 1, "FE <NULL>");
227 goto err5;
228 }
229
230 if (dvb_register_frontend(&mantis->dvb_adapter, mantis->fe)) {
231 dprintk(MANTIS_ERROR, 1, "ERROR: Frontend registration failed");
232
233 if (mantis->fe->ops.release)
234 mantis->fe->ops.release(mantis->fe);
235
236 mantis->fe = NULL;
237 goto err5;
238 }
239 }
240 }
Manu Abrahamd9dd5f72009-12-04 05:07:41 -0300241
Manu Abraham41e840b2009-12-02 21:57:10 -0300242 return 0;
243
Manu Abrahamb3b96142009-12-04 05:41:11 -0300244 /* Error conditions .. */
245err5:
246 tasklet_kill(&mantis->tasklet);
247 dvb_net_release(&mantis->dvbnet);
Manu Abraham68fe2552009-12-04 09:01:35 -0300248 dvb_unregister_frontend(mantis->fe);
249 dvb_frontend_detach(mantis->fe);
Manu Abraham41e840b2009-12-02 21:57:10 -0300250err4:
251 mantis->demux.dmx.remove_frontend(&mantis->demux.dmx, &mantis->fe_mem);
Manu Abraham3e978a82009-12-04 05:56:35 -0300252
Manu Abraham41e840b2009-12-02 21:57:10 -0300253err3:
254 mantis->demux.dmx.remove_frontend(&mantis->demux.dmx, &mantis->fe_hw);
Manu Abraham3e978a82009-12-04 05:56:35 -0300255
Manu Abraham41e840b2009-12-02 21:57:10 -0300256err2:
257 dvb_dmxdev_release(&mantis->dmxdev);
Manu Abraham3e978a82009-12-04 05:56:35 -0300258
Manu Abraham41e840b2009-12-02 21:57:10 -0300259err1:
260 dvb_dmx_release(&mantis->demux);
Manu Abraham3e978a82009-12-04 05:56:35 -0300261
Manu Abraham41e840b2009-12-02 21:57:10 -0300262err0:
263 dvb_unregister_adapter(&mantis->dvb_adapter);
264
265 return result;
266}
Manu Abrahamb3b96142009-12-04 05:41:11 -0300267EXPORT_SYMBOL_GPL(mantis_dvb_init);
Manu Abraham41e840b2009-12-02 21:57:10 -0300268
269int __devexit mantis_dvb_exit(struct mantis_pci *mantis)
270{
Manu Abrahamb3b96142009-12-04 05:41:11 -0300271 int err;
272
Manu Abraham3e978a82009-12-04 05:56:35 -0300273 if (mantis->fe) {
Manu Abrahamf5ae4f62009-12-15 08:47:21 -0300274 /* mantis_ca_exit(mantis); */
Manu Abraham3e978a82009-12-04 05:56:35 -0300275 err = mantis_frontend_shutdown(mantis);
276 if (err != 0)
277 dprintk(MANTIS_ERROR, 1, "Frontend exit while POWER ON! <%d>", err);
Manu Abraham3e978a82009-12-04 05:56:35 -0300278 dvb_unregister_frontend(mantis->fe);
Manu Abrahamc5e598a2009-12-09 19:59:26 -0300279 dvb_frontend_detach(mantis->fe);
Manu Abraham3e978a82009-12-04 05:56:35 -0300280 }
281
Manu Abraham41e840b2009-12-02 21:57:10 -0300282 tasklet_kill(&mantis->tasklet);
283 dvb_net_release(&mantis->dvbnet);
Manu Abraham3e978a82009-12-04 05:56:35 -0300284
Manu Abraham41e840b2009-12-02 21:57:10 -0300285 mantis->demux.dmx.remove_frontend(&mantis->demux.dmx, &mantis->fe_mem);
286 mantis->demux.dmx.remove_frontend(&mantis->demux.dmx, &mantis->fe_hw);
Manu Abraham3e978a82009-12-04 05:56:35 -0300287
Manu Abraham41e840b2009-12-02 21:57:10 -0300288 dvb_dmxdev_release(&mantis->dmxdev);
289 dvb_dmx_release(&mantis->demux);
290
Manu Abrahamb3b96142009-12-04 05:41:11 -0300291 dprintk(MANTIS_DEBUG, 1, "dvb_unregister_adapter");
Manu Abraham41e840b2009-12-02 21:57:10 -0300292 dvb_unregister_adapter(&mantis->dvb_adapter);
293
294 return 0;
295}
Manu Abrahamb3b96142009-12-04 05:41:11 -0300296EXPORT_SYMBOL_GPL(mantis_dvb_exit);