Восстановление данных. Практическое руководство
Практический пример
Рассказ о файловой системе NTFS был бы неполным без практической иллюстрации техники разбора файловой записи вручную. До сих пор мы витали в облаках теоретической абстракции. Пора спускаться на грешную землю.
Воспользовавшись любым дисковым редактором, например, Disk Probe, попробуем декодировать одну файловую запись вручную. Найдем сектор, содержащий сигнатуру
в его начале (не обязательно брать первый встретившийся сектор). Он может выглядеть, например, как в листинге 6.4.FILEЛистинг 6.4. Ручное декодирование файловой записи (разные атрибуты выделены разным цветом)
: 00 01 02 03 04 05 06 07 | 08 09 0A 0B 0C 0D 0E 0F00000000: 46 49 4C 45 2A 00 03 00 | 60 79 1A 04 02 00 00 00 FILE*...`y......00000010: 01 00 01 00 30 00 01 00 | 50 01 00 00 00 04 00 00 ....0...P.......00000020: 00 00 00 00 00 00 00 00 | 04 00 03 00 00 00 00 00 ................00000030: 10 00 00 00 60 00 00 00 | 00 00 00 00 00 00 00 00 ................00000040: 48 00 00 00 18 00 00 00 | B0 D5 C9 2F C6 0B C4 01 H.......░╒╔/╞.─.00000050: E0 5A B3 7B A9 FA C3 01 | 90 90 F1 2F C6 0B C4 01 рZ│{й·├.PPё/╞.─.00000060: 50 7F BC FE C8 0B C4 01 | 20 00 00 00 00 00 00 00 P⌂╝■╚.─. .......00000070: 00 00 00 00 00 00 00 00 | 00 00 00 00 05 01 00 00 ................00000080: 00 00 00 00 00 00 00 00 | 00 00 00 00 00 00 00 00 ................00000090: 30 00 00 00 70 00 00 00 | 00 00 00 00 00 00 02 00 0...p...........000000A0: 54 00 00 00 18 00 01 00 | DB 1A 01 00 00 00 01 00 T.......█.......000000B0: B0 D5 C9 2F C6 0B C4 01 | B0 D5 C9 2F C6 0B C4 01 ░╒╔/╞.─.░╒╔/╞.─.000000C0: B0 D5 C9 2F C6 0B C4 01 | B0 D5 C9 2F C6 CB C4 01 ░╒╔/╞.─.░╒╔/╞.─.000000D0: 00 00 00 00 00 00 00 00 | 00 00 00 00 00 00 00 00 ................000000E0: 20 00 00 00 00 00 00 00 | 09 03 49 00 6C 00 66 00 ..........I.l.f.000000F0: 61 00 6B 00 2E 00 64 00 | 62 00 78 00 00 00 00 00 a.k...d.b.x.....00000100: 80 00 00 00 48 00 00 00 | 01 00 00 00 00 00 03 00 А...H...........00000110: 00 00 00 00 00 00 00 00 | ED 04 00 00 00 00 00 00 ........э.......00000120: 40 00 00 00 00 00 00 00 | 00 E0 4E 00 00 00 00 00 @........рN.....00000130: F0 D1 4E 00 00 00 00 00 | F0 D1 4E 00 00 00 00 00 Ё╤N.....Ё╤N.....00000140: 32 EE 04 D9 91 00 00 81 | FF FF FF FF 82 79 47 11 2ю.┘С..Б ВyG.000001F0: 00 00 00 00 00 00 00 00 | 00 00 00 00 00 00 03 00 ................: 00 01 02 03 04 05 06 07 | 08 09 0A 0B 0C 0D 0F 0FПервым делом необходимо восстановить оригинальное содержимое последовательности обновления. По смещению
от начала сектора лежит 16-разрядный указатель на нее, равный в данном случае04h(значит, это NTFS 3.0 или более ранняя версия). А что у нас лежит по смещению2Ah? Это — пара байт2Ah. Данная последовательность представляет собой номер последовательности обновления. Сверяем его с содержимым двух последних байт этого и следующего секторов (смещения03 00и1FEhсоответственно). Они равны! Следовательно, данная файловая запись цела (по крайней мере, на первый взгляд), и можно переходить к операции ее восстановления. По смещению3FEhрасположен массив, содержащий оригинальные значения последовательности обновления. Количество элементов в нем равно содержимому 16-разрядного поля, расположенному по смещению2Chот начала сектора и уменьшенного на единицу (в данном случае имеем06h). Извлекаем два слова, начиная со смещения03h - 01h == 02h(в данном случае они равны2Chи00 00) и записываем их в конец первого и последнего секторов.00 00Теперь нам необходимо выяснить, используется ли данная файловая запись, или же ассоциированный с ней файл или каталог был удален. 16-разрядное поле, расположенное по смещению
, содержит значение16h. Следовательно, перед нами файл, а не каталог, и этот файл еще не удален. Но является ли эта файловая запись базовой для данного файла или мы имеем дело с ее продолжением? 64-разрядное поле, расположенное по смещению01h, равно нулю, следовательно, данная файловая запись — базовая.20hОчень хорошо, теперь переходим к исследованию атрибутов. 16-разрядное поле, находящееся по смещению
, равно14h, следовательно, заголовок первого атрибута начинается со смещения30hот начала сектора.30hПервое двойное слово атрибута равно
, значит, перед нами атрибут типа10h. 32-разрядное поле длины атрибута, находящееся по смещению$STANDARD_INFORMATIONи равное в нашем случае04hбайт, позволяет нам вычислить смещение следующего атрибута в списке:60h(смещение нашего атрибута)30h(его длина)+ 60h(смещение следующего атрибута). Первое двойное слово следующего атрибута равно== 90h, значит, это атрибут типа30h, и следующее 32-разрядное поле хранит его длину, равную в данном случае$NAME. Сложив длину атрибута с его смещением, мы получим смещение следующего атрибута —70h. Первое двойное слово третьего атрибута равно90h + 70h == 100h, следовательно, это атрибут типа80h, хранящий основные данные файла. Складываем его смещение с длиной —$DATA. И вот здесь мы наткнулись на частокол100h + 32h == 132h, сигнализирующий о том, что атрибутFFFFFFhпоследний в списке.$DATAТеперь, разбив файловую запись на атрибуты, можно приступить к исследованию каждого из атрибутов в отдельности. Начнем с разбора имени. 8-разрядное поле, находящееся по смещению
от начала атрибутного заголовка (и по смещению08hот начала сектора), содержит флаг нерезидентности. В данном случае этот флаг равен нулю. Это значит, атрибут резидентный, и его тело хранится непосредственно в самой файловой записи, что уже хорошо. 16-разрядное поле, расположенное по смещению98hот начала атрибутного заголовка (и по смещению0Сhот начала сектора) равно нулю, следовательно, тело атрибута не сжато и не зашифровано. Таким образом, можно приступать к разбору тела атрибута. 32-разрядное поле, расположенное по смещению9Chот начала атрибутного заголовка (и по смещению10hот начала сектора), содержит длину атрибутного тела, равную в данном случаеA0hбайт. 16-разрядное поле, расположенное по смещению54hот начала атрибутного заголовка и по смещению14hот начала сектора, хранит смещение атрибутного тела, равное в данном случаеA4h. Следовательно, тело атрибута18hрасполагается по смещению$FILE_NAMEот начала сектора.A8h