Commit 6a28bd94 authored by Mauro Carvalho Chehab's avatar Mauro Carvalho Chehab

[media] siano: add support for .poll on debugfs

Implement poll() method for debugfs and be sure that the
debug_data won't be freed on ir or on read().
With this change, poll() will return POLLIN if either data was
filled or if data was read. That allows read() to return 0
to indicate EOF in the latter case.
As poll() is now provided, fix support for non-block mode.
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent 773adad1
...@@ -352,6 +352,14 @@ static int smsdvb_stats_open(struct inode *inode, struct file *file) ...@@ -352,6 +352,14 @@ static int smsdvb_stats_open(struct inode *inode, struct file *file)
return 0; return 0;
} }
static void smsdvb_debugfs_data_release(struct kref *ref)
{
struct smsdvb_debugfs *debug_data;
debug_data = container_of(ref, struct smsdvb_debugfs, refcount);
kfree(debug_data);
}
static int smsdvb_stats_wait_read(struct smsdvb_debugfs *debug_data) static int smsdvb_stats_wait_read(struct smsdvb_debugfs *debug_data)
{ {
int rc = 1; int rc = 1;
...@@ -368,41 +376,73 @@ static int smsdvb_stats_wait_read(struct smsdvb_debugfs *debug_data) ...@@ -368,41 +376,73 @@ static int smsdvb_stats_wait_read(struct smsdvb_debugfs *debug_data)
return rc; return rc;
} }
static unsigned int smsdvb_stats_poll(struct file *file, poll_table *wait)
{
struct smsdvb_debugfs *debug_data = file->private_data;
int rc;
kref_get(&debug_data->refcount);
poll_wait(file, &debug_data->stats_queue, wait);
rc = smsdvb_stats_wait_read(debug_data);
if (rc > 0)
rc = POLLIN | POLLRDNORM;
kref_put(&debug_data->refcount, smsdvb_debugfs_data_release);
return rc;
}
static ssize_t smsdvb_stats_read(struct file *file, char __user *user_buf, static ssize_t smsdvb_stats_read(struct file *file, char __user *user_buf,
size_t nbytes, loff_t *ppos) size_t nbytes, loff_t *ppos)
{ {
int rc = 0; int rc = 0, len;
struct smsdvb_debugfs *debug_data = file->private_data; struct smsdvb_debugfs *debug_data = file->private_data;
kref_get(&debug_data->refcount);
if (file->f_flags & O_NONBLOCK) {
rc = smsdvb_stats_wait_read(debug_data);
if (!rc) {
rc = -EWOULDBLOCK;
goto ret;
}
} else {
rc = wait_event_interruptible(debug_data->stats_queue, rc = wait_event_interruptible(debug_data->stats_queue,
smsdvb_stats_wait_read(debug_data)); smsdvb_stats_wait_read(debug_data));
if (rc < 0) if (rc < 0)
return rc; goto ret;
}
if (debug_data->stats_was_read) {
rc = 0; /* EOF */
goto ret;
}
len = debug_data->stats_count - *ppos;
if (len >= 0)
rc = simple_read_from_buffer(user_buf, nbytes, ppos, rc = simple_read_from_buffer(user_buf, nbytes, ppos,
debug_data->stats_data, debug_data->stats_data, len);
debug_data->stats_count); else
rc = 0;
if (*ppos >= debug_data->stats_count) {
spin_lock(&debug_data->lock); spin_lock(&debug_data->lock);
debug_data->stats_was_read = true; debug_data->stats_was_read = true;
spin_unlock(&debug_data->lock); spin_unlock(&debug_data->lock);
}
ret:
kref_put(&debug_data->refcount, smsdvb_debugfs_data_release);
return rc; return rc;
} }
static void smsdvb_debugfs_data_release(struct kref *ref)
{
struct smsdvb_debugfs *debug_data;
debug_data = container_of(ref, struct smsdvb_debugfs, refcount);
kfree(debug_data);
}
static int smsdvb_stats_release(struct inode *inode, struct file *file) static int smsdvb_stats_release(struct inode *inode, struct file *file)
{ {
struct smsdvb_debugfs *debug_data = file->private_data; struct smsdvb_debugfs *debug_data = file->private_data;
spin_lock(&debug_data->lock); spin_lock(&debug_data->lock);
debug_data->stats_was_read = true; debug_data->stats_was_read = true; /* return EOF to read() */
spin_unlock(&debug_data->lock); spin_unlock(&debug_data->lock);
wake_up_interruptible_sync(&debug_data->stats_queue); wake_up_interruptible_sync(&debug_data->stats_queue);
...@@ -414,6 +454,7 @@ static int smsdvb_stats_release(struct inode *inode, struct file *file) ...@@ -414,6 +454,7 @@ static int smsdvb_stats_release(struct inode *inode, struct file *file)
static const struct file_operations debugfs_stats_ops = { static const struct file_operations debugfs_stats_ops = {
.open = smsdvb_stats_open, .open = smsdvb_stats_open,
.poll = smsdvb_stats_poll,
.read = smsdvb_stats_read, .read = smsdvb_stats_read,
.release = smsdvb_stats_release, .release = smsdvb_stats_release,
.llseek = generic_file_llseek, .llseek = generic_file_llseek,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment