Hi!

Here it is. With crc loop device, you can no longer get silent data
corruption due to disk subsystem...
                                                                Pavel
--- clean/drivers/block/loop.c  Mon Jan  8 22:49:50 2001
+++ linux/drivers/block/loop.c  Mon Jan  8 22:44:09 2001
@@ -61,6 +61,7 @@
 #include <linux/devfs_fs_kernel.h>
 
 #include <asm/uaccess.h>
+#include <asm/checksum.h>
 
 #include <linux/loop.h>                
 
@@ -118,6 +119,101 @@
        return 0;
 }
 
+#define ID printk(KERN_ERR "crc: info about (%s, %d, %d) ", kdevname(lo->lo_device), 
+real_block, blksize);
+
+
+static int transfer_crc(struct loop_device *lo, int cmd, char *raw_buf,
+                 char *loop_buf, int size, int real_block)
+{
+       struct buffer_head *bh;
+       int blksize = 1024, nsect;      /* Size of block on auxilary media */
+       int cksum;
+       u32 *data;
+       nsect = blksize / 4;
+
+       if (!lo->second_device) {
+               ID; printk( "reading from not-yet-setup crc device can result in 
+armagedon. Dont try again.\n" );
+               return -1;
+       }
+       bh = getblk(lo->second_device, 1+real_block/nsect, blksize);
+       if (!bh) {
+               ID; printk( "getblk returned NULL.\n" );
+               return -1;
+       }
+       if (!buffer_uptodate(bh)) {
+               ll_rw_block(READ, 1, &bh);
+               wait_on_buffer(bh);
+               if (!buffer_uptodate(bh)) {
+                       ID; printk(  "could not read block with CRC\n" );
+                       goto error;
+               }
+       }
+
+       data = (u32 *) bh->b_data;
+       if (cmd == READ)
+               cksum = csum_partial_copy_nocheck(raw_buf, loop_buf, size, 0);
+       else
+               cksum = csum_partial_copy_nocheck(loop_buf, raw_buf, size, 0);
+
+       if (cmd == READ) {
+               if (le32_to_cpu(data[real_block%nsect]) != cksum) {
+                       if (lo->lo_encrypt_key_size == 0) {     /* Normal mode */
+                               ID; printk( "wrong checksum reading, is %x, should be 
+%x\n", cksum, 0x1234 );
+                               goto error;
+                       } else { 
+                               ID; printk( "wrong checksum repairing, setting to 
+%x\n", cksum );
+                               goto repair;
+                       }
+               }
+       } else {
+       repair:
+               data[real_block%nsect] = cpu_to_le32(cksum);
+               mark_buffer_uptodate(bh, 1);
+               mark_buffer_dirty(bh);
+       }
+
+       brelse(bh);
+       return 0;
+error:
+       brelse(bh);
+       return -1;
+
+}
+
+static int ioctl_crc(struct loop_device *lo, int cmd, unsigned long arg)
+{
+       struct file     *file;
+       struct inode    *inode;
+       int error;
+
+       printk( "Entering ioctl_crc\n" );
+       if (cmd != LOOP_CRC_SET_FD)
+               return -EINVAL;
+
+       error = -EBADF;
+       file = fget(arg);
+       if (!file)
+               return -EINVAL;
+
+       error = -EINVAL;
+       inode = file->f_dentry->d_inode;
+       if (!inode) {
+               printk(KERN_ERR "ioctl_crc: NULL inode?!?\n");
+               goto out;
+       }
+
+       if (S_ISBLK(inode->i_mode)) {
+               error = blkdev_open(inode, file);
+               lo->second_device = inode->i_rdev;
+               printk( "loop_crc: Registered device %x\n", lo->second_device );
+               return error;
+       } else {
+       out:
+               fput(file);
+               return -EINVAL;
+       }
+}
+
 static int none_status(struct loop_device *lo, struct loop_info *info)
 {
        return 0; 
@@ -142,10 +238,19 @@
        init: xor_status
 };     
 
+struct loop_func_table crc_funcs = { 
+       number: LO_CRYPT_CRC,
+       transfer: transfer_crc,
+       init: none_status,
+       ioctl: ioctl_crc
+};     
+
 /* xfer_funcs[0] is special - its release function is never called */ 
 struct loop_func_table *xfer_funcs[MAX_LO_CRYPT] = {
        &none_funcs,
-       &xor_funcs  
+       &xor_funcs,
+       NULL, NULL, NULL, NULL, NULL,
+       &crc_funcs,
 };
 
 #define MAX_DISK_SIZE 1024*1024*1024
@@ -551,6 +656,7 @@
        lo->transfer = NULL;
        lo->ioctl = NULL;
        lo->lo_device = 0;
+       lo->second_device = 0;
        lo->lo_encrypt_type = 0;
        lo->lo_offset = 0;
        lo->lo_encrypt_key_size = 0;
--- clean/include/linux/loop.h  Mon Nov 23 06:29:54 1998
+++ linux/include/linux/loop.h  Sat Apr 15 13:22:57 2000
@@ -22,6 +22,7 @@
        struct dentry   *lo_dentry;
        int             lo_refcnt;
        kdev_t          lo_device;
+       kdev_t          second_device;
        int             lo_offset;
        int             lo_encrypt_type;
        int             lo_encrypt_key_size;
@@ -94,6 +95,7 @@
 #define LO_CRYPT_BLOW     4
 #define LO_CRYPT_CAST128  5
 #define LO_CRYPT_IDEA     6
+#define LO_CRYPT_CRC     7
 #define LO_CRYPT_DUMMY    9
 #define LO_CRYPT_SKIPJACK 10
 #define MAX_LO_CRYPT   20
@@ -126,5 +128,6 @@
 #define LOOP_CLR_FD    0x4C01
 #define LOOP_SET_STATUS        0x4C02
 #define LOOP_GET_STATUS        0x4C03
+#define LOOP_CRC_SET_FD 0x4C04
 
 #endif

-- 
I'm [EMAIL PROTECTED] "In my country we have almost anarchy and I don't care."
Panos Katsaloulis describing me w.r.t. patents at [EMAIL PROTECTED]


----- End forwarded message -----

-- 
I'm [EMAIL PROTECTED] "In my country we have almost anarchy and I don't care."
Panos Katsaloulis describing me w.r.t. patents at [EMAIL PROTECTED]
-
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