2018-03-03 23:43:14 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
//
|
|
|
|
// em28xx-core.c - driver for Empia EM2800/EM2820/2840 USB video capture devices
|
|
|
|
//
|
|
|
|
// Copyright (C) 2005 Ludovico Cavedon <cavedon@sssup.it>
|
|
|
|
// Markus Rechberger <mrechberger@gmail.com>
|
MAINTAINERS & files: Canonize the e-mails I use at files
From now on, I'll start using my @kernel.org as my development e-mail.
As such, let's remove the entries that point to the old
mchehab@s-opensource.com at MAINTAINERS file.
For the files written with a copyright with mchehab@s-opensource,
let's keep Samsung on their names, using mchehab+samsung@kernel.org,
in order to keep pointing to my employer, with sponsors the work.
For the files written before I join Samsung (on July, 4 2013),
let's just use mchehab@kernel.org.
For bug reports, we can simply point to just kernel.org, as
this will reach my mchehab+samsung inbox anyway.
Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
Signed-off-by: Brian Warner <brian.warner@samsung.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org>
2018-04-25 17:34:48 +08:00
|
|
|
// Mauro Carvalho Chehab <mchehab@kernel.org>
|
2018-03-03 23:43:14 +08:00
|
|
|
// Sascha Sommer <saschasommer@freenet.de>
|
|
|
|
// Copyright (C) 2012 Frank Schäfer <fschaefer.oss@googlemail.com>
|
|
|
|
//
|
|
|
|
// 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.
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2016-10-12 18:26:47 +08:00
|
|
|
#include "em28xx.h"
|
|
|
|
|
2005-11-09 13:37:07 +08:00
|
|
|
#include <linux/init.h>
|
2013-12-27 11:28:57 +08:00
|
|
|
#include <linux/jiffies.h>
|
2005-11-09 13:37:07 +08:00
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/module.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>
|
2005-11-09 13:37:07 +08:00
|
|
|
#include <linux/usb.h>
|
|
|
|
#include <linux/vmalloc.h>
|
2012-06-12 02:17:23 +08:00
|
|
|
#include <sound/ac97_codec.h>
|
2008-12-29 09:18:14 +08:00
|
|
|
#include <media/v4l2-common.h>
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2013-12-23 00:27:02 +08:00
|
|
|
#define DRIVER_AUTHOR "Ludovico Cavedon <cavedon@sssup.it>, " \
|
|
|
|
"Markus Rechberger <mrechberger@gmail.com>, " \
|
MAINTAINERS & files: Canonize the e-mails I use at files
From now on, I'll start using my @kernel.org as my development e-mail.
As such, let's remove the entries that point to the old
mchehab@s-opensource.com at MAINTAINERS file.
For the files written with a copyright with mchehab@s-opensource,
let's keep Samsung on their names, using mchehab+samsung@kernel.org,
in order to keep pointing to my employer, with sponsors the work.
For the files written before I join Samsung (on July, 4 2013),
let's just use mchehab@kernel.org.
For bug reports, we can simply point to just kernel.org, as
this will reach my mchehab+samsung inbox anyway.
Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
Signed-off-by: Brian Warner <brian.warner@samsung.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org>
2018-04-25 17:34:48 +08:00
|
|
|
"Mauro Carvalho Chehab <mchehab@kernel.org>, " \
|
2013-12-23 00:27:02 +08:00
|
|
|
"Sascha Sommer <saschasommer@freenet.de>"
|
|
|
|
|
|
|
|
MODULE_AUTHOR(DRIVER_AUTHOR);
|
|
|
|
MODULE_DESCRIPTION(DRIVER_DESC);
|
2018-03-03 23:43:14 +08:00
|
|
|
MODULE_LICENSE("GPL v2");
|
2013-12-23 00:27:02 +08:00
|
|
|
MODULE_VERSION(EM28XX_VERSION);
|
|
|
|
|
2005-11-09 13:37:07 +08:00
|
|
|
/* #define ENABLE_DEBUG_ISOC_FRAMES */
|
|
|
|
|
2008-04-23 01:41:48 +08:00
|
|
|
static unsigned int core_debug;
|
2009-02-11 10:28:24 +08:00
|
|
|
module_param(core_debug, int, 0644);
|
2016-10-12 18:32:23 +08:00
|
|
|
MODULE_PARM_DESC(core_debug, "enable debug messages [core and isoc]");
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2016-10-20 18:42:03 +08:00
|
|
|
#define em28xx_coredbg(fmt, arg...) do { \
|
|
|
|
if (core_debug) \
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_printk(KERN_DEBUG, &dev->intf->dev, \
|
2016-10-20 18:42:03 +08:00
|
|
|
"core: %s: " fmt, __func__, ## arg); \
|
|
|
|
} while (0)
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2008-04-23 01:41:48 +08:00
|
|
|
static unsigned int reg_debug;
|
2009-02-11 10:28:24 +08:00
|
|
|
module_param(reg_debug, int, 0644);
|
|
|
|
MODULE_PARM_DESC(reg_debug, "enable debug messages [URB reg]");
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2016-10-20 18:42:03 +08:00
|
|
|
#define em28xx_regdbg(fmt, arg...) do { \
|
|
|
|
if (reg_debug) \
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_printk(KERN_DEBUG, &dev->intf->dev, \
|
2016-10-20 18:42:03 +08:00
|
|
|
"reg: %s: " fmt, __func__, ## arg); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/* FIXME: don't abuse core_debug */
|
|
|
|
#define em28xx_isocdbg(fmt, arg...) do { \
|
|
|
|
if (core_debug) \
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_printk(KERN_DEBUG, &dev->intf->dev, \
|
2016-10-20 18:42:03 +08:00
|
|
|
"core: %s: " fmt, __func__, ## arg); \
|
|
|
|
} while (0)
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2005-11-09 13:37:07 +08:00
|
|
|
/*
|
2005-11-09 13:38:27 +08:00
|
|
|
* em28xx_read_reg_req()
|
2005-11-09 13:37:07 +08:00
|
|
|
* reads data from the usb device specifying bRequest
|
|
|
|
*/
|
2005-11-09 13:38:27 +08:00
|
|
|
int em28xx_read_reg_req_len(struct em28xx *dev, u8 req, u16 reg,
|
2014-11-28 19:34:15 +08:00
|
|
|
char *buf, int len)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
2008-11-26 20:58:48 +08:00
|
|
|
int ret;
|
2016-12-08 00:34:22 +08:00
|
|
|
struct usb_device *udev = interface_to_usbdev(dev->intf);
|
|
|
|
int pipe = usb_rcvctrlpipe(udev, 0);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2012-12-28 06:02:43 +08:00
|
|
|
if (dev->disconnected)
|
2008-11-19 01:51:08 +08:00
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
if (len > URB_MAX_CTRL_SIZE)
|
|
|
|
return -EINVAL;
|
2006-02-07 16:49:11 +08:00
|
|
|
|
2008-11-19 17:17:44 +08:00
|
|
|
mutex_lock(&dev->ctrl_urb_lock);
|
2016-12-08 00:34:22 +08:00
|
|
|
ret = usb_control_msg(udev, pipe, req,
|
2005-11-09 13:37:07 +08:00
|
|
|
USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
2008-11-19 01:51:08 +08:00
|
|
|
0x0000, reg, dev->urb_buf, len, HZ);
|
|
|
|
if (ret < 0) {
|
2017-04-27 00:43:12 +08:00
|
|
|
em28xx_regdbg("(pipe 0x%08x): IN: %02x %02x %02x %02x %02x %02x %02x %02x failed with error %i\n",
|
2018-03-04 02:43:34 +08:00
|
|
|
pipe,
|
|
|
|
USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
|
|
|
req, 0, 0,
|
|
|
|
reg & 0xff, reg >> 8,
|
|
|
|
len & 0xff, len >> 8, ret);
|
2008-11-19 17:17:44 +08:00
|
|
|
mutex_unlock(&dev->ctrl_urb_lock);
|
2013-01-04 01:27:05 +08:00
|
|
|
return usb_translate_errors(ret);
|
2008-11-19 01:51:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (len)
|
|
|
|
memcpy(buf, dev->urb_buf, len);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2008-11-19 17:17:44 +08:00
|
|
|
mutex_unlock(&dev->ctrl_urb_lock);
|
|
|
|
|
2017-04-27 00:43:12 +08:00
|
|
|
em28xx_regdbg("(pipe 0x%08x): IN: %02x %02x %02x %02x %02x %02x %02x %02x <<< %*ph\n",
|
2018-03-04 02:43:34 +08:00
|
|
|
pipe, USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
|
|
|
req, 0, 0,
|
|
|
|
reg & 0xff, reg >> 8,
|
|
|
|
len & 0xff, len >> 8, len, buf);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2005-11-09 13:38:27 +08:00
|
|
|
* em28xx_read_reg_req()
|
2005-11-09 13:37:07 +08:00
|
|
|
* reads data from the usb device specifying bRequest
|
|
|
|
*/
|
2005-11-09 13:38:27 +08:00
|
|
|
int em28xx_read_reg_req(struct em28xx *dev, u8 req, u16 reg)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
|
|
|
int ret;
|
2008-11-26 20:58:48 +08:00
|
|
|
u8 val;
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2008-11-26 20:58:48 +08:00
|
|
|
ret = em28xx_read_reg_req_len(dev, req, reg, &val, 1);
|
|
|
|
if (ret < 0)
|
2008-11-19 01:51:08 +08:00
|
|
|
return ret;
|
2005-11-09 13:37:07 +08:00
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2005-11-09 13:38:27 +08:00
|
|
|
int em28xx_read_reg(struct em28xx *dev, u16 reg)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
2005-11-09 13:38:27 +08:00
|
|
|
return em28xx_read_reg_req(dev, USB_REQ_GET_STATUS, reg);
|
2005-11-09 13:37:07 +08:00
|
|
|
}
|
2012-03-26 20:13:31 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_read_reg);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
|
|
|
/*
|
2005-11-09 13:38:27 +08:00
|
|
|
* em28xx_write_regs_req()
|
2005-11-09 13:37:07 +08:00
|
|
|
* sends data to the usb device, specifying bRequest
|
|
|
|
*/
|
2005-11-09 13:38:27 +08:00
|
|
|
int em28xx_write_regs_req(struct em28xx *dev, u8 req, u16 reg, char *buf,
|
2014-11-28 19:34:15 +08:00
|
|
|
int len)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
|
|
|
int ret;
|
2016-12-08 00:34:22 +08:00
|
|
|
struct usb_device *udev = interface_to_usbdev(dev->intf);
|
|
|
|
int pipe = usb_sndctrlpipe(udev, 0);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2012-12-28 06:02:43 +08:00
|
|
|
if (dev->disconnected)
|
2008-04-18 08:48:00 +08:00
|
|
|
return -ENODEV;
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
if (len < 1 || len > URB_MAX_CTRL_SIZE)
|
2008-04-18 08:48:00 +08:00
|
|
|
return -EINVAL;
|
2006-02-07 16:49:11 +08:00
|
|
|
|
2008-11-19 17:17:44 +08:00
|
|
|
mutex_lock(&dev->ctrl_urb_lock);
|
2008-11-19 01:51:08 +08:00
|
|
|
memcpy(dev->urb_buf, buf, len);
|
2016-12-08 00:34:22 +08:00
|
|
|
ret = usb_control_msg(udev, pipe, req,
|
2005-11-09 13:37:07 +08:00
|
|
|
USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
2008-11-19 01:51:08 +08:00
|
|
|
0x0000, reg, dev->urb_buf, len, HZ);
|
2008-11-19 17:17:44 +08:00
|
|
|
mutex_unlock(&dev->ctrl_urb_lock);
|
2008-11-19 01:51:08 +08:00
|
|
|
|
2017-04-27 00:43:12 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
em28xx_regdbg("(pipe 0x%08x): OUT: %02x %02x %02x %02x %02x %02x %02x %02x >>> %*ph failed with error %i\n",
|
|
|
|
pipe,
|
|
|
|
USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
|
|
|
req, 0, 0,
|
|
|
|
reg & 0xff, reg >> 8,
|
|
|
|
len & 0xff, len >> 8, len, buf, ret);
|
2013-01-04 01:27:05 +08:00
|
|
|
return usb_translate_errors(ret);
|
2017-04-27 00:43:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
em28xx_regdbg("(pipe 0x%08x): OUT: %02x %02x %02x %02x %02x %02x %02x %02x >>> %*ph\n",
|
|
|
|
pipe,
|
|
|
|
USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
|
|
|
req, 0, 0,
|
|
|
|
reg & 0xff, reg >> 8,
|
|
|
|
len & 0xff, len >> 8, len, buf);
|
2013-01-04 01:27:05 +08:00
|
|
|
|
2008-04-18 08:42:14 +08:00
|
|
|
if (dev->wait_after_write)
|
|
|
|
msleep(dev->wait_after_write);
|
|
|
|
|
2005-11-09 13:37:07 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2005-11-09 13:38:27 +08:00
|
|
|
int em28xx_write_regs(struct em28xx *dev, u16 reg, char *buf, int len)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
2013-06-04 01:12:05 +08:00
|
|
|
return em28xx_write_regs_req(dev, USB_REQ_GET_STATUS, reg, buf, len);
|
2005-11-09 13:37:07 +08:00
|
|
|
}
|
2012-03-26 20:13:31 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_write_regs);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2008-11-20 20:52:20 +08:00
|
|
|
/* Write a single register */
|
|
|
|
int em28xx_write_reg(struct em28xx *dev, u16 reg, u8 val)
|
|
|
|
{
|
|
|
|
return em28xx_write_regs(dev, reg, &val, 1);
|
|
|
|
}
|
2011-07-04 08:05:06 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_write_reg);
|
2008-11-20 20:52:20 +08:00
|
|
|
|
2005-11-09 13:37:07 +08:00
|
|
|
/*
|
2005-11-09 13:38:27 +08:00
|
|
|
* em28xx_write_reg_bits()
|
2005-11-09 13:37:07 +08:00
|
|
|
* sets only some bits (specified by bitmask) of a register, by first reading
|
|
|
|
* the actual value
|
|
|
|
*/
|
2009-12-05 19:27:49 +08:00
|
|
|
int em28xx_write_reg_bits(struct em28xx *dev, u16 reg, u8 val,
|
2014-11-28 19:34:15 +08:00
|
|
|
u8 bitmask)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
|
|
|
int oldval;
|
|
|
|
u8 newval;
|
2008-04-18 08:41:10 +08:00
|
|
|
|
2013-06-04 01:12:05 +08:00
|
|
|
oldval = em28xx_read_reg(dev, reg);
|
2008-04-18 08:41:10 +08:00
|
|
|
if (oldval < 0)
|
2005-11-09 13:37:07 +08:00
|
|
|
return oldval;
|
2008-04-18 08:41:10 +08:00
|
|
|
|
2014-11-28 19:34:15 +08:00
|
|
|
newval = (((u8)oldval) & ~bitmask) | (val & bitmask);
|
2008-04-18 08:48:00 +08:00
|
|
|
|
2005-11-09 13:38:27 +08:00
|
|
|
return em28xx_write_regs(dev, reg, &newval, 1);
|
2005-11-09 13:37:07 +08:00
|
|
|
}
|
2012-03-26 20:13:31 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_write_reg_bits);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2013-12-02 05:06:56 +08:00
|
|
|
/*
|
|
|
|
* em28xx_toggle_reg_bits()
|
|
|
|
* toggles/inverts the bits (specified by bitmask) of a register
|
|
|
|
*/
|
|
|
|
int em28xx_toggle_reg_bits(struct em28xx *dev, u16 reg, u8 bitmask)
|
|
|
|
{
|
|
|
|
int oldval;
|
|
|
|
u8 newval;
|
|
|
|
|
|
|
|
oldval = em28xx_read_reg(dev, reg);
|
|
|
|
if (oldval < 0)
|
|
|
|
return oldval;
|
|
|
|
|
|
|
|
newval = (~oldval & bitmask) | (oldval & ~bitmask);
|
|
|
|
|
|
|
|
return em28xx_write_reg(dev, reg, newval);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(em28xx_toggle_reg_bits);
|
|
|
|
|
2008-11-19 23:01:33 +08:00
|
|
|
/*
|
|
|
|
* em28xx_is_ac97_ready()
|
|
|
|
* Checks if ac97 is ready
|
|
|
|
*/
|
|
|
|
static int em28xx_is_ac97_ready(struct em28xx *dev)
|
|
|
|
{
|
2013-12-27 11:28:57 +08:00
|
|
|
unsigned long timeout = jiffies + msecs_to_jiffies(EM28XX_AC97_XFER_TIMEOUT);
|
|
|
|
int ret;
|
2008-11-19 23:01:33 +08:00
|
|
|
|
|
|
|
/* Wait up to 50 ms for AC97 command to complete */
|
2013-12-27 11:28:57 +08:00
|
|
|
while (time_is_after_jiffies(timeout)) {
|
2008-11-19 23:01:33 +08:00
|
|
|
ret = em28xx_read_reg(dev, EM28XX_R43_AC97BUSY);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (!(ret & 0x01))
|
|
|
|
return 0;
|
2013-12-27 11:28:57 +08:00
|
|
|
msleep(5);
|
2008-11-19 23:01:33 +08:00
|
|
|
}
|
|
|
|
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"AC97 command still being executed: not handled properly!\n");
|
2008-11-19 23:01:33 +08:00
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* em28xx_read_ac97()
|
|
|
|
* write a 16 bit value to the specified AC97 address (LSB first!)
|
|
|
|
*/
|
2008-12-23 00:18:27 +08:00
|
|
|
int em28xx_read_ac97(struct em28xx *dev, u8 reg)
|
2008-11-19 23:01:33 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 addr = (reg & 0x7f) | 0x80;
|
2014-08-22 03:43:03 +08:00
|
|
|
__le16 val;
|
2008-11-19 23:01:33 +08:00
|
|
|
|
|
|
|
ret = em28xx_is_ac97_ready(dev);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = em28xx_write_regs(dev, EM28XX_R42_AC97ADDR, &addr, 1);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = dev->em28xx_read_reg_req_len(dev, 0, EM28XX_R40_AC97LSB,
|
|
|
|
(u8 *)&val, sizeof(val));
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
return le16_to_cpu(val);
|
|
|
|
}
|
2011-06-20 00:06:40 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_read_ac97);
|
2008-11-19 23:01:33 +08:00
|
|
|
|
2005-11-09 13:37:07 +08:00
|
|
|
/*
|
2005-11-09 13:38:27 +08:00
|
|
|
* em28xx_write_ac97()
|
2005-11-09 13:37:07 +08:00
|
|
|
* write a 16 bit value to the specified AC97 address (LSB first!)
|
|
|
|
*/
|
2008-12-23 00:18:27 +08:00
|
|
|
int em28xx_write_ac97(struct em28xx *dev, u8 reg, u16 val)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
2008-11-19 23:01:33 +08:00
|
|
|
int ret;
|
2005-11-09 13:37:07 +08:00
|
|
|
u8 addr = reg & 0x7f;
|
2008-11-19 23:01:33 +08:00
|
|
|
__le16 value;
|
|
|
|
|
|
|
|
value = cpu_to_le16(val);
|
|
|
|
|
|
|
|
ret = em28xx_is_ac97_ready(dev);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2008-04-18 08:41:10 +08:00
|
|
|
|
2014-11-28 19:34:15 +08:00
|
|
|
ret = em28xx_write_regs(dev, EM28XX_R40_AC97LSB, (u8 *)&value, 2);
|
2008-04-18 08:41:10 +08:00
|
|
|
if (ret < 0)
|
2005-11-09 13:37:07 +08:00
|
|
|
return ret;
|
2008-04-18 08:41:10 +08:00
|
|
|
|
2008-04-18 08:44:58 +08:00
|
|
|
ret = em28xx_write_regs(dev, EM28XX_R42_AC97ADDR, &addr, 1);
|
2008-04-18 08:41:10 +08:00
|
|
|
if (ret < 0)
|
2005-11-09 13:37:07 +08:00
|
|
|
return ret;
|
2008-02-07 05:34:13 +08:00
|
|
|
|
2008-11-19 23:01:33 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2011-06-20 00:06:40 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_write_ac97);
|
2008-04-18 08:41:10 +08:00
|
|
|
|
2011-06-18 18:02:49 +08:00
|
|
|
struct em28xx_vol_itable {
|
2008-11-21 00:39:39 +08:00
|
|
|
enum em28xx_amux mux;
|
2008-11-20 20:06:09 +08:00
|
|
|
u8 reg;
|
|
|
|
};
|
|
|
|
|
2011-06-18 18:02:49 +08:00
|
|
|
static struct em28xx_vol_itable inputs[] = {
|
2012-06-12 02:17:23 +08:00
|
|
|
{ EM28XX_AMUX_VIDEO, AC97_VIDEO },
|
|
|
|
{ EM28XX_AMUX_LINE_IN, AC97_LINE },
|
|
|
|
{ EM28XX_AMUX_PHONE, AC97_PHONE },
|
|
|
|
{ EM28XX_AMUX_MIC, AC97_MIC },
|
|
|
|
{ EM28XX_AMUX_CD, AC97_CD },
|
|
|
|
{ EM28XX_AMUX_AUX, AC97_AUX },
|
|
|
|
{ EM28XX_AMUX_PCM_OUT, AC97_PCM },
|
2008-11-20 20:06:09 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static int set_ac97_input(struct em28xx *dev)
|
2008-11-19 23:01:33 +08:00
|
|
|
{
|
2008-11-20 20:06:09 +08:00
|
|
|
int ret, i;
|
|
|
|
enum em28xx_amux amux = dev->ctl_ainput;
|
2008-11-19 23:01:33 +08:00
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
/*
|
|
|
|
* EM28XX_AMUX_VIDEO2 is a special case used to indicate that
|
|
|
|
* em28xx should point to LINE IN, while AC97 should use VIDEO
|
2008-11-20 20:06:09 +08:00
|
|
|
*/
|
|
|
|
if (amux == EM28XX_AMUX_VIDEO2)
|
2008-11-20 23:10:44 +08:00
|
|
|
amux = EM28XX_AMUX_VIDEO;
|
2008-11-19 23:01:33 +08:00
|
|
|
|
2008-11-20 20:06:09 +08:00
|
|
|
/* Mute all entres but the one that were selected */
|
|
|
|
for (i = 0; i < ARRAY_SIZE(inputs); i++) {
|
2008-11-21 00:39:39 +08:00
|
|
|
if (amux == inputs[i].mux)
|
2008-11-20 20:06:09 +08:00
|
|
|
ret = em28xx_write_ac97(dev, inputs[i].reg, 0x0808);
|
|
|
|
else
|
|
|
|
ret = em28xx_write_ac97(dev, inputs[i].reg, 0x8000);
|
2008-11-19 23:01:33 +08:00
|
|
|
|
2008-11-20 20:06:09 +08:00
|
|
|
if (ret < 0)
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"couldn't setup AC97 register %d\n",
|
|
|
|
inputs[i].reg);
|
2008-11-20 20:06:09 +08:00
|
|
|
}
|
|
|
|
return 0;
|
2005-11-09 13:37:07 +08:00
|
|
|
}
|
|
|
|
|
2008-02-07 05:34:13 +08:00
|
|
|
static int em28xx_set_audio_source(struct em28xx *dev)
|
2008-01-05 20:53:54 +08:00
|
|
|
{
|
2008-02-05 18:37:21 +08:00
|
|
|
int ret;
|
2008-01-05 20:53:54 +08:00
|
|
|
u8 input;
|
|
|
|
|
2008-11-25 20:39:50 +08:00
|
|
|
if (dev->board.is_em2800) {
|
2008-11-20 20:06:09 +08:00
|
|
|
if (dev->ctl_ainput == EM28XX_AMUX_VIDEO)
|
2008-01-05 20:53:54 +08:00
|
|
|
input = EM2800_AUDIO_SRC_TUNER;
|
2008-11-20 20:06:09 +08:00
|
|
|
else
|
|
|
|
input = EM2800_AUDIO_SRC_LINE;
|
2008-01-05 20:53:54 +08:00
|
|
|
|
2008-04-18 08:44:58 +08:00
|
|
|
ret = em28xx_write_regs(dev, EM2800_R08_AUDIOSRC, &input, 1);
|
2008-01-05 20:53:54 +08:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
if (dev->has_msp34xx) {
|
2008-01-05 20:53:54 +08:00
|
|
|
input = EM28XX_AUDIO_SRC_TUNER;
|
2018-03-04 02:43:34 +08:00
|
|
|
} else {
|
2008-01-05 20:53:54 +08:00
|
|
|
switch (dev->ctl_ainput) {
|
|
|
|
case EM28XX_AMUX_VIDEO:
|
|
|
|
input = EM28XX_AUDIO_SRC_TUNER;
|
|
|
|
break;
|
2008-11-19 23:01:33 +08:00
|
|
|
default:
|
2008-01-05 20:53:54 +08:00
|
|
|
input = EM28XX_AUDIO_SRC_LINE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-04 19:27:52 +08:00
|
|
|
if (dev->board.mute_gpio && dev->mute)
|
|
|
|
em28xx_gpio_set(dev, dev->board.mute_gpio);
|
|
|
|
else
|
|
|
|
em28xx_gpio_set(dev, INPUT(dev->ctl_input)->gpio);
|
|
|
|
|
2008-04-18 08:44:58 +08:00
|
|
|
ret = em28xx_write_reg_bits(dev, EM28XX_R0E_AUDIOSRC, input, 0xc0);
|
2008-01-05 20:53:54 +08:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2018-03-04 02:43:34 +08:00
|
|
|
usleep_range(10000, 11000);
|
2008-01-05 20:53:54 +08:00
|
|
|
|
2008-11-19 23:01:33 +08:00
|
|
|
switch (dev->audio_mode.ac97) {
|
|
|
|
case EM28XX_NO_AC97:
|
|
|
|
break;
|
2008-11-20 20:06:09 +08:00
|
|
|
default:
|
|
|
|
ret = set_ac97_input(dev);
|
2008-11-19 23:01:33 +08:00
|
|
|
}
|
2008-01-05 20:53:54 +08:00
|
|
|
|
2008-11-20 20:06:09 +08:00
|
|
|
return ret;
|
2008-01-05 20:53:54 +08:00
|
|
|
}
|
|
|
|
|
2011-06-18 18:02:49 +08:00
|
|
|
struct em28xx_vol_otable {
|
|
|
|
enum em28xx_aout mux;
|
|
|
|
u8 reg;
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct em28xx_vol_otable outputs[] = {
|
2012-06-12 02:17:23 +08:00
|
|
|
{ EM28XX_AOUT_MASTER, AC97_MASTER },
|
|
|
|
{ EM28XX_AOUT_LINE, AC97_HEADPHONE },
|
|
|
|
{ EM28XX_AOUT_MONO, AC97_MASTER_MONO },
|
|
|
|
{ EM28XX_AOUT_LFE, AC97_CENTER_LFE_MASTER },
|
|
|
|
{ EM28XX_AOUT_SURR, AC97_SURROUND_MASTER },
|
2008-11-20 23:40:51 +08:00
|
|
|
};
|
|
|
|
|
2005-11-09 13:38:27 +08:00
|
|
|
int em28xx_audio_analog_set(struct em28xx *dev)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
2008-11-20 23:40:51 +08:00
|
|
|
int ret, i;
|
2008-11-25 18:05:06 +08:00
|
|
|
u8 xclk;
|
2008-01-05 20:53:54 +08:00
|
|
|
|
2014-09-13 16:52:21 +08:00
|
|
|
if (dev->int_audio_type == EM28XX_INT_AUDIO_NONE)
|
2008-11-19 23:01:33 +08:00
|
|
|
return 0;
|
2008-01-05 20:53:54 +08:00
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
/*
|
|
|
|
* It is assumed that all devices use master volume for output.
|
|
|
|
* It would be possible to use also line output.
|
2008-11-20 20:06:09 +08:00
|
|
|
*/
|
2008-11-19 23:01:33 +08:00
|
|
|
if (dev->audio_mode.ac97 != EM28XX_NO_AC97) {
|
2008-11-20 23:40:51 +08:00
|
|
|
/* Mute all outputs */
|
|
|
|
for (i = 0; i < ARRAY_SIZE(outputs); i++) {
|
2008-11-21 00:39:39 +08:00
|
|
|
ret = em28xx_write_ac97(dev, outputs[i].reg, 0x8000);
|
2008-11-20 23:40:51 +08:00
|
|
|
if (ret < 0)
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"couldn't setup AC97 register %d\n",
|
|
|
|
outputs[i].reg);
|
2008-11-20 23:40:51 +08:00
|
|
|
}
|
2008-11-19 23:01:33 +08:00
|
|
|
}
|
2008-01-05 20:53:54 +08:00
|
|
|
|
2008-11-25 20:39:50 +08:00
|
|
|
xclk = dev->board.xclk & 0x7f;
|
2008-01-06 04:01:41 +08:00
|
|
|
if (!dev->mute)
|
2009-01-19 08:44:46 +08:00
|
|
|
xclk |= EM28XX_XCLK_AUDIO_UNMUTE;
|
2008-01-06 04:01:41 +08:00
|
|
|
|
2008-11-25 18:05:06 +08:00
|
|
|
ret = em28xx_write_reg(dev, EM28XX_R0F_XCLK, xclk);
|
2008-01-05 20:53:54 +08:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2018-03-04 02:43:34 +08:00
|
|
|
usleep_range(10000, 11000);
|
2008-01-05 20:53:54 +08:00
|
|
|
|
|
|
|
/* Selects the proper audio input */
|
|
|
|
ret = em28xx_set_audio_source(dev);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2008-11-19 23:01:33 +08:00
|
|
|
/* Sets volume */
|
|
|
|
if (dev->audio_mode.ac97 != EM28XX_NO_AC97) {
|
|
|
|
int vol;
|
|
|
|
|
2012-06-12 02:17:23 +08:00
|
|
|
em28xx_write_ac97(dev, AC97_POWERDOWN, 0x4200);
|
|
|
|
em28xx_write_ac97(dev, AC97_EXTENDED_STATUS, 0x0031);
|
|
|
|
em28xx_write_ac97(dev, AC97_PCM_LR_ADC_RATE, 0xbb80);
|
2009-01-19 08:45:28 +08:00
|
|
|
|
2008-11-19 23:01:33 +08:00
|
|
|
/* LSB: left channel - both channels with the same level */
|
|
|
|
vol = (0x1f - dev->volume) | ((0x1f - dev->volume) << 8);
|
|
|
|
|
|
|
|
/* Mute device, if needed */
|
|
|
|
if (dev->mute)
|
|
|
|
vol |= 0x8000;
|
|
|
|
|
|
|
|
/* Sets volume */
|
2008-11-21 00:39:39 +08:00
|
|
|
for (i = 0; i < ARRAY_SIZE(outputs); i++) {
|
|
|
|
if (dev->ctl_aoutput & outputs[i].mux)
|
|
|
|
ret = em28xx_write_ac97(dev, outputs[i].reg,
|
|
|
|
vol);
|
|
|
|
if (ret < 0)
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"couldn't setup AC97 register %d\n",
|
|
|
|
outputs[i].reg);
|
2008-11-21 00:39:39 +08:00
|
|
|
}
|
2009-01-13 08:50:52 +08:00
|
|
|
|
|
|
|
if (dev->ctl_aoutput & EM28XX_AOUT_PCM_IN) {
|
|
|
|
int sel = ac97_return_record_select(dev->ctl_aoutput);
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
/*
|
|
|
|
* Use the same input for both left and right
|
|
|
|
* channels
|
|
|
|
*/
|
2009-01-13 08:50:52 +08:00
|
|
|
sel |= (sel << 8);
|
|
|
|
|
2012-06-12 02:17:23 +08:00
|
|
|
em28xx_write_ac97(dev, AC97_REC_SEL, sel);
|
2009-01-13 08:50:52 +08:00
|
|
|
}
|
2008-11-19 23:01:33 +08:00
|
|
|
}
|
2008-02-07 05:34:13 +08:00
|
|
|
|
2008-01-05 20:53:54 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(em28xx_audio_analog_set);
|
2005-11-09 13:37:07 +08:00
|
|
|
|
2008-11-19 23:01:33 +08:00
|
|
|
int em28xx_audio_setup(struct em28xx *dev)
|
|
|
|
{
|
|
|
|
int vid1, vid2, feat, cfg;
|
2014-09-29 17:48:16 +08:00
|
|
|
u32 vid = 0;
|
2014-09-13 16:52:19 +08:00
|
|
|
u8 i2s_samplerates;
|
2008-11-19 23:01:33 +08:00
|
|
|
|
2013-12-27 11:16:13 +08:00
|
|
|
if (dev->chip_id == CHIP_ID_EM2870 ||
|
|
|
|
dev->chip_id == CHIP_ID_EM2874 ||
|
|
|
|
dev->chip_id == CHIP_ID_EM28174 ||
|
|
|
|
dev->chip_id == CHIP_ID_EM28178) {
|
|
|
|
/* Digital only device - don't load any alsa module */
|
2014-09-13 16:52:21 +08:00
|
|
|
dev->int_audio_type = EM28XX_INT_AUDIO_NONE;
|
2014-09-13 16:52:20 +08:00
|
|
|
dev->usb_audio_type = EM28XX_USB_AUDIO_NONE;
|
2008-11-19 23:01:33 +08:00
|
|
|
return 0;
|
2013-12-27 11:16:13 +08:00
|
|
|
}
|
|
|
|
|
2008-11-19 23:01:33 +08:00
|
|
|
/* See how this device is configured */
|
|
|
|
cfg = em28xx_read_reg(dev, EM28XX_R00_CHIPCFG);
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_info(&dev->intf->dev, "Config register raw data: 0x%02x\n", cfg);
|
2014-09-13 16:56:46 +08:00
|
|
|
if (cfg < 0) { /* Register read error */
|
|
|
|
/* Be conservative */
|
2014-09-13 16:52:21 +08:00
|
|
|
dev->int_audio_type = EM28XX_INT_AUDIO_AC97;
|
2009-06-10 10:40:39 +08:00
|
|
|
} else if ((cfg & EM28XX_CHIPCFG_AUDIOMASK) == 0x00) {
|
|
|
|
/* The device doesn't have vendor audio at all */
|
2014-09-13 16:52:21 +08:00
|
|
|
dev->int_audio_type = EM28XX_INT_AUDIO_NONE;
|
2014-09-13 16:52:20 +08:00
|
|
|
dev->usb_audio_type = EM28XX_USB_AUDIO_NONE;
|
2009-06-10 10:40:39 +08:00
|
|
|
return 0;
|
2013-12-22 22:17:46 +08:00
|
|
|
} else if ((cfg & EM28XX_CHIPCFG_AUDIOMASK) != EM28XX_CHIPCFG_AC97) {
|
2014-09-13 16:52:21 +08:00
|
|
|
dev->int_audio_type = EM28XX_INT_AUDIO_I2S;
|
2013-12-22 22:17:46 +08:00
|
|
|
if (dev->chip_id < CHIP_ID_EM2860 &&
|
2014-11-28 19:34:15 +08:00
|
|
|
(cfg & EM28XX_CHIPCFG_AUDIOMASK) ==
|
2013-12-22 22:17:46 +08:00
|
|
|
EM2820_CHIPCFG_I2S_1_SAMPRATE)
|
2014-09-13 16:52:19 +08:00
|
|
|
i2s_samplerates = 1;
|
2013-12-22 22:17:46 +08:00
|
|
|
else if (dev->chip_id >= CHIP_ID_EM2860 &&
|
|
|
|
(cfg & EM28XX_CHIPCFG_AUDIOMASK) ==
|
|
|
|
EM2860_CHIPCFG_I2S_5_SAMPRATES)
|
2014-09-13 16:52:19 +08:00
|
|
|
i2s_samplerates = 5;
|
2013-12-22 22:17:46 +08:00
|
|
|
else
|
2014-09-13 16:52:19 +08:00
|
|
|
i2s_samplerates = 3;
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_info(&dev->intf->dev, "I2S Audio (%d sample rate(s))\n",
|
2018-03-04 02:43:34 +08:00
|
|
|
i2s_samplerates);
|
2008-12-30 11:17:09 +08:00
|
|
|
/* Skip the code that does AC97 vendor detection */
|
2008-11-19 23:01:33 +08:00
|
|
|
dev->audio_mode.ac97 = EM28XX_NO_AC97;
|
|
|
|
goto init_audio;
|
2014-09-13 16:52:21 +08:00
|
|
|
} else {
|
|
|
|
dev->int_audio_type = EM28XX_INT_AUDIO_AC97;
|
2008-11-19 23:01:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
dev->audio_mode.ac97 = EM28XX_AC97_OTHER;
|
|
|
|
|
|
|
|
vid1 = em28xx_read_ac97(dev, AC97_VENDOR_ID1);
|
|
|
|
if (vid1 < 0) {
|
2009-11-27 23:52:54 +08:00
|
|
|
/*
|
|
|
|
* Device likely doesn't support AC97
|
|
|
|
* Note: (some) em2800 devices without eeprom reports 0x91 on
|
|
|
|
* CHIPCFG register, even not having an AC97 chip
|
|
|
|
*/
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"AC97 chip type couldn't be determined\n");
|
2009-11-27 23:52:54 +08:00
|
|
|
dev->audio_mode.ac97 = EM28XX_NO_AC97;
|
2014-09-13 16:52:20 +08:00
|
|
|
if (dev->usb_audio_type == EM28XX_USB_AUDIO_VENDOR)
|
|
|
|
dev->usb_audio_type = EM28XX_USB_AUDIO_NONE;
|
2014-09-13 16:52:21 +08:00
|
|
|
dev->int_audio_type = EM28XX_INT_AUDIO_NONE;
|
2008-11-19 23:01:33 +08:00
|
|
|
goto init_audio;
|
|
|
|
}
|
|
|
|
|
|
|
|
vid2 = em28xx_read_ac97(dev, AC97_VENDOR_ID2);
|
|
|
|
if (vid2 < 0)
|
|
|
|
goto init_audio;
|
|
|
|
|
2008-11-20 19:56:19 +08:00
|
|
|
vid = vid1 << 16 | vid2;
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev, "AC97 vendor ID = 0x%08x\n", vid);
|
2008-11-19 23:01:33 +08:00
|
|
|
|
|
|
|
feat = em28xx_read_ac97(dev, AC97_RESET);
|
|
|
|
if (feat < 0)
|
|
|
|
goto init_audio;
|
|
|
|
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev, "AC97 features = 0x%04x\n", feat);
|
2008-11-19 23:01:33 +08:00
|
|
|
|
2008-11-20 19:56:19 +08:00
|
|
|
/* Try to identify what audio processor we have */
|
2018-03-04 02:43:34 +08:00
|
|
|
if ((vid == 0xffffffff || vid == 0x83847650) && feat == 0x6a90)
|
2008-11-19 23:01:33 +08:00
|
|
|
dev->audio_mode.ac97 = EM28XX_AC97_EM202;
|
2008-11-20 23:49:33 +08:00
|
|
|
else if ((vid >> 8) == 0x838476)
|
|
|
|
dev->audio_mode.ac97 = EM28XX_AC97_SIGMATEL;
|
2008-11-19 23:01:33 +08:00
|
|
|
|
|
|
|
init_audio:
|
|
|
|
/* Reports detected AC97 processor */
|
|
|
|
switch (dev->audio_mode.ac97) {
|
|
|
|
case EM28XX_NO_AC97:
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_info(&dev->intf->dev, "No AC97 audio processor\n");
|
2008-11-19 23:01:33 +08:00
|
|
|
break;
|
|
|
|
case EM28XX_AC97_EM202:
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_info(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"Empia 202 AC97 audio processor detected\n");
|
2008-11-19 23:01:33 +08:00
|
|
|
break;
|
2008-11-20 23:49:33 +08:00
|
|
|
case EM28XX_AC97_SIGMATEL:
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_info(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"Sigmatel audio processor detected (stac 97%02x)\n",
|
|
|
|
vid & 0xff);
|
2008-11-20 23:49:33 +08:00
|
|
|
break;
|
2008-11-19 23:01:33 +08:00
|
|
|
case EM28XX_AC97_OTHER:
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_warn(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"Unknown AC97 audio processor detected!\n");
|
2008-11-19 23:01:33 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return em28xx_audio_analog_set(dev);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(em28xx_audio_setup);
|
|
|
|
|
2013-12-02 05:06:55 +08:00
|
|
|
const struct em28xx_led *em28xx_find_led(struct em28xx *dev,
|
|
|
|
enum em28xx_led_role role)
|
|
|
|
{
|
|
|
|
if (dev->board.leds) {
|
|
|
|
u8 k = 0;
|
2014-11-28 19:34:15 +08:00
|
|
|
|
2013-12-02 05:06:55 +08:00
|
|
|
while (dev->board.leds[k].role >= 0 &&
|
2014-11-28 19:34:15 +08:00
|
|
|
dev->board.leds[k].role < EM28XX_NUM_LED_ROLES) {
|
2013-12-02 05:06:55 +08:00
|
|
|
if (dev->board.leds[k].role == role)
|
|
|
|
return &dev->board.leds[k];
|
|
|
|
k++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(em28xx_find_led);
|
|
|
|
|
2005-11-09 13:38:27 +08:00
|
|
|
int em28xx_capture_start(struct em28xx *dev, int start)
|
2005-11-09 13:37:07 +08:00
|
|
|
{
|
2008-04-18 08:37:31 +08:00
|
|
|
int rc;
|
2014-03-04 23:33:45 +08:00
|
|
|
const struct em28xx_led *led = NULL;
|
2008-11-12 13:05:24 +08:00
|
|
|
|
2011-07-04 08:05:06 +08:00
|
|
|
if (dev->chip_id == CHIP_ID_EM2874 ||
|
|
|
|
dev->chip_id == CHIP_ID_EM2884 ||
|
2013-02-25 19:19:04 +08:00
|
|
|
dev->chip_id == CHIP_ID_EM28174 ||
|
|
|
|
dev->chip_id == CHIP_ID_EM28178) {
|
2008-11-12 13:05:24 +08:00
|
|
|
/* The Transport Stream Enable Register moved in em2874 */
|
2018-02-02 06:04:37 +08:00
|
|
|
if (dev->dvb_xfer_bulk) {
|
|
|
|
/* Max Tx Size = 188 * 256 = 48128 - LCM(188,512) * 2 */
|
|
|
|
em28xx_write_reg(dev, (dev->ts == PRIMARY_TS) ?
|
2018-03-04 02:43:34 +08:00
|
|
|
EM2874_R5D_TS1_PKT_SIZE :
|
|
|
|
EM2874_R5E_TS2_PKT_SIZE,
|
2018-03-07 03:14:56 +08:00
|
|
|
0xff);
|
2018-02-02 06:04:37 +08:00
|
|
|
} else {
|
|
|
|
/* ISOC Maximum Transfer Size = 188 * 5 */
|
|
|
|
em28xx_write_reg(dev, (dev->ts == PRIMARY_TS) ?
|
2018-03-04 02:43:34 +08:00
|
|
|
EM2874_R5D_TS1_PKT_SIZE :
|
|
|
|
EM2874_R5E_TS2_PKT_SIZE,
|
|
|
|
dev->dvb_max_pkt_size_isoc / 188);
|
2018-02-02 06:04:37 +08:00
|
|
|
}
|
2018-01-05 08:04:11 +08:00
|
|
|
if (dev->ts == PRIMARY_TS)
|
|
|
|
rc = em28xx_write_reg_bits(dev,
|
2018-03-04 02:43:34 +08:00
|
|
|
EM2874_R5F_TS_ENABLE,
|
|
|
|
start ? EM2874_TS1_CAPTURE_ENABLE : 0x00,
|
|
|
|
EM2874_TS1_CAPTURE_ENABLE);
|
2018-01-05 08:04:11 +08:00
|
|
|
else
|
|
|
|
rc = em28xx_write_reg_bits(dev,
|
2018-03-04 02:43:34 +08:00
|
|
|
EM2874_R5F_TS_ENABLE,
|
|
|
|
start ? EM2874_TS2_CAPTURE_ENABLE : 0x00,
|
|
|
|
EM2874_TS2_CAPTURE_ENABLE);
|
2013-12-02 05:06:51 +08:00
|
|
|
} else {
|
|
|
|
/* FIXME: which is the best order? */
|
|
|
|
/* video registers are sampled by VREF */
|
|
|
|
rc = em28xx_write_reg_bits(dev, EM28XX_R0C_USBSUSP,
|
|
|
|
start ? 0x10 : 0x00, 0x10);
|
|
|
|
if (rc < 0)
|
|
|
|
return rc;
|
2008-11-12 13:05:24 +08:00
|
|
|
|
2013-12-02 05:06:51 +08:00
|
|
|
if (start) {
|
2018-03-01 23:08:42 +08:00
|
|
|
if (dev->is_webcam)
|
2013-12-02 05:06:51 +08:00
|
|
|
rc = em28xx_write_reg(dev, 0x13, 0x0c);
|
2008-11-12 13:05:24 +08:00
|
|
|
|
2013-12-02 05:06:51 +08:00
|
|
|
/* Enable video capture */
|
|
|
|
rc = em28xx_write_reg(dev, 0x48, 0x00);
|
2014-03-04 23:33:45 +08:00
|
|
|
if (rc < 0)
|
|
|
|
return rc;
|
2008-04-18 08:37:31 +08:00
|
|
|
|
2013-12-02 05:06:51 +08:00
|
|
|
if (dev->mode == EM28XX_ANALOG_MODE)
|
|
|
|
rc = em28xx_write_reg(dev,
|
2014-11-28 19:34:15 +08:00
|
|
|
EM28XX_R12_VINENABLE,
|
|
|
|
0x67);
|
2013-12-02 05:06:51 +08:00
|
|
|
else
|
|
|
|
rc = em28xx_write_reg(dev,
|
2014-11-28 19:34:15 +08:00
|
|
|
EM28XX_R12_VINENABLE,
|
|
|
|
0x37);
|
2014-03-04 23:33:45 +08:00
|
|
|
if (rc < 0)
|
|
|
|
return rc;
|
2009-08-08 05:43:00 +08:00
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
usleep_range(10000, 11000);
|
2013-12-02 05:06:51 +08:00
|
|
|
} else {
|
|
|
|
/* disable video capture */
|
|
|
|
rc = em28xx_write_reg(dev, EM28XX_R12_VINENABLE, 0x27);
|
|
|
|
}
|
|
|
|
}
|
2008-04-18 08:40:45 +08:00
|
|
|
|
2014-03-04 23:33:45 +08:00
|
|
|
if (dev->mode == EM28XX_ANALOG_MODE)
|
2013-12-02 05:06:55 +08:00
|
|
|
led = em28xx_find_led(dev, EM28XX_LED_ANALOG_CAPTURING);
|
2014-03-04 23:33:45 +08:00
|
|
|
else
|
|
|
|
led = em28xx_find_led(dev, EM28XX_LED_DIGITAL_CAPTURING);
|
|
|
|
|
|
|
|
if (led)
|
|
|
|
em28xx_write_reg_bits(dev, led->gpio_reg,
|
|
|
|
(!start ^ led->inverted) ?
|
|
|
|
~led->gpio_mask : led->gpio_mask,
|
|
|
|
led->gpio_mask);
|
2008-04-18 08:37:31 +08:00
|
|
|
|
|
|
|
return rc;
|
2005-11-09 13:37:07 +08:00
|
|
|
}
|
|
|
|
|
2018-03-01 23:34:34 +08:00
|
|
|
int em28xx_gpio_set(struct em28xx *dev, const struct em28xx_reg_seq *gpio)
|
2008-04-18 08:48:00 +08:00
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
if (!gpio)
|
|
|
|
return rc;
|
|
|
|
|
2008-11-27 20:10:40 +08:00
|
|
|
if (dev->mode != EM28XX_SUSPEND) {
|
|
|
|
em28xx_write_reg(dev, 0x48, 0x00);
|
|
|
|
if (dev->mode == EM28XX_ANALOG_MODE)
|
|
|
|
em28xx_write_reg(dev, EM28XX_R12_VINENABLE, 0x67);
|
|
|
|
else
|
|
|
|
em28xx_write_reg(dev, EM28XX_R12_VINENABLE, 0x37);
|
2018-03-04 02:43:34 +08:00
|
|
|
usleep_range(10000, 11000);
|
2008-11-27 20:10:40 +08:00
|
|
|
}
|
2008-04-18 08:48:00 +08:00
|
|
|
|
|
|
|
/* Send GPIO reset sequences specified at board entry */
|
|
|
|
while (gpio->sleep >= 0) {
|
|
|
|
if (gpio->reg >= 0) {
|
|
|
|
rc = em28xx_write_reg_bits(dev,
|
|
|
|
gpio->reg,
|
|
|
|
gpio->val,
|
|
|
|
gpio->mask);
|
|
|
|
if (rc < 0)
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
if (gpio->sleep > 0)
|
|
|
|
msleep(gpio->sleep);
|
|
|
|
|
|
|
|
gpio++;
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
2011-07-04 08:05:06 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_gpio_set);
|
2008-04-18 08:48:00 +08:00
|
|
|
|
|
|
|
int em28xx_set_mode(struct em28xx *dev, enum em28xx_mode set_mode)
|
|
|
|
{
|
|
|
|
if (dev->mode == set_mode)
|
|
|
|
return 0;
|
|
|
|
|
2008-11-27 20:10:40 +08:00
|
|
|
if (set_mode == EM28XX_SUSPEND) {
|
2008-04-18 08:48:00 +08:00
|
|
|
dev->mode = set_mode;
|
2008-11-27 20:10:40 +08:00
|
|
|
|
|
|
|
/* FIXME: add suspend support for ac97 */
|
|
|
|
|
|
|
|
return em28xx_gpio_set(dev, dev->board.suspend_gpio);
|
2008-04-18 08:48:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
dev->mode = set_mode;
|
|
|
|
|
|
|
|
if (dev->mode == EM28XX_DIGITAL_MODE)
|
2008-11-27 20:00:00 +08:00
|
|
|
return em28xx_gpio_set(dev, dev->board.dvb_gpio);
|
2008-04-18 08:48:00 +08:00
|
|
|
else
|
2008-11-27 20:00:00 +08:00
|
|
|
return em28xx_gpio_set(dev, INPUT(dev->ctl_input)->gpio);
|
2008-04-18 08:48:00 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(em28xx_set_mode);
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
/*
|
|
|
|
*URB control
|
|
|
|
*/
|
2008-04-18 08:40:16 +08:00
|
|
|
|
|
|
|
/*
|
2012-11-09 01:11:39 +08:00
|
|
|
* URB completion handler for isoc/bulk transfers
|
2008-04-18 08:40:16 +08:00
|
|
|
*/
|
|
|
|
static void em28xx_irq_callback(struct urb *urb)
|
|
|
|
{
|
2009-09-01 12:19:46 +08:00
|
|
|
struct em28xx *dev = urb->context;
|
2011-06-18 07:28:51 +08:00
|
|
|
int i;
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2009-02-09 00:09:11 +08:00
|
|
|
switch (urb->status) {
|
|
|
|
case 0: /* success */
|
|
|
|
case -ETIMEDOUT: /* NAK */
|
|
|
|
break;
|
|
|
|
case -ECONNRESET: /* kill */
|
|
|
|
case -ENOENT:
|
|
|
|
case -ESHUTDOWN:
|
|
|
|
return;
|
|
|
|
default: /* error */
|
2018-03-04 02:43:34 +08:00
|
|
|
em28xx_isocdbg("urb completion error %d.\n", urb->status);
|
2009-02-09 00:09:11 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-04-18 08:40:16 +08:00
|
|
|
/* Copy data from URB */
|
|
|
|
spin_lock(&dev->slock);
|
2012-11-09 01:11:37 +08:00
|
|
|
dev->usb_ctl.urb_data_copy(dev, urb);
|
2008-04-18 08:40:16 +08:00
|
|
|
spin_unlock(&dev->slock);
|
|
|
|
|
|
|
|
/* Reset urb buffers */
|
|
|
|
for (i = 0; i < urb->number_of_packets; i++) {
|
2012-11-09 01:11:39 +08:00
|
|
|
/* isoc only (bulk: number_of_packets = 0) */
|
2008-04-18 08:40:16 +08:00
|
|
|
urb->iso_frame_desc[i].status = 0;
|
|
|
|
urb->iso_frame_desc[i].actual_length = 0;
|
|
|
|
}
|
|
|
|
urb->status = 0;
|
|
|
|
|
|
|
|
urb->status = usb_submit_urb(urb, GFP_ATOMIC);
|
|
|
|
if (urb->status) {
|
2008-04-21 18:01:09 +08:00
|
|
|
em28xx_isocdbg("urb resubmit failed (error=%i)\n",
|
|
|
|
urb->status);
|
2008-04-18 08:40:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Stop and Deallocate URBs
|
|
|
|
*/
|
2012-11-09 01:11:40 +08:00
|
|
|
void em28xx_uninit_usb_xfer(struct em28xx *dev, enum em28xx_mode mode)
|
2008-04-18 08:40:16 +08:00
|
|
|
{
|
|
|
|
struct urb *urb;
|
2012-11-09 01:11:40 +08:00
|
|
|
struct em28xx_usb_bufs *usb_bufs;
|
2008-04-18 08:40:16 +08:00
|
|
|
int i;
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
em28xx_isocdbg("called %s in mode %d\n", __func__, mode);
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
|
|
|
|
if (mode == EM28XX_DIGITAL_MODE)
|
2012-11-09 01:11:40 +08:00
|
|
|
usb_bufs = &dev->usb_ctl.digital_bufs;
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
else
|
2012-11-09 01:11:40 +08:00
|
|
|
usb_bufs = &dev->usb_ctl.analog_bufs;
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2012-11-09 01:11:40 +08:00
|
|
|
for (i = 0; i < usb_bufs->num_bufs; i++) {
|
|
|
|
urb = usb_bufs->urb[i];
|
2008-04-18 08:40:16 +08:00
|
|
|
if (urb) {
|
2009-01-26 00:08:07 +08:00
|
|
|
if (!irqs_disabled())
|
|
|
|
usb_kill_urb(urb);
|
|
|
|
else
|
|
|
|
usb_unlink_urb(urb);
|
|
|
|
|
2008-04-18 08:40:16 +08:00
|
|
|
usb_free_urb(urb);
|
2012-11-09 01:11:40 +08:00
|
|
|
usb_bufs->urb[i] = NULL;
|
2008-04-18 08:40:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-09 01:11:40 +08:00
|
|
|
kfree(usb_bufs->urb);
|
2018-02-27 19:08:09 +08:00
|
|
|
kfree(usb_bufs->buf);
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2012-11-09 01:11:40 +08:00
|
|
|
usb_bufs->urb = NULL;
|
2018-02-27 19:08:09 +08:00
|
|
|
usb_bufs->buf = NULL;
|
2012-11-09 01:11:40 +08:00
|
|
|
usb_bufs->num_bufs = 0;
|
2008-04-18 08:40:16 +08:00
|
|
|
|
|
|
|
em28xx_capture_start(dev, 0);
|
|
|
|
}
|
2012-11-09 01:11:40 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_uninit_usb_xfer);
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2012-03-22 19:48:17 +08:00
|
|
|
/*
|
|
|
|
* Stop URBs
|
|
|
|
*/
|
|
|
|
void em28xx_stop_urbs(struct em28xx *dev)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct urb *urb;
|
2012-11-09 01:11:37 +08:00
|
|
|
struct em28xx_usb_bufs *isoc_bufs = &dev->usb_ctl.digital_bufs;
|
2012-03-22 19:48:17 +08:00
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
em28xx_isocdbg("called %s\n", __func__);
|
2012-03-22 19:48:17 +08:00
|
|
|
|
|
|
|
for (i = 0; i < isoc_bufs->num_bufs; i++) {
|
|
|
|
urb = isoc_bufs->urb[i];
|
|
|
|
if (urb) {
|
|
|
|
if (!irqs_disabled())
|
|
|
|
usb_kill_urb(urb);
|
|
|
|
else
|
|
|
|
usb_unlink_urb(urb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
em28xx_capture_start(dev, 0);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(em28xx_stop_urbs);
|
|
|
|
|
2008-04-18 08:40:16 +08:00
|
|
|
/*
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
* Allocate URBs
|
2008-04-18 08:40:16 +08:00
|
|
|
*/
|
2012-11-09 01:11:41 +08:00
|
|
|
int em28xx_alloc_urbs(struct em28xx *dev, enum em28xx_mode mode, int xfer_bulk,
|
|
|
|
int num_bufs, int max_pkt_size, int packet_multiplier)
|
2008-04-18 08:40:16 +08:00
|
|
|
{
|
2012-11-09 01:11:41 +08:00
|
|
|
struct em28xx_usb_bufs *usb_bufs;
|
2016-12-08 00:34:22 +08:00
|
|
|
struct urb *urb;
|
|
|
|
struct usb_device *udev = interface_to_usbdev(dev->intf);
|
2008-04-18 08:40:16 +08:00
|
|
|
int i;
|
|
|
|
int sb_size, pipe;
|
|
|
|
int j, k;
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
em28xx_isocdbg("em28xx: called %s in mode %d\n", __func__, mode);
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
/*
|
|
|
|
* Check mode and if we have an endpoint for the selected
|
|
|
|
* transfer type, select buffer
|
|
|
|
*/
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
if (mode == EM28XX_DIGITAL_MODE) {
|
|
|
|
if ((xfer_bulk && !dev->dvb_ep_bulk) ||
|
|
|
|
(!xfer_bulk && !dev->dvb_ep_isoc)) {
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_err(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"no endpoint for DVB mode and transfer type %d\n",
|
|
|
|
xfer_bulk > 0);
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2012-11-09 01:11:41 +08:00
|
|
|
usb_bufs = &dev->usb_ctl.digital_bufs;
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
} else if (mode == EM28XX_ANALOG_MODE) {
|
|
|
|
if ((xfer_bulk && !dev->analog_ep_bulk) ||
|
|
|
|
(!xfer_bulk && !dev->analog_ep_isoc)) {
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_err(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"no endpoint for analog mode and transfer type %d\n",
|
|
|
|
xfer_bulk > 0);
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2012-11-09 01:11:41 +08:00
|
|
|
usb_bufs = &dev->usb_ctl.analog_bufs;
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
} else {
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_err(&dev->intf->dev, "invalid mode selected\n");
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2008-04-18 08:40:16 +08:00
|
|
|
|
|
|
|
/* De-allocates all pending stuff */
|
2012-11-09 01:11:40 +08:00
|
|
|
em28xx_uninit_usb_xfer(dev, mode);
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2012-11-09 01:11:41 +08:00
|
|
|
usb_bufs->num_bufs = num_bufs;
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
usb_bufs->urb = kcalloc(num_bufs, sizeof(void *), GFP_KERNEL);
|
2016-10-12 18:26:47 +08:00
|
|
|
if (!usb_bufs->urb)
|
2008-04-18 08:40:16 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
usb_bufs->buf = kcalloc(num_bufs, sizeof(void *), GFP_KERNEL);
|
2018-02-27 19:08:09 +08:00
|
|
|
if (!usb_bufs->buf) {
|
|
|
|
kfree(usb_bufs->buf);
|
2008-04-18 08:40:16 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2012-11-09 01:11:41 +08:00
|
|
|
usb_bufs->max_pkt_size = max_pkt_size;
|
|
|
|
if (xfer_bulk)
|
|
|
|
usb_bufs->num_packets = 0;
|
|
|
|
else
|
|
|
|
usb_bufs->num_packets = packet_multiplier;
|
2012-11-09 01:11:37 +08:00
|
|
|
dev->usb_ctl.vid_buf = NULL;
|
|
|
|
dev->usb_ctl.vbi_buf = NULL;
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2012-11-09 01:11:41 +08:00
|
|
|
sb_size = packet_multiplier * usb_bufs->max_pkt_size;
|
2008-04-18 08:40:16 +08:00
|
|
|
|
|
|
|
/* allocate urbs and transfer buffers */
|
2012-11-09 01:11:41 +08:00
|
|
|
for (i = 0; i < usb_bufs->num_bufs; i++) {
|
|
|
|
urb = usb_alloc_urb(usb_bufs->num_packets, GFP_KERNEL);
|
2008-04-18 08:40:16 +08:00
|
|
|
if (!urb) {
|
2012-11-09 01:11:40 +08:00
|
|
|
em28xx_uninit_usb_xfer(dev, mode);
|
2008-04-18 08:40:16 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2012-11-09 01:11:41 +08:00
|
|
|
usb_bufs->urb[i] = urb;
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2018-02-27 19:08:09 +08:00
|
|
|
usb_bufs->buf[i] = kzalloc(sb_size, GFP_KERNEL);
|
|
|
|
if (!usb_bufs->buf[i]) {
|
2012-11-09 01:11:40 +08:00
|
|
|
em28xx_uninit_usb_xfer(dev, mode);
|
2018-02-27 19:08:09 +08:00
|
|
|
|
|
|
|
for (i--; i >= 0; i--)
|
|
|
|
kfree(usb_bufs->buf[i]);
|
|
|
|
|
|
|
|
kfree(usb_bufs->buf);
|
|
|
|
usb_bufs->buf = NULL;
|
|
|
|
|
2008-04-18 08:40:16 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2018-02-27 19:08:09 +08:00
|
|
|
|
|
|
|
urb->transfer_flags = URB_FREE_BUFFER;
|
2012-11-09 01:11:41 +08:00
|
|
|
|
|
|
|
if (xfer_bulk) { /* bulk */
|
2016-12-08 00:34:22 +08:00
|
|
|
pipe = usb_rcvbulkpipe(udev,
|
2012-11-09 01:11:41 +08:00
|
|
|
mode == EM28XX_ANALOG_MODE ?
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
dev->analog_ep_bulk :
|
|
|
|
dev->dvb_ep_bulk);
|
2018-02-27 19:08:09 +08:00
|
|
|
usb_fill_bulk_urb(urb, udev, pipe, usb_bufs->buf[i],
|
|
|
|
sb_size, em28xx_irq_callback, dev);
|
2012-11-09 01:11:41 +08:00
|
|
|
} else { /* isoc */
|
2016-12-08 00:34:22 +08:00
|
|
|
pipe = usb_rcvisocpipe(udev,
|
2012-11-09 01:11:41 +08:00
|
|
|
mode == EM28XX_ANALOG_MODE ?
|
[media] em28xx: improve USB endpoint logic, also use bulk transfers
The current enpoint logic ignores all bulk endpoints and uses
a fixed mapping between endpint addresses and the supported
data stream types (analog/audio/DVB):
Ep 0x82, isoc => analog
Ep 0x83, isoc => audio
Ep 0x84, isoc => DVB
Now that the code can also do bulk transfers, the endpoint
logic has to be extended to also consider bulk endpoints.
The new logic preserves backwards compatibility and reflects
the endpoint configurations we have seen so far:
Ep 0x82, isoc => analog
Ep 0x82, bulk => analog
Ep 0x83, isoc* => audio
Ep 0x84, isoc => digital
Ep 0x84, bulk => analog or digital**
(*: audio should always be isoc)
(**: analog, if ep 0x82 is isoc, otherwise digital)
[mchehab@redhat.com: Fix a CodingStyle issue: don't break strings
into separate lines]
Signed-off-by: Frank Schäfer <fschaefer.oss@googlemail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-11-09 01:11:52 +08:00
|
|
|
dev->analog_ep_isoc :
|
|
|
|
dev->dvb_ep_isoc);
|
2018-02-27 19:08:09 +08:00
|
|
|
usb_fill_int_urb(urb, udev, pipe, usb_bufs->buf[i],
|
|
|
|
sb_size, em28xx_irq_callback, dev, 1);
|
|
|
|
urb->transfer_flags |= URB_ISO_ASAP;
|
2012-11-09 01:11:41 +08:00
|
|
|
k = 0;
|
|
|
|
for (j = 0; j < usb_bufs->num_packets; j++) {
|
|
|
|
urb->iso_frame_desc[j].offset = k;
|
|
|
|
urb->iso_frame_desc[j].length =
|
|
|
|
usb_bufs->max_pkt_size;
|
|
|
|
k += usb_bufs->max_pkt_size;
|
|
|
|
}
|
2008-04-18 08:40:16 +08:00
|
|
|
}
|
2012-11-09 01:11:41 +08:00
|
|
|
|
|
|
|
urb->number_of_packets = usb_bufs->num_packets;
|
2008-04-18 08:40:16 +08:00
|
|
|
}
|
|
|
|
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2012-11-09 01:11:41 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_alloc_urbs);
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate URBs and start IRQ
|
|
|
|
*/
|
2012-11-09 01:11:42 +08:00
|
|
|
int em28xx_init_usb_xfer(struct em28xx *dev, enum em28xx_mode mode,
|
2014-11-28 19:34:15 +08:00
|
|
|
int xfer_bulk, int num_bufs, int max_pkt_size,
|
2012-11-09 01:11:42 +08:00
|
|
|
int packet_multiplier,
|
2014-11-28 19:34:15 +08:00
|
|
|
int (*urb_data_copy)(struct em28xx *dev, struct urb *urb))
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
{
|
|
|
|
struct em28xx_dmaqueue *dma_q = &dev->vidq;
|
|
|
|
struct em28xx_dmaqueue *vbi_dma_q = &dev->vbiq;
|
2012-11-09 01:11:42 +08:00
|
|
|
struct em28xx_usb_bufs *usb_bufs;
|
2016-12-08 00:34:22 +08:00
|
|
|
struct usb_device *udev = interface_to_usbdev(dev->intf);
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
int i;
|
|
|
|
int rc;
|
|
|
|
int alloc;
|
|
|
|
|
2018-03-04 02:43:34 +08:00
|
|
|
em28xx_isocdbg("em28xx: called %s in mode %d\n", __func__, mode);
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
|
2012-11-09 01:11:42 +08:00
|
|
|
dev->usb_ctl.urb_data_copy = urb_data_copy;
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
|
|
|
|
if (mode == EM28XX_DIGITAL_MODE) {
|
2012-11-09 01:11:42 +08:00
|
|
|
usb_bufs = &dev->usb_ctl.digital_bufs;
|
|
|
|
/* no need to free/alloc usb buffers in digital mode */
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
alloc = 0;
|
|
|
|
} else {
|
2012-11-09 01:11:42 +08:00
|
|
|
usb_bufs = &dev->usb_ctl.analog_bufs;
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
alloc = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (alloc) {
|
2012-11-09 01:11:42 +08:00
|
|
|
rc = em28xx_alloc_urbs(dev, mode, xfer_bulk, num_bufs,
|
|
|
|
max_pkt_size, packet_multiplier);
|
[media] em28xx: pre-allocate DVB isoc transfer buffers
On MIPS/ARM set-top-boxes, as well as old x86 PCs, memory allocation failures
in the em28xx driver are common, due to memory fragmentation over time, that
makes impossible to allocate large chunks of coherent memory.
A typical system with 256/512 MB of RAM fails after just 1 day of uptime (see
the old thread for detailed reports and crashlogs).
In fact, the em28xx driver allocates memory for USB isoc transfers at runtime,
as opposite to the dvb-usb drivers that allocates the USB buffers when the
device is initialized, and frees them when the device is disconnected.
Moreover, in digital mode the USB isoc transfer buffers are freed, allocated
and cleared every time the user selects a new channel, wasting time and
resources.
This patch solves both problems by allocating DVB isoc transfer buffers in
em28xx_usb_probe(), and freeing them in em28xx_usb_disconnect().
In fact, the buffers size and number depend only on the max USB packet size
that is parsed from the USB descriptors in em28xx_usb_probe(), so it can
never change for a given device.
This approach makes no sense in analog mode (as the buffer size depends on
the alternate mode selected at runtime), the patch creates two separate sets
of buffers for digital and analog modes.
For digital-only devices, USB buffers are created when the device is probed
and freed when the device is disconnected.
For analog-only devices, nothing changes: isoc buffers are created at runtime.
For hybrid devices, two sets of buffers are maintained: the digital-mode
buffers are created when the device is probed, and freed when the device is
disconnected; analog-mode buffers are created/destroyed at runtime as before.
So, in analog mode, digital and analog buffers coexists at the same time: this
can be justified by the fact that digital mode is by far more commonly used
nowadays, so it makes sense to optimize the driver for this use case scenario.
The patch has been tested in the last few days on a x86 PC and a MIPS
set-top-box, with the PCTV 290e (digital only) and the Terratec Hybrid XS
(hybrid device). With the latter, I switched several times between analog and
digital mode (Kaffeine/TvTime) with no issue at all.
I unplugged/plugged the devices several times with no problem.
Also, after over 3 days of normal usage in the MPIS set-top-box, the PCTV 290e
was still up and running.
Signed-off-by: Gianluca Gennari <gennarone@gmail.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
2012-02-14 00:59:22 +08:00
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2012-11-09 01:11:43 +08:00
|
|
|
if (xfer_bulk) {
|
2016-12-08 00:34:22 +08:00
|
|
|
rc = usb_clear_halt(udev, usb_bufs->urb[0]->pipe);
|
2012-11-09 01:11:43 +08:00
|
|
|
if (rc < 0) {
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_err(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"failed to clear USB bulk endpoint stall/halt condition (error=%i)\n",
|
2016-10-12 18:32:23 +08:00
|
|
|
rc);
|
2012-11-09 01:11:43 +08:00
|
|
|
em28xx_uninit_usb_xfer(dev, mode);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-18 08:40:16 +08:00
|
|
|
init_waitqueue_head(&dma_q->wq);
|
2009-09-01 12:54:54 +08:00
|
|
|
init_waitqueue_head(&vbi_dma_q->wq);
|
2008-04-18 08:40:16 +08:00
|
|
|
|
2008-04-18 08:48:00 +08:00
|
|
|
em28xx_capture_start(dev, 1);
|
2008-04-18 08:40:16 +08:00
|
|
|
|
|
|
|
/* submit urbs and enables IRQ */
|
2012-11-09 01:11:42 +08:00
|
|
|
for (i = 0; i < usb_bufs->num_bufs; i++) {
|
|
|
|
rc = usb_submit_urb(usb_bufs->urb[i], GFP_ATOMIC);
|
2008-04-18 08:40:16 +08:00
|
|
|
if (rc) {
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_err(&dev->intf->dev,
|
2016-10-20 18:42:03 +08:00
|
|
|
"submit of urb %i failed (error=%i)\n", i, rc);
|
2012-11-09 01:11:40 +08:00
|
|
|
em28xx_uninit_usb_xfer(dev, mode);
|
2008-04-18 08:40:16 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2012-11-09 01:11:42 +08:00
|
|
|
EXPORT_SYMBOL_GPL(em28xx_init_usb_xfer);
|
2008-12-29 09:18:14 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Device control list
|
|
|
|
*/
|
|
|
|
|
|
|
|
static LIST_HEAD(em28xx_devlist);
|
|
|
|
static DEFINE_MUTEX(em28xx_devlist_mutex);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Extension interface
|
|
|
|
*/
|
|
|
|
|
|
|
|
static LIST_HEAD(em28xx_extension_devlist);
|
|
|
|
|
|
|
|
int em28xx_register_extension(struct em28xx_ops *ops)
|
|
|
|
{
|
|
|
|
struct em28xx *dev = NULL;
|
|
|
|
|
|
|
|
mutex_lock(&em28xx_devlist_mutex);
|
|
|
|
list_add_tail(&ops->next, &em28xx_extension_devlist);
|
|
|
|
list_for_each_entry(dev, &em28xx_devlist, devlist) {
|
2018-01-05 08:04:11 +08:00
|
|
|
if (ops->init) {
|
|
|
|
ops->init(dev);
|
2018-03-04 02:43:34 +08:00
|
|
|
if (dev->dev_next)
|
2018-01-05 08:04:11 +08:00
|
|
|
ops->init(dev->dev_next);
|
|
|
|
}
|
2008-12-29 09:18:14 +08:00
|
|
|
}
|
|
|
|
mutex_unlock(&em28xx_devlist_mutex);
|
2016-10-12 18:32:23 +08:00
|
|
|
pr_info("em28xx: Registered (%s) extension\n", ops->name);
|
2008-12-29 09:18:14 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(em28xx_register_extension);
|
|
|
|
|
|
|
|
void em28xx_unregister_extension(struct em28xx_ops *ops)
|
|
|
|
{
|
|
|
|
struct em28xx *dev = NULL;
|
|
|
|
|
|
|
|
mutex_lock(&em28xx_devlist_mutex);
|
|
|
|
list_for_each_entry(dev, &em28xx_devlist, devlist) {
|
2018-01-05 08:04:11 +08:00
|
|
|
if (ops->fini) {
|
2018-03-04 02:43:34 +08:00
|
|
|
if (dev->dev_next)
|
2018-01-05 08:04:11 +08:00
|
|
|
ops->fini(dev->dev_next);
|
|
|
|
ops->fini(dev);
|
|
|
|
}
|
2008-12-29 09:18:14 +08:00
|
|
|
}
|
|
|
|
list_del(&ops->next);
|
|
|
|
mutex_unlock(&em28xx_devlist_mutex);
|
2016-10-20 18:42:03 +08:00
|
|
|
pr_info("em28xx: Removed (%s) extension\n", ops->name);
|
2008-12-29 09:18:14 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(em28xx_unregister_extension);
|
|
|
|
|
|
|
|
void em28xx_init_extension(struct em28xx *dev)
|
|
|
|
{
|
2011-08-20 19:28:17 +08:00
|
|
|
const struct em28xx_ops *ops = NULL;
|
2008-12-29 09:18:14 +08:00
|
|
|
|
2010-04-08 04:07:58 +08:00
|
|
|
mutex_lock(&em28xx_devlist_mutex);
|
2011-08-20 19:28:17 +08:00
|
|
|
list_add_tail(&dev->devlist, &em28xx_devlist);
|
|
|
|
list_for_each_entry(ops, &em28xx_extension_devlist, next) {
|
2018-01-05 08:04:11 +08:00
|
|
|
if (ops->init) {
|
2011-08-20 19:28:17 +08:00
|
|
|
ops->init(dev);
|
2018-03-04 02:43:34 +08:00
|
|
|
if (dev->dev_next)
|
2018-01-05 08:04:11 +08:00
|
|
|
ops->init(dev->dev_next);
|
|
|
|
}
|
2008-12-29 09:18:14 +08:00
|
|
|
}
|
2010-04-08 04:07:58 +08:00
|
|
|
mutex_unlock(&em28xx_devlist_mutex);
|
2008-12-29 09:18:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void em28xx_close_extension(struct em28xx *dev)
|
|
|
|
{
|
2011-09-24 22:02:32 +08:00
|
|
|
const struct em28xx_ops *ops = NULL;
|
2008-12-29 09:18:14 +08:00
|
|
|
|
2010-04-08 04:07:58 +08:00
|
|
|
mutex_lock(&em28xx_devlist_mutex);
|
2011-09-24 22:02:32 +08:00
|
|
|
list_for_each_entry(ops, &em28xx_extension_devlist, next) {
|
2018-01-05 08:04:11 +08:00
|
|
|
if (ops->fini) {
|
2018-03-04 02:43:34 +08:00
|
|
|
if (dev->dev_next)
|
2018-01-05 08:04:11 +08:00
|
|
|
ops->fini(dev->dev_next);
|
2011-09-24 22:02:32 +08:00
|
|
|
ops->fini(dev);
|
2018-01-05 08:04:11 +08:00
|
|
|
}
|
2008-12-29 09:18:14 +08:00
|
|
|
}
|
2011-09-24 22:02:32 +08:00
|
|
|
list_del(&dev->devlist);
|
2010-04-08 04:07:58 +08:00
|
|
|
mutex_unlock(&em28xx_devlist_mutex);
|
2008-12-29 09:18:14 +08:00
|
|
|
}
|
2014-02-22 08:50:13 +08:00
|
|
|
|
|
|
|
int em28xx_suspend_extension(struct em28xx *dev)
|
|
|
|
{
|
|
|
|
const struct em28xx_ops *ops = NULL;
|
|
|
|
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_info(&dev->intf->dev, "Suspending extensions\n");
|
2014-02-22 08:50:13 +08:00
|
|
|
mutex_lock(&em28xx_devlist_mutex);
|
|
|
|
list_for_each_entry(ops, &em28xx_extension_devlist, next) {
|
|
|
|
if (ops->suspend)
|
|
|
|
ops->suspend(dev);
|
2018-03-04 02:43:34 +08:00
|
|
|
if (dev->dev_next)
|
2018-01-05 08:04:11 +08:00
|
|
|
ops->suspend(dev->dev_next);
|
2014-02-22 08:50:13 +08:00
|
|
|
}
|
|
|
|
mutex_unlock(&em28xx_devlist_mutex);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int em28xx_resume_extension(struct em28xx *dev)
|
|
|
|
{
|
|
|
|
const struct em28xx_ops *ops = NULL;
|
|
|
|
|
2016-12-07 23:48:10 +08:00
|
|
|
dev_info(&dev->intf->dev, "Resuming extensions\n");
|
2014-02-22 08:50:13 +08:00
|
|
|
mutex_lock(&em28xx_devlist_mutex);
|
|
|
|
list_for_each_entry(ops, &em28xx_extension_devlist, next) {
|
2018-04-23 19:02:39 +08:00
|
|
|
if (!ops->resume)
|
|
|
|
continue;
|
|
|
|
ops->resume(dev);
|
2018-03-04 02:43:34 +08:00
|
|
|
if (dev->dev_next)
|
2018-01-05 08:04:11 +08:00
|
|
|
ops->resume(dev->dev_next);
|
2014-02-22 08:50:13 +08:00
|
|
|
}
|
|
|
|
mutex_unlock(&em28xx_devlist_mutex);
|
|
|
|
return 0;
|
|
|
|
}
|