2008-09-16 04:18:09 +08:00
|
|
|
/*
|
|
|
|
* DVB USB Linux driver for Afatech AF9015 DVB-T USB2.0 receiver
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 Antti Palosaari <crope@iki.fi>
|
|
|
|
*
|
|
|
|
* Thanks to Afatech who kindly provided information.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2010-01-22 23:10:52 +08:00
|
|
|
#include <linux/hash.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2010-01-22 23:10:52 +08:00
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
#include "af9015.h"
|
|
|
|
#include "af9013.h"
|
|
|
|
#include "mt2060.h"
|
|
|
|
#include "qt1010.h"
|
|
|
|
#include "tda18271.h"
|
|
|
|
#include "mxl5005s.h"
|
2009-02-03 01:59:50 +08:00
|
|
|
#include "mc44s803.h"
|
2010-08-13 14:51:26 +08:00
|
|
|
#include "tda18218.h"
|
2010-09-10 01:59:10 +08:00
|
|
|
#include "mxl5007t.h"
|
2008-09-16 04:18:09 +08:00
|
|
|
|
2008-11-06 03:31:24 +08:00
|
|
|
static int dvb_usb_af9015_debug;
|
2008-09-16 04:18:09 +08:00
|
|
|
module_param_named(debug, dvb_usb_af9015_debug, int, 0644);
|
|
|
|
MODULE_PARM_DESC(debug, "set debugging level" DVB_USB_DEBUG_STATUS);
|
2008-11-06 03:31:24 +08:00
|
|
|
static int dvb_usb_af9015_remote;
|
2008-09-16 04:18:09 +08:00
|
|
|
module_param_named(remote, dvb_usb_af9015_remote, int, 0644);
|
|
|
|
MODULE_PARM_DESC(remote, "select remote");
|
|
|
|
DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);
|
|
|
|
|
|
|
|
static DEFINE_MUTEX(af9015_usb_mutex);
|
|
|
|
|
|
|
|
static struct af9015_config af9015_config;
|
2009-04-09 20:16:12 +08:00
|
|
|
static struct dvb_usb_device_properties af9015_properties[3];
|
2008-11-06 03:31:24 +08:00
|
|
|
static int af9015_properties_count = ARRAY_SIZE(af9015_properties);
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
static struct af9013_config af9015_af9013_config[] = {
|
|
|
|
{
|
|
|
|
.demod_address = AF9015_I2C_DEMOD,
|
|
|
|
.output_mode = AF9013_OUTPUT_MODE_USB,
|
|
|
|
.api_version = { 0, 1, 9, 0 },
|
|
|
|
.gpio[0] = AF9013_GPIO_HI,
|
|
|
|
.gpio[3] = AF9013_GPIO_TUNER_ON,
|
|
|
|
|
|
|
|
}, {
|
|
|
|
.output_mode = AF9013_OUTPUT_MODE_SERIAL,
|
|
|
|
.api_version = { 0, 1, 9, 0 },
|
|
|
|
.gpio[0] = AF9013_GPIO_TUNER_ON,
|
|
|
|
.gpio[1] = AF9013_GPIO_LO,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static int af9015_rw_udev(struct usb_device *udev, struct req_t *req)
|
|
|
|
{
|
2009-09-13 07:46:30 +08:00
|
|
|
#define BUF_LEN 63
|
|
|
|
#define REQ_HDR_LEN 8 /* send header size */
|
|
|
|
#define ACK_HDR_LEN 2 /* rece header size */
|
2008-09-16 04:18:09 +08:00
|
|
|
int act_len, ret;
|
2009-09-13 07:46:30 +08:00
|
|
|
u8 buf[BUF_LEN];
|
2008-09-16 04:18:09 +08:00
|
|
|
u8 write = 1;
|
2009-09-13 07:46:30 +08:00
|
|
|
u8 msg_len = REQ_HDR_LEN;
|
2008-09-16 04:18:09 +08:00
|
|
|
static u8 seq; /* packet sequence number */
|
|
|
|
|
|
|
|
if (mutex_lock_interruptible(&af9015_usb_mutex) < 0)
|
|
|
|
return -EAGAIN;
|
|
|
|
|
|
|
|
buf[0] = req->cmd;
|
|
|
|
buf[1] = seq++;
|
|
|
|
buf[2] = req->i2c_addr;
|
|
|
|
buf[3] = req->addr >> 8;
|
|
|
|
buf[4] = req->addr & 0xff;
|
|
|
|
buf[5] = req->mbox;
|
|
|
|
buf[6] = req->addr_len;
|
|
|
|
buf[7] = req->data_len;
|
|
|
|
|
|
|
|
switch (req->cmd) {
|
|
|
|
case GET_CONFIG:
|
|
|
|
case READ_MEMORY:
|
|
|
|
case RECONNECT_USB:
|
|
|
|
write = 0;
|
|
|
|
break;
|
|
|
|
case READ_I2C:
|
|
|
|
write = 0;
|
|
|
|
buf[2] |= 0x01; /* set I2C direction */
|
|
|
|
case WRITE_I2C:
|
|
|
|
buf[0] = READ_WRITE_I2C;
|
|
|
|
break;
|
|
|
|
case WRITE_MEMORY:
|
|
|
|
if (((req->addr & 0xff00) == 0xff00) ||
|
2009-09-13 08:25:59 +08:00
|
|
|
((req->addr & 0xff00) == 0xae00))
|
2008-09-16 04:18:09 +08:00
|
|
|
buf[0] = WRITE_VIRTUAL_MEMORY;
|
|
|
|
case WRITE_VIRTUAL_MEMORY:
|
|
|
|
case COPY_FIRMWARE:
|
|
|
|
case DOWNLOAD_FIRMWARE:
|
2009-07-28 22:54:52 +08:00
|
|
|
case BOOT:
|
2008-09-16 04:18:09 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
err("unknown command:%d", req->cmd);
|
|
|
|
ret = -1;
|
|
|
|
goto error_unlock;
|
|
|
|
}
|
|
|
|
|
2009-09-13 07:46:30 +08:00
|
|
|
/* buffer overflow check */
|
|
|
|
if ((write && (req->data_len > BUF_LEN - REQ_HDR_LEN)) ||
|
|
|
|
(!write && (req->data_len > BUF_LEN - ACK_HDR_LEN))) {
|
|
|
|
err("too much data; cmd:%d len:%d", req->cmd, req->data_len);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto error_unlock;
|
|
|
|
}
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
/* write requested */
|
|
|
|
if (write) {
|
2009-09-13 07:46:30 +08:00
|
|
|
memcpy(&buf[REQ_HDR_LEN], req->data, req->data_len);
|
2008-09-16 04:18:09 +08:00
|
|
|
msg_len += req->data_len;
|
|
|
|
}
|
2009-09-13 07:46:30 +08:00
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
deb_xfer(">>> ");
|
|
|
|
debug_dump(buf, msg_len, deb_xfer);
|
|
|
|
|
|
|
|
/* send req */
|
|
|
|
ret = usb_bulk_msg(udev, usb_sndbulkpipe(udev, 0x02), buf, msg_len,
|
2009-09-13 07:46:30 +08:00
|
|
|
&act_len, AF9015_USB_TIMEOUT);
|
2008-09-16 04:18:09 +08:00
|
|
|
if (ret)
|
|
|
|
err("bulk message failed:%d (%d/%d)", ret, msg_len, act_len);
|
|
|
|
else
|
|
|
|
if (act_len != msg_len)
|
|
|
|
ret = -1; /* all data is not send */
|
|
|
|
if (ret)
|
|
|
|
goto error_unlock;
|
|
|
|
|
|
|
|
/* no ack for those packets */
|
|
|
|
if (req->cmd == DOWNLOAD_FIRMWARE || req->cmd == RECONNECT_USB)
|
|
|
|
goto exit_unlock;
|
|
|
|
|
2009-09-13 07:46:30 +08:00
|
|
|
/* write receives seq + status = 2 bytes
|
|
|
|
read receives seq + status + data = 2 + N bytes */
|
|
|
|
msg_len = ACK_HDR_LEN;
|
|
|
|
if (!write)
|
|
|
|
msg_len += req->data_len;
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
ret = usb_bulk_msg(udev, usb_rcvbulkpipe(udev, 0x81), buf, msg_len,
|
2009-09-13 07:46:30 +08:00
|
|
|
&act_len, AF9015_USB_TIMEOUT);
|
2008-09-16 04:18:09 +08:00
|
|
|
if (ret) {
|
|
|
|
err("recv bulk message failed:%d", ret);
|
|
|
|
ret = -1;
|
|
|
|
goto error_unlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
deb_xfer("<<< ");
|
|
|
|
debug_dump(buf, act_len, deb_xfer);
|
|
|
|
|
|
|
|
/* check status */
|
|
|
|
if (buf[1]) {
|
|
|
|
err("command failed:%d", buf[1]);
|
|
|
|
ret = -1;
|
|
|
|
goto error_unlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* read request, copy returned data to return buf */
|
|
|
|
if (!write)
|
2009-09-13 07:46:30 +08:00
|
|
|
memcpy(req->data, &buf[ACK_HDR_LEN], req->data_len);
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
error_unlock:
|
|
|
|
exit_unlock:
|
|
|
|
mutex_unlock(&af9015_usb_mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_ctrl_msg(struct dvb_usb_device *d, struct req_t *req)
|
|
|
|
{
|
|
|
|
return af9015_rw_udev(d->udev, req);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_write_regs(struct dvb_usb_device *d, u16 addr, u8 *val,
|
|
|
|
u8 len)
|
|
|
|
{
|
|
|
|
struct req_t req = {WRITE_MEMORY, AF9015_I2C_DEMOD, addr, 0, 0, len,
|
|
|
|
val};
|
|
|
|
return af9015_ctrl_msg(d, &req);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_write_reg(struct dvb_usb_device *d, u16 addr, u8 val)
|
|
|
|
{
|
|
|
|
return af9015_write_regs(d, addr, &val, 1);
|
|
|
|
}
|
|
|
|
|
2010-10-23 05:45:18 +08:00
|
|
|
static int af9015_read_regs(struct dvb_usb_device *d, u16 addr, u8 *val, u8 len)
|
2008-09-16 04:18:09 +08:00
|
|
|
{
|
2010-10-23 05:45:18 +08:00
|
|
|
struct req_t req = {READ_MEMORY, AF9015_I2C_DEMOD, addr, 0, 0, len,
|
|
|
|
val};
|
2008-09-16 04:18:09 +08:00
|
|
|
return af9015_ctrl_msg(d, &req);
|
|
|
|
}
|
|
|
|
|
2010-10-23 05:45:18 +08:00
|
|
|
static int af9015_read_reg(struct dvb_usb_device *d, u16 addr, u8 *val)
|
|
|
|
{
|
|
|
|
return af9015_read_regs(d, addr, val, 1);
|
|
|
|
}
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
static int af9015_write_reg_i2c(struct dvb_usb_device *d, u8 addr, u16 reg,
|
|
|
|
u8 val)
|
|
|
|
{
|
|
|
|
struct req_t req = {WRITE_I2C, addr, reg, 1, 1, 1, &val};
|
|
|
|
|
|
|
|
if (addr == af9015_af9013_config[0].demod_address ||
|
|
|
|
addr == af9015_af9013_config[1].demod_address)
|
|
|
|
req.addr_len = 3;
|
|
|
|
|
|
|
|
return af9015_ctrl_msg(d, &req);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_read_reg_i2c(struct dvb_usb_device *d, u8 addr, u16 reg,
|
|
|
|
u8 *val)
|
|
|
|
{
|
|
|
|
struct req_t req = {READ_I2C, addr, reg, 0, 1, 1, val};
|
|
|
|
|
|
|
|
if (addr == af9015_af9013_config[0].demod_address ||
|
|
|
|
addr == af9015_af9013_config[1].demod_address)
|
|
|
|
req.addr_len = 3;
|
|
|
|
|
|
|
|
return af9015_ctrl_msg(d, &req);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_i2c_xfer(struct i2c_adapter *adap, struct i2c_msg msg[],
|
|
|
|
int num)
|
|
|
|
{
|
|
|
|
struct dvb_usb_device *d = i2c_get_adapdata(adap);
|
|
|
|
int ret = 0, i = 0;
|
|
|
|
u16 addr;
|
2010-10-08 08:46:41 +08:00
|
|
|
u8 uninitialized_var(mbox), addr_len;
|
2008-09-16 04:18:09 +08:00
|
|
|
struct req_t req;
|
|
|
|
|
|
|
|
/* TODO: implement bus lock
|
|
|
|
|
|
|
|
The bus lock is needed because there is two tuners both using same I2C-address.
|
|
|
|
Due to that the only way to select correct tuner is use demodulator I2C-gate.
|
|
|
|
|
|
|
|
................................................
|
|
|
|
. AF9015 includes integrated AF9013 demodulator.
|
|
|
|
. ____________ ____________ . ____________
|
|
|
|
.| uC | | demod | . | tuner |
|
|
|
|
.|------------| |------------| . |------------|
|
|
|
|
.| AF9015 | | AF9013/5 | . | MXL5003 |
|
|
|
|
.| |--+----I2C-------|-----/ -----|-.-----I2C-------| |
|
|
|
|
.| | | | addr 0x38 | . | addr 0xc6 |
|
|
|
|
.|____________| | |____________| . |____________|
|
|
|
|
.................|..............................
|
|
|
|
| ____________ ____________
|
|
|
|
| | demod | | tuner |
|
|
|
|
| |------------| |------------|
|
|
|
|
| | AF9013 | | MXL5003 |
|
|
|
|
+----I2C-------|-----/ -----|-------I2C-------| |
|
|
|
|
| addr 0x3a | | addr 0xc6 |
|
|
|
|
|____________| |____________|
|
|
|
|
*/
|
|
|
|
if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
|
|
|
|
return -EAGAIN;
|
|
|
|
|
|
|
|
while (i < num) {
|
|
|
|
if (msg[i].addr == af9015_af9013_config[0].demod_address ||
|
|
|
|
msg[i].addr == af9015_af9013_config[1].demod_address) {
|
|
|
|
addr = msg[i].buf[0] << 8;
|
|
|
|
addr += msg[i].buf[1];
|
|
|
|
mbox = msg[i].buf[2];
|
|
|
|
addr_len = 3;
|
|
|
|
} else {
|
|
|
|
addr = msg[i].buf[0];
|
|
|
|
addr_len = 1;
|
2010-10-08 08:46:41 +08:00
|
|
|
/* mbox is don't care in that case */
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (num > i + 1 && (msg[i+1].flags & I2C_M_RD)) {
|
2011-06-18 08:16:38 +08:00
|
|
|
if (msg[i].len > 3 || msg[i+1].len > 61) {
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
goto error;
|
|
|
|
}
|
2008-09-16 04:18:09 +08:00
|
|
|
if (msg[i].addr ==
|
|
|
|
af9015_af9013_config[0].demod_address)
|
|
|
|
req.cmd = READ_MEMORY;
|
|
|
|
else
|
|
|
|
req.cmd = READ_I2C;
|
|
|
|
req.i2c_addr = msg[i].addr;
|
|
|
|
req.addr = addr;
|
|
|
|
req.mbox = mbox;
|
|
|
|
req.addr_len = addr_len;
|
|
|
|
req.data_len = msg[i+1].len;
|
|
|
|
req.data = &msg[i+1].buf[0];
|
|
|
|
ret = af9015_ctrl_msg(d, &req);
|
|
|
|
i += 2;
|
2009-02-03 01:59:50 +08:00
|
|
|
} else if (msg[i].flags & I2C_M_RD) {
|
2011-06-18 08:16:38 +08:00
|
|
|
if (msg[i].len > 61) {
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
goto error;
|
|
|
|
}
|
2009-02-03 01:59:50 +08:00
|
|
|
if (msg[i].addr ==
|
2011-06-17 07:02:41 +08:00
|
|
|
af9015_af9013_config[0].demod_address) {
|
|
|
|
ret = -EINVAL;
|
2009-02-03 01:59:50 +08:00
|
|
|
goto error;
|
2011-06-17 07:02:41 +08:00
|
|
|
}
|
|
|
|
req.cmd = READ_I2C;
|
2009-02-03 01:59:50 +08:00
|
|
|
req.i2c_addr = msg[i].addr;
|
|
|
|
req.addr = addr;
|
|
|
|
req.mbox = mbox;
|
|
|
|
req.addr_len = addr_len;
|
|
|
|
req.data_len = msg[i].len;
|
|
|
|
req.data = &msg[i].buf[0];
|
|
|
|
ret = af9015_ctrl_msg(d, &req);
|
|
|
|
i += 1;
|
2008-09-16 04:18:09 +08:00
|
|
|
} else {
|
2011-06-18 08:16:38 +08:00
|
|
|
if (msg[i].len > 21) {
|
|
|
|
ret = -EOPNOTSUPP;
|
|
|
|
goto error;
|
|
|
|
}
|
2008-09-16 04:18:09 +08:00
|
|
|
if (msg[i].addr ==
|
|
|
|
af9015_af9013_config[0].demod_address)
|
|
|
|
req.cmd = WRITE_MEMORY;
|
|
|
|
else
|
|
|
|
req.cmd = WRITE_I2C;
|
|
|
|
req.i2c_addr = msg[i].addr;
|
|
|
|
req.addr = addr;
|
|
|
|
req.mbox = mbox;
|
|
|
|
req.addr_len = addr_len;
|
|
|
|
req.data_len = msg[i].len-addr_len;
|
|
|
|
req.data = &msg[i].buf[addr_len];
|
|
|
|
ret = af9015_ctrl_msg(d, &req);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
}
|
|
|
|
ret = i;
|
|
|
|
|
|
|
|
error:
|
|
|
|
mutex_unlock(&d->i2c_mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static u32 af9015_i2c_func(struct i2c_adapter *adapter)
|
|
|
|
{
|
|
|
|
return I2C_FUNC_I2C;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i2c_algorithm af9015_i2c_algo = {
|
|
|
|
.master_xfer = af9015_i2c_xfer,
|
|
|
|
.functionality = af9015_i2c_func,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int af9015_do_reg_bit(struct dvb_usb_device *d, u16 addr, u8 bit, u8 op)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 val, mask = 0x01;
|
|
|
|
|
|
|
|
ret = af9015_read_reg(d, addr, &val);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
mask <<= bit;
|
|
|
|
if (op) {
|
|
|
|
/* set bit */
|
|
|
|
val |= mask;
|
|
|
|
} else {
|
|
|
|
/* clear bit */
|
|
|
|
mask ^= 0xff;
|
|
|
|
val &= mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
return af9015_write_reg(d, addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_set_reg_bit(struct dvb_usb_device *d, u16 addr, u8 bit)
|
|
|
|
{
|
|
|
|
return af9015_do_reg_bit(d, addr, bit, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_clear_reg_bit(struct dvb_usb_device *d, u16 addr, u8 bit)
|
|
|
|
{
|
|
|
|
return af9015_do_reg_bit(d, addr, bit, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_init_endpoint(struct dvb_usb_device *d)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u16 frame_size;
|
|
|
|
u8 packet_size;
|
|
|
|
deb_info("%s: USB speed:%d\n", __func__, d->udev->speed);
|
|
|
|
|
2009-09-13 00:35:29 +08:00
|
|
|
/* Windows driver uses packet count 21 for USB1.1 and 348 for USB2.0.
|
|
|
|
We use smaller - about 1/4 from the original, 5 and 87. */
|
2008-09-16 04:18:09 +08:00
|
|
|
#define TS_PACKET_SIZE 188
|
|
|
|
|
2009-09-13 00:35:29 +08:00
|
|
|
#define TS_USB20_PACKET_COUNT 87
|
2008-09-16 04:18:09 +08:00
|
|
|
#define TS_USB20_FRAME_SIZE (TS_PACKET_SIZE*TS_USB20_PACKET_COUNT)
|
|
|
|
|
2009-09-13 00:35:29 +08:00
|
|
|
#define TS_USB11_PACKET_COUNT 5
|
2008-09-16 04:18:09 +08:00
|
|
|
#define TS_USB11_FRAME_SIZE (TS_PACKET_SIZE*TS_USB11_PACKET_COUNT)
|
|
|
|
|
|
|
|
#define TS_USB20_MAX_PACKET_SIZE 512
|
|
|
|
#define TS_USB11_MAX_PACKET_SIZE 64
|
|
|
|
|
|
|
|
if (d->udev->speed == USB_SPEED_FULL) {
|
|
|
|
frame_size = TS_USB11_FRAME_SIZE/4;
|
|
|
|
packet_size = TS_USB11_MAX_PACKET_SIZE/4;
|
|
|
|
} else {
|
|
|
|
frame_size = TS_USB20_FRAME_SIZE/4;
|
|
|
|
packet_size = TS_USB20_MAX_PACKET_SIZE/4;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = af9015_set_reg_bit(d, 0xd507, 2); /* assert EP4 reset */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_set_reg_bit(d, 0xd50b, 1); /* assert EP5 reset */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_clear_reg_bit(d, 0xdd11, 5); /* disable EP4 */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_clear_reg_bit(d, 0xdd11, 6); /* disable EP5 */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_set_reg_bit(d, 0xdd11, 5); /* enable EP4 */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
if (af9015_config.dual_mode) {
|
|
|
|
ret = af9015_set_reg_bit(d, 0xdd11, 6); /* enable EP5 */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
ret = af9015_clear_reg_bit(d, 0xdd13, 5); /* disable EP4 NAK */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
if (af9015_config.dual_mode) {
|
|
|
|
ret = af9015_clear_reg_bit(d, 0xdd13, 6); /* disable EP5 NAK */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
/* EP4 xfer length */
|
|
|
|
ret = af9015_write_reg(d, 0xdd88, frame_size & 0xff);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_write_reg(d, 0xdd89, frame_size >> 8);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
/* EP5 xfer length */
|
|
|
|
ret = af9015_write_reg(d, 0xdd8a, frame_size & 0xff);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_write_reg(d, 0xdd8b, frame_size >> 8);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_write_reg(d, 0xdd0c, packet_size); /* EP4 packet size */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_write_reg(d, 0xdd0d, packet_size); /* EP5 packet size */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
ret = af9015_clear_reg_bit(d, 0xd507, 2); /* negate EP4 reset */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
if (af9015_config.dual_mode) {
|
|
|
|
ret = af9015_clear_reg_bit(d, 0xd50b, 1); /* negate EP5 reset */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* enable / disable mp2if2 */
|
|
|
|
if (af9015_config.dual_mode)
|
|
|
|
ret = af9015_set_reg_bit(d, 0xd50b, 0);
|
|
|
|
else
|
|
|
|
ret = af9015_clear_reg_bit(d, 0xd50b, 0);
|
2011-03-19 06:36:42 +08:00
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
error:
|
|
|
|
if (ret)
|
|
|
|
err("endpoint init failed:%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_copy_firmware(struct dvb_usb_device *d)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 fw_params[4];
|
|
|
|
u8 val, i;
|
|
|
|
struct req_t req = {COPY_FIRMWARE, 0, 0x5100, 0, 0, sizeof(fw_params),
|
|
|
|
fw_params };
|
|
|
|
deb_info("%s:\n", __func__);
|
|
|
|
|
|
|
|
fw_params[0] = af9015_config.firmware_size >> 8;
|
|
|
|
fw_params[1] = af9015_config.firmware_size & 0xff;
|
|
|
|
fw_params[2] = af9015_config.firmware_checksum >> 8;
|
|
|
|
fw_params[3] = af9015_config.firmware_checksum & 0xff;
|
|
|
|
|
|
|
|
/* wait 2nd demodulator ready */
|
|
|
|
msleep(100);
|
|
|
|
|
2010-09-13 08:02:55 +08:00
|
|
|
ret = af9015_read_reg_i2c(d,
|
|
|
|
af9015_af9013_config[1].demod_address, 0x98be, &val);
|
2008-09-16 04:18:09 +08:00
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
else
|
|
|
|
deb_info("%s: firmware status:%02x\n", __func__, val);
|
|
|
|
|
|
|
|
if (val == 0x0c) /* fw is running, no need for download */
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
/* set I2C master clock to fast (to speed up firmware copy) */
|
|
|
|
ret = af9015_write_reg(d, 0xd416, 0x04); /* 0x04 * 400ns */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
msleep(50);
|
|
|
|
|
|
|
|
/* copy firmware */
|
|
|
|
ret = af9015_ctrl_msg(d, &req);
|
|
|
|
if (ret)
|
|
|
|
err("firmware copy cmd failed:%d", ret);
|
|
|
|
deb_info("%s: firmware copy done\n", __func__);
|
|
|
|
|
|
|
|
/* set I2C master clock back to normal */
|
|
|
|
ret = af9015_write_reg(d, 0xd416, 0x14); /* 0x14 * 400ns */
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
/* request boot firmware */
|
|
|
|
ret = af9015_write_reg_i2c(d, af9015_af9013_config[1].demod_address,
|
|
|
|
0xe205, 1);
|
|
|
|
deb_info("%s: firmware boot cmd status:%d\n", __func__, ret);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
for (i = 0; i < 15; i++) {
|
|
|
|
msleep(100);
|
|
|
|
|
|
|
|
/* check firmware status */
|
|
|
|
ret = af9015_read_reg_i2c(d,
|
|
|
|
af9015_af9013_config[1].demod_address, 0x98be, &val);
|
|
|
|
deb_info("%s: firmware status cmd status:%d fw status:%02x\n",
|
|
|
|
__func__, ret, val);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (val == 0x0c || val == 0x04) /* success or fail */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (val == 0x04) {
|
|
|
|
err("firmware did not run");
|
|
|
|
ret = -1;
|
|
|
|
} else if (val != 0x0c) {
|
|
|
|
err("firmware boot timeout");
|
|
|
|
ret = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
error:
|
|
|
|
exit:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-01-22 23:10:52 +08:00
|
|
|
/* hash (and dump) eeprom */
|
|
|
|
static int af9015_eeprom_hash(struct usb_device *udev)
|
2008-09-16 04:18:09 +08:00
|
|
|
{
|
2010-01-22 23:10:52 +08:00
|
|
|
static const unsigned int eeprom_size = 256;
|
|
|
|
unsigned int reg;
|
|
|
|
int ret;
|
|
|
|
u8 val, *eeprom;
|
|
|
|
struct req_t req = {READ_I2C, AF9015_I2C_EEPROM, 0, 0, 1, 1, &val};
|
2008-09-16 04:18:09 +08:00
|
|
|
|
2010-01-22 23:10:52 +08:00
|
|
|
eeprom = kmalloc(eeprom_size, GFP_KERNEL);
|
|
|
|
if (eeprom == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for (reg = 0; reg < eeprom_size; reg++) {
|
|
|
|
req.addr = reg;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto free;
|
|
|
|
eeprom[reg] = val;
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
2010-01-22 23:10:52 +08:00
|
|
|
|
|
|
|
if (dvb_usb_af9015_debug & 0x01)
|
|
|
|
print_hex_dump_bytes("", DUMP_PREFIX_OFFSET, eeprom,
|
|
|
|
eeprom_size);
|
|
|
|
|
|
|
|
BUG_ON(eeprom_size % 4);
|
|
|
|
|
|
|
|
af9015_config.eeprom_sum = 0;
|
|
|
|
for (reg = 0; reg < eeprom_size / sizeof(u32); reg++) {
|
|
|
|
af9015_config.eeprom_sum *= GOLDEN_RATIO_PRIME_32;
|
|
|
|
af9015_config.eeprom_sum += le32_to_cpu(((u32 *)eeprom)[reg]);
|
|
|
|
}
|
|
|
|
|
|
|
|
deb_info("%s: eeprom sum=%.8x\n", __func__, af9015_config.eeprom_sum);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
free:
|
|
|
|
kfree(eeprom);
|
|
|
|
return ret;
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_init(struct dvb_usb_device *d)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
deb_info("%s:\n", __func__);
|
|
|
|
|
2011-03-19 06:36:42 +08:00
|
|
|
/* init RC canary */
|
|
|
|
ret = af9015_write_reg(d, 0x98e9, 0xff);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
ret = af9015_init_endpoint(d);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
error:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_pid_filter_ctrl(struct dvb_usb_adapter *adap, int onoff)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
deb_info("%s: onoff:%d\n", __func__, onoff);
|
|
|
|
|
|
|
|
if (onoff)
|
|
|
|
ret = af9015_set_reg_bit(adap->dev, 0xd503, 0);
|
|
|
|
else
|
|
|
|
ret = af9015_clear_reg_bit(adap->dev, 0xd503, 0);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_pid_filter(struct dvb_usb_adapter *adap, int index, u16 pid,
|
|
|
|
int onoff)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 idx;
|
|
|
|
|
|
|
|
deb_info("%s: set pid filter, index %d, pid %x, onoff %d\n",
|
|
|
|
__func__, index, pid, onoff);
|
|
|
|
|
|
|
|
ret = af9015_write_reg(adap->dev, 0xd505, (pid & 0xff));
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
ret = af9015_write_reg(adap->dev, 0xd506, (pid >> 8));
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
idx = ((index & 0x1f) | (1 << 5));
|
|
|
|
ret = af9015_write_reg(adap->dev, 0xd504, idx);
|
|
|
|
|
|
|
|
error:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_download_firmware(struct usb_device *udev,
|
|
|
|
const struct firmware *fw)
|
|
|
|
{
|
2011-03-20 03:51:43 +08:00
|
|
|
int i, len, remaining, ret;
|
2008-09-16 04:18:09 +08:00
|
|
|
struct req_t req = {DOWNLOAD_FIRMWARE, 0, 0, 0, 0, 0, NULL};
|
|
|
|
u16 checksum = 0;
|
|
|
|
|
|
|
|
deb_info("%s:\n", __func__);
|
|
|
|
|
|
|
|
/* calc checksum */
|
|
|
|
for (i = 0; i < fw->size; i++)
|
|
|
|
checksum += fw->data[i];
|
|
|
|
|
|
|
|
af9015_config.firmware_size = fw->size;
|
|
|
|
af9015_config.firmware_checksum = checksum;
|
|
|
|
|
2011-03-20 03:51:43 +08:00
|
|
|
#define FW_ADDR 0x5100 /* firmware start address */
|
|
|
|
#define LEN_MAX 55 /* max packet size */
|
|
|
|
for (remaining = fw->size; remaining > 0; remaining -= LEN_MAX) {
|
|
|
|
len = remaining;
|
|
|
|
if (len > LEN_MAX)
|
|
|
|
len = LEN_MAX;
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
req.data_len = len;
|
2011-03-20 03:51:43 +08:00
|
|
|
req.data = (u8 *) &fw->data[fw->size - remaining];
|
|
|
|
req.addr = FW_ADDR + fw->size - remaining;
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret) {
|
2011-03-20 03:51:43 +08:00
|
|
|
err("firmware download failed:%d", ret);
|
2008-09-16 04:18:09 +08:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* firmware loaded, request boot */
|
|
|
|
req.cmd = BOOT;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret) {
|
|
|
|
err("firmware boot failed:%d", ret);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
error:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
struct af9015_rc_setup {
|
2010-01-22 23:10:54 +08:00
|
|
|
unsigned int id;
|
2010-10-13 04:22:32 +08:00
|
|
|
char *rc_codes;
|
2010-01-22 23:10:54 +08:00
|
|
|
};
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
static char *af9015_rc_setup_match(unsigned int id,
|
|
|
|
const struct af9015_rc_setup *table)
|
2010-01-22 23:10:54 +08:00
|
|
|
{
|
2010-10-13 04:22:32 +08:00
|
|
|
for (; table->rc_codes; table++)
|
2010-01-22 23:10:54 +08:00
|
|
|
if (table->id == id)
|
2010-10-13 04:22:32 +08:00
|
|
|
return table->rc_codes;
|
2010-01-22 23:10:54 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
static const struct af9015_rc_setup af9015_rc_setup_modparam[] = {
|
|
|
|
{ AF9015_REMOTE_A_LINK_DTU_M, RC_MAP_ALINK_DTU_M },
|
|
|
|
{ AF9015_REMOTE_MSI_DIGIVOX_MINI_II_V3, RC_MAP_MSI_DIGIVOX_II },
|
|
|
|
{ AF9015_REMOTE_MYGICTV_U718, RC_MAP_TOTAL_MEDIA_IN_HAND },
|
|
|
|
{ AF9015_REMOTE_DIGITTRADE_DVB_T, RC_MAP_DIGITTRADE },
|
|
|
|
{ AF9015_REMOTE_AVERMEDIA_KS, RC_MAP_AVERMEDIA_RM_KS },
|
2010-01-22 23:10:54 +08:00
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
static const struct af9015_rc_setup af9015_rc_setup_hashes[] = {
|
|
|
|
{ 0xb8feb708, RC_MAP_MSI_DIGIVOX_II },
|
|
|
|
{ 0xa3703d00, RC_MAP_ALINK_DTU_M },
|
|
|
|
{ 0x9b7dc64e, RC_MAP_TOTAL_MEDIA_IN_HAND }, /* MYGICTV U718 */
|
2011-06-13 04:25:12 +08:00
|
|
|
{ 0x5d49e3db, RC_MAP_DIGITTRADE }, /* LC-Power LC-USB-DVBT */
|
2010-01-22 23:10:54 +08:00
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
static const struct af9015_rc_setup af9015_rc_setup_usbids[] = {
|
2011-03-20 01:25:36 +08:00
|
|
|
{ (USB_VID_TERRATEC << 16) + USB_PID_TERRATEC_CINERGY_T_STICK_RC,
|
|
|
|
RC_MAP_TERRATEC_SLIM_2 },
|
2010-10-13 04:22:32 +08:00
|
|
|
{ (USB_VID_TERRATEC << 16) + USB_PID_TERRATEC_CINERGY_T_STICK_DUAL_RC,
|
|
|
|
RC_MAP_TERRATEC_SLIM },
|
|
|
|
{ (USB_VID_VISIONPLUS << 16) + USB_PID_AZUREWAVE_AD_TU700,
|
|
|
|
RC_MAP_AZUREWAVE_AD_TU700 },
|
|
|
|
{ (USB_VID_VISIONPLUS << 16) + USB_PID_TINYTWIN,
|
|
|
|
RC_MAP_AZUREWAVE_AD_TU700 },
|
|
|
|
{ (USB_VID_MSI_2 << 16) + USB_PID_MSI_DIGI_VOX_MINI_III,
|
|
|
|
RC_MAP_MSI_DIGIVOX_III },
|
2011-06-15 22:29:47 +08:00
|
|
|
{ (USB_VID_MSI_2 << 16) + USB_PID_MSI_DIGIVOX_DUO,
|
|
|
|
RC_MAP_MSI_DIGIVOX_III },
|
2010-10-13 04:22:32 +08:00
|
|
|
{ (USB_VID_LEADTEK << 16) + USB_PID_WINFAST_DTV_DONGLE_GOLD,
|
|
|
|
RC_MAP_LEADTEK_Y04G0051 },
|
|
|
|
{ (USB_VID_AVERMEDIA << 16) + USB_PID_AVERMEDIA_VOLAR_X,
|
|
|
|
RC_MAP_AVERMEDIA_M135A },
|
|
|
|
{ (USB_VID_AFATECH << 16) + USB_PID_TREKSTOR_DVBT,
|
|
|
|
RC_MAP_TREKSTOR },
|
2010-10-23 07:37:11 +08:00
|
|
|
{ (USB_VID_KWORLD_2 << 16) + USB_PID_TINYTWIN_2,
|
|
|
|
RC_MAP_DIGITALNOW_TINYTWIN },
|
2010-10-23 18:35:31 +08:00
|
|
|
{ (USB_VID_GTEK << 16) + USB_PID_TINYTWIN_3,
|
|
|
|
RC_MAP_DIGITALNOW_TINYTWIN },
|
2010-01-22 23:10:55 +08:00
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2010-01-22 23:10:53 +08:00
|
|
|
static void af9015_set_remote_config(struct usb_device *udev,
|
|
|
|
struct dvb_usb_device_properties *props)
|
|
|
|
{
|
2010-10-13 04:22:32 +08:00
|
|
|
u16 vid = le16_to_cpu(udev->descriptor.idVendor);
|
|
|
|
u16 pid = le16_to_cpu(udev->descriptor.idProduct);
|
|
|
|
|
|
|
|
/* try to load remote based module param */
|
|
|
|
props->rc.core.rc_codes = af9015_rc_setup_match(
|
|
|
|
dvb_usb_af9015_remote, af9015_rc_setup_modparam);
|
|
|
|
|
|
|
|
/* try to load remote based eeprom hash */
|
|
|
|
if (!props->rc.core.rc_codes)
|
|
|
|
props->rc.core.rc_codes = af9015_rc_setup_match(
|
|
|
|
af9015_config.eeprom_sum, af9015_rc_setup_hashes);
|
|
|
|
|
|
|
|
/* try to load remote based USB ID */
|
|
|
|
if (!props->rc.core.rc_codes)
|
|
|
|
props->rc.core.rc_codes = af9015_rc_setup_match(
|
|
|
|
(vid << 16) + pid, af9015_rc_setup_usbids);
|
|
|
|
|
|
|
|
/* try to load remote based USB iManufacturer string */
|
|
|
|
if (!props->rc.core.rc_codes && vid == USB_VID_AFATECH) {
|
|
|
|
/* Check USB manufacturer and product strings and try
|
|
|
|
to determine correct remote in case of chip vendor
|
|
|
|
reference IDs are used.
|
|
|
|
DO NOT ADD ANYTHING NEW HERE. Use hashes instead. */
|
|
|
|
char manufacturer[10];
|
|
|
|
memset(manufacturer, 0, sizeof(manufacturer));
|
|
|
|
usb_string(udev, udev->descriptor.iManufacturer,
|
|
|
|
manufacturer, sizeof(manufacturer));
|
|
|
|
if (!strcmp("MSI", manufacturer)) {
|
|
|
|
/* iManufacturer 1 MSI
|
|
|
|
iProduct 2 MSI K-VOX */
|
|
|
|
props->rc.core.rc_codes = af9015_rc_setup_match(
|
|
|
|
AF9015_REMOTE_MSI_DIGIVOX_MINI_II_V3,
|
|
|
|
af9015_rc_setup_modparam);
|
|
|
|
}
|
2010-01-22 23:10:53 +08:00
|
|
|
}
|
2010-10-23 05:45:18 +08:00
|
|
|
|
|
|
|
/* finally load "empty" just for leaving IR receiver enabled */
|
|
|
|
if (!props->rc.core.rc_codes)
|
|
|
|
props->rc.core.rc_codes = RC_MAP_EMPTY;
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
return;
|
2010-01-22 23:10:53 +08:00
|
|
|
}
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
static int af9015_read_config(struct usb_device *udev)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 val, i, offset = 0;
|
|
|
|
struct req_t req = {READ_I2C, AF9015_I2C_EEPROM, 0, 0, 1, 1, &val};
|
|
|
|
|
|
|
|
/* IR remote controller */
|
|
|
|
req.addr = AF9015_EEPROM_IR_MODE;
|
2009-01-21 01:56:20 +08:00
|
|
|
/* first message will timeout often due to possible hw bug */
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (!ret)
|
|
|
|
break;
|
|
|
|
}
|
2008-09-16 04:18:09 +08:00
|
|
|
if (ret)
|
|
|
|
goto error;
|
2010-01-22 23:10:52 +08:00
|
|
|
|
|
|
|
ret = af9015_eeprom_hash(udev);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
deb_info("%s: IR mode:%d\n", __func__, val);
|
|
|
|
for (i = 0; i < af9015_properties_count; i++) {
|
2010-10-23 05:45:18 +08:00
|
|
|
if (val == AF9015_IR_MODE_DISABLED)
|
|
|
|
af9015_properties[i].rc.core.rc_codes = NULL;
|
|
|
|
else
|
2010-01-22 23:10:53 +08:00
|
|
|
af9015_set_remote_config(udev, &af9015_properties[i]);
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* TS mode - one or two receivers */
|
|
|
|
req.addr = AF9015_EEPROM_TS_MODE;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
af9015_config.dual_mode = val;
|
|
|
|
deb_info("%s: TS mode:%d\n", __func__, af9015_config.dual_mode);
|
|
|
|
|
2009-01-14 00:08:29 +08:00
|
|
|
/* Set adapter0 buffer size according to USB port speed, adapter1 buffer
|
|
|
|
size can be static because it is enabled only USB2.0 */
|
2008-09-16 04:18:09 +08:00
|
|
|
for (i = 0; i < af9015_properties_count; i++) {
|
|
|
|
/* USB1.1 set smaller buffersize and disable 2nd adapter */
|
|
|
|
if (udev->speed == USB_SPEED_FULL) {
|
2009-01-14 00:08:29 +08:00
|
|
|
af9015_properties[i].adapter[0].stream.u.bulk.buffersize
|
2009-09-13 00:35:29 +08:00
|
|
|
= TS_USB11_FRAME_SIZE;
|
2008-09-16 04:18:09 +08:00
|
|
|
/* disable 2nd adapter because we don't have
|
|
|
|
PID-filters */
|
|
|
|
af9015_config.dual_mode = 0;
|
|
|
|
} else {
|
2009-01-14 00:08:29 +08:00
|
|
|
af9015_properties[i].adapter[0].stream.u.bulk.buffersize
|
2009-09-12 20:51:36 +08:00
|
|
|
= TS_USB20_FRAME_SIZE;
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (af9015_config.dual_mode) {
|
|
|
|
/* read 2nd demodulator I2C address */
|
|
|
|
req.addr = AF9015_EEPROM_DEMOD2_I2C;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
af9015_af9013_config[1].demod_address = val;
|
|
|
|
|
|
|
|
/* enable 2nd adapter */
|
|
|
|
for (i = 0; i < af9015_properties_count; i++)
|
|
|
|
af9015_properties[i].num_adapters = 2;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
/* disable 2nd adapter */
|
|
|
|
for (i = 0; i < af9015_properties_count; i++)
|
|
|
|
af9015_properties[i].num_adapters = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < af9015_properties[0].num_adapters; i++) {
|
|
|
|
if (i == 1)
|
|
|
|
offset = AF9015_EEPROM_OFFSET;
|
|
|
|
/* xtal */
|
|
|
|
req.addr = AF9015_EEPROM_XTAL_TYPE1 + offset;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
switch (val) {
|
|
|
|
case 0:
|
|
|
|
af9015_af9013_config[i].adc_clock = 28800;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
af9015_af9013_config[i].adc_clock = 20480;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
af9015_af9013_config[i].adc_clock = 28000;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
af9015_af9013_config[i].adc_clock = 25000;
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
deb_info("%s: [%d] xtal:%d set adc_clock:%d\n", __func__, i,
|
|
|
|
val, af9015_af9013_config[i].adc_clock);
|
|
|
|
|
|
|
|
/* tuner IF */
|
|
|
|
req.addr = AF9015_EEPROM_IF1H + offset;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
af9015_af9013_config[i].tuner_if = val << 8;
|
|
|
|
req.addr = AF9015_EEPROM_IF1L + offset;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
af9015_af9013_config[i].tuner_if += val;
|
|
|
|
deb_info("%s: [%d] IF1:%d\n", __func__, i,
|
|
|
|
af9015_af9013_config[0].tuner_if);
|
|
|
|
|
|
|
|
/* MT2060 IF1 */
|
|
|
|
req.addr = AF9015_EEPROM_MT2060_IF1H + offset;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
af9015_config.mt2060_if1[i] = val << 8;
|
|
|
|
req.addr = AF9015_EEPROM_MT2060_IF1L + offset;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
af9015_config.mt2060_if1[i] += val;
|
|
|
|
deb_info("%s: [%d] MT2060 IF1:%d\n", __func__, i,
|
|
|
|
af9015_config.mt2060_if1[i]);
|
|
|
|
|
|
|
|
/* tuner */
|
|
|
|
req.addr = AF9015_EEPROM_TUNER_ID1 + offset;
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
switch (val) {
|
|
|
|
case AF9013_TUNER_ENV77H11D5:
|
|
|
|
case AF9013_TUNER_MT2060:
|
|
|
|
case AF9013_TUNER_QT1010:
|
|
|
|
case AF9013_TUNER_UNKNOWN:
|
|
|
|
case AF9013_TUNER_MT2060_2:
|
|
|
|
case AF9013_TUNER_TDA18271:
|
|
|
|
case AF9013_TUNER_QT1010A:
|
2010-08-13 14:51:26 +08:00
|
|
|
case AF9013_TUNER_TDA18218:
|
2008-09-16 04:18:09 +08:00
|
|
|
af9015_af9013_config[i].rf_spec_inv = 1;
|
|
|
|
break;
|
|
|
|
case AF9013_TUNER_MXL5003D:
|
|
|
|
case AF9013_TUNER_MXL5005D:
|
|
|
|
case AF9013_TUNER_MXL5005R:
|
2010-09-10 01:59:10 +08:00
|
|
|
case AF9013_TUNER_MXL5007T:
|
2008-09-16 04:18:09 +08:00
|
|
|
af9015_af9013_config[i].rf_spec_inv = 0;
|
|
|
|
break;
|
2009-02-03 01:59:50 +08:00
|
|
|
case AF9013_TUNER_MC44S803:
|
|
|
|
af9015_af9013_config[i].gpio[1] = AF9013_GPIO_LO;
|
|
|
|
af9015_af9013_config[i].rf_spec_inv = 1;
|
|
|
|
break;
|
2008-09-16 04:18:09 +08:00
|
|
|
default:
|
|
|
|
warn("tuner id:%d not supported, please report!", val);
|
|
|
|
return -ENODEV;
|
|
|
|
};
|
|
|
|
|
|
|
|
af9015_af9013_config[i].tuner = val;
|
|
|
|
deb_info("%s: [%d] tuner id:%d\n", __func__, i, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
error:
|
|
|
|
if (ret)
|
|
|
|
err("eeprom read failed:%d", ret);
|
|
|
|
|
2009-04-01 04:01:02 +08:00
|
|
|
/* AverMedia AVerTV Volar Black HD (A850) device have bad EEPROM
|
2010-10-02 03:55:43 +08:00
|
|
|
content :-( Override some wrong values here. Ditto for the
|
|
|
|
AVerTV Red HD+ (A850T) device. */
|
2009-04-01 04:01:02 +08:00
|
|
|
if (le16_to_cpu(udev->descriptor.idVendor) == USB_VID_AVERMEDIA &&
|
2010-10-08 08:37:06 +08:00
|
|
|
((le16_to_cpu(udev->descriptor.idProduct) ==
|
|
|
|
USB_PID_AVERMEDIA_A850) ||
|
|
|
|
(le16_to_cpu(udev->descriptor.idProduct) ==
|
|
|
|
USB_PID_AVERMEDIA_A850T))) {
|
2009-04-01 04:01:02 +08:00
|
|
|
deb_info("%s: AverMedia A850: overriding config\n", __func__);
|
|
|
|
/* disable dual mode */
|
|
|
|
af9015_config.dual_mode = 0;
|
|
|
|
/* disable 2nd adapter */
|
|
|
|
for (i = 0; i < af9015_properties_count; i++)
|
|
|
|
af9015_properties[i].num_adapters = 1;
|
|
|
|
|
|
|
|
/* set correct IF */
|
|
|
|
af9015_af9013_config[0].tuner_if = 4570;
|
|
|
|
}
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_identify_state(struct usb_device *udev,
|
|
|
|
struct dvb_usb_device_properties *props,
|
|
|
|
struct dvb_usb_device_description **desc,
|
|
|
|
int *cold)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 reply;
|
|
|
|
struct req_t req = {GET_CONFIG, 0, 0, 0, 0, 1, &reply};
|
|
|
|
|
|
|
|
ret = af9015_rw_udev(udev, &req);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
deb_info("%s: reply:%02x\n", __func__, reply);
|
|
|
|
if (reply == 0x02)
|
|
|
|
*cold = 0;
|
|
|
|
else
|
|
|
|
*cold = 1;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
static int af9015_rc_query(struct dvb_usb_device *d)
|
2008-09-16 04:18:09 +08:00
|
|
|
{
|
2010-09-13 00:31:56 +08:00
|
|
|
struct af9015_state *priv = d->priv;
|
|
|
|
int ret;
|
2011-03-19 06:23:05 +08:00
|
|
|
u8 buf[17];
|
2008-09-16 04:18:09 +08:00
|
|
|
|
2010-09-13 00:31:56 +08:00
|
|
|
/* read registers needed to detect remote controller code */
|
2010-10-23 05:45:18 +08:00
|
|
|
ret = af9015_read_regs(d, 0x98d9, buf, sizeof(buf));
|
2010-09-13 00:31:56 +08:00
|
|
|
if (ret)
|
|
|
|
goto error;
|
2008-09-16 04:18:09 +08:00
|
|
|
|
2011-03-19 06:23:05 +08:00
|
|
|
/* If any of these are non-zero, assume invalid data */
|
|
|
|
if (buf[1] || buf[2] || buf[3])
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* Check for repeat of previous code */
|
|
|
|
if ((priv->rc_repeat != buf[6] || buf[0]) &&
|
|
|
|
!memcmp(&buf[12], priv->rc_last, 4)) {
|
|
|
|
deb_rc("%s: key repeated\n", __func__);
|
|
|
|
rc_keydown(d->rc_dev, priv->rc_keycode, 0);
|
|
|
|
priv->rc_repeat = buf[6];
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Only process key if canary killed */
|
|
|
|
if (buf[16] != 0xff && buf[0] != 0x01) {
|
2010-10-23 05:45:18 +08:00
|
|
|
deb_rc("%s: key pressed %02x %02x %02x %02x\n", __func__,
|
|
|
|
buf[12], buf[13], buf[14], buf[15]);
|
2010-09-13 00:31:56 +08:00
|
|
|
|
2011-03-19 06:36:42 +08:00
|
|
|
/* Reset the canary */
|
|
|
|
ret = af9015_write_reg(d, 0x98e9, 0xff);
|
|
|
|
if (ret)
|
|
|
|
goto error;
|
|
|
|
|
2011-03-19 06:23:05 +08:00
|
|
|
/* Remember this key */
|
|
|
|
memcpy(priv->rc_last, &buf[12], 4);
|
2010-10-23 05:45:18 +08:00
|
|
|
if (buf[14] == (u8) ~buf[15]) {
|
|
|
|
if (buf[12] == (u8) ~buf[13]) {
|
2010-10-13 04:22:32 +08:00
|
|
|
/* NEC */
|
2010-10-23 05:45:18 +08:00
|
|
|
priv->rc_keycode = buf[12] << 8 | buf[14];
|
2010-10-13 04:22:32 +08:00
|
|
|
} else {
|
|
|
|
/* NEC extended*/
|
2010-10-23 05:45:18 +08:00
|
|
|
priv->rc_keycode = buf[12] << 16 |
|
|
|
|
buf[13] << 8 | buf[14];
|
2010-10-13 04:22:32 +08:00
|
|
|
}
|
|
|
|
} else {
|
2011-03-19 06:36:42 +08:00
|
|
|
/* 32 bit NEC */
|
2011-03-19 06:23:05 +08:00
|
|
|
priv->rc_keycode = buf[12] << 24 | buf[13] << 16 |
|
|
|
|
buf[14] << 8 | buf[15];
|
2010-10-13 04:22:32 +08:00
|
|
|
}
|
2010-11-18 00:53:11 +08:00
|
|
|
rc_keydown(d->rc_dev, priv->rc_keycode, 0);
|
2010-09-13 00:31:56 +08:00
|
|
|
} else {
|
|
|
|
deb_rc("%s: no key press\n", __func__);
|
2011-03-19 06:23:05 +08:00
|
|
|
/* Invalidate last keypress */
|
|
|
|
/* Not really needed, but helps with debug */
|
|
|
|
priv->rc_last[2] = priv->rc_last[3];
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
|
|
|
|
2010-10-23 05:45:18 +08:00
|
|
|
priv->rc_repeat = buf[6];
|
2010-09-13 00:31:56 +08:00
|
|
|
|
|
|
|
error:
|
|
|
|
if (ret)
|
|
|
|
err("%s: failed:%d", __func__, ret);
|
|
|
|
|
|
|
|
return ret;
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* init 2nd I2C adapter */
|
2008-11-06 03:31:24 +08:00
|
|
|
static int af9015_i2c_init(struct dvb_usb_device *d)
|
2008-09-16 04:18:09 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct af9015_state *state = d->priv;
|
|
|
|
deb_info("%s:\n", __func__);
|
|
|
|
|
|
|
|
strncpy(state->i2c_adap.name, d->desc->name,
|
|
|
|
sizeof(state->i2c_adap.name));
|
|
|
|
state->i2c_adap.algo = d->props.i2c_algo;
|
|
|
|
state->i2c_adap.algo_data = NULL;
|
|
|
|
state->i2c_adap.dev.parent = &d->udev->dev;
|
|
|
|
|
|
|
|
i2c_set_adapdata(&state->i2c_adap, d);
|
|
|
|
|
|
|
|
ret = i2c_add_adapter(&state->i2c_adap);
|
|
|
|
if (ret < 0)
|
|
|
|
err("could not add i2c adapter");
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int af9015_af9013_frontend_attach(struct dvb_usb_adapter *adap)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct af9015_state *state = adap->dev->priv;
|
|
|
|
struct i2c_adapter *i2c_adap;
|
|
|
|
|
|
|
|
if (adap->id == 0) {
|
|
|
|
/* select I2C adapter */
|
|
|
|
i2c_adap = &adap->dev->i2c_adap;
|
|
|
|
|
|
|
|
deb_info("%s: init I2C\n", __func__);
|
|
|
|
ret = af9015_i2c_init(adap->dev);
|
|
|
|
} else {
|
|
|
|
/* select I2C adapter */
|
|
|
|
i2c_adap = &state->i2c_adap;
|
|
|
|
|
|
|
|
/* copy firmware to 2nd demodulator */
|
|
|
|
if (af9015_config.dual_mode) {
|
|
|
|
ret = af9015_copy_firmware(adap->dev);
|
|
|
|
if (ret) {
|
|
|
|
err("firmware copy to 2nd frontend " \
|
|
|
|
"failed, will disable it");
|
|
|
|
af9015_config.dual_mode = 0;
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* attach demodulator */
|
|
|
|
adap->fe = dvb_attach(af9013_attach, &af9015_af9013_config[adap->id],
|
|
|
|
i2c_adap);
|
|
|
|
|
|
|
|
return adap->fe == NULL ? -ENODEV : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct mt2060_config af9015_mt2060_config = {
|
|
|
|
.i2c_address = 0xc0,
|
|
|
|
.clock_out = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct qt1010_config af9015_qt1010_config = {
|
|
|
|
.i2c_address = 0xc4,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct tda18271_config af9015_tda18271_config = {
|
|
|
|
.gate = TDA18271_GATE_DIGITAL,
|
2010-10-28 00:55:34 +08:00
|
|
|
.small_i2c = TDA18271_16_BYTE_CHUNK_INIT,
|
2008-09-16 04:18:09 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct mxl5005s_config af9015_mxl5003_config = {
|
|
|
|
.i2c_address = 0xc6,
|
|
|
|
.if_freq = IF_FREQ_4570000HZ,
|
|
|
|
.xtal_freq = CRYSTAL_FREQ_16000000HZ,
|
|
|
|
.agc_mode = MXL_SINGLE_AGC,
|
|
|
|
.tracking_filter = MXL_TF_DEFAULT,
|
2008-09-23 00:59:25 +08:00
|
|
|
.rssi_enable = MXL_RSSI_ENABLE,
|
2008-09-16 04:18:09 +08:00
|
|
|
.cap_select = MXL_CAP_SEL_ENABLE,
|
|
|
|
.div_out = MXL_DIV_OUT_4,
|
|
|
|
.clock_out = MXL_CLOCK_OUT_DISABLE,
|
|
|
|
.output_load = MXL5005S_IF_OUTPUT_LOAD_200_OHM,
|
|
|
|
.top = MXL5005S_TOP_25P2,
|
|
|
|
.mod_mode = MXL_DIGITAL_MODE,
|
|
|
|
.if_mode = MXL_ZERO_IF,
|
|
|
|
.AgcMasterByte = 0x00,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct mxl5005s_config af9015_mxl5005_config = {
|
|
|
|
.i2c_address = 0xc6,
|
|
|
|
.if_freq = IF_FREQ_4570000HZ,
|
|
|
|
.xtal_freq = CRYSTAL_FREQ_16000000HZ,
|
|
|
|
.agc_mode = MXL_SINGLE_AGC,
|
|
|
|
.tracking_filter = MXL_TF_OFF,
|
2008-09-23 00:59:25 +08:00
|
|
|
.rssi_enable = MXL_RSSI_ENABLE,
|
2008-09-16 04:18:09 +08:00
|
|
|
.cap_select = MXL_CAP_SEL_ENABLE,
|
|
|
|
.div_out = MXL_DIV_OUT_4,
|
|
|
|
.clock_out = MXL_CLOCK_OUT_DISABLE,
|
|
|
|
.output_load = MXL5005S_IF_OUTPUT_LOAD_200_OHM,
|
|
|
|
.top = MXL5005S_TOP_25P2,
|
|
|
|
.mod_mode = MXL_DIGITAL_MODE,
|
|
|
|
.if_mode = MXL_ZERO_IF,
|
|
|
|
.AgcMasterByte = 0x00,
|
|
|
|
};
|
|
|
|
|
2009-02-03 01:59:50 +08:00
|
|
|
static struct mc44s803_config af9015_mc44s803_config = {
|
|
|
|
.i2c_address = 0xc0,
|
|
|
|
.dig_out = 1,
|
|
|
|
};
|
|
|
|
|
2010-08-13 14:51:26 +08:00
|
|
|
static struct tda18218_config af9015_tda18218_config = {
|
|
|
|
.i2c_address = 0xc0,
|
|
|
|
.i2c_wr_max = 21, /* max wr bytes AF9015 I2C adap can handle at once */
|
|
|
|
};
|
|
|
|
|
2010-09-10 01:59:10 +08:00
|
|
|
static struct mxl5007t_config af9015_mxl5007t_config = {
|
|
|
|
.xtal_freq_hz = MxL_XTAL_24_MHZ,
|
|
|
|
.if_freq_hz = MxL_IF_4_57_MHZ,
|
|
|
|
};
|
|
|
|
|
2008-09-16 04:18:09 +08:00
|
|
|
static int af9015_tuner_attach(struct dvb_usb_adapter *adap)
|
|
|
|
{
|
|
|
|
struct af9015_state *state = adap->dev->priv;
|
|
|
|
struct i2c_adapter *i2c_adap;
|
|
|
|
int ret;
|
2010-10-08 08:37:06 +08:00
|
|
|
deb_info("%s:\n", __func__);
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
/* select I2C adapter */
|
|
|
|
if (adap->id == 0)
|
|
|
|
i2c_adap = &adap->dev->i2c_adap;
|
|
|
|
else
|
|
|
|
i2c_adap = &state->i2c_adap;
|
|
|
|
|
|
|
|
switch (af9015_af9013_config[adap->id].tuner) {
|
|
|
|
case AF9013_TUNER_MT2060:
|
|
|
|
case AF9013_TUNER_MT2060_2:
|
|
|
|
ret = dvb_attach(mt2060_attach, adap->fe, i2c_adap,
|
|
|
|
&af9015_mt2060_config,
|
|
|
|
af9015_config.mt2060_if1[adap->id])
|
|
|
|
== NULL ? -ENODEV : 0;
|
|
|
|
break;
|
|
|
|
case AF9013_TUNER_QT1010:
|
|
|
|
case AF9013_TUNER_QT1010A:
|
|
|
|
ret = dvb_attach(qt1010_attach, adap->fe, i2c_adap,
|
|
|
|
&af9015_qt1010_config) == NULL ? -ENODEV : 0;
|
|
|
|
break;
|
|
|
|
case AF9013_TUNER_TDA18271:
|
|
|
|
ret = dvb_attach(tda18271_attach, adap->fe, 0xc0, i2c_adap,
|
|
|
|
&af9015_tda18271_config) == NULL ? -ENODEV : 0;
|
|
|
|
break;
|
2010-08-13 14:51:26 +08:00
|
|
|
case AF9013_TUNER_TDA18218:
|
|
|
|
ret = dvb_attach(tda18218_attach, adap->fe, i2c_adap,
|
|
|
|
&af9015_tda18218_config) == NULL ? -ENODEV : 0;
|
|
|
|
break;
|
2008-09-16 04:18:09 +08:00
|
|
|
case AF9013_TUNER_MXL5003D:
|
|
|
|
ret = dvb_attach(mxl5005s_attach, adap->fe, i2c_adap,
|
|
|
|
&af9015_mxl5003_config) == NULL ? -ENODEV : 0;
|
|
|
|
break;
|
|
|
|
case AF9013_TUNER_MXL5005D:
|
|
|
|
case AF9013_TUNER_MXL5005R:
|
|
|
|
ret = dvb_attach(mxl5005s_attach, adap->fe, i2c_adap,
|
|
|
|
&af9015_mxl5005_config) == NULL ? -ENODEV : 0;
|
|
|
|
break;
|
|
|
|
case AF9013_TUNER_ENV77H11D5:
|
|
|
|
ret = dvb_attach(dvb_pll_attach, adap->fe, 0xc0, i2c_adap,
|
|
|
|
DVB_PLL_TDA665X) == NULL ? -ENODEV : 0;
|
|
|
|
break;
|
|
|
|
case AF9013_TUNER_MC44S803:
|
2009-02-03 01:59:50 +08:00
|
|
|
ret = dvb_attach(mc44s803_attach, adap->fe, i2c_adap,
|
|
|
|
&af9015_mc44s803_config) == NULL ? -ENODEV : 0;
|
2008-09-16 04:18:09 +08:00
|
|
|
break;
|
2010-09-10 01:59:10 +08:00
|
|
|
case AF9013_TUNER_MXL5007T:
|
|
|
|
ret = dvb_attach(mxl5007t_attach, adap->fe, i2c_adap,
|
|
|
|
0xc0, &af9015_mxl5007t_config) == NULL ? -ENODEV : 0;
|
|
|
|
break;
|
2008-09-16 04:18:09 +08:00
|
|
|
case AF9013_TUNER_UNKNOWN:
|
|
|
|
default:
|
|
|
|
ret = -ENODEV;
|
|
|
|
err("Unknown tuner id:%d",
|
|
|
|
af9015_af9013_config[adap->id].tuner);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct usb_device_id af9015_usb_table[] = {
|
|
|
|
/* 0 */{USB_DEVICE(USB_VID_AFATECH, USB_PID_AFATECH_AF9015_9015)},
|
|
|
|
{USB_DEVICE(USB_VID_AFATECH, USB_PID_AFATECH_AF9015_9016)},
|
|
|
|
{USB_DEVICE(USB_VID_LEADTEK, USB_PID_WINFAST_DTV_DONGLE_GOLD)},
|
|
|
|
{USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_PCTV71E)},
|
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_399U)},
|
|
|
|
/* 5 */{USB_DEVICE(USB_VID_VISIONPLUS,
|
|
|
|
USB_PID_TINYTWIN)},
|
|
|
|
{USB_DEVICE(USB_VID_VISIONPLUS,
|
|
|
|
USB_PID_AZUREWAVE_AD_TU700)},
|
|
|
|
{USB_DEVICE(USB_VID_TERRATEC, USB_PID_TERRATEC_CINERGY_T_USB_XE_REV2)},
|
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_PC160_2T)},
|
|
|
|
{USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_VOLAR_X)},
|
|
|
|
/* 10 */{USB_DEVICE(USB_VID_XTENSIONS, USB_PID_XTENSIONS_XD_380)},
|
|
|
|
{USB_DEVICE(USB_VID_MSI_2, USB_PID_MSI_DIGIVOX_DUO)},
|
|
|
|
{USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_VOLAR_X_2)},
|
2008-09-20 05:34:06 +08:00
|
|
|
{USB_DEVICE(USB_VID_TELESTAR, USB_PID_TELESTAR_STARSTICK_2)},
|
2008-09-22 23:32:37 +08:00
|
|
|
{USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_A309)},
|
2008-10-07 21:06:36 +08:00
|
|
|
/* 15 */{USB_DEVICE(USB_VID_MSI_2, USB_PID_MSI_DIGI_VOX_MINI_III)},
|
2008-11-04 23:57:47 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_395U)},
|
2009-01-13 23:47:28 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_395U_2)},
|
2009-03-27 07:41:05 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_395U_3)},
|
2009-03-27 08:07:18 +08:00
|
|
|
{USB_DEVICE(USB_VID_AFATECH, USB_PID_TREKSTOR_DVBT)},
|
2009-04-10 02:14:18 +08:00
|
|
|
/* 20 */{USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_A850)},
|
|
|
|
{USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_A805)},
|
2009-04-10 04:16:41 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_CONCEPTRONIC_CTVDIGRCU)},
|
2009-04-28 12:11:22 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_MC810)},
|
2009-06-01 04:07:01 +08:00
|
|
|
{USB_DEVICE(USB_VID_KYE, USB_PID_GENIUS_TVGO_DVB_T03)},
|
2009-07-25 00:45:41 +08:00
|
|
|
/* 25 */{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_399U_2)},
|
2009-09-19 00:37:57 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_PC160_T)},
|
2009-11-14 10:13:34 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_SVEON_STV20)},
|
2010-02-11 07:05:48 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_TINYTWIN_2)},
|
2010-02-11 07:07:30 +08:00
|
|
|
{USB_DEVICE(USB_VID_LEADTEK, USB_PID_WINFAST_DTV2000DS)},
|
2010-03-02 00:50:40 +08:00
|
|
|
/* 30 */{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_UB383_T)},
|
2010-03-02 01:06:52 +08:00
|
|
|
{USB_DEVICE(USB_VID_KWORLD_2, USB_PID_KWORLD_395U_4)},
|
2010-05-24 05:26:37 +08:00
|
|
|
{USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_A815M)},
|
V4L/DVB: af9015: add USB ID for Terratec Cinergy T Stick RC MKII
Adding the USB ID for my TerraTec Electronic GmbH Cinergy T RC MKII
[0ccd:0097] and hooking it up into af9015, on top of your new NXP TDA18218
patches, makes it work for me.
Just the shipped IR remote control doesn't seem to create keycode events
yet (tested with different remote=%d parameters), are there any hints to
add support for that?
[ 2.250022] usb 1-10: new high speed USB device using ehci_hcd and address 5
[ 2.369287] usb 1-10: New USB device found, idVendor=0ccd, idProduct=0097
[ 2.369290] usb 1-10: New USB device strings: Mfr=1, Product=2, SerialNumber=3
[ 2.369293] usb 1-10: Product: USB2.0 DVB-T TV Stick
[ 2.369294] usb 1-10: Manufacturer: NEWMI
[ 2.369296] usb 1-10: SerialNumber: 010101010600001
[ 2.534023] usbcore: registered new interface driver hiddev
[ 2.537235] input: NEWMI USB2.0 DVB-T TV Stick as /devices/pci0000:00/0000:00:02.1/usb1/1-10/1-10:1.1/input/input0
[ 2.537323] generic-usb 0003:0CCD:0097.0001: input,hidraw0: USB HID v1.01 Keyboard [NEWMI USB2.0 DVB-T TV Stick] on usb-0000:00:02.1-10/input1
[ 2.537349] usbcore: registered new interface driver usbhid
[ 2.537351] usbhid: USB HID core driver
[ 3.263177] generic-usb 0003:04D9:1603.0002: input,hidraw1: USB HID v1.10 Keyboard [ USB Keyboard] on usb-0000:00:02.0-8.1/input0
[ 3.286946] generic-usb 0003:04D9:1603.0003: input,hidraw2: USB HID v1.10 Device [ USB Keyboard] on usb-0000:00:02.0-8.1/input1
[ 3.467136] generic-usb 0003:046D:C050.0004: input,hidraw3: USB HID v1.10 Mouse [Logitech USB-PS/2 Optical Mouse] on usb-0000:00:02.0-8.2/input0
[ 3.660890] generic-usb 0003:10D5:000D.0005: input,hidraw4: USB HID v1.10 Keyboard [No brand SP02-A1] on usb-0000:00:02.0-8.3/input0
[ 5.567632] dvb-usb: found a 'TerraTec Cinergy T Stick RC' in cold state, will try to load a firmware
[ 5.693497] dvb-usb: downloading firmware from file 'dvb-usb-af9015.fw'
[ 5.773109] dvb-usb: found a 'TerraTec Cinergy T Stick RC' in warm state.
[ 5.773168] dvb-usb: will pass the complete MPEG2 transport stream to the software demuxer.
[ 5.774290] DVB: registering new adapter (TerraTec Cinergy T Stick RC)
[ 6.007696] af9013: firmware version:5.1.0
[ 6.010843] DVB: registering adapter 0 frontend 0 (Afatech AF9013 DVB-T)...
[ 6.032697] tda18218: NXP TDA18218HN successfully identified.
[ 6.034442] dvb-usb: TerraTec Cinergy T Stick RC successfully initialized and connected.
[ 6.040612] usbcore: registered new interface driver dvb_usb_af9015
[mchehab@redhat.com: Fix merge conflict with another board addition]
Signed-off-by: Stefan Lippers-Hollmann <s.l-h@gmx.de>
Acked-by: Antti Palosaari <crope@iki.fi>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2010-08-25 21:08:48 +08:00
|
|
|
{USB_DEVICE(USB_VID_TERRATEC, USB_PID_TERRATEC_CINERGY_T_STICK_RC)},
|
2010-09-10 01:59:10 +08:00
|
|
|
{USB_DEVICE(USB_VID_TERRATEC,
|
|
|
|
USB_PID_TERRATEC_CINERGY_T_STICK_DUAL_RC)},
|
2010-10-02 03:55:43 +08:00
|
|
|
/* 35 */{USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_A850T)},
|
2010-10-23 18:35:31 +08:00
|
|
|
{USB_DEVICE(USB_VID_GTEK, USB_PID_TINYTWIN_3)},
|
2008-09-16 04:18:09 +08:00
|
|
|
{0},
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(usb, af9015_usb_table);
|
|
|
|
|
2010-09-13 00:31:56 +08:00
|
|
|
#define AF9015_RC_INTERVAL 500
|
2008-09-16 04:18:09 +08:00
|
|
|
static struct dvb_usb_device_properties af9015_properties[] = {
|
|
|
|
{
|
|
|
|
.caps = DVB_USB_IS_AN_I2C_ADAPTER,
|
|
|
|
|
|
|
|
.usb_ctrl = DEVICE_SPECIFIC,
|
|
|
|
.download_firmware = af9015_download_firmware,
|
|
|
|
.firmware = "dvb-usb-af9015.fw",
|
2008-11-14 01:14:18 +08:00
|
|
|
.no_reconnect = 1,
|
2008-09-16 04:18:09 +08:00
|
|
|
|
2009-09-17 07:33:03 +08:00
|
|
|
.size_of_priv = sizeof(struct af9015_state),
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
.num_adapters = 2,
|
|
|
|
.adapter = {
|
|
|
|
{
|
|
|
|
.caps = DVB_USB_ADAP_HAS_PID_FILTER |
|
|
|
|
DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF,
|
|
|
|
|
|
|
|
.pid_filter_count = 32,
|
|
|
|
.pid_filter = af9015_pid_filter,
|
|
|
|
.pid_filter_ctrl = af9015_pid_filter_ctrl,
|
|
|
|
|
|
|
|
.frontend_attach =
|
|
|
|
af9015_af9013_frontend_attach,
|
|
|
|
.tuner_attach = af9015_tuner_attach,
|
|
|
|
.stream = {
|
|
|
|
.type = USB_BULK,
|
|
|
|
.count = 6,
|
|
|
|
.endpoint = 0x84,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.frontend_attach =
|
|
|
|
af9015_af9013_frontend_attach,
|
|
|
|
.tuner_attach = af9015_tuner_attach,
|
|
|
|
.stream = {
|
|
|
|
.type = USB_BULK,
|
|
|
|
.count = 6,
|
|
|
|
.endpoint = 0x85,
|
2009-01-14 00:08:29 +08:00
|
|
|
.u = {
|
|
|
|
.bulk = {
|
|
|
|
.buffersize =
|
2009-09-12 20:51:36 +08:00
|
|
|
TS_USB20_FRAME_SIZE,
|
2009-01-14 00:08:29 +08:00
|
|
|
}
|
|
|
|
}
|
2008-09-16 04:18:09 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
.identify_state = af9015_identify_state,
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
.rc.core = {
|
2010-11-18 01:20:52 +08:00
|
|
|
.protocol = RC_TYPE_NEC,
|
2010-10-13 04:22:32 +08:00
|
|
|
.module_name = "af9015",
|
2010-10-23 05:45:18 +08:00
|
|
|
.rc_query = af9015_rc_query,
|
2010-09-13 00:31:56 +08:00
|
|
|
.rc_interval = AF9015_RC_INTERVAL,
|
2010-11-18 01:20:52 +08:00
|
|
|
.allowed_protos = RC_TYPE_NEC,
|
2010-08-01 05:04:09 +08:00
|
|
|
},
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
.i2c_algo = &af9015_i2c_algo,
|
|
|
|
|
2010-10-02 03:55:43 +08:00
|
|
|
.num_device_descs = 12, /* check max from dvb-usb.h */
|
2008-09-16 04:18:09 +08:00
|
|
|
.devices = {
|
|
|
|
{
|
|
|
|
.name = "Afatech AF9015 DVB-T USB2.0 stick",
|
|
|
|
.cold_ids = {&af9015_usb_table[0],
|
|
|
|
&af9015_usb_table[1], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "Leadtek WinFast DTV Dongle Gold",
|
|
|
|
.cold_ids = {&af9015_usb_table[2], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "Pinnacle PCTV 71e",
|
|
|
|
.cold_ids = {&af9015_usb_table[3], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "KWorld PlusTV Dual DVB-T Stick " \
|
|
|
|
"(DVB-T 399U)",
|
2009-07-25 00:45:41 +08:00
|
|
|
.cold_ids = {&af9015_usb_table[4],
|
|
|
|
&af9015_usb_table[25], NULL},
|
2008-09-16 04:18:09 +08:00
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "DigitalNow TinyTwin DVB-T Receiver",
|
2010-02-11 07:05:48 +08:00
|
|
|
.cold_ids = {&af9015_usb_table[5],
|
2010-10-23 18:35:31 +08:00
|
|
|
&af9015_usb_table[28],
|
|
|
|
&af9015_usb_table[36], NULL},
|
2008-09-16 04:18:09 +08:00
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "TwinHan AzureWave AD-TU700(704J)",
|
|
|
|
.cold_ids = {&af9015_usb_table[6], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "TerraTec Cinergy T USB XE",
|
|
|
|
.cold_ids = {&af9015_usb_table[7], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "KWorld PlusTV Dual DVB-T PCI " \
|
|
|
|
"(DVB-T PC160-2T)",
|
|
|
|
.cold_ids = {&af9015_usb_table[8], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "AVerMedia AVerTV DVB-T Volar X",
|
|
|
|
.cold_ids = {&af9015_usb_table[9], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2010-09-09 23:10:50 +08:00
|
|
|
{
|
|
|
|
.name = "TerraTec Cinergy T Stick RC",
|
|
|
|
.cold_ids = {&af9015_usb_table[33], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2010-09-10 01:59:10 +08:00
|
|
|
{
|
|
|
|
.name = "TerraTec Cinergy T Stick Dual RC",
|
|
|
|
.cold_ids = {&af9015_usb_table[34], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2010-10-02 03:55:43 +08:00
|
|
|
{
|
|
|
|
.name = "AverMedia AVerTV Red HD+ (A850T)",
|
|
|
|
.cold_ids = {&af9015_usb_table[35], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
|
|
|
}, {
|
|
|
|
.caps = DVB_USB_IS_AN_I2C_ADAPTER,
|
|
|
|
|
|
|
|
.usb_ctrl = DEVICE_SPECIFIC,
|
|
|
|
.download_firmware = af9015_download_firmware,
|
|
|
|
.firmware = "dvb-usb-af9015.fw",
|
2008-11-14 01:14:18 +08:00
|
|
|
.no_reconnect = 1,
|
2008-09-16 04:18:09 +08:00
|
|
|
|
2009-09-17 07:33:03 +08:00
|
|
|
.size_of_priv = sizeof(struct af9015_state),
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
.num_adapters = 2,
|
|
|
|
.adapter = {
|
|
|
|
{
|
|
|
|
.caps = DVB_USB_ADAP_HAS_PID_FILTER |
|
|
|
|
DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF,
|
|
|
|
|
|
|
|
.pid_filter_count = 32,
|
|
|
|
.pid_filter = af9015_pid_filter,
|
|
|
|
.pid_filter_ctrl = af9015_pid_filter_ctrl,
|
|
|
|
|
|
|
|
.frontend_attach =
|
|
|
|
af9015_af9013_frontend_attach,
|
|
|
|
.tuner_attach = af9015_tuner_attach,
|
|
|
|
.stream = {
|
|
|
|
.type = USB_BULK,
|
|
|
|
.count = 6,
|
|
|
|
.endpoint = 0x84,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.frontend_attach =
|
|
|
|
af9015_af9013_frontend_attach,
|
|
|
|
.tuner_attach = af9015_tuner_attach,
|
|
|
|
.stream = {
|
|
|
|
.type = USB_BULK,
|
|
|
|
.count = 6,
|
|
|
|
.endpoint = 0x85,
|
2009-01-14 00:08:29 +08:00
|
|
|
.u = {
|
|
|
|
.bulk = {
|
|
|
|
.buffersize =
|
2009-09-12 20:51:36 +08:00
|
|
|
TS_USB20_FRAME_SIZE,
|
2009-01-14 00:08:29 +08:00
|
|
|
}
|
|
|
|
}
|
2008-09-16 04:18:09 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
.identify_state = af9015_identify_state,
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
.rc.core = {
|
2010-11-18 01:20:52 +08:00
|
|
|
.protocol = RC_TYPE_NEC,
|
2010-10-13 04:22:32 +08:00
|
|
|
.module_name = "af9015",
|
2010-10-23 05:45:18 +08:00
|
|
|
.rc_query = af9015_rc_query,
|
2010-09-13 00:31:56 +08:00
|
|
|
.rc_interval = AF9015_RC_INTERVAL,
|
2010-11-18 01:20:52 +08:00
|
|
|
.allowed_protos = RC_TYPE_NEC,
|
2010-08-01 05:04:09 +08:00
|
|
|
},
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
.i2c_algo = &af9015_i2c_algo,
|
|
|
|
|
2010-09-09 23:05:31 +08:00
|
|
|
.num_device_descs = 9, /* check max from dvb-usb.h */
|
2008-09-16 04:18:09 +08:00
|
|
|
.devices = {
|
|
|
|
{
|
|
|
|
.name = "Xtensions XD-380",
|
|
|
|
.cold_ids = {&af9015_usb_table[10], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "MSI DIGIVOX Duo",
|
|
|
|
.cold_ids = {&af9015_usb_table[11], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "Fujitsu-Siemens Slim Mobile USB DVB-T",
|
|
|
|
.cold_ids = {&af9015_usb_table[12], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2008-09-20 05:26:05 +08:00
|
|
|
{
|
2008-09-20 05:34:06 +08:00
|
|
|
.name = "Telestar Starstick 2",
|
2008-09-20 05:26:05 +08:00
|
|
|
.cold_ids = {&af9015_usb_table[13], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2008-09-22 23:32:37 +08:00
|
|
|
{
|
|
|
|
.name = "AVerMedia A309",
|
|
|
|
.cold_ids = {&af9015_usb_table[14], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2008-10-07 21:06:36 +08:00
|
|
|
{
|
|
|
|
.name = "MSI Digi VOX mini III",
|
|
|
|
.cold_ids = {&af9015_usb_table[15], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2008-11-04 23:57:47 +08:00
|
|
|
{
|
|
|
|
.name = "KWorld USB DVB-T TV Stick II " \
|
|
|
|
"(VS-DVB-T 395U)",
|
2009-01-13 23:47:28 +08:00
|
|
|
.cold_ids = {&af9015_usb_table[16],
|
2009-03-27 07:41:05 +08:00
|
|
|
&af9015_usb_table[17],
|
2010-03-02 01:06:52 +08:00
|
|
|
&af9015_usb_table[18],
|
|
|
|
&af9015_usb_table[31], NULL},
|
2008-11-04 23:57:47 +08:00
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-03-27 08:07:18 +08:00
|
|
|
{
|
|
|
|
.name = "TrekStor DVB-T USB Stick",
|
|
|
|
.cold_ids = {&af9015_usb_table[19], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-04-01 04:01:02 +08:00
|
|
|
{
|
|
|
|
.name = "AverMedia AVerTV Volar Black HD " \
|
|
|
|
"(A850)",
|
|
|
|
.cold_ids = {&af9015_usb_table[20], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2008-09-16 04:18:09 +08:00
|
|
|
}
|
2009-04-09 20:16:12 +08:00
|
|
|
}, {
|
|
|
|
.caps = DVB_USB_IS_AN_I2C_ADAPTER,
|
|
|
|
|
|
|
|
.usb_ctrl = DEVICE_SPECIFIC,
|
|
|
|
.download_firmware = af9015_download_firmware,
|
|
|
|
.firmware = "dvb-usb-af9015.fw",
|
|
|
|
.no_reconnect = 1,
|
|
|
|
|
2009-09-17 07:33:03 +08:00
|
|
|
.size_of_priv = sizeof(struct af9015_state),
|
2009-04-09 20:16:12 +08:00
|
|
|
|
|
|
|
.num_adapters = 2,
|
|
|
|
.adapter = {
|
|
|
|
{
|
|
|
|
.caps = DVB_USB_ADAP_HAS_PID_FILTER |
|
|
|
|
DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF,
|
|
|
|
|
|
|
|
.pid_filter_count = 32,
|
|
|
|
.pid_filter = af9015_pid_filter,
|
|
|
|
.pid_filter_ctrl = af9015_pid_filter_ctrl,
|
|
|
|
|
|
|
|
.frontend_attach =
|
|
|
|
af9015_af9013_frontend_attach,
|
|
|
|
.tuner_attach = af9015_tuner_attach,
|
|
|
|
.stream = {
|
|
|
|
.type = USB_BULK,
|
|
|
|
.count = 6,
|
|
|
|
.endpoint = 0x84,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.frontend_attach =
|
|
|
|
af9015_af9013_frontend_attach,
|
|
|
|
.tuner_attach = af9015_tuner_attach,
|
|
|
|
.stream = {
|
|
|
|
.type = USB_BULK,
|
|
|
|
.count = 6,
|
|
|
|
.endpoint = 0x85,
|
|
|
|
.u = {
|
|
|
|
.bulk = {
|
|
|
|
.buffersize =
|
2009-09-12 20:51:36 +08:00
|
|
|
TS_USB20_FRAME_SIZE,
|
2009-04-09 20:16:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
.identify_state = af9015_identify_state,
|
|
|
|
|
2010-10-13 04:22:32 +08:00
|
|
|
.rc.core = {
|
2010-11-18 01:20:52 +08:00
|
|
|
.protocol = RC_TYPE_NEC,
|
2010-10-13 04:22:32 +08:00
|
|
|
.module_name = "af9015",
|
2010-10-23 05:45:18 +08:00
|
|
|
.rc_query = af9015_rc_query,
|
2010-09-13 00:31:56 +08:00
|
|
|
.rc_interval = AF9015_RC_INTERVAL,
|
2010-11-18 01:20:52 +08:00
|
|
|
.allowed_protos = RC_TYPE_NEC,
|
2010-08-01 05:04:09 +08:00
|
|
|
},
|
2009-04-09 20:16:12 +08:00
|
|
|
|
|
|
|
.i2c_algo = &af9015_i2c_algo,
|
|
|
|
|
2010-09-09 23:05:31 +08:00
|
|
|
.num_device_descs = 9, /* check max from dvb-usb.h */
|
2009-04-09 20:16:12 +08:00
|
|
|
.devices = {
|
2009-04-10 02:14:18 +08:00
|
|
|
{
|
|
|
|
.name = "AverMedia AVerTV Volar GPS 805 (A805)",
|
|
|
|
.cold_ids = {&af9015_usb_table[21], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-04-10 04:16:41 +08:00
|
|
|
{
|
|
|
|
.name = "Conceptronic USB2.0 DVB-T CTVDIGRCU " \
|
|
|
|
"V3.0",
|
|
|
|
.cold_ids = {&af9015_usb_table[22], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-04-28 12:11:22 +08:00
|
|
|
{
|
|
|
|
.name = "KWorld Digial MC-810",
|
|
|
|
.cold_ids = {&af9015_usb_table[23], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-06-01 04:07:01 +08:00
|
|
|
{
|
|
|
|
.name = "Genius TVGo DVB-T03",
|
|
|
|
.cold_ids = {&af9015_usb_table[24], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-09-19 00:37:57 +08:00
|
|
|
{
|
|
|
|
.name = "KWorld PlusTV DVB-T PCI Pro Card " \
|
|
|
|
"(DVB-T PC160-T)",
|
|
|
|
.cold_ids = {&af9015_usb_table[26], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-11-14 10:13:34 +08:00
|
|
|
{
|
|
|
|
.name = "Sveon STV20 Tuner USB DVB-T HDTV",
|
|
|
|
.cold_ids = {&af9015_usb_table[27], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2010-02-11 07:07:30 +08:00
|
|
|
{
|
|
|
|
.name = "Leadtek WinFast DTV2000DS",
|
|
|
|
.cold_ids = {&af9015_usb_table[29], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2010-03-02 00:50:40 +08:00
|
|
|
{
|
|
|
|
.name = "KWorld USB DVB-T Stick Mobile " \
|
|
|
|
"(UB383-T)",
|
|
|
|
.cold_ids = {&af9015_usb_table[30], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2010-05-24 05:26:37 +08:00
|
|
|
{
|
|
|
|
.name = "AverMedia AVerTV Volar M (A815Mac)",
|
|
|
|
.cold_ids = {&af9015_usb_table[32], NULL},
|
|
|
|
.warm_ids = {NULL},
|
|
|
|
},
|
2009-04-09 20:16:12 +08:00
|
|
|
}
|
|
|
|
},
|
2008-09-16 04:18:09 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static int af9015_usb_probe(struct usb_interface *intf,
|
|
|
|
const struct usb_device_id *id)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
struct dvb_usb_device *d = NULL;
|
|
|
|
struct usb_device *udev = interface_to_usbdev(intf);
|
|
|
|
u8 i;
|
|
|
|
|
|
|
|
deb_info("%s: interface:%d\n", __func__,
|
|
|
|
intf->cur_altsetting->desc.bInterfaceNumber);
|
|
|
|
|
|
|
|
/* interface 0 is used by DVB-T receiver and
|
|
|
|
interface 1 is for remote controller (HID) */
|
|
|
|
if (intf->cur_altsetting->desc.bInterfaceNumber == 0) {
|
|
|
|
ret = af9015_read_config(udev);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
for (i = 0; i < af9015_properties_count; i++) {
|
|
|
|
ret = dvb_usb_device_init(intf, &af9015_properties[i],
|
|
|
|
THIS_MODULE, &d, adapter_nr);
|
|
|
|
if (!ret)
|
|
|
|
break;
|
|
|
|
if (ret != -ENODEV)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (d)
|
|
|
|
ret = af9015_init(d);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2008-11-06 03:31:24 +08:00
|
|
|
static void af9015_i2c_exit(struct dvb_usb_device *d)
|
2008-09-16 04:18:09 +08:00
|
|
|
{
|
|
|
|
struct af9015_state *state = d->priv;
|
2010-10-08 08:37:06 +08:00
|
|
|
deb_info("%s:\n", __func__);
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
/* remove 2nd I2C adapter */
|
|
|
|
if (d->state & DVB_USB_STATE_I2C)
|
|
|
|
i2c_del_adapter(&state->i2c_adap);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void af9015_usb_device_exit(struct usb_interface *intf)
|
|
|
|
{
|
|
|
|
struct dvb_usb_device *d = usb_get_intfdata(intf);
|
2010-10-08 08:37:06 +08:00
|
|
|
deb_info("%s:\n", __func__);
|
2008-09-16 04:18:09 +08:00
|
|
|
|
|
|
|
/* remove 2nd I2C adapter */
|
|
|
|
if (d != NULL && d->desc != NULL)
|
|
|
|
af9015_i2c_exit(d);
|
|
|
|
|
|
|
|
dvb_usb_device_exit(intf);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* usb specific object needed to register this driver with the usb subsystem */
|
|
|
|
static struct usb_driver af9015_usb_driver = {
|
|
|
|
.name = "dvb_usb_af9015",
|
|
|
|
.probe = af9015_usb_probe,
|
|
|
|
.disconnect = af9015_usb_device_exit,
|
|
|
|
.id_table = af9015_usb_table,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* module stuff */
|
|
|
|
static int __init af9015_usb_module_init(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
ret = usb_register(&af9015_usb_driver);
|
|
|
|
if (ret)
|
|
|
|
err("module init failed:%d", ret);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit af9015_usb_module_exit(void)
|
|
|
|
{
|
|
|
|
/* deregister this driver from the USB subsystem */
|
|
|
|
usb_deregister(&af9015_usb_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(af9015_usb_module_init);
|
|
|
|
module_exit(af9015_usb_module_exit);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Antti Palosaari <crope@iki.fi>");
|
|
|
|
MODULE_DESCRIPTION("Driver for Afatech AF9015 DVB-T");
|
|
|
|
MODULE_LICENSE("GPL");
|