DataRun.cs 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. /* Copyright (C) 2014-2018 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
  2. *
  3. * You can redistribute this program and/or modify it under the terms of
  4. * the GNU Lesser Public License as published by the Free Software Foundation,
  5. * either version 3 of the License, or (at your option) any later version.
  6. */
  7. using System;
  8. using System.Collections.Generic;
  9. using System.IO;
  10. namespace DiskAccessLibrary.FileSystems.NTFS
  11. {
  12. public class DataRun
  13. {
  14. // The maximum NTFS file size is 2^64 bytes, so total number of file clusters can be represented using long
  15. // http://technet.microsoft.com/en-us/library/cc938937.aspx
  16. public long RunLength; // In clusters
  17. public long RunOffset; // In clusters, relative to previous data run start LCN
  18. public bool IsSparse;
  19. /// <returns>Record length</returns>
  20. public int Read(byte[] buffer, int offset)
  21. {
  22. int runOffsetSize = buffer[offset] >> 4;
  23. int runLengthSize = buffer[offset] & 0x0F;
  24. RunLength = ReadVarLong(ref buffer, offset + 1, runLengthSize);
  25. if (RunLength < 0)
  26. {
  27. throw new InvalidDataException("Invalid Data Run record");
  28. }
  29. RunOffset = ReadVarLong(ref buffer, offset + 1 + runLengthSize, runOffsetSize);
  30. IsSparse = (runOffsetSize == 0);
  31. return 1 + runLengthSize + runOffsetSize;
  32. }
  33. public byte[] GetBytes()
  34. {
  35. if (IsSparse)
  36. {
  37. RunOffset = 0;
  38. }
  39. byte[] buffer = new byte[RecordLength];
  40. int runLengthSize = WriteVarLong(buffer, 1, RunLength);
  41. int runOffsetSize;
  42. if (IsSparse)
  43. {
  44. runOffsetSize = 0;
  45. }
  46. else
  47. {
  48. runOffsetSize = WriteVarLong(buffer, 1 + runLengthSize, RunOffset);
  49. }
  50. buffer[0] = (byte)((runLengthSize & 0x0F) | ((runOffsetSize << 4) & 0xF0));
  51. return buffer;
  52. }
  53. private static long ReadVarLong(ref byte[] buffer, int offset, int size)
  54. {
  55. ulong val = 0;
  56. bool signExtend = false;
  57. for (int i = 0; i < size; ++i)
  58. {
  59. byte b = buffer[offset + i];
  60. val = val | (((ulong)b) << (i * 8));
  61. signExtend = (b & 0x80) != 0;
  62. }
  63. if (signExtend)
  64. {
  65. for (int i = size; i < 8; ++i)
  66. {
  67. val = val | (((ulong)0xFF) << (i * 8));
  68. }
  69. }
  70. return (long)val;
  71. }
  72. private static int WriteVarLong(byte[] buffer, int offset, long val)
  73. {
  74. bool isPositive = val >= 0;
  75. int pos = 0;
  76. do
  77. {
  78. buffer[offset + pos] = (byte)(val & 0xFF);
  79. val >>= 8;
  80. pos++;
  81. }
  82. while (val != 0 && val != -1);
  83. // Avoid appearing to have a negative number that is actually positive,
  84. // record an extra empty byte if needed.
  85. if (isPositive && (buffer[offset + pos - 1] & 0x80) != 0)
  86. {
  87. buffer[offset + pos] = 0;
  88. pos++;
  89. }
  90. else if (!isPositive && (buffer[offset + pos - 1] & 0x80) != 0x80)
  91. {
  92. buffer[offset + pos] = 0xFF;
  93. pos++;
  94. }
  95. return pos;
  96. }
  97. private static int VarLongSize(long val)
  98. {
  99. bool isPositive = val >= 0;
  100. bool lastByteHighBitSet = false;
  101. int len = 0;
  102. do
  103. {
  104. lastByteHighBitSet = (val & 0x80) != 0;
  105. val >>= 8;
  106. len++;
  107. }
  108. while (val != 0 && val != -1);
  109. if ((isPositive && lastByteHighBitSet) || (!isPositive && !lastByteHighBitSet))
  110. {
  111. len++;
  112. }
  113. return len;
  114. }
  115. /// <summary>
  116. /// Length of the DataRun record inside the non-resident attribute record
  117. /// </summary>
  118. public int RecordLength
  119. {
  120. get
  121. {
  122. int runLengthSize = VarLongSize(RunLength);
  123. int runOffsetSize = VarLongSize(RunOffset);
  124. return 1 + runLengthSize + runOffsetSize;
  125. }
  126. }
  127. }
  128. }