summaryrefslogtreecommitdiff
path: root/linux/drivers/media
diff options
context:
space:
mode:
Diffstat (limited to 'linux/drivers/media')
-rw-r--r--linux/drivers/media/dvb/dvb-core/dvb_ca_en50221.c76
1 files changed, 22 insertions, 54 deletions
diff --git a/linux/drivers/media/dvb/dvb-core/dvb_ca_en50221.c b/linux/drivers/media/dvb/dvb-core/dvb_ca_en50221.c
index e205a6964..cb2e7d6ba 100644
--- a/linux/drivers/media/dvb/dvb-core/dvb_ca_en50221.c
+++ b/linux/drivers/media/dvb/dvb-core/dvb_ca_en50221.c
@@ -111,9 +111,6 @@ struct dvb_ca_slot {
/* size of the buffer to use when talking to the CAM */
int link_buf_size;
- /* spinlock for syncing access to slot structure */
- rwlock_t lock;
-
/* buffer for incoming packets */
struct dvb_ringbuffer rx_buffer;
@@ -595,7 +592,6 @@ static int dvb_ca_en50221_read_data(struct dvb_ca_private *ca, int slot, u8 * eb
int status;
u8 buf[HOST_LINK_BUF_SIZE];
int i;
- unsigned long irqsave;
dprintk("%s\n", __FUNCTION__);
@@ -603,14 +599,11 @@ static int dvb_ca_en50221_read_data(struct dvb_ca_private *ca, int slot, u8 * eb
if (ebuf == NULL) {
int buf_free;
- read_lock_irqsave(&ca->slot_info[slot].lock, irqsave);
if (ca->slot_info[slot].rx_buffer.data == NULL) {
- read_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
status = -EIO;
goto exit;
}
buf_free = dvb_ringbuffer_free(&ca->slot_info[slot].rx_buffer);
- read_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
if (buf_free < (ca->slot_info[slot].link_buf_size + DVB_RINGBUFFER_PKTHDRSIZE)) {
status = -EAGAIN;
@@ -681,14 +674,11 @@ static int dvb_ca_en50221_read_data(struct dvb_ca_private *ca, int slot, u8 * eb
/* OK, add it to the receive buffer, or copy into external buffer if supplied */
if (ebuf == NULL) {
- read_lock_irqsave(&ca->slot_info[slot].lock, irqsave);
if (ca->slot_info[slot].rx_buffer.data == NULL) {
- read_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
status = -EIO;
goto exit;
}
dvb_ringbuffer_pkt_write(&ca->slot_info[slot].rx_buffer, buf, bytes_read);
- read_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
} else {
memcpy(ebuf, buf, bytes_read);
}
@@ -801,16 +791,10 @@ EXPORT_SYMBOL(dvb_ca_en50221_camchange_irq);
*/
static int dvb_ca_en50221_slot_shutdown(struct dvb_ca_private *ca, int slot)
{
- unsigned long irqsave;
-
dprintk("%s\n", __FUNCTION__);
- write_lock_irqsave(&ca->slot_info[slot].lock, irqsave);
ca->pub->slot_shutdown(ca->pub, slot);
ca->slot_info[slot].slot_state = DVB_CA_SLOTSTATE_NONE;
- vfree(ca->slot_info[slot].rx_buffer.data);
- ca->slot_info[slot].rx_buffer.data = NULL;
- write_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
/* need to wake up all processes to check if they're now
trying to write to a defunct CAM */
@@ -896,7 +880,7 @@ void dvb_ca_en50221_frda_irq(struct dvb_ca_en50221 *pubca, int slot)
case DVB_CA_SLOTSTATE_RUNNING:
if (ca->open)
- dvb_ca_en50221_read_data(ca, slot, NULL, 0);
+ dvb_ca_en50221_thread_wakeup(ca);
break;
}
}
@@ -1004,7 +988,6 @@ static int dvb_ca_en50221_thread(void *data)
int status;
int pktcount;
void *rxbuf;
- unsigned long irqsave;
dprintk("%s\n", __FUNCTION__);
@@ -1131,16 +1114,16 @@ static int dvb_ca_en50221_thread(void *data)
break;
}
- rxbuf = vmalloc(RX_BUFFER_SIZE);
- if (rxbuf == NULL) {
- printk("dvb_ca adapter %d: Unable to allocate CAM rx buffer :(\n", ca->dvbdev->adapter->num);
- ca->slot_info[slot].slot_state = DVB_CA_SLOTSTATE_INVALID;
- dvb_ca_en50221_thread_update_delay(ca);
- break;
+ if (ca->slot_info[slot].rx_buffer.data == NULL) {
+ rxbuf = vmalloc(RX_BUFFER_SIZE);
+ if (rxbuf == NULL) {
+ printk("dvb_ca adapter %d: Unable to allocate CAM rx buffer :(\n", ca->dvbdev->adapter->num);
+ ca->slot_info[slot].slot_state = DVB_CA_SLOTSTATE_INVALID;
+ dvb_ca_en50221_thread_update_delay(ca);
+ break;
+ }
+ dvb_ringbuffer_init(&ca->slot_info[slot].rx_buffer, rxbuf, RX_BUFFER_SIZE);
}
- write_lock_irqsave(&ca->slot_info[slot].lock, irqsave);
- dvb_ringbuffer_init(&ca->slot_info[slot].rx_buffer, rxbuf, RX_BUFFER_SIZE);
- write_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
ca->pub->slot_ts_enable(ca->pub, slot);
ca->slot_info[slot].slot_state = DVB_CA_SLOTSTATE_RUNNING;
@@ -1152,11 +1135,7 @@ static int dvb_ca_en50221_thread(void *data)
if (!ca->open)
continue;
- // no need to poll if the CAM supports IRQs
- if (ca->slot_info[slot].da_irq_supported)
- break;
-
- // poll mode
+ // poll slots for data
pktcount = 0;
while ((status = dvb_ca_en50221_read_data(ca, slot, NULL, 0)) > 0) {
if (!ca->open)
@@ -1372,13 +1351,12 @@ exit:
* Condition for waking up in dvb_ca_en50221_io_read_condition
*/
static int dvb_ca_en50221_io_read_condition(struct dvb_ca_private *ca,
- int *result, int *_slot,
- unsigned long *irqsave)
+ int *result, int *_slot)
{
int slot;
int slot_count = 0;
int idx;
- int fraglen;
+ size_t fraglen;
int connection_id = -1;
int found = 0;
u8 hdr[2];
@@ -1388,10 +1366,7 @@ static int dvb_ca_en50221_io_read_condition(struct dvb_ca_private *ca,
if (ca->slot_info[slot].slot_state != DVB_CA_SLOTSTATE_RUNNING)
goto nextslot;
- read_lock_irqsave(&ca->slot_info[slot].lock, *irqsave);
-
if (ca->slot_info[slot].rx_buffer.data == NULL) {
- read_unlock_irqrestore(&ca->slot_info[slot].lock, *irqsave);
return 0;
}
@@ -1409,9 +1384,6 @@ static int dvb_ca_en50221_io_read_condition(struct dvb_ca_private *ca,
idx = dvb_ringbuffer_pkt_next(&ca->slot_info[slot].rx_buffer, idx, &fraglen);
}
- if (!found)
- read_unlock_irqrestore(&ca->slot_info[slot].lock, *irqsave);
-
nextslot:
slot = (slot + 1) % ca->slot_count;
slot_count++;
@@ -1447,7 +1419,6 @@ static ssize_t dvb_ca_en50221_io_read(struct file *file, char __user * buf,
size_t fraglen;
int pktlen;
int dispose = 0;
- unsigned long irqsave;
dprintk("%s\n", __FUNCTION__);
@@ -1456,7 +1427,7 @@ static ssize_t dvb_ca_en50221_io_read(struct file *file, char __user * buf,
return -EINVAL;
/* wait for some data */
- if ((status = dvb_ca_en50221_io_read_condition(ca, &result, &slot, &irqsave)) == 0) {
+ if ((status = dvb_ca_en50221_io_read_condition(ca, &result, &slot)) == 0) {
/* if we're in nonblocking mode, exit immediately */
if (file->f_flags & O_NONBLOCK)
@@ -1465,7 +1436,7 @@ static ssize_t dvb_ca_en50221_io_read(struct file *file, char __user * buf,
/* wait for some data */
status = wait_event_interruptible(ca->wait_queue,
dvb_ca_en50221_io_read_condition
- (ca, &result, &slot, &irqsave));
+ (ca, &result, &slot));
}
if ((status < 0) || (result < 0)) {
if (result)
@@ -1519,7 +1490,6 @@ static ssize_t dvb_ca_en50221_io_read(struct file *file, char __user * buf,
status = pktlen;
exit:
- read_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
return status;
}
@@ -1538,7 +1508,6 @@ static int dvb_ca_en50221_io_open(struct inode *inode, struct file *file)
struct dvb_ca_private *ca = dvbdev->priv;
int err;
int i;
- unsigned long irqsave;
dprintk("%s\n", __FUNCTION__);
@@ -1552,11 +1521,11 @@ static int dvb_ca_en50221_io_open(struct inode *inode, struct file *file)
for (i = 0; i < ca->slot_count; i++) {
if (ca->slot_info[i].slot_state == DVB_CA_SLOTSTATE_RUNNING) {
- read_lock_irqsave(&ca->slot_info[i].lock, irqsave);
if (ca->slot_info[i].rx_buffer.data != NULL) {
+ /* it is safe to call this here without locks because
+ * ca->open == 0. Data is not read in this case */
dvb_ringbuffer_flush(&ca->slot_info[i].rx_buffer);
}
- read_unlock_irqrestore(&ca->slot_info[i].lock, irqsave);
}
}
@@ -1611,12 +1580,10 @@ static unsigned int dvb_ca_en50221_io_poll(struct file *file, poll_table * wait)
unsigned int mask = 0;
int slot;
int result = 0;
- unsigned long irqsave;
dprintk("%s\n", __FUNCTION__);
- if (dvb_ca_en50221_io_read_condition(ca, &result, &slot, &irqsave) == 1) {
- read_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
+ if (dvb_ca_en50221_io_read_condition(ca, &result, &slot) == 1) {
mask |= POLLIN;
}
@@ -1627,8 +1594,7 @@ static unsigned int dvb_ca_en50221_io_poll(struct file *file, poll_table * wait)
/* wait for something to happen */
poll_wait(file, &ca->wait_queue, wait);
- if (dvb_ca_en50221_io_read_condition(ca, &result, &slot, &irqsave) == 1) {
- read_unlock_irqrestore(&ca->slot_info[slot].lock, irqsave);
+ if (dvb_ca_en50221_io_read_condition(ca, &result, &slot) == 1) {
mask |= POLLIN;
}
@@ -1718,7 +1684,6 @@ int dvb_ca_en50221_init(struct dvb_adapter *dvb_adapter,
ca->slot_info[i].slot_state = DVB_CA_SLOTSTATE_NONE;
atomic_set(&ca->slot_info[i].camchange_count, 0);
ca->slot_info[i].camchange_type = DVB_CA_EN50221_CAMCHANGE_REMOVED;
- ca->slot_info[i].lock = RW_LOCK_UNLOCKED;
}
if (signal_pending(current)) {
@@ -1780,6 +1745,9 @@ void dvb_ca_en50221_release(struct dvb_ca_en50221 *pubca)
for (i = 0; i < ca->slot_count; i++) {
dvb_ca_en50221_slot_shutdown(ca, i);
+ if (ca->slot_info[i].rx_buffer.data != NULL) {
+ vfree(ca->slot_info[i].rx_buffer.data);
+ }
}
kfree(ca->slot_info);
dvb_unregister_device(ca->dvbdev);