IMG архив — различия между версиями
Материал из GTAModding.ru
Dageron (обсуждение | вклад) |
Seemann (обсуждение | вклад) (→Версия 2 - GTA SA: для чего GOTO то?!) |
||
Строка 49: | Строка 49: | ||
ID:Array [0..3] of AnsiChar; //Для идентификатора формата | ID:Array [0..3] of AnsiChar; //Для идентификатора формата | ||
FileName, DirName : String; //Для названия IMG и DIR | FileName, DirName : String; //Для названия IMG и DIR | ||
− | |||
− | |||
procedure TForm1.Close1Click(Sender: TObject); //Процедура открытия архива | procedure TForm1.Close1Click(Sender: TObject); //Процедура открытия архива | ||
Строка 59: | Строка 57: | ||
DestFile : TFileStream; // Сюда будем загружать файл для сохранения | DestFile : TFileStream; // Сюда будем загружать файл для сохранения | ||
Label VER1,VER2,MAINEND; | Label VER1,VER2,MAINEND; | ||
− | |||
begin | begin | ||
Строка 74: | Строка 71: | ||
Stream := TFileStream.Create(FileName, fmOpenRead); // Открываем архив для чтения | Stream := TFileStream.Create(FileName, fmOpenRead); // Открываем архив для чтения | ||
Stream.Read(ID, 4); // Записываем первые три байта чтобы проверить верисию архива | Stream.Read(ID, 4); // Записываем первые три байта чтобы проверить верисию архива | ||
− | if ID='VER2' then | + | if ID='VER2' then |
− | + | ||
− | + | ||
begin | begin | ||
label2.Caption:='Загружен архив версии №2 - SanAndreas'; | label2.Caption:='Загружен архив версии №2 - SanAndreas'; | ||
Строка 90: | Строка 85: | ||
ListItem.SubItems.Add(IntToStr(Offset * 2048)); // | ListItem.SubItems.Add(IntToStr(Offset * 2048)); // | ||
end; | end; | ||
− | end | + | end |
− | + | else begin | |
− | + | if FileExists(DirName) then | |
− | + | label2.Caption:='Загружен архив версии №1 - GTAIII/ViceCity' | |
− | + | else | |
− | + | label2.Caption:='Ошибка загрузки архива версии №2 - не найден DIR файл'; | |
− | + | Stream.Read(ID, 3); //Считываем первые три байта | |
− | + | DIRStream := TFileStream.Create(DirName, fmOpenRead); //Создаем область памяти куда будем записывать DIR | |
− | + | FileNum:=DIRStream.Size div 32; //Высчитыввем число файлов | |
//ProgressBar1.Step:=1; | //ProgressBar1.Step:=1; | ||
//ProgressBar1.Max:=FileNum; | //ProgressBar1.Max:=FileNum; | ||
Строка 111: | Строка 106: | ||
ListItem.SubItems.Add(IntToStr(Offset * 2048)); // | ListItem.SubItems.Add(IntToStr(Offset * 2048)); // | ||
end; | end; | ||
− | + | end; | |
− | + | ||
− | + | ||
end; | end; | ||
end; | end; |
Версия 17:24, 30 января 2009
В IMG архивах GTA в основном хранятся игровые модели, текстуры, транспортные и пешеходные пути, анимации, а так же внешние скрипты. IMG архив имеет расширение .img
. Эти архивы разделяются на версии и имеют схожий между собой формат (за исключением версии, используемой GTA 4).
Содержание |
Структура
Версия 1 - GTA III & VC
Архивы первой версии, используемые в GTA III и Vice City, разделены на два файла: директория содержимого архива .dir
и сам архив, содержащий файлы .img
.
Директория содержит список файлов, их имена, смещение и размер. Изначально формат IMG оптимизировался для использования на DVD или CD, поэтому размер и смещение файлов кратны сектору DVD (2048 байт). Даже если файл имеет размер менее 2-х КБ (например, 218 байт), то в архиве он будет занимать 2 КБ. Файлы (обычно) не сортированы, не сжаты и располагаются "друг за другом" (никакой древовидной иерархии).
Чтобы найти общее количество элементов (файлов) в IMG архиве, необходимо разделить размер .dir файла на 32.
Файл директории не имеет заголовка, просто массив элементов по 32 байта на каждый:
Элемент (повторяется n-раз, где n - общее количество элементов): 4 байта - DWORD - смещение файла (в блоках) в .img архиве, относительно его начала 4 байта - DWORD - размер файла (в блоках) в .img архиве 24 байта - CHAR[24] - имя файла
Сам IMG архив не имеет заголовка, просто файлы, как говорилось ранее, расположенные в блоках по 2 КБ.
Версия 2 - GTA SA
Архивы второй версии, используемые в GTA San Andreas, содержат .dir
и .img
файлы в одном IMG файле. Директория имеет такой же формат, как и в формате 1-ой версии, но располагается в начале IMG файла. Файлы так же размещены в блоках по 2 КБ. Смещения файлов задаются от начала самого архива, а не от конца списка файлов.
Заголовок: 4 байта - CHAR[4] - сигнатура архива, всегда имеет значение "VER2" 4 байта - DWORD - общее количество элементов (файлов) Элемент: (повторяется n-раз, где n - общее количество элементов): 4 байта - DWORD - смещение файла (в блоках) в архиве 4 байта - DWORD - размер файла (в блоках) 24 байта - CHAR[24] - имя файла
Недостаток этого формата - это его расширяемость. Если вы добавите слишком много файлов, директория начнёт записываться поверх первых файлов в архиве, поэтому вам нужно будет записывать первые файлы в конец архива.
Пример программы на Delphi по работе с форматом GTAIII/VC/SA
{В глобальные переменные}
Stream, DIRStream: TFileStream; //Stream - переменная для загрузки архива, DIRStream - переменная для загрузки DIR файла
Size, Offset, FileNum, A : Integer; //Целые данные - размер, смещение, количество файлов
ID:Array [0..3] of AnsiChar; //Для идентификатора формата
FileName, DirName : String; //Для названия IMG и DIR
procedure TForm1.Close1Click(Sender: TObject); //Процедура открытия архива
var
ListItem : TListItem;
Name: Array [0..23] of AnsiChar; //Переменная для сохранения имен
Z:Integer;
DestFile : TFileStream; // Сюда будем загружать файл для сохранения
Label VER1,VER2,MAINEND;
begin
if OpenDialog1.Execute then //Вызываем диалог открытия
begin
CFileName.Text := OpenDialog1.FileName;
DirName:= CFileName.Text; //Присваиваем DirName название самого архива
Delete (DirName, Length (DirName)-3, 4); //Удаляем расширение
DirName:=DirName+'.dir'; //Ставим расширение ".dir"
Size := 0;
Offset := 0;
CFileList.Clear;
FileName := OpenDialog1.FileName;
Stream := TFileStream.Create(FileName, fmOpenRead); // Открываем архив для чтения
Stream.Read(ID, 4); // Записываем первые три байта чтобы проверить верисию архива
if ID='VER2' then
begin
label2.Caption:='Загружен архив версии №2 - SanAndreas';
Stream.Read(FileNum, 4); // Считываем количество файлов
for A := 0 to FileNum - 1 do // Цикл считывания всех файлов
begin
Stream.Read(Offset, 4); // Считываем смещение файла
Stream.Read(Size, 4); // Считываем размер файла
Stream.Read(Name, 24); // Считываем имя файла
ListItem := CFileList.Items.Add; //
ListItem.Caption := Name; // Добавляем файл со всеми
ListItem.SubItems.Add(IntToStr(Size * 2048)); // параметрами в список файлов
ListItem.SubItems.Add(IntToStr(Offset * 2048)); //
end;
end
else begin
if FileExists(DirName) then
label2.Caption:='Загружен архив версии №1 - GTAIII/ViceCity'
else
label2.Caption:='Ошибка загрузки архива версии №2 - не найден DIR файл';
Stream.Read(ID, 3); //Считываем первые три байта
DIRStream := TFileStream.Create(DirName, fmOpenRead); //Создаем область памяти куда будем записывать DIR
FileNum:=DIRStream.Size div 32; //Высчитыввем число файлов
//ProgressBar1.Step:=1;
//ProgressBar1.Max:=FileNum;
for A := 0 to FileNum - 1 do // Цикл считывания всех файлов
begin
DIRStream.Read(Offset, 4); // Считываем смещение файла
DIRStream.Read(Size, 4); // Считываем размер файла
DIRStream.Read(Name, 24); // Считываем имя файла
ListItem := CFileList.Items.Add; //
ListItem.Caption := Name; // Добавляем файл со всеми
ListItem.SubItems.Add(IntToStr(Size * 2048)); // параметрами в список файлов
ListItem.SubItems.Add(IntToStr(Offset * 2048)); //
end;
end;
end;
end;
procedure TForm1.N6Click(Sender: TObject); //Процедура извлечения файлов
var
DestFile : TFileStream; // Сюда будем загружать файл для сохранения
begin
if CFileList.SelCount > 0 then // Если что-либо выделено в списке файлов
begin
SaveDialog1.FileName := CFileList.Selected.Caption;
if SaveDialog1.Execute then //Вызываем диалог сохранения
begin
DestFile := TFileStream.Create(SaveDialog1.FileName, fmCreate); // Создаём файл
DestFile.Seek(0, 0); // Смещаемся в начало файла.
Size := StrToInt(CFileList.Selected.SubItems.Strings[0]);// Узнаём размер выделенного файла
Offset := StrToInt(CFileList.Selected.SubItems.Strings[1]);// Узнаём смещение выделенного файла
Stream.Seek(Offset, 0); // Смещаемся к смещению (то есть, к началу) файла в архиве
DestFile.CopyFrom(Stream, Size); // Копируем количество байт, равным размеру файла в архиве
DestFile.Free; // Закрываем файл
end;
end;
end;
Версия 2А - GTA:LCS и GTA:VCS
Архивы портативных GTA делятся на два типа - сжатые и несжатые.
Структура несжатого архива почти не отличается от структуры версии 1. Отсутствует .dir
файл. Имена файлов, их смещение и размер содержатся в GAME.DTZ архиве (ближе к началу файла). То есть для того чтобы открыть несжатый архив, нужно либо извлечь данные из GAME.DTZ, либо сделать .dir
таблицу файлов самостоятельно (причем второй вариант оказался гораздо более простым и умеет делать это YAIE).
Сжатые архивы имеют гораздо более сложную структуру. В таком .img
файле нет разделителей и читать данные о смещении и размере неоткуда. На смену .dir
пришел .lvz
- новый, неизвестный формат по своей сути полностью отличающийся от предшественника.
Версия 3 - GTA IV
Данная версия архива имеет более сложную структуру. Общим со старыми версиями осталось только то, что файлы выравнены по размеру сектора (2048 байт). Каталог архива может быть зашифрован.
Заголовок
Заголовок файла всегда равен 20 байтам.
4 byte - DWORD - Идентификатор (0xA94E2A52) Если, это поле имеет другое значение, архив зашифрован 4 byte - DWORD - Версия (всегда 3) 4 byte - DWORD - Количество объектов 4 byte - DWORD - Размер каталога (в байтах) 2 byte - WORD - Размер элемента каталога (всегда должен быть равен 0x10) 2 byte - WORD - Неизвестно
Каталог архива
Каталог архива состоит из двух частей: описателей файлов, содержащих смещение и размер файла внутри архива и имен файлов.
В начале идет массив описателей по 16 байт:
4 byte - DWORD - Размер объекта и флаги 4 byte - DWORD - Версия ресурса (только для RSC, для остальных объектов не используется) 4 byte - DWORD - Смещение от начала IMG в блоках (2048 байт) 2 byte - WORD - Количество используемых блоков 2 byte - WORD - Размер неиспользуемого места в последнем блоке
За описателями следуют имена файлов, разделенные '\x0', в том же порядке, что и описатели.
Чтение зашифрованных архивов
Если идентификатор архива не равен 0xA94E2A52, архив считается зашифрованным. Для чтения такого архива, следует прочитать заголовок (20 байт), но дешифровать только первые 16 байт. После этого, следует снова проверить идентификатор. Если идентификатор корректный, следует считать каталог (размер берется из расшифрованного заголовка) и дешифровать его, округлив размер до числа кратного 16, вниз. (Например, если указан размер 259 байт, дешифруются только первые 256).