* conf/common.rmk (grub_probe_SOURCES): Add disk/mdraid_linux.c. (grub_fstest_SOURCES): Add disk/raid5_recover.c, disk/raid6_recover.c, disk/mdraid_linux.c and disk/dmraid_nvidia.c and lib/crc.c. (pkglib_MODULES): Add raid5rec.mod, raid6rec.mod, mdraid.mod and dm_nv.mod. (raid5rec_mod_SOURCES): New macro. (raid5rec_mod_CFLAGS): Likewise. (raid5rec_mod_LDFLAGS): Likewise. (raid6rec_mod_SOURCES): Likewise. (raid6rec_mod_CFLAGS): Likewise. (raid6rec_mod_LDFLAGS): Likewise. (mdraid_mod_SOURCES): Likewise. (mdraid_mod_CFLAGS): Likewise. (mdraid_mod_LDFLAGS): Likewise. (dm_nv_mod_SOURCES): Likewise. (dm_nv_mod_CFLAGS): Likewise. (dm_nv_mod_LDFLAGS): Likewise. * conf/i386-pc.rmk (grub_setup_SOURCES): Add disk/mdraid_linux.c. (grub_emu_SOURCES): Add disk/raid5_recover.c, disk/raid6_recover.c, disk/mdraid_linux.c and disk/dmraid_nvidia.c. * conf/i386-coreboot.rmk (grub_emu_SOURCES): Add disk/raid5_recover.c, disk/raid6_recover.c, disk/mdraid_linux.c and disk/dmraid_nvidia.c. * conf/i386-efi.rmk (grub_emu_SOURCES): Likewise. * conf/x86_64-efi.rmk (grub_emu_SOURCES): Likewise. * conf/i386-ieee1275.rmk (grub_emu_SOURCES): Likewise. * conf/powerpc-ieee1275.rmk (grub_emu_SOURCES): Likewise. * disk/raid5_recover.c: New file. * disk/raid6_recover.c: Likewise. * disk/mdraid_linux.c: Likewise. * disk/dmraid_nvidia.c: Likewise. * disk/i386/pc/biosdisk.c: Set total_sectors of cdrom device to ULONG_MAX. * disk/raid.c (grub_raid_open): Use the size of the smallest disk to calculate the size of raid device. (grub_raid_read): Simplify raid0 code. Support raid4, raid6 and four different layout of raid5. (grub_raid_scan_device): Remove code specific to mdraid. (grub_raid_list): New variable. (free_array): New function. (grub_raid_register): Likewise. (grub_raid_unregister): Likewise. (grub_raid_rescan): Likewise. (GRUB_MOD_INIT): Don't iterate device here. (GRUB_MOD_FINI): Use free_array to release resource. * include/grub/raid.h: Remove macro and structure specific to mdraid. (grub_raid5_recover_func_t): New function variable type. (grub_raid6_recover_func_t): Likewise. (grub_raid5_recover_func): New variable. (grub_raid6_recover_func): Likewise. (grub_raid_register): New function. (grub_raid_unregister): Likewise. (grub_raid_rescan): Likewise. (grub_raid_block_xor): Likewise. * util/grub-fstest.c: Add #include <grub/raid.h> and <grub/lib/crc.h>. (CMD_CRC): New macro. (part): Removed. (read_file): Handle device as well as file. (cmd_crc): New function. (fstest): Handle multiple disks. (options): Remove part, raw and long, add root and diskcount. (usage): Add crc, remove -p, -r, -l, add -r and -c. (main): Find the first non option entry and ignore subsequence options, add handling for the new options, support multiple disks. * util/grub-probe.c (probe): Add mdraid to abstraction_name.
		
			
				
	
	
		
			216 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			216 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* raid6_recover.c - module to recover from faulty RAID6 arrays.  */
 | |
| /*
 | |
|  *  GRUB  --  GRand Unified Bootloader
 | |
|  *  Copyright (C) 2006,2007,2008  Free Software Foundation, Inc.
 | |
|  *
 | |
|  *  GRUB is free software: you can redistribute it and/or modify
 | |
|  *  it under the terms of the GNU General Public License as published by
 | |
|  *  the Free Software Foundation, either version 3 of the License, or
 | |
|  *  (at your option) any later version.
 | |
|  *
 | |
|  *  GRUB is distributed in the hope that it will be useful,
 | |
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  *  GNU General Public License for more details.
 | |
|  *
 | |
|  *  You should have received a copy of the GNU General Public License
 | |
|  *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #include <grub/dl.h>
 | |
| #include <grub/disk.h>
 | |
| #include <grub/mm.h>
 | |
| #include <grub/err.h>
 | |
| #include <grub/misc.h>
 | |
| #include <grub/raid.h>
 | |
| 
 | |
| static grub_uint8_t raid6_table1[256][256];
 | |
| static grub_uint8_t raid6_table2[256][256];
 | |
| 
 | |
| static void
 | |
| grub_raid_block_mul (grub_uint8_t mul, char *buf, int size)
 | |
| {
 | |
|   int i;
 | |
|   grub_uint8_t *p;
 | |
| 
 | |
|   p = (grub_uint8_t *) buf;
 | |
|   for (i = 0; i < size; i++, p++)
 | |
|     *p = raid6_table1[mul][*p];
 | |
| }
 | |
| 
 | |
| static void
 | |
| grub_raid6_init_table (void)
 | |
| {
 | |
|   int i, j;
 | |
| 
 | |
|   for (i = 0; i < 256; i++)
 | |
|     raid6_table1[i][1] = raid6_table1[1][i] = i;
 | |
| 
 | |
|   for (i = 2; i < 256; i++)
 | |
|     for (j = i; j < 256; j++)
 | |
|       {
 | |
|         int n;
 | |
|         grub_uint8_t c;
 | |
| 
 | |
|         n = i >> 1;
 | |
| 
 | |
|         c = raid6_table1[n][j];
 | |
|         c = (c << 1) ^ ((c & 0x80) ? 0x1d : 0);
 | |
|         if (i & 1)
 | |
|           c ^= j;
 | |
| 
 | |
|         raid6_table1[j][i] = raid6_table1[i][j] = c;
 | |
|       }
 | |
| 
 | |
|   raid6_table2[0][0] = 1;
 | |
|   for (i = 1; i < 256; i++)
 | |
|     raid6_table2[i][i] = raid6_table1[raid6_table2[i - 1][i - 1]][2];
 | |
| 
 | |
|   for (i = 0; i < 254; i++)
 | |
|     for (j = 0; j < 254; j++)
 | |
|       {
 | |
|         grub_uint8_t c, n;
 | |
|         int k;
 | |
| 
 | |
|         if (i == j)
 | |
|           continue;
 | |
| 
 | |
|         k = i - j;
 | |
|         if (k < 0)
 | |
|           k += 255;
 | |
| 
 | |
|         c = n = raid6_table2[k][k] ^ 1;
 | |
|         for (k = 0; k < 253; k++)
 | |
|           c = raid6_table1[c][n];
 | |
| 
 | |
|         raid6_table2[i][j] = raid6_table1[raid6_table2[255 - j][255 - j]][c];
 | |
|       }
 | |
| }
 | |
| 
 | |
| static grub_err_t
 | |
| grub_raid6_recover (struct grub_raid_array *array, int disknr, int p,
 | |
|                     char *buf, grub_disk_addr_t sector, int size)
 | |
| {
 | |
|   int i, q, pos;
 | |
|   int err[2], nerr;
 | |
|   char *pbuf = 0, *qbuf = 0;
 | |
| 
 | |
|   size <<= GRUB_DISK_SECTOR_BITS;
 | |
|   pbuf = grub_malloc (size);
 | |
|   if (!pbuf)
 | |
|     goto quit;
 | |
| 
 | |
|   qbuf = grub_malloc (size);
 | |
|   if (!qbuf)
 | |
|     goto quit;
 | |
| 
 | |
|   q = p + 1;
 | |
|   if (q == (int) array->total_devs)
 | |
|     q = 0;
 | |
| 
 | |
|   grub_memset (pbuf, 0, size);
 | |
|   grub_memset (qbuf, 0, size);
 | |
| 
 | |
|   pos = q + 1;
 | |
|   if (pos == (int) array->total_devs)
 | |
|     pos = 0;
 | |
| 
 | |
|   nerr = 1;
 | |
|   for (i = 0; i < (int) array->total_devs - 2; i++)
 | |
|     {
 | |
|       if (pos == disknr)
 | |
|         err[0] = i;
 | |
|       else
 | |
|         {
 | |
|           if ((array->device[pos]) &&
 | |
|               (! grub_disk_read (array->device[pos], sector, 0, size, buf)))
 | |
|             {
 | |
|               grub_raid_block_xor (pbuf, buf, size);
 | |
|               grub_raid_block_mul (raid6_table2[i][i], buf, size);
 | |
|               grub_raid_block_xor (qbuf, buf, size);
 | |
|             }
 | |
|           else
 | |
|             {
 | |
|               if (nerr >= 2)
 | |
|                 goto quit;
 | |
| 
 | |
|               err[nerr++] = i;
 | |
|               grub_errno = GRUB_ERR_NONE;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|       pos++;
 | |
|       if (pos == (int) array->total_devs)
 | |
|         pos = 0;
 | |
|     }
 | |
| 
 | |
|   if (nerr == 1)
 | |
|     {
 | |
|       if ((array->device[p]) &&
 | |
|           (! grub_disk_read (array->device[p], sector, 0, size, buf)))
 | |
|         {
 | |
|           grub_raid_block_xor (buf, pbuf, size);
 | |
|           goto quit;
 | |
|         }
 | |
| 
 | |
|       if (! array->device[q])
 | |
|         {
 | |
|           grub_error (GRUB_ERR_READ_ERROR, "Not enough disk to restore");
 | |
|           goto quit;
 | |
|         }
 | |
| 
 | |
|       grub_errno = GRUB_ERR_NONE;
 | |
|       if (grub_disk_read (array->device[q], sector, 0, size, buf))
 | |
|         goto quit;
 | |
| 
 | |
|       grub_raid_block_xor (buf, qbuf, size);
 | |
|       grub_raid_block_mul (raid6_table2[255 - err[0]][255 - err[0]], buf,
 | |
|                            size);
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       grub_uint8_t c;
 | |
| 
 | |
|       if ((! array->device[p]) || (! array->device[q]))
 | |
|         {
 | |
|           grub_error (GRUB_ERR_READ_ERROR, "Not enough disk to restore");
 | |
|           goto quit;
 | |
|         }
 | |
| 
 | |
|       if (grub_disk_read (array->device[p], sector, 0, size, buf))
 | |
|         goto quit;
 | |
| 
 | |
|       grub_raid_block_xor (pbuf, buf, size);
 | |
| 
 | |
|       if (grub_disk_read (array->device[q], sector, 0, size, buf))
 | |
|         goto quit;
 | |
| 
 | |
|       grub_raid_block_xor (qbuf, buf, size);
 | |
| 
 | |
|       c = raid6_table2[err[1]][err[0]];
 | |
|       grub_raid_block_mul (c, qbuf, size);
 | |
| 
 | |
|       c = raid6_table1[raid6_table2[err[1]][err[1]]][c];
 | |
|       grub_raid_block_mul (c, pbuf, size);
 | |
| 
 | |
|       grub_raid_block_xor (pbuf, qbuf, size);
 | |
|       grub_memcpy (buf, pbuf, size);
 | |
|     }
 | |
| 
 | |
| quit:
 | |
|   grub_free (pbuf);
 | |
|   grub_free (qbuf);
 | |
| 
 | |
|   return grub_errno;
 | |
| }
 | |
| 
 | |
| GRUB_MOD_INIT(raid6rec)
 | |
| {
 | |
|   grub_raid6_init_table ();
 | |
|   grub_raid6_recover_func = grub_raid6_recover;
 | |
| }
 | |
| 
 | |
| GRUB_MOD_FINI(raid6rec)
 | |
| {
 | |
|   grub_raid6_recover_func = 0;
 | |
| }
 |