blob: 4e9ea5d48689dab07499ea9465f3c4ac19f80554 [file] [log] [blame]
// 7zUpdate.cpp
#include "StdAfx.h"
#include "../../../../C/CpuArch.h"
#include "../../../Common/MyLinux.h"
#include "../../../Common/StringToInt.h"
#include "../../../Common/Wildcard.h"
#include "../../Common/CreateCoder.h"
#include "../../Common/LimitedStreams.h"
#include "../../Common/ProgressUtils.h"
#include "../../Compress/CopyCoder.h"
#include "../Common/ItemNameUtils.h"
#include "7zDecode.h"
#include "7zEncode.h"
#include "7zFolderInStream.h"
#include "7zHandler.h"
#include "7zOut.h"
#include "7zUpdate.h"
namespace NArchive {
namespace N7z {
#define k_X86 k_BCJ
struct CFilterMode
{
UInt32 Id;
UInt32 Delta; // required File Size alignment, if Id is not k_Delta.
// (Delta == 0) means unknown alignment
UInt32 Offset; // for k_ARM64
// UInt32 AlignSizeOpt; // for k_ARM64
CFilterMode():
Id(0),
Delta(0),
Offset(0)
// , AlignSizeOpt(0)
{}
void ClearFilterMode()
{
Id = 0;
Delta = 0;
Offset = 0;
// AlignSizeOpt = 0;
}
// it sets Delta as Align value, if Id is exe filter
// in another cases it sets Delta = 0, that
void SetDelta()
{
if (Id == k_IA64)
Delta = 16;
else if (Id == k_ARM64 || Id == k_ARM || Id == k_PPC || Id == k_SPARC)
Delta = 4;
else if (Id == k_ARMT)
Delta = 2;
else if (Id == k_BCJ || Id == k_BCJ2)
Delta = 1; // do we need it?
else
Delta = 0;
}
};
/* ---------- PE ---------- */
#define MZ_SIG 0x5A4D
#define PE_SIG 0x00004550
#define PE_OptHeader_Magic_32 0x10B
#define PE_OptHeader_Magic_64 0x20B
// #define PE_SectHeaderSize 40
// #define PE_SECT_EXECUTE 0x20000000
static int Parse_EXE(const Byte *buf, size_t size, CFilterMode *filterMode)
{
if (size < 512 || GetUi16(buf) != MZ_SIG)
return 0;
const Byte *p;
UInt32 peOffset, optHeaderSize, filterId;
peOffset = GetUi32(buf + 0x3C);
if (peOffset >= 0x1000 || peOffset + 512 > size || (peOffset & 7) != 0)
return 0;
p = buf + peOffset;
if (GetUi32(p) != PE_SIG)
return 0;
p += 4;
switch (GetUi16(p))
{
case 0x014C:
case 0x8664: filterId = k_X86; break;
case 0xAA64: filterId = k_ARM64; break;
/*
IMAGE_FILE_MACHINE_ARM 0x01C0 // ARM LE
IMAGE_FILE_MACHINE_THUMB 0x01C2 // ARM Thumb / Thumb-2 LE
IMAGE_FILE_MACHINE_ARMNT 0x01C4 // ARM Thumb-2, LE
Note: We use ARM filter for 0x01C2. (WinCE 5 - 0x01C2) files mostly contain ARM code (not Thumb/Thumb-2).
*/
case 0x01C0: // WinCE old
case 0x01C2: filterId = k_ARM; break; // WinCE new
case 0x01C4: filterId = k_ARMT; break; // WinRT
case 0x0200: filterId = k_IA64; break;
default: return 0;
}
// const UInt32 numSections = GetUi16(p + 2);
optHeaderSize = GetUi16(p + 16);
if (optHeaderSize > (1 << 10))
return 0;
p += 20; /* headerSize */
switch (GetUi16(p))
{
case PE_OptHeader_Magic_32:
case PE_OptHeader_Magic_64:
break;
default:
return 0;
}
/*
// Windows exe file sizes are not aligned for 4 KiB.
// So we can't use (CFilterMode::Offset != 0) in solid archives.
// So we just don't set Offset here.
#define NUM_SCAN_SECTIONS_MAX (1 << 6)
#define EXE_SECTION_OFFSET_MAX (1 << 27)
#define EXE_SECTION_SIZE_MIN (1 << 8)
#define EXE_SECTION_SIZE_MAX (1 << 27)
#define PE_SectHeaderSize 40
#define PE_SECT_EXECUTE 0x20000000
if (numSections > NUM_SCAN_SECTIONS_MAX)
return 0;
p += optHeaderSize;
// UInt32 numExeSections = 0;
// bool execute_finded = false;
// UInt32 sect_va = 0;
// UInt32 sect_size = 0;
// UInt32 sect_offset = 0;
for (UInt32 i = 0; i < numSections
// && numExeSections < numSectionsMax
; i++, p += PE_SectHeaderSize)
{
UInt32 characts, rawSize, offset;
if ((UInt32)(p - buf) + PE_SectHeaderSize > size)
return 0;
rawSize = GetUi32(p + 16);
offset = GetUi32(p + 20);
characts = GetUi32(p + 36);
if (rawSize >= EXE_SECTION_SIZE_MIN &&
rawSize <= EXE_SECTION_SIZE_MAX &&
offset <= EXE_SECTION_OFFSET_MAX &&
// offset < limit &&
offset > 0)
{
if ((characts & PE_SECT_EXECUTE) != 0)
{
// execute_finded = true;
// sect_va = GetUi32(p + 12);
// sect_size = rawSize;
// sect_offset = offset;
break;
}
}
}
filterMode->Offset = 0;
if (filterId == k_ARM64)
{
// filterMode->AlignSizeOpt = (1 << 12);
// const UInt32 offs = (sect_va - sect_offset) & 0xFFF;
// if (offs != 0)
// filterMode->Offset = offs; // change it
}
*/
filterMode->Id = filterId;
return 1;
}
/* ---------- ELF ---------- */
#define ELF_SIG 0x464C457F
#define ELF_CLASS_32 1
#define ELF_CLASS_64 2
#define ELF_DATA_2LSB 1
#define ELF_DATA_2MSB 2
static UInt16 Get16(const Byte *p, BoolInt be) { if (be) return (UInt16)GetBe16(p); return (UInt16)GetUi16(p); }
static UInt32 Get32(const Byte *p, BoolInt be) { if (be) return GetBe32(p); return GetUi32(p); }
// static UInt64 Get64(const Byte *p, BoolInt be) { if (be) return GetBe64(p); return GetUi64(p); }
static int Parse_ELF(const Byte *buf, size_t size, CFilterMode *filterMode)
{
BoolInt /* is32, */ be;
UInt32 filterId;
if (size < 512 || buf[6] != 1) /* ver */
return 0;
if (GetUi32(buf) != ELF_SIG)
return 0;
switch (buf[4])
{
case ELF_CLASS_32: /* is32 = True; */ break;
case ELF_CLASS_64: /* is32 = False; */ break;
default: return 0;
}
switch (buf[5])
{
case ELF_DATA_2LSB: be = False; break;
case ELF_DATA_2MSB: be = True; break;
default: return 0;
}
switch (Get16(buf + 0x12, be))
{
case 3:
case 6:
case 62: filterId = k_X86; break;
case 2:
case 18:
case 43: filterId = k_SPARC; break;
case 20:
case 21: if (!be) return 0; filterId = k_PPC; break;
case 40: if ( be) return 0; filterId = k_ARM; break;
case 183: if (be) return 0; filterId = k_ARM64; break;
/* Some IA-64 ELF executables have size that is not aligned for 16 bytes.
So we don't use IA-64 filter for IA-64 ELF */
// case 50: if ( be) return 0; filterId = k_IA64; break;
default: return 0;
}
filterMode->Id = filterId;
return 1;
}
/* ---------- Mach-O ---------- */
#define MACH_SIG_BE_32 0xCEFAEDFE
#define MACH_SIG_BE_64 0xCFFAEDFE
#define MACH_SIG_LE_32 0xFEEDFACE
#define MACH_SIG_LE_64 0xFEEDFACF
#define MACH_ARCH_ABI64 (1 << 24)
#define MACH_MACHINE_386 7
#define MACH_MACHINE_ARM 12
#define MACH_MACHINE_SPARC 14
#define MACH_MACHINE_PPC 18
#define MACH_MACHINE_PPC64 (MACH_ARCH_ABI64 | MACH_MACHINE_PPC)
#define MACH_MACHINE_AMD64 (MACH_ARCH_ABI64 | MACH_MACHINE_386)
#define MACH_MACHINE_ARM64 (MACH_ARCH_ABI64 | MACH_MACHINE_ARM)
static unsigned Parse_MACH(const Byte *buf, size_t size, CFilterMode *filterMode)
{
UInt32 filterId, numCommands, commandsSize;
if (size < 512)
return 0;
BoolInt /* mode64, */ be;
switch (GetUi32(buf))
{
case MACH_SIG_BE_32: /* mode64 = False; */ be = True; break;
case MACH_SIG_BE_64: /* mode64 = True; */ be = True; break;
case MACH_SIG_LE_32: /* mode64 = False; */ be = False; break;
case MACH_SIG_LE_64: /* mode64 = True; */ be = False; break;
default: return 0;
}
switch (Get32(buf + 4, be))
{
case MACH_MACHINE_386:
case MACH_MACHINE_AMD64: filterId = k_X86; break;
case MACH_MACHINE_ARM: if ( be) return 0; filterId = k_ARM; break;
case MACH_MACHINE_SPARC: if (!be) return 0; filterId = k_SPARC; break;
case MACH_MACHINE_PPC:
case MACH_MACHINE_PPC64: if (!be) return 0; filterId = k_PPC; break;
case MACH_MACHINE_ARM64: if ( be) return 0; filterId = k_ARM64; break;
default: return 0;
}
numCommands = Get32(buf + 0x10, be);
commandsSize = Get32(buf + 0x14, be);
if (commandsSize > (1 << 24) || numCommands > (1 << 18))
return 0;
filterMode->Id = filterId;
return 1;
}
/* ---------- WAV ---------- */
#define WAV_SUBCHUNK_fmt 0x20746D66
#define WAV_SUBCHUNK_data 0x61746164
#define RIFF_SIG 0x46464952
static BoolInt Parse_WAV(const Byte *buf, size_t size, CFilterMode *filterMode)
{
UInt32 subChunkSize, pos;
if (size < 0x2C)
return False;
if (GetUi32(buf + 0) != RIFF_SIG ||
GetUi32(buf + 8) != 0x45564157 || // WAVE
GetUi32(buf + 0xC) != WAV_SUBCHUNK_fmt)
return False;
subChunkSize = GetUi32(buf + 0x10);
/* [0x14 = format] = 1 (PCM) */
if (subChunkSize < 0x10 || subChunkSize > 0x12 || GetUi16(buf + 0x14) != 1)
return False;
const unsigned numChannels = GetUi16(buf + 0x16);
const unsigned bitsPerSample = GetUi16(buf + 0x22);
if ((bitsPerSample & 0x7) != 0)
return False;
const UInt32 delta = (UInt32)numChannels * (bitsPerSample >> 3);
if (delta == 0 || delta > 256)
return False;
pos = 0x14 + subChunkSize;
const int kNumSubChunksTests = 10;
// Do we need to scan more than 3 sub-chunks?
for (int i = 0; i < kNumSubChunksTests; i++)
{
if (pos + 8 > size)
return False;
subChunkSize = GetUi32(buf + pos + 4);
if (GetUi32(buf + pos) == WAV_SUBCHUNK_data)
{
filterMode->Id = k_Delta;
filterMode->Delta = delta;
return True;
}
if (subChunkSize > (1 << 16))
return False;
pos += subChunkSize + 8;
}
return False;
}
/*
filterMode->Delta will be set as:
= delta value : [1, 256] : for k_Delta
= 0 for another filters (branch filters)
*/
static BoolInt ParseFile(const Byte *buf, size_t size, CFilterMode *filterMode)
{
filterMode->ClearFilterMode();
if (Parse_EXE(buf, size, filterMode)) return True;
if (Parse_ELF(buf, size, filterMode)) return True;
if (Parse_MACH(buf, size, filterMode)) return True;
return Parse_WAV(buf, size, filterMode);
}
struct CFilterMode2: public CFilterMode
{
bool Encrypted;
unsigned GroupIndex;
CFilterMode2(): Encrypted(false) {}
int Compare(const CFilterMode2 &m) const
{
if (!Encrypted)
{
if (m.Encrypted)
return -1;
}
else if (!m.Encrypted)
return 1;
const UInt32 id1 = Id;
const UInt32 id2 = m.Id;
/*
// we can change the order to place k_ARM64 files close to another exe files
if (id1 <= k_SPARC &&
id2 <= k_SPARC)
{
#define k_ARM64_FOR_SORT 0x3030901
if (id1 == k_ARM64) id1 = k_ARM64_FOR_SORT;
if (id2 == k_ARM64) id2 = k_ARM64_FOR_SORT;
}
*/
if (id1 < id2) return -1;
if (id1 > id2) return 1;
if (Delta < m.Delta) return -1;
if (Delta > m.Delta) return 1;
if (Offset < m.Offset) return -1;
if (Offset > m.Offset) return 1;
/* we don't go here, because GetGroup()
and operator ==(const CFilterMode2 &m)
add only unique CFilterMode2:: { Id, Delta, Offset, Encrypted } items.
*/
/*
if (GroupIndex < m.GroupIndex) return -1;
if (GroupIndex > m.GroupIndex) return 1;
*/
return 0;
}
bool operator ==(const CFilterMode2 &m) const
{
return Id == m.Id
&& Delta == m.Delta
&& Offset == m.Offset
&& Encrypted == m.Encrypted;
}
};
static unsigned GetGroup(CRecordVector<CFilterMode2> &filters, const CFilterMode2 &m)
{
unsigned i;
for (i = 0; i < filters.Size(); i++)
{
const CFilterMode2 &m2 = filters[i];
if (m == m2)
return i;
/*
if (m.Encrypted != m2.Encrypted)
{
if (!m.Encrypted)
break;
continue;
}
if (m.Id < m2.Id) break;
if (m.Id != m2.Id) continue;
if (m.Delta < m2.Delta) break;
if (m.Delta != m2.Delta) continue;
*/
}
// filters.Insert(i, m);
// return i;
return filters.Add(m);
}
static inline bool Is86Filter(CMethodId m)
{
return (m == k_BCJ || m == k_BCJ2);
}
static inline bool IsExeFilter(CMethodId m)
{
switch (m)
{
case k_ARM64:
case k_BCJ:
case k_BCJ2:
case k_ARM:
case k_ARMT:
case k_PPC:
case k_SPARC:
case k_IA64:
return true;
}
return false;
}
static unsigned Get_FilterGroup_for_Folder(
CRecordVector<CFilterMode2> &filters, const CFolderEx &f, bool extractFilter)
{
CFilterMode2 m;
// m.Id = 0;
// m.Delta = 0;
// m.Offset = 0;
m.Encrypted = f.IsEncrypted();
if (extractFilter)
{
const CCoderInfo &coder = f.Coders[f.UnpackCoder];
if (coder.MethodID == k_Delta)
{
if (coder.Props.Size() == 1)
{
m.Delta = (unsigned)coder.Props[0] + 1;
m.Id = k_Delta;
}
}
else if (IsExeFilter(coder.MethodID))
{
m.Id = (UInt32)coder.MethodID;
if (m.Id == k_BCJ2)
m.Id = k_BCJ;
m.SetDelta();
if (m.Id == k_ARM64)
if (coder.Props.Size() == 4)
m.Offset = GetUi32(coder.Props);
}
}
return GetGroup(filters, m);
}
static HRESULT WriteRange(IInStream *inStream, ISequentialOutStream *outStream,
UInt64 position, UInt64 size, ICompressProgressInfo *progress)
{
RINOK(InStream_SeekSet(inStream, position))
CLimitedSequentialInStream *streamSpec = new CLimitedSequentialInStream;
CMyComPtr<ISequentialInStream> inStreamLimited(streamSpec);
streamSpec->SetStream(inStream);
streamSpec->Init(size);
NCompress::CCopyCoder *copyCoderSpec = new NCompress::CCopyCoder;
CMyComPtr<ICompressCoder> copyCoder = copyCoderSpec;
RINOK(copyCoder->Code(inStreamLimited, outStream, NULL, NULL, progress))
return (copyCoderSpec->TotalSize == size ? S_OK : E_FAIL);
}
/*
unsigned CUpdateItem::GetExtensionPos() const
{
int slashPos = Name.ReverseFind_PathSepar();
int dotPos = Name.ReverseFind_Dot();
if (dotPos <= slashPos)
return Name.Len();
return dotPos + 1;
}
UString CUpdateItem::GetExtension() const
{
return Name.Ptr(GetExtensionPos());
}
*/
#define RINOZ(x) { const int _t_ = (x); if (_t_ != 0) return _t_; }
#define RINOZ_COMP(a, b) RINOZ(MyCompare(a, b))
/*
static int CompareBuffers(const CByteBuffer &a1, const CByteBuffer &a2)
{
size_t c1 = a1.GetCapacity();
size_t c2 = a2.GetCapacity();
RINOZ_COMP(c1, c2);
for (size_t i = 0; i < c1; i++)
RINOZ_COMP(a1[i], a2[i]);
return 0;
}
static int CompareCoders(const CCoderInfo &c1, const CCoderInfo &c2)
{
RINOZ_COMP(c1.NumInStreams, c2.NumInStreams);
RINOZ_COMP(c1.NumOutStreams, c2.NumOutStreams);
RINOZ_COMP(c1.MethodID, c2.MethodID);
return CompareBuffers(c1.Props, c2.Props);
}
static int CompareBonds(const CBond &b1, const CBond &b2)
{
RINOZ_COMP(b1.InIndex, b2.InIndex);
return MyCompare(b1.OutIndex, b2.OutIndex);
}
static int CompareFolders(const CFolder &f1, const CFolder &f2)
{
int s1 = f1.Coders.Size();
int s2 = f2.Coders.Size();
RINOZ_COMP(s1, s2);
int i;
for (i = 0; i < s1; i++)
RINOZ(CompareCoders(f1.Coders[i], f2.Coders[i]));
s1 = f1.Bonds.Size();
s2 = f2.Bonds.Size();
RINOZ_COMP(s1, s2);
for (i = 0; i < s1; i++)
RINOZ(CompareBonds(f1.Bonds[i], f2.Bonds[i]));
return 0;
}
*/
/*
static int CompareFiles(const CFileItem &f1, const CFileItem &f2)
{
return CompareFileNames(f1.Name, f2.Name);
}
*/
struct CFolderRepack
{
unsigned FolderIndex;
CNum NumCopyFiles;
};
/*
static int CompareFolderRepacks(const CFolderRepack *p1, const CFolderRepack *p2, void *)
{
int i1 = p1->FolderIndex;
int i2 = p2->FolderIndex;
// In that version we don't want to parse folders here, so we don't compare folders
// probably it must be improved in future
// const CDbEx &db = *(const CDbEx *)param;
// RINOZ(CompareFolders(
// db.Folders[i1],
// db.Folders[i2]));
return MyCompare(i1, i2);
// RINOZ_COMP(
// db.NumUnpackStreamsVector[i1],
// db.NumUnpackStreamsVector[i2]);
// if (db.NumUnpackStreamsVector[i1] == 0)
// return 0;
// return CompareFiles(
// db.Files[db.FolderStartFileIndex[i1]],
// db.Files[db.FolderStartFileIndex[i2]]);
}
*/
/*
we sort empty files and dirs in such order:
- Dir.NonAnti (name sorted)
- File.NonAnti (name sorted)
- File.Anti (name sorted)
- Dir.Anti (reverse name sorted)
*/
static int CompareEmptyItems(const unsigned *p1, const unsigned *p2, void *param)
{
const CObjectVector<CUpdateItem> &updateItems = *(const CObjectVector<CUpdateItem> *)param;
const CUpdateItem &u1 = updateItems[*p1];
const CUpdateItem &u2 = updateItems[*p2];
// NonAnti < Anti
if (u1.IsAnti != u2.IsAnti)
return (u1.IsAnti ? 1 : -1);
if (u1.IsDir != u2.IsDir)
{
// Dir.NonAnti < File < Dir.Anti
if (u1.IsDir)
return (u1.IsAnti ? 1 : -1);
return (u2.IsAnti ? -1 : 1);
}
int n = CompareFileNames(u1.Name, u2.Name);
return (u1.IsDir && u1.IsAnti) ? -n : n;
}
static const char *g_Exts =
" 7z xz lzma ace arc arj bz tbz bz2 tbz2 cab deb gz tgz ha lha lzh lzo lzx pak rar rpm sit zoo"
" zip jar ear war msi"
" 3gp avi mov mpeg mpg mpe wmv"
" aac ape fla flac la mp3 m4a mp4 ofr ogg pac ra rm rka shn swa tta wv wma wav"
" swf"
" chm hxi hxs"
" gif jpeg jpg jp2 png tiff bmp ico psd psp"
" awg ps eps cgm dxf svg vrml wmf emf ai md"
" cad dwg pps key sxi"
" max 3ds"
" iso bin nrg mdf img pdi tar cpio xpi"
" vfd vhd vud vmc vsv"
" vmdk dsk nvram vmem vmsd vmsn vmss vmtm"
" inl inc idl acf asa"
" h hpp hxx c cpp cxx m mm go swift"
" rc java cs rs pas bas vb cls ctl frm dlg def"
" f77 f f90 f95"
" asm s"
" sql manifest dep"
" mak clw csproj vcproj sln dsp dsw"
" class"
" bat cmd bash sh"
" xml xsd xsl xslt hxk hxc htm html xhtml xht mht mhtml htw asp aspx css cgi jsp shtml"
" awk sed hta js json php php3 php4 php5 phptml pl pm py pyo rb tcl ts vbs"
" text txt tex ans asc srt reg ini doc docx mcw dot rtf hlp xls xlr xlt xlw ppt pdf"
" sxc sxd sxi sxg sxw stc sti stw stm odt ott odg otg odp otp ods ots odf"
" abw afp cwk lwp wpd wps wpt wrf wri"
" abf afm bdf fon mgf otf pcf pfa snf ttf"
" dbf mdb nsf ntf wdb db fdb gdb"
" exe dll ocx vbx sfx sys tlb awx com obj lib out o so"
" pdb pch idb ncb opt";
static unsigned GetExtIndex(const char *ext)
{
unsigned extIndex = 1;
const char *p = g_Exts;
for (;;)
{
char c = *p++;
if (c == 0)
return extIndex;
if (c == ' ')
continue;
unsigned pos = 0;
for (;;)
{
char c2 = ext[pos++];
if (c2 == 0 && (c == 0 || c == ' '))
return extIndex;
if (c != c2)
break;
c = *p++;
}
extIndex++;
for (;;)
{
if (c == 0)
return extIndex;
if (c == ' ')
break;
c = *p++;
}
}
}
struct CRefItem
{
const CUpdateItem *UpdateItem;
UInt32 Index;
unsigned ExtensionPos;
unsigned NamePos;
unsigned ExtensionIndex;
CRefItem() {}
CRefItem(UInt32 index, const CUpdateItem &ui, bool sortByType):
UpdateItem(&ui),
Index(index),
ExtensionPos(0),
NamePos(0),
ExtensionIndex(0)
{
if (sortByType)
{
int slashPos = ui.Name.ReverseFind_PathSepar();
NamePos = (unsigned)(slashPos + 1);
int dotPos = ui.Name.ReverseFind_Dot();
if (dotPos <= slashPos)
ExtensionPos = ui.Name.Len();
else
{
ExtensionPos = (unsigned)(dotPos + 1);
if (ExtensionPos != ui.Name.Len())
{
AString s;
for (unsigned pos = ExtensionPos;; pos++)
{
wchar_t c = ui.Name[pos];
if (c >= 0x80)
break;
if (c == 0)
{
ExtensionIndex = GetExtIndex(s);
break;
}
s += (char)MyCharLower_Ascii((char)c);
}
}
}
}
}
};
struct CSortParam
{
// const CObjectVector<CTreeFolder> *TreeFolders;
bool SortByType;
};
/*
we sort files in such order:
- Dir.NonAnti (name sorted)
- alt streams
- Dirs
- Dir.Anti (reverse name sorted)
*/
static int CompareUpdateItems(const CRefItem *p1, const CRefItem *p2, void *param)
{
const CRefItem &a1 = *p1;
const CRefItem &a2 = *p2;
const CUpdateItem &u1 = *a1.UpdateItem;
const CUpdateItem &u2 = *a2.UpdateItem;
/*
if (u1.IsAltStream != u2.IsAltStream)
return u1.IsAltStream ? 1 : -1;
*/
// Actually there are no dirs that time. They were stored in other steps
// So that code is unused?
if (u1.IsDir != u2.IsDir)
return u1.IsDir ? 1 : -1;
if (u1.IsDir)
{
if (u1.IsAnti != u2.IsAnti)
return (u1.IsAnti ? 1 : -1);
int n = CompareFileNames(u1.Name, u2.Name);
return -n;
}
// bool sortByType = *(bool *)param;
const CSortParam *sortParam = (const CSortParam *)param;
const bool sortByType = sortParam->SortByType;
if (sortByType)
{
RINOZ_COMP(a1.ExtensionIndex, a2.ExtensionIndex)
RINOZ(CompareFileNames(u1.Name.Ptr(a1.ExtensionPos), u2.Name.Ptr(a2.ExtensionPos)))
RINOZ(CompareFileNames(u1.Name.Ptr(a1.NamePos), u2.Name.Ptr(a2.NamePos)))
if (!u1.MTimeDefined && u2.MTimeDefined) return 1;
if (u1.MTimeDefined && !u2.MTimeDefined) return -1;
if (u1.MTimeDefined && u2.MTimeDefined) RINOZ_COMP(u1.MTime, u2.MTime)
RINOZ_COMP(u1.Size, u2.Size)
}
/*
int par1 = a1.UpdateItem->ParentFolderIndex;
int par2 = a2.UpdateItem->ParentFolderIndex;
const CTreeFolder &tf1 = (*sortParam->TreeFolders)[par1];
const CTreeFolder &tf2 = (*sortParam->TreeFolders)[par2];
int b1 = tf1.SortIndex, e1 = tf1.SortIndexEnd;
int b2 = tf2.SortIndex, e2 = tf2.SortIndexEnd;
if (b1 < b2)
{
if (e1 <= b2)
return -1;
// p2 in p1
int par = par2;
for (;;)
{
const CTreeFolder &tf = (*sortParam->TreeFolders)[par];
par = tf.Parent;
if (par == par1)
{
RINOZ(CompareFileNames(u1.Name, tf.Name));
break;
}
}
}
else if (b2 < b1)
{
if (e2 <= b1)
return 1;
// p1 in p2
int par = par1;
for (;;)
{
const CTreeFolder &tf = (*sortParam->TreeFolders)[par];
par = tf.Parent;
if (par == par2)
{
RINOZ(CompareFileNames(tf.Name, u2.Name));
break;
}
}
}
*/
// RINOZ_COMP(a1.UpdateItem->ParentSortIndex, a2.UpdateItem->ParentSortIndex);
RINOK(CompareFileNames(u1.Name, u2.Name))
RINOZ_COMP(a1.UpdateItem->IndexInClient, a2.UpdateItem->IndexInClient)
RINOZ_COMP(a1.UpdateItem->IndexInArchive, a2.UpdateItem->IndexInArchive)
return 0;
}
struct CSolidGroup
{
CRecordVector<UInt32> Indices;
CRecordVector<CFolderRepack> folderRefs;
};
static const char * const g_Exe_Exts[] =
{
"dll"
, "exe"
, "ocx"
, "sfx"
, "sys"
};
static const char * const g_ExeUnix_Exts[] =
{
"so"
, "dylib"
};
static bool IsExt_Exe(const wchar_t *ext)
{
for (unsigned i = 0; i < Z7_ARRAY_SIZE(g_Exe_Exts); i++)
if (StringsAreEqualNoCase_Ascii(ext, g_Exe_Exts[i]))
return true;
return false;
}
/*
static bool IsExt_ExeUnix(const wchar_t *ext)
{
for (unsigned i = 0; i < Z7_ARRAY_SIZE(g_ExeUnix_Exts); i++)
if (StringsAreEqualNoCase_Ascii(ext, g_ExeUnix_Exts[i]))
return true;
return false;
}
*/
// we try to find "so" extension in such name: libstdc++.so.6.0.29
static bool IsExt_ExeUnix_NumericAllowed(const UString &path)
{
unsigned pos = path.Len();
unsigned dotPos = pos;
for (;;)
{
if (pos == 0)
return false;
const wchar_t c = path[--pos];
if (IS_PATH_SEPAR(c))
return false;
if (c == '.')
{
const unsigned num = (dotPos - pos) - 1;
if (num < 1)
return false;
const wchar_t *cur = path.Ptr(pos + 1);
for (unsigned i = 0; i < Z7_ARRAY_SIZE(g_ExeUnix_Exts); i++)
{
const char *ext = g_ExeUnix_Exts[i];
if (num == MyStringLen(ext))
if (IsString1PrefixedByString2_NoCase_Ascii(cur, ext))
return true;
}
const wchar_t *end;
ConvertStringToUInt32(cur, &end);
if ((size_t)(end - cur) != num)
return false;
dotPos = pos;
}
}
}
struct CAnalysis
{
CMyComPtr<IArchiveUpdateCallbackFile> Callback;
CByteBuffer Buffer;
bool ParseWav;
bool ParseExe;
bool ParseExeUnix;
bool ParseNoExt;
bool ParseAll;
/*
bool Need_ATime;
bool ATime_Defined;
FILETIME ATime;
*/
CAnalysis():
ParseWav(false),
ParseExe(false),
ParseExeUnix(false),
ParseNoExt(false),
ParseAll(false)
/*
, Need_ATime(false)
, ATime_Defined(false)
*/
{}
HRESULT GetFilterGroup(UInt32 index, const CUpdateItem &ui, CFilterMode &filterMode);
};
static const size_t kAnalysisBufSize = 1 << 14;
HRESULT CAnalysis::GetFilterGroup(UInt32 index, const CUpdateItem &ui, CFilterMode &filterMode)
{
filterMode.Id = 0;
filterMode.Delta = 0;
filterMode.Offset = 0;
CFilterMode filterModeTemp = filterMode;
const int slashPos = ui.Name.ReverseFind_PathSepar();
const int dotPos = ui.Name.ReverseFind_Dot();
// if (dotPos > slashPos)
{
bool needReadFile = ParseAll;
/* if (Callback) is not supported by client,
we still try to use file name extension to detect executable file */
bool probablyIsSameIsa = false;
if (!needReadFile || !Callback)
{
const wchar_t *ext = NULL;
if (dotPos > slashPos)
ext = ui.Name.Ptr((unsigned)(dotPos + 1));
// 7-zip stores posix attributes in high 16 bits and sets (0x8000) flag
if (ui.Attrib & 0x8000)
{
const unsigned st_mode = ui.Attrib >> 16;
/* note: executable ".so" can be without execute permission,
and symbolic link to such ".so" file is possible */
// st_mode = 00111; // for debug
/* in Linux we expect such permissions:
0755 : for most executables
0644 : for some ".so" files
0777 : in WSL for all files.
We can try to exclude some such 0777 cases from analysis,
if there is non-executable extension.
*/
if ((st_mode & (
MY_LIN_S_IXUSR |
MY_LIN_S_IXGRP |
MY_LIN_S_IXOTH)) != 0
&& MY_LIN_S_ISREG(st_mode)
&& (ui.Size >= (1u << 11)))
{
#ifndef _WIN32
probablyIsSameIsa = true;
#endif
needReadFile = true;
}
}
if (!needReadFile)
{
if (!ext)
needReadFile = ParseNoExt;
else
{
bool isUnixExt = false;
if (ParseExeUnix)
isUnixExt = IsExt_ExeUnix_NumericAllowed(ui.Name);
if (isUnixExt)
{
needReadFile = true;
#ifndef _WIN32
probablyIsSameIsa = true;
#endif
}
else if (IsExt_Exe(ext))
{
needReadFile = ParseExe;
#ifdef _WIN32
probablyIsSameIsa = true;
#endif
}
else if (StringsAreEqualNoCase_Ascii(ext, "wav"))
{
if (!needReadFile)
needReadFile = ParseWav;
}
}
}
}
if (needReadFile)
{
BoolInt parseRes = false;
if (Callback)
{
if (Buffer.Size() != kAnalysisBufSize)
Buffer.Alloc(kAnalysisBufSize);
CMyComPtr<ISequentialInStream> stream;
HRESULT result = Callback->GetStream2(index, &stream, NUpdateNotifyOp::kAnalyze);
if (result == S_OK && stream)
{
/*
if (Need_ATime)
{
// access time could be changed in analysis pass
CMyComPtr<IStreamGetProps> getProps;
stream.QueryInterface(IID_IStreamGetProps, (void **)&getProps);
if (getProps)
if (getProps->GetProps(NULL, NULL, &ATime, NULL, NULL) == S_OK)
ATime_Defined = true;
}
*/
size_t size = kAnalysisBufSize;
result = ReadStream(stream, Buffer, &size);
stream.Release();
// RINOK(Callback->SetOperationResult2(index, NUpdate::NOperationResult::kOK));
if (result == S_OK)
{
parseRes = ParseFile(Buffer, size, &filterModeTemp);
}
}
} // Callback
else if (probablyIsSameIsa)
{
#ifdef MY_CPU_X86_OR_AMD64
filterModeTemp.Id = k_X86;
#endif
#ifdef MY_CPU_ARM64
filterModeTemp.Id = k_ARM64;
#endif
parseRes = true;
}
if (parseRes
&& filterModeTemp.Id != k_Delta
&& filterModeTemp.Delta == 0)
{
/* ParseFile() sets (filterModeTemp.Delta == 0) for all
methods except of k_Delta. */
// it's not k_Delta
// So we call SetDelta() to set Delta
filterModeTemp.SetDelta();
if (filterModeTemp.Delta > 1)
{
/* If file Size is not aligned, then branch filter
will not work for next file in solid block.
Maybe we should allow filter for non-aligned-size file in non-solid archives ?
*/
if (ui.Size % filterModeTemp.Delta != 0)
parseRes = false;
// windows exe files are not aligned for 4 KiB.
/*
else if (filterModeTemp.Id == k_ARM64 && filterModeTemp.Offset != 0)
{
if (ui.Size % (1 << 12) != 0)
{
// If Size is not aligned for 4 KiB, then Offset will not work for next file in solid block.
// so we place such file in group with (Offset==0).
filterModeTemp.Offset = 0;
}
}
*/
}
}
if (!parseRes)
filterModeTemp.ClearFilterMode();
}
}
filterMode = filterModeTemp;
return S_OK;
}
static inline void GetMethodFull(UInt64 methodID, UInt32 numStreams, CMethodFull &m)
{
m.Id = methodID;
m.NumStreams = numStreams;
}
static HRESULT AddBondForFilter(CCompressionMethodMode &mode)
{
for (unsigned c = 1; c < mode.Methods.Size(); c++)
{
if (!mode.IsThereBond_to_Coder(c))
{
CBond2 bond;
bond.OutCoder = 0;
bond.OutStream = 0;
bond.InCoder = c;
mode.Bonds.Add(bond);
return S_OK;
}
}
return E_INVALIDARG;
}
static HRESULT AddFilterBond(CCompressionMethodMode &mode)
{
if (!mode.Bonds.IsEmpty())
return AddBondForFilter(mode);
return S_OK;
}
static HRESULT AddBcj2Methods(CCompressionMethodMode &mode)
{
// mode.Methods[0] must be k_BCJ2 method !
CMethodFull m;
GetMethodFull(k_LZMA, 1, m);
m.AddProp32(NCoderPropID::kDictionarySize, 1 << 20);
m.AddProp32(NCoderPropID::kNumFastBytes, 128);
m.AddProp32(NCoderPropID::kNumThreads, 1);
m.AddProp32(NCoderPropID::kLitPosBits, 2);
m.AddProp32(NCoderPropID::kLitContextBits, 0);
// m.AddProp_Ascii(NCoderPropID::kMatchFinder, "BT2");
unsigned methodIndex = mode.Methods.Size();
if (mode.Bonds.IsEmpty())
{
for (unsigned i = 1; i + 1 < mode.Methods.Size(); i++)
{
CBond2 bond;
bond.OutCoder = i;
bond.OutStream = 0;
bond.InCoder = i + 1;
mode.Bonds.Add(bond);
}
}
mode.Methods.Add(m);
mode.Methods.Add(m);
RINOK(AddBondForFilter(mode))
CBond2 bond;
bond.OutCoder = 0;
bond.InCoder = methodIndex; bond.OutStream = 1; mode.Bonds.Add(bond);
bond.InCoder = methodIndex + 1; bond.OutStream = 2; mode.Bonds.Add(bond);
return S_OK;
}
static HRESULT MakeExeMethod(CCompressionMethodMode &mode,
const CFilterMode &filterMode, /* bool addFilter, */ bool bcj2Filter)
{
if (mode.Filter_was_Inserted)
{
const CMethodFull &m = mode.Methods[0];
const CMethodId id = m.Id;
if (id == k_BCJ2)
return AddBcj2Methods(mode);
if (!m.IsSimpleCoder())
return E_NOTIMPL;
// if (Bonds.IsEmpty()) we can create bonds later
return AddFilterBond(mode);
}
if (filterMode.Id == 0)
return S_OK;
CMethodFull &m = mode.Methods.InsertNew(0);
{
FOR_VECTOR (k, mode.Bonds)
{
CBond2 &bond = mode.Bonds[k];
bond.InCoder++;
bond.OutCoder++;
}
}
HRESULT res;
if (bcj2Filter && Is86Filter(filterMode.Id))
{
GetMethodFull(k_BCJ2, 4, m);
res = AddBcj2Methods(mode);
}
else
{
GetMethodFull(filterMode.Id, 1, m);
if (filterMode.Id == k_Delta)
m.AddProp32(NCoderPropID::kDefaultProp, filterMode.Delta);
else if (filterMode.Id == k_ARM64)
{
// if (filterMode.Offset != 0)
m.AddProp32(
NCoderPropID::kDefaultProp,
// NCoderPropID::kBranchOffset,
filterMode.Offset);
}
res = AddFilterBond(mode);
int alignBits = -1;
{
const UInt32 delta = filterMode.Delta;
if (delta == 0 || delta > 16)
{
// if (delta == 0) alignBits = GetAlignForFilterMethod(filterMode.Id);
}
else if ((delta & ((1 << 4) - 1)) == 0) alignBits = 4;
else if ((delta & ((1 << 3) - 1)) == 0) alignBits = 3;
else if ((delta & ((1 << 2) - 1)) == 0) alignBits = 2;
else if ((delta & ((1 << 1) - 1)) == 0) alignBits = 1;
// else alignBits = 0;
/* alignBits=0 is default mode for lzma/lzma2.
So we don't set alignBits=0 here. */
}
if (res == S_OK && alignBits > 0)
{
unsigned nextCoder = 1;
if (!mode.Bonds.IsEmpty())
{
nextCoder = mode.Bonds.Back().InCoder;
}
if (nextCoder < mode.Methods.Size())
{
CMethodFull &nextMethod = mode.Methods[nextCoder];
if (nextMethod.Id == k_LZMA || nextMethod.Id == k_LZMA2)
{
if (!nextMethod.Are_Lzma_Model_Props_Defined())
{
if (alignBits != 0)
{
if (alignBits > 2 || filterMode.Id == k_Delta)
nextMethod.AddProp32(NCoderPropID::kPosStateBits, (unsigned)alignBits);
unsigned lc = 0;
if (alignBits < 3)
lc = (unsigned)(3 - alignBits);
nextMethod.AddProp32(NCoderPropID::kLitContextBits, lc);
nextMethod.AddProp32(NCoderPropID::kLitPosBits, (unsigned)alignBits);
}
}
}
}
}
}
return res;
}
static void UpdateItem_To_FileItem2(const CUpdateItem &ui, CFileItem2 &file2)
{
file2.Attrib = ui.Attrib; file2.AttribDefined = ui.AttribDefined;
file2.CTime = ui.CTime; file2.CTimeDefined = ui.CTimeDefined;
file2.ATime = ui.ATime; file2.ATimeDefined = ui.ATimeDefined;
file2.MTime = ui.MTime; file2.MTimeDefined = ui.MTimeDefined;
file2.IsAnti = ui.IsAnti;
// file2.IsAux = false;
file2.StartPosDefined = false;
// file2.StartPos = 0;
}
static void UpdateItem_To_FileItem(const CUpdateItem &ui,
CFileItem &file, CFileItem2 &file2)
{
UpdateItem_To_FileItem2(ui, file2);
file.Size = ui.Size;
file.IsDir = ui.IsDir;
file.HasStream = ui.HasStream();
// file.IsAltStream = ui.IsAltStream;
}
Z7_CLASS_IMP_COM_2(
CRepackInStreamWithSizes
, ISequentialInStream
, ICompressGetSubStreamSize
)
CMyComPtr<ISequentialInStream> _stream;
UInt64 _size;
const CBoolVector *_extractStatuses;
UInt32 _startIndex;
public:
const CDbEx *_db;
void Init(ISequentialInStream *stream, UInt32 startIndex, const CBoolVector *extractStatuses)
{
_startIndex = startIndex;
_extractStatuses = extractStatuses;
_size = 0;
_stream = stream;
}
UInt64 GetSize() const { return _size; }
};
Z7_COM7F_IMF(CRepackInStreamWithSizes::Read(void *data, UInt32 size, UInt32 *processedSize))
{
UInt32 realProcessedSize;
const HRESULT result = _stream->Read(data, size, &realProcessedSize);
_size += realProcessedSize;
if (processedSize)
*processedSize = realProcessedSize;
return result;
}
Z7_COM7F_IMF(CRepackInStreamWithSizes::GetSubStreamSize(UInt64 subStream, UInt64 *value))
{
*value = 0;
if (subStream >= _extractStatuses->Size())
return S_FALSE; // E_FAIL;
const unsigned index = (unsigned)subStream;
if ((*_extractStatuses)[index])
{
const CFileItem &fi = _db->Files[_startIndex + index];
if (fi.HasStream)
*value = fi.Size;
}
return S_OK;
}
class CRepackStreamBase
{
protected:
bool _needWrite;
bool _fileIsOpen;
bool _calcCrc;
UInt32 _crc;
UInt64 _rem;
const CBoolVector *_extractStatuses;
UInt32 _startIndex;
unsigned _currentIndex;
HRESULT OpenFile();
HRESULT CloseFile();
HRESULT ProcessEmptyFiles();
public:
const CDbEx *_db;
CMyComPtr<IArchiveUpdateCallbackFile> _opCallback;
CMyComPtr<IArchiveExtractCallbackMessage2> _extractCallback;
HRESULT Init(UInt32 startIndex, const CBoolVector *extractStatuses);
HRESULT CheckFinishedState() const { return (_currentIndex == _extractStatuses->Size()) ? S_OK: E_FAIL; }
};
HRESULT CRepackStreamBase::Init(UInt32 startIndex, const CBoolVector *extractStatuses)
{
_startIndex = startIndex;
_extractStatuses = extractStatuses;
_currentIndex = 0;
_fileIsOpen = false;
return ProcessEmptyFiles();
}
HRESULT CRepackStreamBase::OpenFile()
{
UInt32 arcIndex = _startIndex + _currentIndex;
const CFileItem &fi = _db->Files[arcIndex];
_needWrite = (*_extractStatuses)[_currentIndex];
if (_opCallback)
{
RINOK(_opCallback->ReportOperation(
NEventIndexType::kInArcIndex, arcIndex,
_needWrite ?
NUpdateNotifyOp::kRepack :
NUpdateNotifyOp::kSkip))
}
_crc = CRC_INIT_VAL;
_calcCrc = (fi.CrcDefined && !fi.IsDir);
_fileIsOpen = true;
_rem = fi.Size;
return S_OK;
}
const HRESULT k_My_HRESULT_CRC_ERROR = 0x20000002;
HRESULT CRepackStreamBase::CloseFile()
{
UInt32 arcIndex = _startIndex + _currentIndex;
const CFileItem &fi = _db->Files[arcIndex];
_fileIsOpen = false;
_currentIndex++;
if (!_calcCrc || fi.Crc == CRC_GET_DIGEST(_crc))
return S_OK;
if (_extractCallback)
{
RINOK(_extractCallback->ReportExtractResult(
NEventIndexType::kInArcIndex, arcIndex,
NExtract::NOperationResult::kCRCError))
}
// return S_FALSE;
return k_My_HRESULT_CRC_ERROR;
}
HRESULT CRepackStreamBase::ProcessEmptyFiles()
{
while (_currentIndex < _extractStatuses->Size() && _db->Files[_startIndex + _currentIndex].Size == 0)
{
RINOK(OpenFile())
RINOK(CloseFile())
}
return S_OK;
}
#ifndef Z7_ST
class CFolderOutStream2 Z7_final:
public CRepackStreamBase,
public ISequentialOutStream,
public CMyUnknownImp
{
Z7_COM_UNKNOWN_IMP_0
Z7_IFACE_COM7_IMP(ISequentialOutStream)
public:
CMyComPtr<ISequentialOutStream> _stream;
};
Z7_COM7F_IMF(CFolderOutStream2::Write(const void *data, UInt32 size, UInt32 *processedSize))
{
if (processedSize)
*processedSize = 0;
while (size != 0)
{
if (_fileIsOpen)
{
UInt32 cur = (size < _rem ? size : (UInt32)_rem);
HRESULT result = S_OK;
if (_needWrite)
result = _stream->Write(data, cur, &cur);
if (_calcCrc)
_crc = CrcUpdate(_crc, data, cur);
if (processedSize)
*processedSize += cur;
data = (const Byte *)data + cur;
size -= cur;
_rem -= cur;
if (_rem == 0)
{
RINOK(CloseFile())
RINOK(ProcessEmptyFiles())
}
RINOK(result)
if (cur == 0)
break;
continue;
}
RINOK(ProcessEmptyFiles())
if (_currentIndex == _extractStatuses->Size())
{
// we don't support write cut here
return E_FAIL;
}
RINOK(OpenFile())
}
return S_OK;
}
#endif
static const UInt32 kTempBufSize = 1 << 16;
class CFolderInStream2 Z7_final:
public CRepackStreamBase,
public ISequentialInStream,
public CMyUnknownImp
{
Z7_COM_UNKNOWN_IMP_0
Z7_IFACE_COM7_IMP(ISequentialInStream)
Byte *_buf;
public:
CMyComPtr<ISequentialInStream> _inStream;
HRESULT Result;
CFolderInStream2():
Result(S_OK)
{
_buf = new Byte[kTempBufSize];
}
~CFolderInStream2()
{
delete []_buf;
}
void Init() { Result = S_OK; }
};
Z7_COM7F_IMF(CFolderInStream2::Read(void *data, UInt32 size, UInt32 *processedSize))
{
if (processedSize)
*processedSize = 0;
while (size != 0)
{
if (_fileIsOpen)
{
UInt32 cur = (size < _rem ? size : (UInt32)_rem);
void *buf;
if (_needWrite)
buf = data;
else
{
buf = _buf;
if (cur > kTempBufSize)
cur = kTempBufSize;
}
const HRESULT result = _inStream->Read(buf, cur, &cur);
_crc = CrcUpdate(_crc, buf, cur);
_rem -= cur;
if (_needWrite)
{
data = (Byte *)data + cur;
size -= cur;
if (processedSize)
*processedSize += cur;
}
if (result != S_OK)
Result = result;
if (_rem == 0)
{
RINOK(CloseFile())
RINOK(ProcessEmptyFiles())
}
RINOK(result)
if (cur == 0)
return E_FAIL;
continue;
}
RINOK(ProcessEmptyFiles())
if (_currentIndex == _extractStatuses->Size())
{
return S_OK;
}
RINOK(OpenFile())
}
return S_OK;
}
class CThreadDecoder Z7_final
#ifndef Z7_ST
: public CVirtThread
#endif
{
public:
CDecoder Decoder;
CThreadDecoder(bool multiThreadMixer):
Decoder(multiThreadMixer)
{
#ifndef Z7_ST
if (multiThreadMixer)
{
MtMode = false;
NumThreads = 1;
FosSpec = new CFolderOutStream2;
Fos = FosSpec;
Result = E_FAIL;
}
#endif
// UnpackSize = 0;
// send_UnpackSize = false;
}
#ifndef Z7_ST
bool dataAfterEnd_Error;
HRESULT Result;
CMyComPtr<IInStream> InStream;
CFolderOutStream2 *FosSpec;
CMyComPtr<ISequentialOutStream> Fos;
UInt64 StartPos;
const CFolders *Folders;
unsigned FolderIndex;
// bool send_UnpackSize;
// UInt64 UnpackSize;
#ifndef Z7_NO_CRYPTO
CMyComPtr<ICryptoGetTextPassword> getTextPassword;
#endif
DECL_EXTERNAL_CODECS_LOC_VARS_DECL
#ifndef Z7_ST
bool MtMode;
UInt32 NumThreads;
#endif
~CThreadDecoder() Z7_DESTRUCTOR_override
{
/* WaitThreadFinish() will be called in ~CVirtThread().
But we need WaitThreadFinish() call before
destructors of this class members.
*/
CVirtThread::WaitThreadFinish();
}
private:
virtual void Execute() Z7_override;
#endif
};
#ifndef Z7_ST
void CThreadDecoder::Execute()
{
try
{
#ifndef Z7_NO_CRYPTO
bool isEncrypted = false;
bool passwordIsDefined = false;
UString password;
#endif
dataAfterEnd_Error = false;
Result = Decoder.Decode(
EXTERNAL_CODECS_LOC_VARS
InStream,
StartPos,
*Folders, FolderIndex,
// send_UnpackSize ? &UnpackSize : NULL,
NULL, // unpackSize : FULL unpack
Fos,
NULL, // compressProgress
NULL // *inStreamMainRes
, dataAfterEnd_Error
Z7_7Z_DECODER_CRYPRO_VARS
#ifndef Z7_ST
, MtMode, NumThreads,
0 // MemUsage
#endif
);
}
catch(...)
{
Result = E_FAIL;
}
/*
if (Result == S_OK)
Result = FosSpec->CheckFinishedState();
*/
FosSpec->_stream.Release();
}
#endif
#ifndef Z7_NO_CRYPTO
Z7_CLASS_IMP_NOQIB_1(
CCryptoGetTextPassword
, ICryptoGetTextPassword
)
public:
UString Password;
};
Z7_COM7F_IMF(CCryptoGetTextPassword::CryptoGetTextPassword(BSTR *password))
{
return StringToBstr(Password, password);
}
#endif
static void GetFile(const CDatabase &inDb, unsigned index, CFileItem &file, CFileItem2 &file2)
{
file = inDb.Files[index];
file2.CTimeDefined = inDb.CTime.GetItem(index, file2.CTime);
file2.ATimeDefined = inDb.ATime.GetItem(index, file2.ATime);
file2.MTimeDefined = inDb.MTime.GetItem(index, file2.MTime);
file2.StartPosDefined = inDb.StartPos.GetItem(index, file2.StartPos);
file2.AttribDefined = inDb.Attrib.GetItem(index, file2.Attrib);
file2.IsAnti = inDb.IsItemAnti(index);
// file2.IsAux = inDb.IsItemAux(index);
}
HRESULT Update(
DECL_EXTERNAL_CODECS_LOC_VARS
IInStream *inStream,
const CDbEx *db,
CObjectVector<CUpdateItem> &updateItems,
// const CObjectVector<CTreeFolder> &treeFolders,
// const CUniqBlocks &secureBlocks,
ISequentialOutStream *seqOutStream,
IArchiveUpdateCallback *updateCallback,
const CUpdateOptions &options)
{
UInt64 numSolidFiles = options.NumSolidFiles;
if (numSolidFiles == 0)
numSolidFiles = 1;
Z7_DECL_CMyComPtr_QI_FROM(
IArchiveUpdateCallbackFile,
opCallback, updateCallback)
Z7_DECL_CMyComPtr_QI_FROM(
IArchiveExtractCallbackMessage2,
extractCallback, updateCallback)
/*
Z7_DECL_CMyComPtr_QI_FROM(
IArchiveUpdateCallbackArcProp,
reportArcProp, updateCallback)
*/
// size_t totalSecureDataSize = (size_t)secureBlocks.GetTotalSizeInBytes();
CMyComPtr<IStreamSetRestriction> v_StreamSetRestriction;
{
Z7_DECL_CMyComPtr_QI_FROM(
IOutStream,
outStream, seqOutStream)
if (!outStream)
return E_NOTIMPL;
const UInt64 sfxBlockSize = (db && !options.RemoveSfxBlock) ?
db->ArcInfo.StartPosition: 0;
seqOutStream->QueryInterface(IID_IStreamSetRestriction, (void **)&v_StreamSetRestriction);
if (v_StreamSetRestriction)
{
UInt64 offset = 0;
RINOK(outStream->Seek(0, STREAM_SEEK_CUR, &offset))
RINOK(v_StreamSetRestriction->SetRestriction(
outStream ? offset + sfxBlockSize : 0,
outStream ? offset + sfxBlockSize + k_StartHeadersRewriteSize : 0))
}
outStream.Release();
if (sfxBlockSize != 0)
{
RINOK(WriteRange(inStream, seqOutStream, 0, sfxBlockSize, NULL))
}
}
CIntArr fileIndexToUpdateIndexMap;
UInt64 complexity = 0;
UInt64 inSizeForReduce2 = 0;
#ifndef Z7_NO_CRYPTO
bool needEncryptedRepack = false;
#endif
CRecordVector<CFilterMode2> filters;
CObjectVector<CSolidGroup> groups;
#ifndef Z7_ST
bool thereAreRepacks = false;
#endif
bool useFilters = options.UseFilters;
if (useFilters)
{
const CCompressionMethodMode &method = *options.Method;
FOR_VECTOR (i, method.Methods)
{
/* IsFilterMethod() knows only built-in codecs
FIXME: we should check IsFilter status for external filters too */
if (IsFilterMethod(method.Methods[i].Id))
{
useFilters = false;
break;
}
}
}
if (db)
{
fileIndexToUpdateIndexMap.Alloc(db->Files.Size());
unsigned i;
for (i = 0; i < db->Files.Size(); i++)
fileIndexToUpdateIndexMap[i] = -1;
for (i = 0; i < updateItems.Size(); i++)
{
int index = updateItems[i].IndexInArchive;
if (index != -1)
fileIndexToUpdateIndexMap[(unsigned)index] = (int)i;
}
for (i = 0; i < db->NumFolders; i++)
{
CNum indexInFolder = 0;
CNum numCopyItems = 0;
const CNum numUnpackStreams = db->NumUnpackStreamsVector[i];
UInt64 repackSize = 0;
for (CNum fi = db->FolderStartFileIndex[i]; indexInFolder < numUnpackStreams; fi++)
{
if (fi >= db->Files.Size())
return E_FAIL;
const CFileItem &file = db->Files[fi];
if (file.HasStream)
{
indexInFolder++;
const int updateIndex = fileIndexToUpdateIndexMap[fi];
if (updateIndex >= 0 && !updateItems[(unsigned)updateIndex].NewData)
{
numCopyItems++;
repackSize += file.Size;
}
}
}
if (numCopyItems == 0)
continue;
CFolderRepack rep;
rep.FolderIndex = i;
rep.NumCopyFiles = numCopyItems;
CFolderEx f;
db->ParseFolderEx(i, f);
#ifndef Z7_NO_CRYPTO
const bool isEncrypted = f.IsEncrypted();
#endif
const bool needCopy = (numCopyItems == numUnpackStreams);
const bool extractFilter = (useFilters || needCopy);
const unsigned groupIndex = Get_FilterGroup_for_Folder(filters, f, extractFilter);
while (groupIndex >= groups.Size())
groups.AddNew();
groups[groupIndex].folderRefs.Add(rep);
if (needCopy)
complexity += db->GetFolderFullPackSize(i);
else
{
#ifndef Z7_ST
thereAreRepacks = true;
#endif
complexity += repackSize;
if (inSizeForReduce2 < repackSize)
inSizeForReduce2 = repackSize;
#ifndef Z7_NO_CRYPTO
if (isEncrypted)
needEncryptedRepack = true;
#endif
}
}
}
UInt64 inSizeForReduce = 0;
{
bool isSolid = (numSolidFiles > 1 && options.NumSolidBytes != 0);
FOR_VECTOR (i, updateItems)
{
const CUpdateItem &ui = updateItems[i];
if (ui.NewData)
{
complexity += ui.Size;
if (isSolid)
inSizeForReduce += ui.Size;
else if (inSizeForReduce < ui.Size)
inSizeForReduce = ui.Size;
}
}
}
if (inSizeForReduce < inSizeForReduce2)
inSizeForReduce = inSizeForReduce2;
RINOK(updateCallback->SetTotal(complexity))
CLocalProgress *lps = new CLocalProgress;
CMyComPtr<ICompressProgressInfo> progress = lps;
lps->Init(updateCallback, true);
#ifndef Z7_ST
CStreamBinder sb;
/*
if (options.MultiThreadMixer)
{
RINOK(sb.CreateEvents());
}
*/
#endif
CThreadDecoder threadDecoder(options.MultiThreadMixer);
#ifndef Z7_ST
if (options.MultiThreadMixer && thereAreRepacks)
{
#ifdef Z7_EXTERNAL_CODECS
threadDecoder._externalCodecs = _externalCodecs;
#endif
const WRes wres = threadDecoder.Create();
if (wres != 0)
return HRESULT_FROM_WIN32(wres);
}
#endif
{
CAnalysis analysis;
// analysis.Need_ATime = options.Need_ATime;
int analysisLevel = options.AnalysisLevel;
// (analysisLevel < 0) means default level (5)
if (analysisLevel < 0)
analysisLevel = 5;
if (analysisLevel != 0)
{
analysis.Callback = opCallback;
analysis.ParseWav = true;
if (analysisLevel >= 5)
{
analysis.ParseExe = true;
analysis.ParseExeUnix = true;
// analysis.ParseNoExt = true;
if (analysisLevel >= 7)
{
analysis.ParseNoExt = true;
if (analysisLevel >= 9)
analysis.ParseAll = true;
}
}
}
// ---------- Split files to groups ----------
const CCompressionMethodMode &method = *options.Method;
FOR_VECTOR (i, updateItems)
{
const CUpdateItem &ui = updateItems[i];
if (!ui.NewData || !ui.HasStream())
continue;
CFilterMode2 fm;
if (useFilters)
{
// analysis.ATime_Defined = false;
RINOK(analysis.GetFilterGroup(i, ui, fm))
/*
if (analysis.ATime_Defined)
{
ui.ATime = FILETIME_To_UInt64(analysis.ATime);
ui.ATime_WasReadByAnalysis = true;
}
*/
}
fm.Encrypted = method.PasswordIsDefined;
const unsigned groupIndex = GetGroup(filters, fm);
while (groupIndex >= groups.Size())
groups.AddNew();
groups[groupIndex].Indices.Add(i);
}
}
#ifndef Z7_NO_CRYPTO
CCryptoGetTextPassword *getPasswordSpec = NULL;
CMyComPtr<ICryptoGetTextPassword> getTextPassword;
if (needEncryptedRepack)
{
getPasswordSpec = new CCryptoGetTextPassword;
getTextPassword = getPasswordSpec;
#ifndef Z7_ST
threadDecoder.getTextPassword = getPasswordSpec;
#endif
if (options.Method->PasswordIsDefined)
getPasswordSpec->Password = options.Method->Password;
else
{
Z7_DECL_CMyComPtr_QI_FROM(
ICryptoGetTextPassword,
getDecoderPassword, updateCallback)
if (!getDecoderPassword)
return E_NOTIMPL;
CMyComBSTR password;
RINOK(getDecoderPassword->CryptoGetTextPassword(&password))
if (password)
getPasswordSpec->Password = password;
}
}
#endif
// ---------- Compress ----------
COutArchive archive;
CArchiveDatabaseOut newDatabase;
RINOK(archive.Create_and_WriteStartPrefix(seqOutStream))
/*
CIntVector treeFolderToArcIndex;
treeFolderToArcIndex.Reserve(treeFolders.Size());
for (i = 0; i < treeFolders.Size(); i++)
treeFolderToArcIndex.Add(-1);
// ---------- Write Tree (only AUX dirs) ----------
for (i = 1; i < treeFolders.Size(); i++)
{
const CTreeFolder &treeFolder = treeFolders[i];
CFileItem file;
CFileItem2 file2;
file2.Init();
int secureID = 0;
if (treeFolder.UpdateItemIndex < 0)
{
// we can store virtual dir item wuthout attrib, but we want all items have attrib.
file.SetAttrib(FILE_ATTRIBUTE_DIRECTORY);
file2.IsAux = true;
}
else
{
const CUpdateItem &ui = updateItems[treeFolder.UpdateItemIndex];
// if item is not dir, then it's parent for alt streams.
// we will write such items later
if (!ui.IsDir)
continue;
secureID = ui.SecureIndex;
if (ui.NewProps)
UpdateItem_To_FileItem(ui, file, file2);
else
GetFile(*db, ui.IndexInArchive, file, file2);
}
file.Size = 0;
file.HasStream = false;
file.IsDir = true;
file.Parent = treeFolder.Parent;
treeFolderToArcIndex[i] = newDatabase.Files.Size();
newDatabase.AddFile(file, file2, treeFolder.Name);
if (totalSecureDataSize != 0)
newDatabase.SecureIDs.Add(secureID);
}
*/
{
/* ---------- Write non-AUX dirs and Empty files ---------- */
CUIntVector emptyRefs;
unsigned i;
for (i = 0; i < updateItems.Size(); i++)
{
const CUpdateItem &ui = updateItems[i];
if (ui.NewData)
{
if (ui.HasStream())
continue;
}
else if (ui.IndexInArchive != -1 && db->Files[(unsigned)ui.IndexInArchive].HasStream)
continue;
/*
if (ui.TreeFolderIndex >= 0)
continue;
*/
emptyRefs.Add(i);
}
emptyRefs.Sort(CompareEmptyItems, (void *)&updateItems);
for (i = 0; i < emptyRefs.Size(); i++)
{
const CUpdateItem &ui = updateItems[emptyRefs[i]];
CFileItem file;
CFileItem2 file2;
UString name;
if (ui.NewProps)
{
UpdateItem_To_FileItem(ui, file, file2);
file.CrcDefined = false;
name = ui.Name;
}
else
{
GetFile(*db, (unsigned)ui.IndexInArchive, file, file2);
db->GetPath((unsigned)ui.IndexInArchive, name);
}
/*
if (totalSecureDataSize != 0)
newDatabase.SecureIDs.Add(ui.SecureIndex);
file.Parent = ui.ParentFolderIndex;
*/
newDatabase.AddFile(file, file2, name);
}
}
lps->ProgressOffset = 0;
{
// ---------- Sort Filters ----------
FOR_VECTOR (i, filters)
{
filters[i].GroupIndex = i;
}
filters.Sort2();
}
for (unsigned groupIndex = 0; groupIndex < filters.Size(); groupIndex++)
{
const CFilterMode2 &filterMode = filters[groupIndex];
CCompressionMethodMode method = *options.Method;
{
const HRESULT res = MakeExeMethod(method, filterMode,
#ifdef Z7_ST
false
#else
options.MaxFilter && options.MultiThreadMixer
#endif
);
RINOK(res)
}
if (filterMode.Encrypted)
{
if (!method.PasswordIsDefined)
{
#ifndef Z7_NO_CRYPTO
if (getPasswordSpec)
method.Password = getPasswordSpec->Password;
#endif
method.PasswordIsDefined = true;
}
}
else
{
method.PasswordIsDefined = false;
method.Password.Empty();
}
CEncoder encoder(method);
// ---------- Repack and copy old solid blocks ----------
const CSolidGroup &group = groups[filterMode.GroupIndex];
FOR_VECTOR (folderRefIndex, group.folderRefs)
{
const CFolderRepack &rep = group.folderRefs[folderRefIndex];
const unsigned folderIndex = rep.FolderIndex;
const CNum numUnpackStreams = db->NumUnpackStreamsVector[folderIndex];
if (rep.NumCopyFiles == numUnpackStreams)
{
if (opCallback)
{
RINOK(opCallback->ReportOperation(
NEventIndexType::kBlockIndex, (UInt32)folderIndex,
NUpdateNotifyOp::kReplicate))
// ---------- Copy old solid block ----------
{
CNum indexInFolder = 0;
for (CNum fi = db->FolderStartFileIndex[folderIndex]; indexInFolder < numUnpackStreams; fi++)
{
if (db->Files[fi].HasStream)
{
indexInFolder++;
RINOK(opCallback->ReportOperation(
NEventIndexType::kInArcIndex, (UInt32)fi,
NUpdateNotifyOp::kReplicate))
}
}
}
}
const UInt64 packSize = db->GetFolderFullPackSize(folderIndex);
RINOK(WriteRange(inStream, archive.SeqStream,
db->GetFolderStreamPos(folderIndex, 0), packSize, progress))
lps->ProgressOffset += packSize;
const unsigned folderIndex_New = newDatabase.Folders.Size();
CFolder &folder = newDatabase.Folders.AddNew();
// v23.01: we copy FolderCrc, if FolderCrc was used
if (db->FolderCRCs.ValidAndDefined(folderIndex))
newDatabase.FolderUnpackCRCs.SetItem(folderIndex_New,
true, db->FolderCRCs.Vals[folderIndex]);
db->ParseFolderInfo(folderIndex, folder);
const CNum startIndex = db->FoStartPackStreamIndex[folderIndex];
FOR_VECTOR (j, folder.PackStreams)
{
newDatabase.PackSizes.Add(db->GetStreamPackSize(startIndex + j));
// newDatabase.PackCRCsDefined.Add(db.PackCRCsDefined[startIndex + j]);
// newDatabase.PackCRCs.Add(db.PackCRCs[startIndex + j]);
}
size_t indexStart = db->FoToCoderUnpackSizes[folderIndex];
const size_t indexEnd = db->FoToCoderUnpackSizes[folderIndex + 1];
for (; indexStart < indexEnd; indexStart++)
newDatabase.CoderUnpackSizes.Add(db->CoderUnpackSizes[indexStart]);
}
else
{
// ---------- Repack old solid block ----------
CBoolVector extractStatuses;
CNum indexInFolder = 0;
if (opCallback)
{
RINOK(opCallback->ReportOperation(
NEventIndexType::kBlockIndex, (UInt32)folderIndex,
NUpdateNotifyOp::kRepack))
}
/* We could reduce data size of decoded folder, if we don't need to repack
last files in folder. But the gain in speed is small in most cases.
So we unpack full folder. */
UInt64 sizeToEncode = 0;
/*
UInt64 importantUnpackSize = 0;
unsigned numImportantFiles = 0;
UInt64 decodeSize = 0;
*/
for (CNum fi = db->FolderStartFileIndex[folderIndex]; indexInFolder < numUnpackStreams; fi++)
{
bool needExtract = false;
const CFileItem &file = db->Files[fi];
if (file.HasStream)
{
indexInFolder++;
const int updateIndex = fileIndexToUpdateIndexMap[fi];
if (updateIndex >= 0 && !updateItems[(unsigned)updateIndex].NewData)
needExtract = true;
// decodeSize += file.Size;
}
extractStatuses.Add(needExtract);
if (needExtract)
{
sizeToEncode += file.Size;
/*
numImportantFiles = extractStatuses.Size();
importantUnpackSize = decodeSize;
*/
}
}
// extractStatuses.DeleteFrom(numImportantFiles);
unsigned startPackIndex = newDatabase.PackSizes.Size();
UInt64 curUnpackSize;
{
CMyComPtr<ISequentialInStream> sbInStream;
CRepackStreamBase *repackBase;
CFolderInStream2 *FosSpec2 = NULL;
CRepackInStreamWithSizes *inStreamSizeCountSpec = new CRepackInStreamWithSizes;
CMyComPtr<ISequentialInStream> inStreamSizeCount = inStreamSizeCountSpec;
{
#ifndef Z7_ST
if (options.MultiThreadMixer)
{
repackBase = threadDecoder.FosSpec;
CMyComPtr<ISequentialOutStream> sbOutStream;
sb.CreateStreams2(sbInStream, sbOutStream);
RINOK(sb.Create_ReInit())
threadDecoder.FosSpec->_stream = sbOutStream;
threadDecoder.InStream = inStream;
threadDecoder.StartPos = db->ArcInfo.DataStartPosition; // db->GetFolderStreamPos(folderIndex, 0);
threadDecoder.Folders = (const CFolders *)db;
threadDecoder.FolderIndex = folderIndex;
// threadDecoder.UnpackSize = importantUnpackSize;
// threadDecoder.send_UnpackSize = true;
}
else
#endif
{
FosSpec2 = new CFolderInStream2;
FosSpec2->Init();
sbInStream = FosSpec2;
repackBase = FosSpec2;
#ifndef Z7_NO_CRYPTO
bool isEncrypted = false;
bool passwordIsDefined = false;
UString password;
#endif
CMyComPtr<ISequentialInStream> decodedStream;
bool dataAfterEnd_Error = false;
const HRESULT res = threadDecoder.Decoder.Decode(
EXTERNAL_CODECS_LOC_VARS
inStream,
db->ArcInfo.DataStartPosition, // db->GetFolderStreamPos(folderIndex, 0);,
*db, folderIndex,
// &importantUnpackSize, // *unpackSize
NULL, // *unpackSize : FULL unpack
NULL, // *outStream
NULL, // *compressProgress
&decodedStream
, dataAfterEnd_Error
Z7_7Z_DECODER_CRYPRO_VARS
#ifndef Z7_ST
, false // mtMode
, 1 // numThreads
, 0 // memUsage
#endif
);
RINOK(res)
if (!decodedStream)
return E_FAIL;
FosSpec2->_inStream = decodedStream;
}
repackBase->_db = db;
repackBase->_opCallback = opCallback;
repackBase->_extractCallback = extractCallback;
UInt32 startIndex = db->FolderStartFileIndex[folderIndex];
RINOK(repackBase->Init(startIndex, &extractStatuses))
inStreamSizeCountSpec->_db = db;
inStreamSizeCountSpec->Init(sbInStream, startIndex, &extractStatuses);
#ifndef Z7_ST
if (options.MultiThreadMixer)
{
WRes wres = threadDecoder.Start();
if (wres != 0)
return HRESULT_FROM_WIN32(wres);
}
#endif
}
// curUnpackSize = sizeToEncode;
HRESULT encodeRes = encoder.Encode1(
EXTERNAL_CODECS_LOC_VARS
inStreamSizeCount,
// NULL,
&inSizeForReduce,
sizeToEncode, // expectedDataSize
newDatabase.Folders.AddNew(),
// newDatabase.CoderUnpackSizes, curUnpackSize,
archive.SeqStream, newDatabase.PackSizes, progress);
if (encodeRes == k_My_HRESULT_CRC_ERROR)
return E_FAIL;
curUnpackSize = inStreamSizeCountSpec->GetSize();
if (encodeRes == S_OK)
{
encoder.Encode_Post(curUnpackSize, newDatabase.CoderUnpackSizes);
}
#ifndef Z7_ST
if (options.MultiThreadMixer)
{
// 16.00: hang was fixed : for case if decoding was not finished.
// We close CBinderInStream and it calls CStreamBinder::CloseRead()
inStreamSizeCount.Release();
sbInStream.Release();
{
const WRes wres = threadDecoder.WaitExecuteFinish();
if (wres != 0)
return HRESULT_FROM_WIN32(wres);
}
const HRESULT decodeRes = threadDecoder.Result;
// if (res == k_My_HRESULT_CRC_ERROR)
if (decodeRes == S_FALSE || threadDecoder.dataAfterEnd_Error)
{
if (extractCallback)
{
RINOK(extractCallback->ReportExtractResult(
NEventIndexType::kInArcIndex, db->FolderStartFileIndex[folderIndex],
// NEventIndexType::kBlockIndex, (UInt32)folderIndex,
(decodeRes != S_OK ?
NExtract::NOperationResult::kDataError :
NExtract::NOperationResult::kDataAfterEnd)))
}
if (decodeRes != S_OK)
return E_FAIL;
}
RINOK(decodeRes)
if (encodeRes == S_OK)
if (sb.ProcessedSize != sizeToEncode)
encodeRes = E_FAIL;
}
else
#endif
{
if (FosSpec2->Result == S_FALSE)
{
if (extractCallback)
{
RINOK(extractCallback->ReportExtractResult(
NEventIndexType::kBlockIndex, (UInt32)folderIndex,
NExtract::NOperationResult::kDataError))
}
return E_FAIL;
}
RINOK(FosSpec2->Result)
}
RINOK(encodeRes)
RINOK(repackBase->CheckFinishedState())
if (curUnpackSize != sizeToEncode)
return E_FAIL;
}
for (; startPackIndex < newDatabase.PackSizes.Size(); startPackIndex++)
lps->OutSize += newDatabase.PackSizes[startPackIndex];
lps->InSize += curUnpackSize;
}
newDatabase.NumUnpackStreamsVector.Add(rep.NumCopyFiles);
CNum indexInFolder = 0;
for (CNum fi = db->FolderStartFileIndex[folderIndex]; indexInFolder < numUnpackStreams; fi++)
{
if (db->Files[fi].HasStream)
{
indexInFolder++;
const int updateIndex = fileIndexToUpdateIndexMap[fi];
if (updateIndex >= 0)
{
const CUpdateItem &ui = updateItems[(unsigned)updateIndex];
if (ui.NewData)
continue;
UString name;
CFileItem file;
CFileItem2 file2;
GetFile(*db, fi, file, file2);
if (ui.NewProps)
{
UpdateItem_To_FileItem2(ui, file2);
file.IsDir = ui.IsDir;
name = ui.Name;
}
else
db->GetPath(fi, name);
/*
file.Parent = ui.ParentFolderIndex;
if (ui.TreeFolderIndex >= 0)
treeFolderToArcIndex[ui.TreeFolderIndex] = newDatabase.Files.Size();
if (totalSecureDataSize != 0)
newDatabase.SecureIDs.Add(ui.SecureIndex);
*/
newDatabase.AddFile(file, file2, name);
}
}
}
}
// ---------- Compress files to new solid blocks ----------
const unsigned numFiles = group.Indices.Size();
if (numFiles == 0)
continue;
CRecordVector<CRefItem> refItems;
refItems.ClearAndSetSize(numFiles);
// bool sortByType = (options.UseTypeSorting && isSoid); // numSolidFiles > 1
const bool sortByType = options.UseTypeSorting;
unsigned i;
for (i = 0; i < numFiles; i++)
refItems[i] = CRefItem(group.Indices[i], updateItems[group.Indices[i]], sortByType);
CSortParam sortParam;
// sortParam.TreeFolders = &treeFolders;
sortParam.SortByType = sortByType;
refItems.Sort(CompareUpdateItems, (void *)&sortParam);
CObjArray<UInt32> indices(numFiles);
for (i = 0; i < numFiles; i++)
{
const UInt32 index = refItems[i].Index;
indices[i] = index;
/*
const CUpdateItem &ui = updateItems[index];
CFileItem file;
if (ui.NewProps)
UpdateItem_To_FileItem(ui, file);
else
file = db.Files[ui.IndexInArchive];
if (file.IsAnti || file.IsDir)
return E_FAIL;
newDatabase.Files.Add(file);
*/
}
for (i = 0; i < numFiles;)
{
UInt64 totalSize = 0;
unsigned numSubFiles;
const wchar_t *prevExtension = NULL;
for (numSubFiles = 0; i + numSubFiles < numFiles && numSubFiles < numSolidFiles; numSubFiles++)
{
const CUpdateItem &ui = updateItems[indices[i + numSubFiles]];
totalSize += ui.Size;
if (totalSize > options.NumSolidBytes)
break;
if (options.SolidExtension)
{
const int slashPos = ui.Name.ReverseFind_PathSepar();
const int dotPos = ui.Name.ReverseFind_Dot();
const wchar_t *ext = ui.Name.Ptr(dotPos <= slashPos ? ui.Name.Len() : (unsigned)(dotPos + 1));
if (numSubFiles == 0)
prevExtension = ext;
else if (!StringsAreEqualNoCase(ext, prevExtension))
break;
}
}
if (numSubFiles < 1)
numSubFiles = 1;
RINOK(lps->SetCur())
/*
const unsigned folderIndex = newDatabase.NumUnpackStreamsVector.Size();
if (opCallback)
{
RINOK(opCallback->ReportOperation(
NEventIndexType::kBlockIndex, (UInt32)folderIndex,
NUpdateNotifyOp::kAdd));
}
*/
CFolderInStream *inStreamSpec = new CFolderInStream;
CMyComPtr<ISequentialInStream> solidInStream(inStreamSpec);
// inStreamSpec->_reportArcProp = reportArcProp;
inStreamSpec->Need_CTime = options.Need_CTime;
inStreamSpec->Need_ATime = options.Need_ATime;
inStreamSpec->Need_MTime = options.Need_MTime;
inStreamSpec->Need_Attrib = options.Need_Attrib;
// inStreamSpec->Need_Crc = options.Need_Crc;
inStreamSpec->Init(updateCallback, &indices[i], numSubFiles);
unsigned startPackIndex = newDatabase.PackSizes.Size();
// UInt64 curFolderUnpackSize = totalSize;
// curFolderUnpackSize = (UInt64)(Int64)-1; // for debug
const UInt64 expectedDataSize = totalSize;
// const unsigned folderIndex_New = newDatabase.Folders.Size();
RINOK(encoder.Encode1(
EXTERNAL_CODECS_LOC_VARS
solidInStream,
// NULL,
&inSizeForReduce,
expectedDataSize, // expected size
newDatabase.Folders.AddNew(),
// newDatabase.CoderUnpackSizes, curFolderUnpackSize,
archive.SeqStream, newDatabase.PackSizes, progress))
if (!inStreamSpec->WasFinished())
return E_FAIL;
/*
if (inStreamSpec->Need_FolderCrc)
newDatabase.FolderUnpackCRCs.SetItem(folderIndex_New,
true, inStreamSpec->GetFolderCrc());
*/
const UInt64 curFolderUnpackSize = inStreamSpec->Get_TotalSize_for_Coder();
encoder.Encode_Post(curFolderUnpackSize, newDatabase.CoderUnpackSizes);
UInt64 packSize = 0;
// const UInt32 numStreams = newDatabase.PackSizes.Size() - startPackIndex;
for (; startPackIndex < newDatabase.PackSizes.Size(); startPackIndex++)
packSize += newDatabase.PackSizes[startPackIndex];
lps->OutSize += packSize;
// for ()
// newDatabase.PackCRCsDefined.Add(false);
// newDatabase.PackCRCs.Add(0);
CNum numUnpackStreams = 0;
UInt64 skippedSize = 0;
UInt64 procSize = 0;
// unsigned numProcessedFiles = 0;
for (unsigned subIndex = 0; subIndex < numSubFiles; subIndex++)
{
const CUpdateItem &ui = updateItems[indices[i + subIndex]];
CFileItem file;
CFileItem2 file2;
UString name;
if (ui.NewProps)
{
UpdateItem_To_FileItem(ui, file, file2);
name = ui.Name;
}
else
{
GetFile(*db, (unsigned)ui.IndexInArchive, file, file2);
db->GetPath((unsigned)ui.IndexInArchive, name);
}
if (file2.IsAnti || file.IsDir)
return E_FAIL;
/*
CFileItem &file = newDatabase.Files[
startFileIndexInDatabase + i + subIndex];
*/
if (!inStreamSpec->Processed[subIndex])
{
// we don't add file here
skippedSize += ui.Size;
continue; // comment it for debug
// name += ".locked"; // for debug
}
// if (inStreamSpec->Need_Crc)
file.Crc = inStreamSpec->CRCs[subIndex];
file.Size = inStreamSpec->Sizes[subIndex];
procSize += file.Size;
// if (file.Size >= 0) // for debug: test purposes
if (file.Size != 0)
{
file.CrcDefined = true; // inStreamSpec->Need_Crc;
file.HasStream = true;
numUnpackStreams++;
}
else
{
file.CrcDefined = false;
file.HasStream = false;
}
if (inStreamSpec->TimesDefined[subIndex])
{
if (inStreamSpec->Need_CTime)
{ file2.CTimeDefined = true; file2.CTime = inStreamSpec->CTimes[subIndex]; }
if (inStreamSpec->Need_ATime
// && !ui.ATime_WasReadByAnalysis
)
{ file2.ATimeDefined = true; file2.ATime = inStreamSpec->ATimes[subIndex]; }
if (inStreamSpec->Need_MTime)
{ file2.MTimeDefined = true; file2.MTime = inStreamSpec->MTimes[subIndex]; }
if (inStreamSpec->Need_Attrib)
{
file2.AttribDefined = true;
file2.Attrib = inStreamSpec->Attribs[subIndex];
}
}
/*
file.Parent = ui.ParentFolderIndex;
if (ui.TreeFolderIndex >= 0)
treeFolderToArcIndex[ui.TreeFolderIndex] = newDatabase.Files.Size();
if (totalSecureDataSize != 0)
newDatabase.SecureIDs.Add(ui.SecureIndex);
*/
/*
if (reportArcProp)
{
RINOK(ReportItemProps(reportArcProp, ui.IndexInClient, file.Size,
file.CrcDefined ? &file.Crc : NULL))
}
*/
// numProcessedFiles++;
newDatabase.AddFile(file, file2, name);
}
/*
// for debug:
// we can write crc to folders area, if folder contains only one file
if (numUnpackStreams == 1 && numSubFiles == 1)
{
const CFileItem &file = newDatabase.Files.Back();
if (file.CrcDefined)
newDatabase.FolderUnpackCRCs.SetItem(folderIndex_New, true, file.Crc);
}
*/
/*
// it's optional check to ensure that sizes are correct
if (inStreamSpec->TotalSize_for_Coder != curFolderUnpackSize)
return E_FAIL;
*/
// if (inStreamSpec->AlignLog == 0)
{
if (procSize != curFolderUnpackSize)
return E_FAIL;
}
// else
{
/*
{
const CFolder &old = newDatabase.Folders.Back();
CFolder &folder = newDatabase.Folders.AddNew();
{
const unsigned numBonds = old.Bonds.Size();
folder.Bonds.SetSize(numBonds + 1);
for (unsigned k = 0; k < numBonds; k++)
folder.Bonds[k] = old.Bonds[k];
CBond &bond = folder.Bonds[numBonds];
bond.PackIndex = 0;
bond.UnpackIndex = 0;
}
{
const unsigned numCoders = old.Coders.Size();
folder.Coders.SetSize(numCoders + 1);
for (unsigned k = 0; k < numCoders; k++)
folder.Coders[k] = old.Coders[k];
CCoderInfo &cod = folder.Coders[numCoders];
cod.Props.Alloc(1);
cod.Props[0] = (Byte)inStreamSpec->AlignLog;
cod.NumStreams = 1;
}
{
const unsigned numPackStreams = old.Coders.Size();
folder.Coders.SetSize(numPackStreams);
for (unsigned k = 0; k < numPackStreams; k++)
folder.PackStreams[k] = old.PackStreams[k];
}
}
newDatabase.Folders.Delete(newDatabase.Folders.Size() - 2);
*/
}
lps->InSize += procSize;
// lps->InSize += curFolderUnpackSize;
// numUnpackStreams = 0 is very bad case for locked files
// v3.13 doesn't understand it.
newDatabase.NumUnpackStreamsVector.Add(numUnpackStreams);
i += numSubFiles;
if (skippedSize != 0 && complexity >= skippedSize)
{
complexity -= skippedSize;
RINOK(updateCallback->SetTotal(complexity))
}
/*
if (reportArcProp)
{
PROPVARIANT prop;
prop.vt = VT_EMPTY;
prop.wReserved1 = 0;
{
NWindows::NCOM::PropVarEm_Set_UInt32(&prop, numProcessedFiles);
RINOK(reportArcProp->ReportProp(
NEventIndexType::kBlockIndex, (UInt32)folderIndex, kpidNumSubFiles, &prop));
}
{
NWindows::NCOM::PropVarEm_Set_UInt64(&prop, curFolderUnpackSize);
RINOK(reportArcProp->ReportProp(
NEventIndexType::kBlockIndex, (UInt32)folderIndex, kpidSize, &prop));
}
{
NWindows::NCOM::PropVarEm_Set_UInt64(&prop, packSize);
RINOK(reportArcProp->ReportProp(
NEventIndexType::kBlockIndex, (UInt32)folderIndex, kpidPackSize, &prop));
}
{
NWindows::NCOM::PropVarEm_Set_UInt32(&prop, numStreams);
RINOK(reportArcProp->ReportProp(
NEventIndexType::kBlockIndex, (UInt32)folderIndex, kpidNumStreams, &prop));
}
RINOK(reportArcProp->ReportFinished(NEventIndexType::kBlockIndex, (UInt32)folderIndex, NUpdate::NOperationResult::kOK));
}
*/
/*
if (opCallback)
{
RINOK(opCallback->ReportOperation(
NEventIndexType::kBlockIndex, (UInt32)folderIndex,
NUpdateNotifyOp::kOpFinished));
}
*/
}
}
RINOK(lps->SetCur())
/*
fileIndexToUpdateIndexMap.ClearAndFree();
groups.ClearAndFree();
*/
/*
for (i = 0; i < newDatabase.Files.Size(); i++)
{
CFileItem &file = newDatabase.Files[i];
file.Parent = treeFolderToArcIndex[file.Parent];
}
if (totalSecureDataSize != 0)
{
newDatabase.SecureBuf.SetCapacity(totalSecureDataSize);
size_t pos = 0;
newDatabase.SecureSizes.Reserve(secureBlocks.Sorted.Size());
for (i = 0; i < secureBlocks.Sorted.Size(); i++)
{
const CByteBuffer &buf = secureBlocks.Bufs[secureBlocks.Sorted[i]];
size_t size = buf.GetCapacity();
if (size != 0)
memcpy(newDatabase.SecureBuf + pos, buf, size);
newDatabase.SecureSizes.Add((UInt32)size);
pos += size;
}
}
*/
{
const unsigned numFolders = newDatabase.Folders.Size();
if (newDatabase.NumUnpackStreamsVector.Size() != numFolders
|| newDatabase.FolderUnpackCRCs.Defs.Size() > numFolders)
return E_FAIL;
newDatabase.FolderUnpackCRCs.if_NonEmpty_FillResedue_with_false(numFolders);
}
updateItems.ClearAndFree();
newDatabase.ReserveDown();
if (opCallback)
RINOK(opCallback->ReportOperation(NEventIndexType::kNoIndex, (UInt32)(Int32)-1, NUpdateNotifyOp::kHeader))
RINOK(archive.WriteDatabase(EXTERNAL_CODECS_LOC_VARS
newDatabase, options.HeaderMethod, options.HeaderOptions))
if (v_StreamSetRestriction)
RINOK(v_StreamSetRestriction->SetRestriction(0, 0))
return S_OK;
}
}}