[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20110208191143.GC10125@ovro.caltech.edu>
Date: Tue, 8 Feb 2011 11:11:44 -0800
From: "Ira W. Snyder" <iws@...o.caltech.edu>
To: Dmitry Torokhov <dmitry.torokhov@...il.com>
Cc: linuxppc-dev@...ts.ozlabs.org, linux-kernel@...r.kernel.org
Subject: Re: [PATCH 1/2] misc: add CARMA DATA-FPGA Access Driver
On Tue, Feb 08, 2011 at 09:50:29AM -0800, Dmitry Torokhov wrote:
> On Tue, Feb 08, 2011 at 09:20:46AM -0800, Ira W. Snyder wrote:
> > On Mon, Feb 07, 2011 at 11:33:10PM -0800, Dmitry Torokhov wrote:
> > > > +static void data_free_buffer(struct device *dev, struct data_buf *buf)
> > > > +{
> > > > + /* It is ok to free a NULL buffer */
> > > > + if (!buf)
> > > > + return;
> > > > +
> > > > + /* Make sure the buffer is not on any list */
> > > > + list_del_init(&buf->entry);
> > >
> > > And what happens if it is? Should it be WARN_ON(!list_empty()) instead?
> > >
> >
> > This was only defensive programming. Everywhere this function is called,
> > the buffer has already been removed from the list.
>
> I am concerned as sometimes defencive programming is the sign that we
> arenot quite sure how the code works. I believe defensive programming
> should be used when providing library-like code, not in local cases.
>
Ok.
> > > > +
> > > > + list_for_each_entry_safe(buf, tmp, &priv->free, entry) {
> > > > + list_del_init(&buf->entry);
> > > > + spin_unlock_irq(&priv->lock);
> > > > + data_free_buffer(priv->dev, buf);
> > > > + spin_lock_irq(&priv->lock);
> > > > + }
> > >
> > > This is messed up. If there is concurrent access to the free list then
> > > it is not safe to continue iterating list after releasing the lock, you
> > > need to do:
> > >
> > > spin_lock_irq(&priv->lock);
> > > while (!list_empty(&priv->free)) {
> > > buf = list_first_entry(&priv->free, struct data_buf, entry);
> > > list_del_init(&buf->entry);
> > > spin_unlock_irq(&priv->lock);
> > > data_free_buffer(priv->dev, buf);
> > > spin_lock_irq(&priv->lock);
> > > }
> > >
> > > BUT, the function is only called when you disable (or fail to enable) device
> > > which, at this point, should be quiesced, thus all this locking is not
> > > really needed.
> > >
> >
> > Correct.
> >
> > I thought it would be clearer to reviewers if I always used the lock to
> > protect a data structure, even when it isn't technically needed.
>
> No, locks should protect wehat needs to be protected. The rest just
> muddles water.
>
Ok.
> > > > +
> > > > + spin_lock_irq(&priv->lock);
> > > > + while (!list_empty(list)) {
> > > > + spin_unlock_irq(&priv->lock);
> > > > +
> > > > + ret = wait_event_interruptible(priv->wait, list_empty(list));
> > > > + if (ret)
> > > > + return -ERESTARTSYS;
> > > > +
> > > > + spin_lock_irq(&priv->lock);
> > > > + }
> > > > + spin_unlock_irq(&priv->lock);
> > >
> > > Locking is not needed - if you disable interrupyts what would put more
> > > stuff on the list?
> > >
> >
> > The locking is definitely needed.
> >
> > You've missed a critical piece of information. There are *two* devices
> > we are interacting with here, and BOTH generate interrupts.
> >
>
> No, I did not miss this fact. The point is that when we get to this code
> the device _putting_ items on wauiting list is stopped and we only need
> to wait for the list to drain. Nobody puts more stuff on it. You can
> check fir list_empty() condition without locking.
>
> And if someone _is_ putting more stuff on the list - you are screwed
> since list may become non-empty the moment you release the lock.
>
Ok, I understand what you mean now. You are correct, nothing else can
add things to the list. Thanks for clarifying this for me. :)
> > > > +
> > > > +static ssize_t data_num_buffers_show(struct device *dev,
> > > > + struct device_attribute *attr, char *buf)
> > > > +{
> > > > + struct fpga_device *priv = dev_get_drvdata(dev);
> > > > + unsigned int num;
> > > > +
> > > > + spin_lock_irq(&priv->lock);
> > > > + num = priv->num_buffers;
> > > > + spin_unlock_irq(&priv->lock);
> > >
> > > This spin lock is pointless, priv->num_buffers might be already changed
> > > here, you can't guarantee that you show accurate data.
> > >
> >
> > Correct, I know this. I just wanted to protect the data structure at all
> > points of use in the driver.
>
> Protect from what? integer reads are guaranteed to be complete and you
> are not concerned with missing updates as information is obsolete the
> moment you release trhe lock.
>
> > Would an atomic_t be better for this, or
> > should I just remove the locking completely?
>
> Just remove the locking.
>
Ok.
> > > > +
> > > > + if (mutex_lock_interruptible(&priv->mutex))
> > > > + return -ERESTARTSYS;
> > >
> > > Why don't
> > >
> > > error = mutex_lock_interruptible(&priv->mutex);
> > > if (error)
> > > return error;
> > >
> > > - do not clobber perfectly valid error codes.
> > >
> >
> > That's what the Linux Device Drivers 3rd Edition book does. See page
> > 112. I will change it to fix the return code.
> >
>
> LDD3 is quite old by now...
>
I know, but it is still the best written reference I have. Reviewers
like yourself are better, but I can't look up your advice in a book. :)
I'll return the error code.
> > > > +
> > > > +static struct attribute *data_sysfs_attrs[] = {
> > > > + &dev_attr_num_buffers.attr,
> > > > + &dev_attr_buffer_size.attr,
> > > > + &dev_attr_num_inflight.attr,
> > > > + &dev_attr_num_free.attr,
> > > > + &dev_attr_num_used.attr,
> > > > + &dev_attr_num_dropped.attr,
> > > > + &dev_attr_enable.attr,
> > > > + NULL,
> > > > +};
> > >
> > > Are all of these really needed or most of them are for debug?
> > >
> >
> > Most are for debugging. They have proved useful a few times in
> > production to track down bugs.
> >
>
> Consider switching over to debugfs then. Or, if you believe the device
> is sufficiently bug-free just cut the code.
>
I've never used debugfs. I'll look into it. I think the device is almost
bug free, though.
> > > > +
> > > > +static ssize_t data_read(struct file *filp, char __user *ubuf, size_t count,
> > > > + loff_t *f_pos)
> > > > +{
> > > > + struct fpga_reader *reader = filp->private_data;
> > > > + struct fpga_device *priv = reader->priv;
> > > > + struct list_head *used = &priv->used;
> > > > + struct data_buf *dbuf;
> > > > + size_t avail;
> > > > + void *data;
> > > > + int ret;
> > > > +
> > > > + /* check if we already have a partial buffer */
> > > > + if (reader->buf) {
> > > > + dbuf = reader->buf;
> > > > + goto have_buffer;
> > > > + }
> > > > +
> > > > + spin_lock_irq(&priv->lock);
> > > > +
> > > > + /* Block until there is at least one buffer on the used list */
> > > > + while (list_empty(used)) {
> > > > + spin_unlock_irq(&priv->lock);
> > > > +
> > > > + if (filp->f_flags & O_NONBLOCK)
> > > > + return -EAGAIN;
> > > > +
> > > > + if (wait_event_interruptible(priv->wait, !list_empty(used)))
> > > > + return -ERESTARTSYS;
> > > > +
> > >
> > > And somebody grabs that entry here...
> > >
> > > > + spin_lock_irq(&priv->lock);
> > > > + }
> > > > +
> > > > + /* Grab the first buffer off of the used list */
> > > > + dbuf = list_first_entry(used, struct data_buf, entry);
> > >
> > > And list is empty so you grabgarbage.
> > >
> > > > + list_del_init(&dbuf->entry);
> > > > +
> > > > + spin_unlock_irq(&priv->lock);
> > >
> > > Shoudl be:
> > >
> > > struct data_buf *dbuf = NULL;
> > > ...
> > >
> > > if (list_empty(&priv->used) && (filp->f_flags & O_NONBLOCK))
> > > return -EAGAIN;
> > >
> > > error = wait_event_interruptible(priv->wait, !list_empty(&priv->used);
> > > if (error)
> > > return error;
> > >
> > > spin_lock_irq(&priv->lock);
> > > if (!list_empty(&priv->used)) {
> > > buf = list_first_entry(&priv->used, struct data_buf, entry);
> > > list_del_init(&dbuf->entry);
> > > }
> > > spin_unlock_irq(&priv->lock);
> > >
> > > if (dbuf) {
> > > .. deal with the buffer
> > > }
> > >
> >
> > I'm pretty sure you're wrong.
>
> And I am certain I am correct. Given that I've even given example how you
> can lose your list entry I don't know why you think I did not understand
> your code.
>
> > Go back and re-think my loop. This is a
> > common idiom straight of out LDD3 pages 153-154.
> >
> > You should note that it is only possible to exit the loop with the lock
> > held AND !list_empty(used). The lock protects the used list, and
> > therefore, there must be a buffer on the list.
>
> No, because you are woken up while not holding the lock so another
> reader is free to take it off the list.
>
Correct. But then I go around the loop and check list_empty() again
before exiting the loop. The list MUST NOT be empty before the loop will
terminate.
I still don't understand. Both of our solutions are equivalent, but
yours has worse error handling. Let's examine your example, line by
line:
> + spin_lock_irq(&priv->lock);
Locked.
> +
> + /* Block until there is at least one buffer on the used list */
> + while (list_empty(used)) {
We ONLY enter the loop if the list is empty.
> + spin_unlock_irq(&priv->lock);
> +
Unlock.
> + if (filp->f_flags & O_NONBLOCK)
> + return -EAGAIN;
> +
> + if (wait_event_interruptible(priv->wait, !list_empty(used)))
> + return -ERESTARTSYS;
> +
Block until the the list is NOT empty. Assume at this point that there
is another reader racing against us. The other reader wins, and the list
is empty BEFORE I grab the lock on the next line. This is exactly what
you describe above.
> + spin_lock_irq(&priv->lock);
Locked. Now we re-check the loop condition. The list is EMPTY and start
the loop again.
> + }
> +
When we get here, there are two true statements:
1) we hold priv->lock spinlock
2) list_empty(used) == false
> + /* Grab the first buffer off of the used list */
> + dbuf = list_first_entry(used, struct data_buf, entry);
> + list_del_init(&dbuf->entry);
> +
Therefore list_first_entry() CAN NOT return garbage. We MUST have
something in the list.
> + spin_unlock_irq(&priv->lock);
Perhaps if I write the loop a different way, it will become more clear?
This loop is exactly equivalent to the loop in my code.
while (1) {
spin_lock_irq(&priv->lock);
if (!list_empty(used))
break;
spin_unlock_irq(&priv->lock);
if (filp->f_flags & O_NONBLOCK)
return -EAGAIN;
if (wait_event_interruptible(priv->wait, !list_empty(used))
return -ERESTARTSYS;
}
/* grab the first buffer off the used list */
dbuf = list_first_entry(used, struct data_buf, entry);
list_del_init(&dbuf->entry);
spin_unlock_irq(&priv->lock);
I must not understand some part of the code. In my best understanding,
my original code cannot fail in the way you expect. This is because of
the while loop, which rechecks its condition before exiting. Can you
please try to clarify where my misunderstanding is?
(I'm not trying to argue. Hopefully this will make my future work
better.)
> > > > +
> > > > +static unsigned int data_poll(struct file *filp, struct poll_table_struct *tbl)
> > > > +{
> > > > + struct fpga_reader *reader = filp->private_data;
> > > > + struct fpga_device *priv = reader->priv;
> > > > + unsigned int mask = 0;
> > > > +
> > > > + poll_wait(filp, &priv->wait, tbl);
> > > > +
> > > > + spin_lock_irq(&priv->lock);
> > > > +
> > > > + if (!list_empty(&priv->used))
> > > > + mask |= POLLIN | POLLRDNORM;
> > > > +
> > > > + spin_unlock_irq(&priv->lock);
> > >
> > > No lock is needed.
> > >
> >
> > Why not? For example, there are two readers:
> > 1) blocked in poll()
> > 2) blocked in poll()
> >
> > A single buffer gets added to the used list. I should only unblock one
> > of them with (POLLIN | POLLRDNORM), correct? Otherwise one of them must
> > have a spurious wakeup. I think that's incorrect (or undesirable)
> > behavior.
>
> And how your lock prevents both of them wakign up? You are not resetting
> wakeup condition in any way.
>
> >
> > Again, it seems very inconsistent and confusing to me to protect the
> > used list with a spinlock in some places and not in others.
>
> Please try to think about what exactly you protecting and whether lock
> adds any protection or not and it will put your mind at ease.
Ok.
Thanks again for the comments,
Ira
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists