Difference between revisions of "Esempio didattico su RAID6"
		
		
		
		
		
		Jump to navigation
		Jump to search
		
				
		
		
	
|  (Created page with "Il codice python che segue consente di provare sperimentalemnte le metodologie di fault tolerance descritte nell'articolo [http://www.cs.unibo.it/~renzo//doc/papers/p245-blaum...") | m | ||
| (One intermediate revision by the same user not shown) | |||
| Line 1: | Line 1: | ||
| Il codice python che segue consente di provare sperimentalemnte le metodologie di fault tolerance descritte nell'articolo | Il codice python che segue consente di provare sperimentalemnte le metodologie di fault tolerance descritte nell'articolo | ||
| − | [http://www.cs.unibo.it/~renzo//doc/papers/p245-blaum.pdf"EVENODD: An Optimal Scheme for Tolerating Double Disk Failures" | + | [http://www.cs.unibo.it/~renzo//doc/papers/p245-blaum.pdf"EVENODD: An Optimal Scheme for Tolerating Double Disk Failures"] di Mario Blaurn et al. | 
| − | + | ||
| + | <source lang=python> | ||
| + | #!/usr/bin/env python3 | ||
| + | |||
| + | def compute_parity(tab, nrow, ncol): | ||
| + |   for i in range(nrow): | ||
| + |     p = 0 | ||
| + |     for j in range(ncol): | ||
| + |       p ^= tab[i][j] | ||
| + |     tab[i][ncol] = p | ||
| + | |||
| + | def compute_dparity(tab, nrow, ncol): | ||
| + |   p = [0] * ncol | ||
| + |   for i in range(nrow): | ||
| + |     for j in range(ncol): | ||
| + |       p[(i+j) % ncol] ^= tab[i][j] | ||
| + | # print(p) | ||
| + |   s = p[ncol - 1] | ||
| + |   for i in range(nrow): | ||
| + |     tab[i][ncol + 1] = p[i] ^ s | ||
| + | |||
| + | def print_table(tab, nrow, ncol): | ||
| + |   for i in range(nrow): | ||
| + |     for j in range(ncol): | ||
| + |       print(tab[i][j], end=" ") | ||
| + |     print() | ||
| + |   print() | ||
| + | |||
| + | def fail(tab, nrow, f): | ||
| + |   for i in range(nrow): | ||
| + |     tab[i][f] = 'x' | ||
| + | |||
| + | def restorep(tab, nrow, ncol, f): | ||
| + |   for i in range(nrow): | ||
| + |     p = tab[i][ncol] | ||
| + |     for j in range(ncol): | ||
| + |       if j != f: | ||
| + |         p ^= tab[i][j] | ||
| + |     tab[i][f] = p | ||
| + | |||
| + | def restored(tab, nrow, ncol, f): | ||
| + |   p = [tab[i][ncol + 1] for i in range(nrow)] + [0] | ||
| + |   for i in range(nrow): | ||
| + |     for j in range(ncol): | ||
| + |       if j != f: | ||
| + |         p[(i+j) % ncol] ^= tab[i][j] | ||
| + |   s = p[(f - 1) % ncol] | ||
| + | # print(p, s) | ||
| + |   for i in range(nrow): | ||
| + |     tab[i][f] = s ^ p[(f + i) % ncol] | ||
| + | |||
| + | def restore2(tab, nrow, ncol, f1, f2): | ||
| + |   s = 0 | ||
| + |   for i in range(nrow): | ||
| + |     s ^= tab[i][ncol] ^ tab[i][ncol + 1] | ||
| + |   p = [s ^ tab[i][ncol + 1] for i in range(nrow)] + [s] | ||
| + | # print(s, p) | ||
| + |   nindex = (nrow + f2) % ncol | ||
| + |   for _ in range(nrow): | ||
| + |     v = p[nindex] | ||
| + | #   print(nindex, (nindex + 1) % ncol) | ||
| + |     for i in range(nrow): | ||
| + |       j = (nindex - i) % ncol | ||
| + |       if j != f1: | ||
| + |         v ^= tab[i][j] | ||
| + |       else: | ||
| + |         row = i | ||
| + | #   print(row) | ||
| + |     tab[row][f1] = v | ||
| + |     nindex = (row + f2) % ncol | ||
| + |     v = tab[row][ncol] | ||
| + |     for j in range(ncol): | ||
| + |       if j != f2: | ||
| + |         v ^= tab[row][j] | ||
| + |       tab[row][f2] = v | ||
| + | |||
| + | def main(): | ||
| + |   tab=[] | ||
| + |   while True: | ||
| + |     line = input("") | ||
| + |     if len(line) <= 1: | ||
| + |       break | ||
| + |     tab.append([0 if x == "0" else 1 for x in line.split()] + [0, 0]) | ||
| + |   nrow = len(tab) | ||
| + |   ncol = len(tab[0]) - 2 | ||
| + | # print_table(tab, nrow, ncol) | ||
| + |   compute_parity(tab, nrow, ncol) | ||
| + | # print_table(tab, nrow, ncol + 1) | ||
| + |   compute_dparity(tab, nrow, ncol) | ||
| + |   print_table(tab, nrow, ncol + 2) | ||
| + | |||
| + |   savetab = tab | ||
| + | |||
| + |   while True: | ||
| + |     line = input("failure ") | ||
| + |     s = line.split() | ||
| + |     if len(s) == 2: | ||
| + |       f1, f2 = int(s[0]), int(s[1]) | ||
| + |     elif len(s) == 1: | ||
| + |       f1 = f2 = int(s[0]) | ||
| + |     else: | ||
| + |       return | ||
| + | |||
| + |     fail(tab, nrow, f1) | ||
| + |     fail(tab, nrow, f2) | ||
| + |     print_table(tab, nrow, ncol + 2) | ||
| + |     if f1 == f2: | ||
| + |       print("1 faulty", f1) | ||
| + |       if f1 == ncol: | ||
| + |         compute_parity(tab, nrow, ncol) | ||
| + |       elif (f1 == ncol + 1): | ||
| + |         compute_dparity(tab, nrow, ncol) | ||
| + |       else: | ||
| + |         restorep(tab, nrow, ncol, f1) | ||
| + |     else: | ||
| + |       if f1 > f2: | ||
| + |         f1, f2 = f2, f1 | ||
| + |       print("2 faulty", f1, f2) | ||
| + |       if f1 == ncol: | ||
| + |         compute_parity(tab, nrow, ncol) | ||
| + |         compute_dparity(tab, nrow, ncol) | ||
| + |       elif f2 == ncol + 1: | ||
| + |         restorep(tab, nrow, ncol, f1) | ||
| + |         compute_dparity(tab, nrow, ncol) | ||
| + |       elif f2 == ncol: | ||
| + |         restored(tab, nrow, ncol, f1) | ||
| + |         compute_parity(tab, nrow, ncol) | ||
| + |         pass | ||
| + |       else: | ||
| + |         restore2(tab, nrow, ncol, f1, f2) | ||
| + |     print_table(tab, nrow, ncol + 2) | ||
| + |     if tab == savetab: | ||
| + |       print('OK') | ||
| + |     else: | ||
| + |       print('ERR!!!!') | ||
| + | |||
| + | if __name__ == "__main__": | ||
| + |   main() | ||
| + | </source> | ||
Latest revision as of 15:36, 4 April 2023
Il codice python che segue consente di provare sperimentalemnte le metodologie di fault tolerance descritte nell'articolo "EVENODD: An Optimal Scheme for Tolerating Double Disk Failures" di Mario Blaurn et al.
#!/usr/bin/env python3
def compute_parity(tab, nrow, ncol):
  for i in range(nrow):
    p = 0
    for j in range(ncol):
      p ^= tab[i][j]
    tab[i][ncol] = p
def compute_dparity(tab, nrow, ncol):
  p = [0] * ncol
  for i in range(nrow):
    for j in range(ncol):
      p[(i+j) % ncol] ^= tab[i][j]
# print(p)
  s = p[ncol - 1]
  for i in range(nrow):
    tab[i][ncol + 1] = p[i] ^ s
def print_table(tab, nrow, ncol):
  for i in range(nrow):
    for j in range(ncol):
      print(tab[i][j], end=" ")
    print()
  print()
def fail(tab, nrow, f):
  for i in range(nrow):
    tab[i][f] = 'x'
def restorep(tab, nrow, ncol, f):
  for i in range(nrow):
    p = tab[i][ncol]
    for j in range(ncol):
      if j != f:
        p ^= tab[i][j]
    tab[i][f] = p
def restored(tab, nrow, ncol, f):
  p = [tab[i][ncol + 1] for i in range(nrow)] + [0]
  for i in range(nrow):
    for j in range(ncol):
      if j != f:
        p[(i+j) % ncol] ^= tab[i][j]
  s = p[(f - 1) % ncol]
# print(p, s)
  for i in range(nrow):
    tab[i][f] = s ^ p[(f + i) % ncol]
def restore2(tab, nrow, ncol, f1, f2):
  s = 0
  for i in range(nrow):
    s ^= tab[i][ncol] ^ tab[i][ncol + 1]
  p = [s ^ tab[i][ncol + 1] for i in range(nrow)] + [s]
# print(s, p)
  nindex = (nrow + f2) % ncol
  for _ in range(nrow):
    v = p[nindex]
#   print(nindex, (nindex + 1) % ncol)
    for i in range(nrow):
      j = (nindex - i) % ncol
      if j != f1:
        v ^= tab[i][j]
      else:
        row = i
#   print(row)
    tab[row][f1] = v
    nindex = (row + f2) % ncol
    v = tab[row][ncol]
    for j in range(ncol):
      if j != f2:
        v ^= tab[row][j]
      tab[row][f2] = v
def main():
  tab=[]
  while True:
    line = input("")
    if len(line) <= 1:
      break
    tab.append([0 if x == "0" else 1 for x in line.split()] + [0, 0])
  nrow = len(tab)
  ncol = len(tab[0]) - 2
# print_table(tab, nrow, ncol)
  compute_parity(tab, nrow, ncol)
# print_table(tab, nrow, ncol + 1)
  compute_dparity(tab, nrow, ncol)
  print_table(tab, nrow, ncol + 2)
  savetab = tab
  while True:
    line = input("failure ")
    s = line.split()
    if len(s) == 2:
      f1, f2 = int(s[0]), int(s[1])
    elif len(s) == 1:
      f1 = f2 = int(s[0])
    else:
      return
    fail(tab, nrow, f1)
    fail(tab, nrow, f2)
    print_table(tab, nrow, ncol + 2)
    if f1 == f2:
      print("1 faulty", f1)
      if f1 == ncol:
        compute_parity(tab, nrow, ncol)
      elif (f1 == ncol + 1):
        compute_dparity(tab, nrow, ncol)
      else:
        restorep(tab, nrow, ncol, f1)
    else:
      if f1 > f2:
        f1, f2 = f2, f1
      print("2 faulty", f1, f2)
      if f1 == ncol:
        compute_parity(tab, nrow, ncol)
        compute_dparity(tab, nrow, ncol)
      elif f2 == ncol + 1:
        restorep(tab, nrow, ncol, f1)
        compute_dparity(tab, nrow, ncol)
      elif f2 == ncol:
        restored(tab, nrow, ncol, f1)
        compute_parity(tab, nrow, ncol)
        pass
      else:
        restore2(tab, nrow, ncol, f1, f2)
    print_table(tab, nrow, ncol + 2)
    if tab == savetab:
      print('OK')
    else:
      print('ERR!!!!')
if __name__ == "__main__":
  main()