On Tue, 5 Sep 2000, Linus Torvalds wrote:

> 
> 
> On Tue, 5 Sep 2000, Alexander Viro wrote:
> > 
> > WTF? <thinks> Erm... Linus, I suspect that we are losing the thing on the
> > very simple effect: readpage() gets buffer_heads, all right, but then
> > we lose the lock. That's your window for losing the buffer_head ring of
> > that page.
> 
> Note that it's not even a small window: we get an up-to-date page, but
> readpage() can have happened hours, days, years before. The buffer-head
> ring _has_ been there at some earlier time, but there's nothing that
> guarantees that it still is there.
> 
> Fixing this is not hard, happily. I'll do it. The only issue is whether to
> just copy one of the existing functions and make changes, or try to share
> code and have helper functions. The loop over the page doing the
> get_block() is certainly something that all of the functions for
> reading/writing need to do, and I suspect that I really should split
> things up a bit and share more code than we do now.
> 
> If you WANT to do it, holler. I'm not glutton for punishment, but it
> doesn't look that bad.

Me neither, but IMO I have a cop-out:

--- buffer.c    Tue Sep  5 21:23:41 2000
+++ buffer.c.new        Tue Sep  5 21:28:53 2000
@@ -1724,12 +1724,24 @@
        return 0;
 }
 
+/*
+ * If it would be '74 that would go into libc...
+ */
+int mem_is_zero(char *p, unsigned len)
+{
+       while (len--)
+               if (*p++)
+                       return 0;
+       return 1;
+}
+
 int block_zero_page(struct address_space *mapping, loff_t from, unsigned length)
 {
        unsigned long index = from >> PAGE_CACHE_SHIFT;
        unsigned offset = from & (PAGE_CACHE_SIZE-1);
        struct inode *inode = (struct inode *)mapping->host;
        struct page *page;
+       char *kaddr;
        int err;
 
        if (!length)
@@ -1745,9 +1757,18 @@
        if (!Page_Uptodate(page))
                goto unlock;
 
-       memset((char *) kmap(page) + offset, 0, length);
+       kaddr = (char*)kmap(page);
+       err = 0;
+       if (mem_is_zero(kaddr+offset, length))
+               goto unmap;
+       err = mapping->a_ops->prepare_write(NULL, page, offset, offset+length);
+       if (err)
+               goto unmap;
+       memset(kaddr+offset, 0, length);
        flush_dcache_page(page);
        __block_commit_write(inode, page, offset, offset+length);
+       kunmap(page);
+unmap:
        kunmap(page);
        err = 0;
 

-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [EMAIL PROTECTED]
Please read the FAQ at http://www.tux.org/lkml/

Reply via email to