xref: /linux/drivers/media/usb/as102/as102_drv.c (revision 58e16d792a6a8c6b750f637a4649967fcac853dc)
1*3e0a4e85SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
241b44e04SPierrick Hascoet /*
341b44e04SPierrick Hascoet  * Abilis Systems Single DVB-T Receiver
441b44e04SPierrick Hascoet  * Copyright (C) 2008 Pierrick Hascoet <pierrick.hascoet@abilis.com>
5c6ccdca9SDevin Heitmueller  * Copyright (C) 2010 Devin Heitmueller <dheitmueller@kernellabs.com>
641b44e04SPierrick Hascoet  */
741b44e04SPierrick Hascoet #include <linux/kernel.h>
841b44e04SPierrick Hascoet #include <linux/errno.h>
941b44e04SPierrick Hascoet #include <linux/slab.h>
1041b44e04SPierrick Hascoet #include <linux/module.h>
1141b44e04SPierrick Hascoet #include <linux/mm.h>
1241b44e04SPierrick Hascoet #include <linux/kref.h>
133288e201SSylwester Nawrocki #include <linux/uaccess.h>
1441b44e04SPierrick Hascoet #include <linux/usb.h>
1541b44e04SPierrick Hascoet 
16b47acf2aSJustin P. Mattock /* header file for usb device driver*/
1741b44e04SPierrick Hascoet #include "as102_drv.h"
1847f79129SMauro Carvalho Chehab #include "as10x_cmd.h"
1947f79129SMauro Carvalho Chehab #include "as102_fe.h"
2041b44e04SPierrick Hascoet #include "as102_fw.h"
21fada1935SMauro Carvalho Chehab #include <media/dvbdev.h>
2241b44e04SPierrick Hascoet 
23c6ccdca9SDevin Heitmueller int dual_tuner;
2441b44e04SPierrick Hascoet module_param_named(dual_tuner, dual_tuner, int, 0644);
25c6ccdca9SDevin Heitmueller MODULE_PARM_DESC(dual_tuner, "Activate Dual-Tuner config (default: off)");
2641b44e04SPierrick Hascoet 
2741b44e04SPierrick Hascoet static int fw_upload = 1;
2841b44e04SPierrick Hascoet module_param_named(fw_upload, fw_upload, int, 0644);
2941b44e04SPierrick Hascoet MODULE_PARM_DESC(fw_upload, "Turn on/off default FW upload (default: on)");
3041b44e04SPierrick Hascoet 
31c6ccdca9SDevin Heitmueller static int pid_filtering;
3241b44e04SPierrick Hascoet module_param_named(pid_filtering, pid_filtering, int, 0644);
3341b44e04SPierrick Hascoet MODULE_PARM_DESC(pid_filtering, "Activate HW PID filtering (default: off)");
3441b44e04SPierrick Hascoet 
35c6ccdca9SDevin Heitmueller static int ts_auto_disable;
3641b44e04SPierrick Hascoet module_param_named(ts_auto_disable, ts_auto_disable, int, 0644);
3741b44e04SPierrick Hascoet MODULE_PARM_DESC(ts_auto_disable, "Stream Auto Enable on FW (default: off)");
3841b44e04SPierrick Hascoet 
3941b44e04SPierrick Hascoet int elna_enable = 1;
4041b44e04SPierrick Hascoet module_param_named(elna_enable, elna_enable, int, 0644);
4141b44e04SPierrick Hascoet MODULE_PARM_DESC(elna_enable, "Activate eLNA (default: on)");
4241b44e04SPierrick Hascoet 
4341b44e04SPierrick Hascoet DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);
4441b44e04SPierrick Hascoet 
as102_stop_stream(struct as102_dev_t * dev)45c6ccdca9SDevin Heitmueller static void as102_stop_stream(struct as102_dev_t *dev)
46c6ccdca9SDevin Heitmueller {
4734490a0aSSylwester Nawrocki 	struct as10x_bus_adapter_t *bus_adap;
4841b44e04SPierrick Hascoet 
4941b44e04SPierrick Hascoet 	if (dev != NULL)
5041b44e04SPierrick Hascoet 		bus_adap = &dev->bus_adap;
5141b44e04SPierrick Hascoet 	else
5241b44e04SPierrick Hascoet 		return;
5341b44e04SPierrick Hascoet 
5441b44e04SPierrick Hascoet 	if (bus_adap->ops->stop_stream != NULL)
5541b44e04SPierrick Hascoet 		bus_adap->ops->stop_stream(dev);
5641b44e04SPierrick Hascoet 
5741b44e04SPierrick Hascoet 	if (ts_auto_disable) {
5841b44e04SPierrick Hascoet 		if (mutex_lock_interruptible(&dev->bus_adap.lock))
5941b44e04SPierrick Hascoet 			return;
6041b44e04SPierrick Hascoet 
61c6ccdca9SDevin Heitmueller 		if (as10x_cmd_stop_streaming(bus_adap) < 0)
622179de60SMartin Kepplinger 			dev_dbg(&dev->bus_adap.usb_dev->dev,
632179de60SMartin Kepplinger 				"as10x_cmd_stop_streaming failed\n");
6441b44e04SPierrick Hascoet 
6541b44e04SPierrick Hascoet 		mutex_unlock(&dev->bus_adap.lock);
6641b44e04SPierrick Hascoet 	}
6741b44e04SPierrick Hascoet }
6841b44e04SPierrick Hascoet 
as102_start_stream(struct as102_dev_t * dev)69c6ccdca9SDevin Heitmueller static int as102_start_stream(struct as102_dev_t *dev)
70c6ccdca9SDevin Heitmueller {
7134490a0aSSylwester Nawrocki 	struct as10x_bus_adapter_t *bus_adap;
7241b44e04SPierrick Hascoet 	int ret = -EFAULT;
7341b44e04SPierrick Hascoet 
7441b44e04SPierrick Hascoet 	if (dev != NULL)
7541b44e04SPierrick Hascoet 		bus_adap = &dev->bus_adap;
7641b44e04SPierrick Hascoet 	else
7741b44e04SPierrick Hascoet 		return ret;
7841b44e04SPierrick Hascoet 
79c6ccdca9SDevin Heitmueller 	if (bus_adap->ops->start_stream != NULL)
8041b44e04SPierrick Hascoet 		ret = bus_adap->ops->start_stream(dev);
8141b44e04SPierrick Hascoet 
8241b44e04SPierrick Hascoet 	if (ts_auto_disable) {
8341b44e04SPierrick Hascoet 		if (mutex_lock_interruptible(&dev->bus_adap.lock))
8441b44e04SPierrick Hascoet 			return -EFAULT;
8541b44e04SPierrick Hascoet 
8641b44e04SPierrick Hascoet 		ret = as10x_cmd_start_streaming(bus_adap);
8741b44e04SPierrick Hascoet 
8841b44e04SPierrick Hascoet 		mutex_unlock(&dev->bus_adap.lock);
8941b44e04SPierrick Hascoet 	}
9041b44e04SPierrick Hascoet 
9141b44e04SPierrick Hascoet 	return ret;
9241b44e04SPierrick Hascoet }
9341b44e04SPierrick Hascoet 
as10x_pid_filter(struct as102_dev_t * dev,int index,u16 pid,int onoff)9441b44e04SPierrick Hascoet static int as10x_pid_filter(struct as102_dev_t *dev,
9541b44e04SPierrick Hascoet 			    int index, u16 pid, int onoff) {
9641b44e04SPierrick Hascoet 
9734490a0aSSylwester Nawrocki 	struct as10x_bus_adapter_t *bus_adap = &dev->bus_adap;
9841b44e04SPierrick Hascoet 	int ret = -EFAULT;
9941b44e04SPierrick Hascoet 
10041b44e04SPierrick Hascoet 	if (mutex_lock_interruptible(&dev->bus_adap.lock)) {
1012179de60SMartin Kepplinger 		dev_dbg(&dev->bus_adap.usb_dev->dev,
1022179de60SMartin Kepplinger 			"amutex_lock_interruptible(lock) failed !\n");
10341b44e04SPierrick Hascoet 		return -EBUSY;
10441b44e04SPierrick Hascoet 	}
10541b44e04SPierrick Hascoet 
10641b44e04SPierrick Hascoet 	switch (onoff) {
10741b44e04SPierrick Hascoet 	case 0:
10841b44e04SPierrick Hascoet 		ret = as10x_cmd_del_PID_filter(bus_adap, (uint16_t) pid);
1092179de60SMartin Kepplinger 		dev_dbg(&dev->bus_adap.usb_dev->dev,
1102179de60SMartin Kepplinger 			"DEL_PID_FILTER([%02d] 0x%04x) ret = %d\n",
11141b44e04SPierrick Hascoet 			index, pid, ret);
11241b44e04SPierrick Hascoet 		break;
11341b44e04SPierrick Hascoet 	case 1:
11441b44e04SPierrick Hascoet 	{
11541b44e04SPierrick Hascoet 		struct as10x_ts_filter filter;
11641b44e04SPierrick Hascoet 
11741b44e04SPierrick Hascoet 		filter.type = TS_PID_TYPE_TS;
11841b44e04SPierrick Hascoet 		filter.idx = 0xFF;
11941b44e04SPierrick Hascoet 		filter.pid = pid;
12041b44e04SPierrick Hascoet 
12141b44e04SPierrick Hascoet 		ret = as10x_cmd_add_PID_filter(bus_adap, &filter);
1222179de60SMartin Kepplinger 		dev_dbg(&dev->bus_adap.usb_dev->dev,
123f52e9828SMonam Agarwal 			"ADD_PID_FILTER([%02d -> %02d], 0x%04x) ret = %d\n",
12441b44e04SPierrick Hascoet 			index, filter.idx, filter.pid, ret);
12541b44e04SPierrick Hascoet 		break;
12641b44e04SPierrick Hascoet 	}
12741b44e04SPierrick Hascoet 	}
12841b44e04SPierrick Hascoet 
12941b44e04SPierrick Hascoet 	mutex_unlock(&dev->bus_adap.lock);
13041b44e04SPierrick Hascoet 	return ret;
13141b44e04SPierrick Hascoet }
13241b44e04SPierrick Hascoet 
as102_dvb_dmx_start_feed(struct dvb_demux_feed * dvbdmxfeed)133c6ccdca9SDevin Heitmueller static int as102_dvb_dmx_start_feed(struct dvb_demux_feed *dvbdmxfeed)
134c6ccdca9SDevin Heitmueller {
13541b44e04SPierrick Hascoet 	int ret = 0;
13641b44e04SPierrick Hascoet 	struct dvb_demux *demux = dvbdmxfeed->demux;
13741b44e04SPierrick Hascoet 	struct as102_dev_t *as102_dev = demux->priv;
13841b44e04SPierrick Hascoet 
13941b44e04SPierrick Hascoet 	if (mutex_lock_interruptible(&as102_dev->sem))
14041b44e04SPierrick Hascoet 		return -ERESTARTSYS;
14141b44e04SPierrick Hascoet 
14214e0e4bfSSylwester Nawrocki 	if (pid_filtering)
14314e0e4bfSSylwester Nawrocki 		as10x_pid_filter(as102_dev, dvbdmxfeed->index,
14414e0e4bfSSylwester Nawrocki 				 dvbdmxfeed->pid, 1);
14541b44e04SPierrick Hascoet 
146c6ccdca9SDevin Heitmueller 	if (as102_dev->streaming++ == 0)
14741b44e04SPierrick Hascoet 		ret = as102_start_stream(as102_dev);
14841b44e04SPierrick Hascoet 
14941b44e04SPierrick Hascoet 	mutex_unlock(&as102_dev->sem);
15041b44e04SPierrick Hascoet 	return ret;
15141b44e04SPierrick Hascoet }
15241b44e04SPierrick Hascoet 
as102_dvb_dmx_stop_feed(struct dvb_demux_feed * dvbdmxfeed)153c6ccdca9SDevin Heitmueller static int as102_dvb_dmx_stop_feed(struct dvb_demux_feed *dvbdmxfeed)
154c6ccdca9SDevin Heitmueller {
15541b44e04SPierrick Hascoet 	struct dvb_demux *demux = dvbdmxfeed->demux;
15641b44e04SPierrick Hascoet 	struct as102_dev_t *as102_dev = demux->priv;
15741b44e04SPierrick Hascoet 
15841b44e04SPierrick Hascoet 	if (mutex_lock_interruptible(&as102_dev->sem))
15941b44e04SPierrick Hascoet 		return -ERESTARTSYS;
16041b44e04SPierrick Hascoet 
161c6ccdca9SDevin Heitmueller 	if (--as102_dev->streaming == 0)
16241b44e04SPierrick Hascoet 		as102_stop_stream(as102_dev);
16341b44e04SPierrick Hascoet 
16414e0e4bfSSylwester Nawrocki 	if (pid_filtering)
16514e0e4bfSSylwester Nawrocki 		as10x_pid_filter(as102_dev, dvbdmxfeed->index,
16614e0e4bfSSylwester Nawrocki 				 dvbdmxfeed->pid, 0);
16741b44e04SPierrick Hascoet 
16841b44e04SPierrick Hascoet 	mutex_unlock(&as102_dev->sem);
16941b44e04SPierrick Hascoet 	return 0;
17041b44e04SPierrick Hascoet }
17141b44e04SPierrick Hascoet 
as102_set_tune(void * priv,struct as10x_tune_args * tune_args)17247f79129SMauro Carvalho Chehab static int as102_set_tune(void *priv, struct as10x_tune_args *tune_args)
17347f79129SMauro Carvalho Chehab {
17447f79129SMauro Carvalho Chehab 	struct as10x_bus_adapter_t *bus_adap = priv;
17547f79129SMauro Carvalho Chehab 	int ret;
17647f79129SMauro Carvalho Chehab 
17747f79129SMauro Carvalho Chehab 	/* Set frontend arguments */
17847f79129SMauro Carvalho Chehab 	if (mutex_lock_interruptible(&bus_adap->lock))
17947f79129SMauro Carvalho Chehab 		return -EBUSY;
18047f79129SMauro Carvalho Chehab 
18147f79129SMauro Carvalho Chehab 	ret =  as10x_cmd_set_tune(bus_adap, tune_args);
18247f79129SMauro Carvalho Chehab 	if (ret != 0)
18347f79129SMauro Carvalho Chehab 		dev_dbg(&bus_adap->usb_dev->dev,
18447f79129SMauro Carvalho Chehab 			"as10x_cmd_set_tune failed. (err = %d)\n", ret);
18547f79129SMauro Carvalho Chehab 
18647f79129SMauro Carvalho Chehab 	mutex_unlock(&bus_adap->lock);
18747f79129SMauro Carvalho Chehab 
18847f79129SMauro Carvalho Chehab 	return ret;
18947f79129SMauro Carvalho Chehab }
19047f79129SMauro Carvalho Chehab 
as102_get_tps(void * priv,struct as10x_tps * tps)19147f79129SMauro Carvalho Chehab static int as102_get_tps(void *priv, struct as10x_tps *tps)
19247f79129SMauro Carvalho Chehab {
19347f79129SMauro Carvalho Chehab 	struct as10x_bus_adapter_t *bus_adap = priv;
19447f79129SMauro Carvalho Chehab 	int ret;
19547f79129SMauro Carvalho Chehab 
19647f79129SMauro Carvalho Chehab 	if (mutex_lock_interruptible(&bus_adap->lock))
19747f79129SMauro Carvalho Chehab 		return -EBUSY;
19847f79129SMauro Carvalho Chehab 
19947f79129SMauro Carvalho Chehab 	/* send abilis command: GET_TPS */
20047f79129SMauro Carvalho Chehab 	ret = as10x_cmd_get_tps(bus_adap, tps);
20147f79129SMauro Carvalho Chehab 
20247f79129SMauro Carvalho Chehab 	mutex_unlock(&bus_adap->lock);
20347f79129SMauro Carvalho Chehab 
20447f79129SMauro Carvalho Chehab 	return ret;
20547f79129SMauro Carvalho Chehab }
20647f79129SMauro Carvalho Chehab 
as102_get_status(void * priv,struct as10x_tune_status * tstate)20747f79129SMauro Carvalho Chehab static int as102_get_status(void *priv, struct as10x_tune_status *tstate)
20847f79129SMauro Carvalho Chehab {
20947f79129SMauro Carvalho Chehab 	struct as10x_bus_adapter_t *bus_adap = priv;
21047f79129SMauro Carvalho Chehab 	int ret;
21147f79129SMauro Carvalho Chehab 
21247f79129SMauro Carvalho Chehab 	if (mutex_lock_interruptible(&bus_adap->lock))
21347f79129SMauro Carvalho Chehab 		return -EBUSY;
21447f79129SMauro Carvalho Chehab 
21547f79129SMauro Carvalho Chehab 	/* send abilis command: GET_TUNE_STATUS */
21647f79129SMauro Carvalho Chehab 	ret = as10x_cmd_get_tune_status(bus_adap, tstate);
21747f79129SMauro Carvalho Chehab 	if (ret < 0) {
21847f79129SMauro Carvalho Chehab 		dev_dbg(&bus_adap->usb_dev->dev,
21947f79129SMauro Carvalho Chehab 			"as10x_cmd_get_tune_status failed (err = %d)\n",
22047f79129SMauro Carvalho Chehab 			ret);
22147f79129SMauro Carvalho Chehab 	}
22247f79129SMauro Carvalho Chehab 
22347f79129SMauro Carvalho Chehab 	mutex_unlock(&bus_adap->lock);
22447f79129SMauro Carvalho Chehab 
22547f79129SMauro Carvalho Chehab 	return ret;
22647f79129SMauro Carvalho Chehab }
22747f79129SMauro Carvalho Chehab 
as102_get_stats(void * priv,struct as10x_demod_stats * demod_stats)22847f79129SMauro Carvalho Chehab static int as102_get_stats(void *priv, struct as10x_demod_stats *demod_stats)
22947f79129SMauro Carvalho Chehab {
23047f79129SMauro Carvalho Chehab 	struct as10x_bus_adapter_t *bus_adap = priv;
23147f79129SMauro Carvalho Chehab 	int ret;
23247f79129SMauro Carvalho Chehab 
23347f79129SMauro Carvalho Chehab 	if (mutex_lock_interruptible(&bus_adap->lock))
23447f79129SMauro Carvalho Chehab 		return -EBUSY;
23547f79129SMauro Carvalho Chehab 
23647f79129SMauro Carvalho Chehab 	/* send abilis command: GET_TUNE_STATUS */
23747f79129SMauro Carvalho Chehab 	ret = as10x_cmd_get_demod_stats(bus_adap, demod_stats);
23847f79129SMauro Carvalho Chehab 	if (ret < 0) {
23947f79129SMauro Carvalho Chehab 		dev_dbg(&bus_adap->usb_dev->dev,
24047f79129SMauro Carvalho Chehab 			"as10x_cmd_get_demod_stats failed (probably not tuned)\n");
24147f79129SMauro Carvalho Chehab 	} else {
24247f79129SMauro Carvalho Chehab 		dev_dbg(&bus_adap->usb_dev->dev,
24347f79129SMauro Carvalho Chehab 			"demod status: fc: 0x%08x, bad fc: 0x%08x, bytes corrected: 0x%08x , MER: 0x%04x\n",
24447f79129SMauro Carvalho Chehab 			demod_stats->frame_count,
24547f79129SMauro Carvalho Chehab 			demod_stats->bad_frame_count,
24647f79129SMauro Carvalho Chehab 			demod_stats->bytes_fixed_by_rs,
24747f79129SMauro Carvalho Chehab 			demod_stats->mer);
24847f79129SMauro Carvalho Chehab 	}
24947f79129SMauro Carvalho Chehab 	mutex_unlock(&bus_adap->lock);
25047f79129SMauro Carvalho Chehab 
25147f79129SMauro Carvalho Chehab 	return ret;
25247f79129SMauro Carvalho Chehab }
25347f79129SMauro Carvalho Chehab 
as102_stream_ctrl(void * priv,int acquire,uint32_t elna_cfg)25447f79129SMauro Carvalho Chehab static int as102_stream_ctrl(void *priv, int acquire, uint32_t elna_cfg)
25547f79129SMauro Carvalho Chehab {
25647f79129SMauro Carvalho Chehab 	struct as10x_bus_adapter_t *bus_adap = priv;
25747f79129SMauro Carvalho Chehab 	int ret;
25847f79129SMauro Carvalho Chehab 
25947f79129SMauro Carvalho Chehab 	if (mutex_lock_interruptible(&bus_adap->lock))
26047f79129SMauro Carvalho Chehab 		return -EBUSY;
26147f79129SMauro Carvalho Chehab 
26247f79129SMauro Carvalho Chehab 	if (acquire) {
26347f79129SMauro Carvalho Chehab 		if (elna_enable)
26447f79129SMauro Carvalho Chehab 			as10x_cmd_set_context(bus_adap,
26547f79129SMauro Carvalho Chehab 					      CONTEXT_LNA, elna_cfg);
26647f79129SMauro Carvalho Chehab 
26747f79129SMauro Carvalho Chehab 		ret = as10x_cmd_turn_on(bus_adap);
26847f79129SMauro Carvalho Chehab 	} else {
26947f79129SMauro Carvalho Chehab 		ret = as10x_cmd_turn_off(bus_adap);
27047f79129SMauro Carvalho Chehab 	}
27147f79129SMauro Carvalho Chehab 
27247f79129SMauro Carvalho Chehab 	mutex_unlock(&bus_adap->lock);
27347f79129SMauro Carvalho Chehab 
27447f79129SMauro Carvalho Chehab 	return ret;
27547f79129SMauro Carvalho Chehab }
27647f79129SMauro Carvalho Chehab 
27747f79129SMauro Carvalho Chehab static const struct as102_fe_ops as102_fe_ops = {
27847f79129SMauro Carvalho Chehab 	.set_tune = as102_set_tune,
27947f79129SMauro Carvalho Chehab 	.get_tps  = as102_get_tps,
28047f79129SMauro Carvalho Chehab 	.get_status = as102_get_status,
28147f79129SMauro Carvalho Chehab 	.get_stats = as102_get_stats,
28247f79129SMauro Carvalho Chehab 	.stream_ctrl = as102_stream_ctrl,
28347f79129SMauro Carvalho Chehab };
28447f79129SMauro Carvalho Chehab 
as102_dvb_register(struct as102_dev_t * as102_dev)285c6ccdca9SDevin Heitmueller int as102_dvb_register(struct as102_dev_t *as102_dev)
286c6ccdca9SDevin Heitmueller {
28788010289SSylwester Nawrocki 	struct device *dev = &as102_dev->bus_adap.usb_dev->dev;
28888010289SSylwester Nawrocki 	int ret;
28941b44e04SPierrick Hascoet 
29041b44e04SPierrick Hascoet 	ret = dvb_register_adapter(&as102_dev->dvb_adap,
29188010289SSylwester Nawrocki 			   as102_dev->name, THIS_MODULE,
29288010289SSylwester Nawrocki 			   dev, adapter_nr);
29341b44e04SPierrick Hascoet 	if (ret < 0) {
29488010289SSylwester Nawrocki 		dev_err(dev, "%s: dvb_register_adapter() failed: %d\n",
295c6ccdca9SDevin Heitmueller 			__func__, ret);
29688010289SSylwester Nawrocki 		return ret;
29741b44e04SPierrick Hascoet 	}
29841b44e04SPierrick Hascoet 
29941b44e04SPierrick Hascoet 	as102_dev->dvb_dmx.priv = as102_dev;
30041b44e04SPierrick Hascoet 	as102_dev->dvb_dmx.filternum = pid_filtering ? 16 : 256;
30141b44e04SPierrick Hascoet 	as102_dev->dvb_dmx.feednum = 256;
30241b44e04SPierrick Hascoet 	as102_dev->dvb_dmx.start_feed = as102_dvb_dmx_start_feed;
30341b44e04SPierrick Hascoet 	as102_dev->dvb_dmx.stop_feed = as102_dvb_dmx_stop_feed;
30441b44e04SPierrick Hascoet 
30541b44e04SPierrick Hascoet 	as102_dev->dvb_dmx.dmx.capabilities = DMX_TS_FILTERING |
30641b44e04SPierrick Hascoet 					      DMX_SECTION_FILTERING;
30741b44e04SPierrick Hascoet 
30841b44e04SPierrick Hascoet 	as102_dev->dvb_dmxdev.filternum = as102_dev->dvb_dmx.filternum;
30941b44e04SPierrick Hascoet 	as102_dev->dvb_dmxdev.demux = &as102_dev->dvb_dmx.dmx;
31041b44e04SPierrick Hascoet 	as102_dev->dvb_dmxdev.capabilities = 0;
31141b44e04SPierrick Hascoet 
312c6ccdca9SDevin Heitmueller 	ret = dvb_dmx_init(&as102_dev->dvb_dmx);
313c6ccdca9SDevin Heitmueller 	if (ret < 0) {
3141ec9a35eSSylwester Nawrocki 		dev_err(dev, "%s: dvb_dmx_init() failed: %d\n", __func__, ret);
31588010289SSylwester Nawrocki 		goto edmxinit;
31641b44e04SPierrick Hascoet 	}
31741b44e04SPierrick Hascoet 
31841b44e04SPierrick Hascoet 	ret = dvb_dmxdev_init(&as102_dev->dvb_dmxdev, &as102_dev->dvb_adap);
31941b44e04SPierrick Hascoet 	if (ret < 0) {
32088010289SSylwester Nawrocki 		dev_err(dev, "%s: dvb_dmxdev_init() failed: %d\n",
32188010289SSylwester Nawrocki 			__func__, ret);
32288010289SSylwester Nawrocki 		goto edmxdinit;
32341b44e04SPierrick Hascoet 	}
32441b44e04SPierrick Hascoet 
325b601d9a5SMauro Carvalho Chehab 	/* Attach the frontend */
326b601d9a5SMauro Carvalho Chehab 	as102_dev->dvb_fe = dvb_attach(as102_attach, as102_dev->name,
32747f79129SMauro Carvalho Chehab 				       &as102_fe_ops,
328b601d9a5SMauro Carvalho Chehab 				       &as102_dev->bus_adap,
329b601d9a5SMauro Carvalho Chehab 				       as102_dev->elna_cfg);
330b601d9a5SMauro Carvalho Chehab 	if (!as102_dev->dvb_fe) {
331bfe40b79SJulia Lawall 		ret = -ENODEV;
332b601d9a5SMauro Carvalho Chehab 		dev_err(dev, "%s: as102_attach() failed: %d",
333b601d9a5SMauro Carvalho Chehab 		    __func__, ret);
334b601d9a5SMauro Carvalho Chehab 		goto efereg;
335b601d9a5SMauro Carvalho Chehab 	}
336b601d9a5SMauro Carvalho Chehab 
337b601d9a5SMauro Carvalho Chehab 	ret =  dvb_register_frontend(&as102_dev->dvb_adap, as102_dev->dvb_fe);
33841b44e04SPierrick Hascoet 	if (ret < 0) {
33988010289SSylwester Nawrocki 		dev_err(dev, "%s: as102_dvb_register_frontend() failed: %d",
340c6ccdca9SDevin Heitmueller 		    __func__, ret);
34188010289SSylwester Nawrocki 		goto efereg;
34241b44e04SPierrick Hascoet 	}
34341b44e04SPierrick Hascoet 
34441b44e04SPierrick Hascoet 	/* init bus mutex for token locking */
34541b44e04SPierrick Hascoet 	mutex_init(&as102_dev->bus_adap.lock);
34641b44e04SPierrick Hascoet 
34741b44e04SPierrick Hascoet 	/* init start / stop stream mutex */
34841b44e04SPierrick Hascoet 	mutex_init(&as102_dev->sem);
34941b44e04SPierrick Hascoet 
35041b44e04SPierrick Hascoet 	/*
35141b44e04SPierrick Hascoet 	 * try to load as102 firmware. If firmware upload failed, we'll be
35241b44e04SPierrick Hascoet 	 * able to upload it later.
35341b44e04SPierrick Hascoet 	 */
35441b44e04SPierrick Hascoet 	if (fw_upload)
35541b44e04SPierrick Hascoet 		try_then_request_module(as102_fw_upload(&as102_dev->bus_adap),
35641b44e04SPierrick Hascoet 				"firmware_class");
35788010289SSylwester Nawrocki 
35888010289SSylwester Nawrocki 	pr_info("Registered device %s", as102_dev->name);
35988010289SSylwester Nawrocki 	return 0;
36088010289SSylwester Nawrocki 
36188010289SSylwester Nawrocki efereg:
36288010289SSylwester Nawrocki 	dvb_dmxdev_release(&as102_dev->dvb_dmxdev);
36388010289SSylwester Nawrocki edmxdinit:
36488010289SSylwester Nawrocki 	dvb_dmx_release(&as102_dev->dvb_dmx);
36588010289SSylwester Nawrocki edmxinit:
36688010289SSylwester Nawrocki 	dvb_unregister_adapter(&as102_dev->dvb_adap);
36741b44e04SPierrick Hascoet 	return ret;
36841b44e04SPierrick Hascoet }
36941b44e04SPierrick Hascoet 
as102_dvb_unregister(struct as102_dev_t * as102_dev)370c6ccdca9SDevin Heitmueller void as102_dvb_unregister(struct as102_dev_t *as102_dev)
371c6ccdca9SDevin Heitmueller {
37241b44e04SPierrick Hascoet 	/* unregister as102 frontend */
373b601d9a5SMauro Carvalho Chehab 	dvb_unregister_frontend(as102_dev->dvb_fe);
374b601d9a5SMauro Carvalho Chehab 
375b601d9a5SMauro Carvalho Chehab 	/* detach frontend */
376b601d9a5SMauro Carvalho Chehab 	dvb_frontend_detach(as102_dev->dvb_fe);
37741b44e04SPierrick Hascoet 
37841b44e04SPierrick Hascoet 	/* unregister demux device */
37941b44e04SPierrick Hascoet 	dvb_dmxdev_release(&as102_dev->dvb_dmxdev);
38041b44e04SPierrick Hascoet 	dvb_dmx_release(&as102_dev->dvb_dmx);
38141b44e04SPierrick Hascoet 
38241b44e04SPierrick Hascoet 	/* unregister dvb adapter */
38341b44e04SPierrick Hascoet 	dvb_unregister_adapter(&as102_dev->dvb_adap);
384ff7029f5SSylwester Nawrocki 
38588010289SSylwester Nawrocki 	pr_info("Unregistered device %s", as102_dev->name);
38641b44e04SPierrick Hascoet }
38741b44e04SPierrick Hascoet 
3889451df0eSGreg Kroah-Hartman module_usb_driver(as102_usb_driver);
38941b44e04SPierrick Hascoet 
39041b44e04SPierrick Hascoet /* modinfo details */
39141b44e04SPierrick Hascoet MODULE_DESCRIPTION(DRIVER_FULL_NAME);
39241b44e04SPierrick Hascoet MODULE_LICENSE("GPL");
39341b44e04SPierrick Hascoet MODULE_AUTHOR("Pierrick Hascoet <pierrick.hascoet@abilis.com>");
394