9#include "loggingcategory.h"
12#include <QCryptographicHash>
17#include <qplatformdefs.h>
19#include "kcompressiondevice.h"
20#include "klimitediodevice_p.h"
21#include <kfilterbase.h>
28#if HAVE_OPENSSL_SUPPORT
29#include <openssl/evp.h>
33#define QT_STAT_LNK 0120000
40#define BUFFER_SIZE 8 * 1024
42static const unsigned char k7zip_signature[6] = {
'7',
'z', 0xBC, 0xAF, 0x27, 0x1C};
46static QChar GetUi16(
const char *p)
48 return QChar(
static_cast<unsigned char>(p[0])
49 | (
static_cast<unsigned char>(p[1]) << 8));
52static quint32 GetUi32(
const char *p)
54 return (
static_cast<unsigned char>(p[0])
55 | (
static_cast<unsigned char>(p[1]) << 8)
56 | (
static_cast<unsigned char>(p[2]) << 16)
57 | (
static_cast<unsigned char>(p[3]) << 24));
60static quint64 GetUi64(
const char *p)
63 | (
static_cast<quint64
>(GetUi32(p + 4)) << 32));
66static quint32 lzma2_dic_size_from_prop(
int p)
68 return ((
static_cast<quint32
>(2) | (p & 1)) << ((p / 2) + 11));
73#define FILE_ATTRIBUTE_READONLY 1
74#define FILE_ATTRIBUTE_HIDDEN 2
75#define FILE_ATTRIBUTE_SYSTEM 4
76#define FILE_ATTRIBUTE_DIRECTORY 16
77#define FILE_ATTRIBUTE_ARCHIVE 32
78#define FILE_ATTRIBUTE_DEVICE 64
79#define FILE_ATTRIBUTE_NORMAL 128
80#define FILE_ATTRIBUTE_TEMPORARY 256
81#define FILE_ATTRIBUTE_SPARSE_FILE 512
82#define FILE_ATTRIBUTE_REPARSE_POINT 1024
83#define FILE_ATTRIBUTE_COMPRESSED 2048
84#define FILE_ATTRIBUTE_OFFLINE 0x1000
85#define FILE_ATTRIBUTE_ENCRYPTED 0x4000
86#define FILE_ATTRIBUTE_UNIX_EXTENSION 0x8000
95 kAdditionalStreamsInfo,
137static const quint64 k_LZMA2 = 0x21;
140static const quint64 k_LZMA = 0x030101;
141static const quint64 k_BCJ = 0x03030103;
142static const quint64 k_BCJ2 = 0x0303011B;
150static const quint64 k_PPMD = 0x030401;
161static const quint64 k_BZip2 = 0x040202;
172static const quint64 k_AES = 0x06F10701;
180 K7ZipFileEntry(K7Zip *zip,
183 const QDateTime &
date,
185 const QString &
group,
186 const QString &symlink,
189 const QByteArray &data);
191 ~K7ZipFileEntry()
override;
197 QByteArray data()
const override;
209 QIODevice *createDevice()
const override;
212 const QByteArray m_data;
216K7ZipFileEntry::K7ZipFileEntry(
K7Zip *zip,
230 m_buffer->setData(m_data);
234K7ZipFileEntry::~K7ZipFileEntry()
244QIODevice *K7ZipFileEntry::createDevice()
const
246 return new KLimitedIODevice(m_buffer,
position(),
size());
256 , attribDefined(false)
286 bool isSimpleCoder()
const
288 return (numInStreams == 1) && (numOutStreams == 1);
293 QList<unsigned char> properties;
298 : unpackCRCDefined(false)
305 qDeleteAll(folderInfos);
308 Q_DISABLE_COPY(Folder)
310 quint64 getUnpackSize()
const
312 if (unpackSizes.isEmpty()) {
315 for (
int i = unpackSizes.size() - 1; i >= 0; i--) {
316 if (findBindPairForOutStream(i) < 0) {
317 return unpackSizes.at(i);
323 int getNumOutStreams()
const
326 for (
int i = 0; i < folderInfos.size(); i++) {
327 result += folderInfos.at(i)->numOutStreams;
332 quint32 getCoderInStreamIndex(quint32 coderIndex)
const
334 quint32 streamIndex = 0;
335 for (quint32 i = 0; i < coderIndex; i++) {
336 streamIndex += folderInfos.at(i)->numInStreams;
341 quint32 getCoderOutStreamIndex(quint32 coderIndex)
const
343 quint32 streamIndex = 0;
344 for (quint32 i = 0; i < coderIndex; i++) {
345 streamIndex += folderInfos.at(i)->numOutStreams;
350 int findBindPairForInStream(
size_t inStreamIndex)
const
352 for (
int i = 0; i < inIndexes.size(); i++) {
353 if (inIndexes[i] == inStreamIndex) {
360 int findBindPairForOutStream(
size_t outStreamIndex)
const
362 for (
int i = 0; i < outIndexes.size(); i++) {
363 if (outIndexes[i] == outStreamIndex) {
370 int findPackStreamArrayIndex(
size_t inStreamIndex)
const
372 for (
int i = 0; i < packedStreams.size(); i++) {
373 if (packedStreams[i] == inStreamIndex) {
380 void findInStream(quint32 streamIndex, quint32 &coderIndex, quint32 &coderStreamIndex)
const
382 for (coderIndex = 0; coderIndex < (quint32)folderInfos.size(); coderIndex++) {
383 quint32 curSize = folderInfos[coderIndex]->numInStreams;
384 if (streamIndex < curSize) {
385 coderStreamIndex = streamIndex;
388 streamIndex -= curSize;
392 void findOutStream(quint32 streamIndex, quint32 &coderIndex, quint32 &coderStreamIndex)
const
394 for (coderIndex = 0; coderIndex < (quint32)folderInfos.size(); coderIndex++) {
395 quint32 curSize = folderInfos[coderIndex]->numOutStreams;
396 if (streamIndex < curSize) {
397 coderStreamIndex = streamIndex;
400 streamIndex -= curSize;
404 bool isEncrypted()
const
406 for (
int i = folderInfos.size() - 1; i >= 0; i--) {
407 if (folderInfos.at(i)->methodID == k_AES) {
416 bool unpackCRCDefined;
418 QList<FolderInfo *> folderInfos;
419 QList<quint64> inIndexes;
420 QList<quint64> outIndexes;
421 QList<quint64> packedStreams;
422 QList<quint64> unpackSizes;
425class Q_DECL_HIDDEN
K7Zip::K7ZipPrivate
428 K7ZipPrivate(K7Zip *parent)
437 , m_currentFile(nullptr)
444 qDeleteAll(fileInfos);
449 QList<bool> packCRCsDefined;
450 QList<quint32> packCRCs;
451 QList<quint64> numUnpackStreamsInFolders;
453 QList<Folder *> folders;
454 QList<FileInfo *> fileInfos;
456 QList<bool> cTimesDefined;
457 QList<quint64> cTimes;
458 QList<bool> aTimesDefined;
459 QList<quint64> aTimes;
460 QList<bool> mTimesDefined;
461 QList<quint64> mTimes;
462 QList<bool> startPositionsDefined;
463 QList<quint64> startPositions;
464 QList<int> fileInfoPopIDs;
467 quint64 numPackStreams;
468 QList<quint64> packSizes;
469 QList<quint64> unpackSizes;
470 QList<bool> digestsDefined;
471 QList<quint32> digests;
486 K7ZipFileEntry *m_currentFile;
487 QList<KArchiveEntry *> m_entryList;
491 packCRCsDefined.
clear();
493 numUnpackStreamsInFolders.
clear();
496 qDeleteAll(fileInfos);
498 cTimesDefined.
clear();
500 aTimesDefined.
clear();
502 mTimesDefined.
clear();
504 startPositionsDefined.
clear();
505 startPositions.
clear();
506 fileInfoPopIDs.
clear();
509 digestsDefined.
clear();
522 quint32 readUInt32();
523 quint64 readUInt64();
524 quint64 readNumber();
525 QString readString();
526 void readHashDigests(
int numItems, QList<bool> &digestsDefined, QList<quint32> &digests);
527 void readBoolVector(
int numItems, QList<bool> &v);
528 void readBoolVector2(
int numItems, QList<bool> &v);
529 void skipData(
int size);
530 bool findAttribute(
int attribute);
531 bool readUInt64DefVector(
int numFiles, QList<quint64> &values, QList<bool> &defined);
534 bool readMainStreamsInfo();
536 bool readUnpackInfo();
537 bool readSubStreamsInfo();
538 KFilterBase *getFilter(
const Folder *folder,
const Folder::FolderInfo *coder,
const int currentCoderIndex, QByteArray &deflatedData, QList<QByteArray> &inflatedDatas);
539 QByteArray readAndDecodePackedStreams(
bool readMainStreamInfo =
true);
542 void createItemsFromEntities(
const KArchiveDirectory *,
const QString &, QByteArray &);
543 void writeByte(
unsigned char b);
544 void writeNumber(quint64 value);
545 void writeBoolVector(
const QList<bool> &boolVector);
546 void writeUInt32(quint32 value);
547 void writeUInt64(quint64 value);
548 void writeHashDigests(
const QList<bool> &digestsDefined,
const QList<quint32> &digests);
549 void writeAlignedBoolHeader(
const QList<bool> &v,
int numDefined,
int type,
unsigned itemSize);
550 void writeUInt64DefVector(
const QList<quint64> &v,
const QList<bool> &defined,
int type);
551 void writeFolder(
const Folder *folder);
552 void writePackInfo(quint64 dataOffset, QList<quint64> &packedSizes, QList<bool> &packedCRCsDefined, QList<quint32> &packedCRCs);
553 void writeUnpackInfo(
const QList<Folder *> &folderItems);
554 void writeSubStreamsInfo(
const QList<quint64> &unpackSizes,
const QList<bool> &digestsDefined,
const QList<quint32> &digests);
555 void writeHeader(quint64 &headerOffset);
556 void writeSignature();
557 void writeStartHeader(
const quint64 nextHeaderSize,
const quint32 nextHeaderCRC,
const quint64 nextHeaderOffset);
558 QByteArray encodeStream(QList<quint64> &packSizes, QList<Folder *> &folds);
563 , d(new K7ZipPrivate(this))
569 , d(new K7ZipPrivate(this))
584 d->password = password;
589 for (
int i = 0; i < d->folders.size(); i++) {
590 if (d->folders.at(i)->isEncrypted()) {
598int K7Zip::K7ZipPrivate::readByte()
600 if (!buffer || pos + 1 > end) {
603 return buffer[pos++];
606quint32 K7Zip::K7ZipPrivate::readUInt32()
608 if (!buffer || (quint64)(pos + 4) > end) {
609 qCDebug(KArchiveLog) <<
"error size";
613 quint32 res = GetUi32(buffer + pos);
618quint64 K7Zip::K7ZipPrivate::readUInt64()
620 if (!buffer || (quint64)(pos + 8) > end) {
621 qCDebug(KArchiveLog) <<
"error size";
625 quint64 res = GetUi64(buffer + pos);
630quint64 K7Zip::K7ZipPrivate::readNumber()
632 if (!buffer || pos >= end) {
636 unsigned char firstByte = buffer[pos++];
637 unsigned char mask = 0x80;
639 for (
int i = 0; i < 8; i++) {
640 if ((firstByte & mask) == 0) {
641 quint64 highPart = firstByte & (mask - 1);
642 value += (highPart << (i * 8));
650 value |= ((
unsigned char)buffer[pos++] << (8 * i));
656QString K7Zip::K7ZipPrivate::readString()
662 const char *buf = buffer + pos;
663 size_t rem = (
end - pos) / 2 * 2;
666 for (i = 0; i < rem; i += 2) {
667 if (buf[i] == 0 && buf[i + 1] == 0) {
672 qCDebug(KArchiveLog) <<
"read string error";
678 int len = (int)(rem / 2);
679 if (len < 0 || (
size_t)len * 2 != rem) {
680 qCDebug(KArchiveLog) <<
"read string unsupported";
685 for (
int i = 0; i < len; i++, buf += 2) {
693void K7Zip::K7ZipPrivate::skipData(
int size)
695 if (!buffer || pos + size > end) {
701bool K7Zip::K7ZipPrivate::findAttribute(
int attribute)
708 int type = readByte();
709 if (type == attribute) {
715 skipData(readNumber());
719void K7Zip::K7ZipPrivate::readBoolVector(
int numItems, QList<bool> &v)
726 unsigned char mask = 0;
727 for (
int i = 0; i < numItems; i++) {
732 v.
append((b & mask) != 0);
737void K7Zip::K7ZipPrivate::readBoolVector2(
int numItems, QList<bool> &v)
743 int allAreDefined = readByte();
744 if (allAreDefined == 0) {
745 readBoolVector(numItems, v);
749 for (
int i = 0; i < numItems; i++) {
754void K7Zip::K7ZipPrivate::readHashDigests(
int numItems, QList<bool> &digestsDefined, QList<quint32> &digests)
760 readBoolVector2(numItems, digestsDefined);
761 for (
int i = 0; i < numItems; i++) {
763 if (digestsDefined[i]) {
764 crc = GetUi32(buffer + pos);
771Folder *K7Zip::K7ZipPrivate::folderItem()
778 int numCoders = readNumber();
780 quint64 numInStreamsTotal = 0;
781 quint64 numOutStreamsTotal = 0;
782 for (
int i = 0; i < numCoders; ++i) {
792 unsigned char coderInfo = readByte();
793 int codecIdSize = (coderInfo & 0xF);
794 if (codecIdSize > 8) {
795 qCDebug(KArchiveLog) <<
"unsupported codec id size";
799 Folder::FolderInfo *info =
new Folder::FolderInfo();
800 std::unique_ptr<unsigned char[]> codecID(
new unsigned char[codecIdSize]);
801 for (
int i = 0; i < codecIdSize; ++i) {
802 codecID[i] = readByte();
806 for (
int j = 0; j < codecIdSize; j++) {
807 id |= codecID[codecIdSize - 1 - j] << (8 * j);
812 if ((coderInfo & 0x10) != 0) {
813 info->numInStreams = readNumber();
814 info->numOutStreams = readNumber();
816 info->numInStreams = 1;
817 info->numOutStreams = 1;
821 if ((coderInfo & 0x20) != 0) {
822 int propertiesSize = readNumber();
823 for (
int i = 0; i < propertiesSize; ++i) {
824 info->properties.
append(readByte());
828 if ((coderInfo & 0x80) != 0) {
829 qCDebug(KArchiveLog) <<
"unsupported";
835 numInStreamsTotal += info->numInStreams;
836 numOutStreamsTotal += info->numOutStreams;
837 folder->folderInfos.
append(info);
840 int numBindPairs = numOutStreamsTotal - 1;
841 for (
int i = 0; i < numBindPairs; i++) {
842 folder->inIndexes.
append(readNumber());
843 folder->outIndexes.
append(readNumber());
846 int numPackedStreams = numInStreamsTotal - numBindPairs;
847 if (numPackedStreams > 1) {
848 for (
int i = 0; i < numPackedStreams; ++i) {
849 folder->packedStreams.
append(readNumber());
852 if (numPackedStreams == 1) {
853 for (quint64 i = 0; i < numInStreamsTotal; i++) {
854 if (folder->findBindPairForInStream(i) < 0) {
855 folder->packedStreams.
append(i);
859 if (folder->packedStreams.
size() != 1) {
868bool K7Zip::K7ZipPrivate::readUInt64DefVector(
int numFiles, QList<quint64> &values, QList<bool> &defined)
874 readBoolVector2(numFiles, defined);
876 int external = readByte();
878 int dataIndex = readNumber();
879 if (dataIndex < 0 /*|| dataIndex >= dataVector->Size()*/) {
880 qCDebug(KArchiveLog) <<
"wrong data index";
887 for (
int i = 0; i < numFiles; i++) {
897bool K7Zip::K7ZipPrivate::readPackInfo()
903 packPos = readNumber();
904 numPackStreams = readNumber();
907 packCRCsDefined.
clear();
910 if (!findAttribute(kSize)) {
911 qCDebug(KArchiveLog) <<
"kSize not found";
915 for (quint64 i = 0; i < numPackStreams; ++i) {
916 packSizes.
append(readNumber());
920 int type = readByte();
925 readHashDigests(numPackStreams, packCRCsDefined, packCRCs);
928 skipData(readNumber());
932 for (quint64 i = 0; i < numPackStreams; ++i) {
933 packCRCsDefined.
append(
false);
940bool K7Zip::K7ZipPrivate::readUnpackInfo()
946 if (!findAttribute(kFolder)) {
947 qCDebug(KArchiveLog) <<
"kFolder not found";
951 int numFolders = readNumber();
954 int external = readByte();
957 for (
int i = 0; i < numFolders; ++i) {
958 folders.
append(folderItem());
963 int dataIndex = readNumber();
964 if (dataIndex < 0 /*|| dataIndex >= dataVector->Size()*/) {
965 qCDebug(KArchiveLog) <<
"wrong data index";
971 qCDebug(KArchiveLog) <<
"external error";
975 if (!findAttribute(kCodersUnpackSize)) {
976 qCDebug(KArchiveLog) <<
"kCodersUnpackSize not found";
980 for (
int i = 0; i < numFolders; ++i) {
982 int numOutStreams = folder->getNumOutStreams();
983 for (
int j = 0; j < numOutStreams; ++j) {
984 folder->unpackSizes.
append(readNumber());
989 int type = readByte();
994 QList<bool> crcsDefined;
996 readHashDigests(numFolders, crcsDefined, crcs);
997 for (
int i = 0; i < numFolders; i++) {
999 folder->unpackCRCDefined = crcsDefined[i];
1000 folder->unpackCRC = crcs[i];
1004 skipData(readNumber());
1009bool K7Zip::K7ZipPrivate::readSubStreamsInfo()
1015 numUnpackStreamsInFolders.
clear();
1020 if (type == kNumUnpackStream) {
1021 for (
int i = 0; i < folders.
size(); i++) {
1022 numUnpackStreamsInFolders.
append(readNumber());
1026 if (type == kCRC || type == kSize) {
1032 skipData(readNumber());
1035 if (numUnpackStreamsInFolders.
isEmpty()) {
1036 for (
int i = 0; i < folders.
size(); i++) {
1037 numUnpackStreamsInFolders.
append(1);
1041 for (
int i = 0; i < numUnpackStreamsInFolders.
size(); i++) {
1042 quint64 numSubstreams = numUnpackStreamsInFolders.
at(i);
1043 if (numSubstreams == 0) {
1047 for (quint64 j = 1; j < numSubstreams; j++) {
1048 if (type == kSize) {
1049 int size = readNumber();
1050 unpackSizes.
append(size);
1054 unpackSizes.
append(folders.
at(i)->getUnpackSize() - sum);
1057 if (type == kSize) {
1062 int numDigestsTotal = 0;
1063 for (
int i = 0; i < folders.
size(); i++) {
1064 quint64 numSubstreams = numUnpackStreamsInFolders.
at(i);
1065 if (numSubstreams != 1 || !folders.
at(i)->unpackCRCDefined) {
1066 numDigests += numSubstreams;
1068 numDigestsTotal += numSubstreams;
1073 QList<bool> digestsDefined2;
1074 QList<quint32> digests2;
1075 readHashDigests(numDigests, digestsDefined2, digests2);
1076 int digestIndex = 0;
1077 for (
int i = 0; i < folders.
size(); i++) {
1078 quint64 numSubstreams = numUnpackStreamsInFolders.
at(i);
1079 const Folder *folder = folders.
at(i);
1080 if (numSubstreams == 1 && folder->unpackCRCDefined) {
1081 digestsDefined.
append(
true);
1082 digests.
append(folder->unpackCRC);
1084 for (quint64 j = 0; j < numSubstreams; j++, digestIndex++) {
1085 digestsDefined.
append(digestsDefined2[digestIndex]);
1086 digests.
append(digests2[digestIndex]);
1090 }
else if (type == kEnd) {
1091 if (digestsDefined.
isEmpty()) {
1092 for (
int i = 0; i < numDigestsTotal; i++) {
1093 digestsDefined.
append(
false);
1100 skipData(readNumber());
1108#define TICKSPERSEC 10000000
1109#define TICKSPERMSEC 10000
1110#define SECSPERDAY 86400
1111#define SECSPERHOUR 3600
1112#define SECSPERMIN 60
1113#define EPOCHWEEKDAY 1
1114#define DAYSPERWEEK 7
1115#define DAYSPERQUADRICENTENNIUM (365 * 400 + 97)
1116#define DAYSPERNORMALQUADRENNIUM (365 * 4 + 1)
1117#define TICKS_1601_TO_1970 (SECS_1601_TO_1970 * TICKSPERSEC)
1118#define SECS_1601_TO_1970 ((369 * 365 + 89) * (unsigned long long)SECSPERDAY)
1120static uint toTimeT(
const long long liTime)
1122 long long time = liTime / TICKSPERSEC;
1128 long int days = time / SECSPERDAY;
1129 int secondsInDay = time % SECSPERDAY;
1132 short hour = (short)(secondsInDay / SECSPERHOUR);
1133 secondsInDay = secondsInDay % SECSPERHOUR;
1134 short minute = (short)(secondsInDay / SECSPERMIN);
1135 short second = (short)(secondsInDay % SECSPERMIN);
1138 long int cleaps = (3 * ((4 * days + 1227) / DAYSPERQUADRICENTENNIUM) + 3) / 4;
1139 days += 28188 + cleaps;
1140 long int years = (20 * days - 2442) / (5 * DAYSPERNORMALQUADRENNIUM);
1141 long int yearday = days - (years * DAYSPERNORMALQUADRENNIUM) / 4;
1142 long int months = (64 * yearday) / 1959;
1150 month = (short)(months - 1);
1151 year = (short)(years + 1524);
1153 month = (short)(months - 13);
1154 year = (short)(years + 1525);
1159 short day = (short)(yearday - (1959 * months) / 64);
1161 QDateTime t(QDate(year, month, day), QTime(hour, minute, second));
1163 return t.toSecsSinceEpoch();
1166long long rtlSecondsSince1970ToSpecTime(quint32 seconds)
1168 long long secs = seconds * (
long long)TICKSPERSEC + TICKS_1601_TO_1970;
1172bool K7Zip::K7ZipPrivate::readMainStreamsInfo()
1181 if (type > ((quint32)1 << 30)) {
1182 qCDebug(KArchiveLog) <<
"type error";
1189 if (!readPackInfo()) {
1190 qCDebug(KArchiveLog) <<
"error during read pack information";
1196 if (!readUnpackInfo()) {
1197 qCDebug(KArchiveLog) <<
"error during read pack information";
1202 case kSubStreamsInfo: {
1203 if (!readSubStreamsInfo()) {
1204 qCDebug(KArchiveLog) <<
"error during read substreams information";
1210 qCDebug(KArchiveLog) <<
"Wrong type";
1215 qCDebug(KArchiveLog) <<
"should not reach";
1219static bool getInStream(
const Folder *folder, quint32 streamIndex,
int &seqInStream, quint32 &coderIndex)
1221 for (
int i = 0; i < folder->packedStreams.
size(); i++) {
1222 if (folder->packedStreams[i] == streamIndex) {
1228 int binderIndex = folder->findBindPairForInStream(streamIndex);
1229 if (binderIndex < 0) {
1233 quint32 coderStreamIndex;
1234 folder->findOutStream(folder->outIndexes[binderIndex], coderIndex, coderStreamIndex);
1236 quint32 startIndex = folder->getCoderInStreamIndex(coderIndex);
1238 if (folder->folderInfos[coderIndex]->numInStreams > 1) {
1242 for (
int i = 0; i < (int)folder->folderInfos[coderIndex]->numInStreams; i++) {
1243 getInStream(folder, startIndex + i, seqInStream, coderIndex);
1249static bool getOutStream(
const Folder *folder, quint32 streamIndex,
int &seqOutStream)
1251 QList<quint32> outStreams;
1252 quint32 outStreamIndex = 0;
1253 for (
int i = 0; i < folder->folderInfos.
size(); i++) {
1254 const Folder::FolderInfo *coderInfo = folder->folderInfos.
at(i);
1256 for (
int j = 0; j < coderInfo->numOutStreams; j++, outStreamIndex++) {
1257 if (folder->findBindPairForOutStream(outStreamIndex) < 0) {
1258 outStreams.
append(outStreamIndex);
1263 for (
int i = 0; i < outStreams.
size(); i++) {
1264 if (outStreams[i] == streamIndex) {
1270 int binderIndex = folder->findBindPairForOutStream(streamIndex);
1271 if (binderIndex < 0) {
1276 quint32 coderStreamIndex;
1277 folder->findInStream(folder->inIndexes[binderIndex], coderIndex, coderStreamIndex);
1279 quint32 startIndex = folder->getCoderOutStreamIndex(coderIndex);
1281 if (folder->folderInfos[coderIndex]->numOutStreams > 1) {
1285 for (
int i = 0; i < (int)folder->folderInfos[coderIndex]->numOutStreams; i++) {
1286 getOutStream(folder, startIndex + i, seqOutStream);
1292static const int catCycle = 6;
1294static QByteArray calculateKey(
const QByteArray &password, quint32 numCyclesPower,
const QByteArray &salt)
1296 quint32 rounds, stages;
1298 if (numCyclesPower > catCycle) {
1299 rounds = 1 << catCycle;
1300 stages = 1 << (numCyclesPower - catCycle);
1302 rounds = 1 << numCyclesPower;
1306 QByteArray saltPassword = salt + password;
1311 for (quint32 i = 0; i < stages; i++) {
1315 for (quint32 j = 0; j < rounds; j++) {
1316 result += saltPassword;
1318 quint64 value = s + j;
1319 for (
int k = 0; k < 8; k++) {
1320 result.
append(value >> (k * 8));
1324 hash.addData(result);
1328 return hash.result();
1331#if HAVE_OPENSSL_SUPPORT
1333static QByteArray decryptAES(
const QList<quint8> &coderProperties,
const QString &password, QByteArray &encryptedData)
1336 const QByteArray passwordBytes = toUtf16LE(password);
1338 quint8 firstByte = coderProperties[0];
1339 quint32 numCyclesPower = firstByte & 0x3F;
1341 if ((firstByte & 0xC0) == 0) {
1342 qCDebug(KArchiveLog) <<
"Unsupported AES properties";
1343 return QByteArray();
1346 int saltSize = ((firstByte >> 7) & 1) + (coderProperties[1] >> 4);
1347 int ivSize = ((firstByte >> 6) & 1) + (coderProperties[1] & 0x0F);
1349 QByteArray salt((
const char *)coderProperties.
data() + 2, saltSize);
1350 QByteArray iv((
const char *)coderProperties.
data() + 2 + saltSize, ivSize);
1353 iv.append(16 - ivSize,
'\x00');
1356 const QByteArray key = calculateKey(passwordBytes, numCyclesPower, salt);
1357 if (key.
size() != 32) {
1358 qCDebug(KArchiveLog) <<
"Failed to calculate key";
1359 return QByteArray();
1362 EVP_CIPHER_CTX *ctx = EVP_CIPHER_CTX_new();
1364 qCDebug(KArchiveLog) <<
"Failed to create OpenSSL cipher context";
1365 return QByteArray();
1368 const auto ctxCleanupGuard = qScopeGuard([&ctx] {
1369 EVP_CIPHER_CTX_free(ctx);
1373 int padLen = encryptedData.
size() % 16;
1375 encryptedData.
append(16 - padLen,
'\x00');
1378 QByteArray decryptedData;
1379 int len, plainTextLen = 0;
1382 if (EVP_DecryptInit_ex(ctx, EVP_aes_256_cbc(),
nullptr, (
const unsigned char *)key.
constData(), (
const unsigned char *)iv.constData()) != 1) {
1383 qCDebug(KArchiveLog) <<
"EVP_DecryptInit_ex failed";
1384 return QByteArray();
1388 EVP_CIPHER_CTX_set_padding(ctx, 0);
1390 if (EVP_DecryptUpdate(ctx, (
unsigned char *)decryptedData.
data(), &len, (
const unsigned char *)encryptedData.
constData(), encryptedData.
size()) != 1) {
1391 qCDebug(KArchiveLog) <<
"EVP_DecryptUpdate failed";
1392 return QByteArray();
1394 plainTextLen += len;
1396 if (EVP_DecryptFinal_ex(ctx, (
unsigned char *)decryptedData.
data() + len, &len) != 1) {
1397 qCDebug(KArchiveLog) <<
"EVP_DecryptFinal_ex failed";
1398 return QByteArray();
1400 plainTextLen += len;
1402 decryptedData.
resize(plainTextLen);
1403 return decryptedData;
1408const int kNumTopBits = 24;
1409const quint32 kTopValue = (1 << kNumTopBits);
1420 RangeDecoder(
const QByteArray &s)
1426 for (
int i = 0; i < 5; i++) {
1427 code = (code << 8) | readByte();
1431 unsigned char readByte()
1433 if (pos >= stream.
size()) {
1436 return stream[pos++];
1441 while (range < kTopValue) {
1442 code = (code << 8) | readByte();
1447 quint32 getThreshold(quint32 total)
1449 return (code) / (range /= total);
1452 void decode(quint32
start, quint32 size)
1454 code -=
start * range;
1459 quint32 decodeDirectBits(
int numTotalBits)
1464 for (
int i = numTotalBits; i != 0; i--) {
1466 quint32 t = (c - r) >> 31;
1468 result = (result << 1) | (1 - t);
1470 if (r < kTopValue) {
1471 c = (c << 8) | readByte();
1480 quint32 DecodeBit(quint32 size0, quint32 numTotalBits)
1482 quint32 newBound = (range >> numTotalBits) * size0;
1484 if (code < newBound) {
1497const int kNumBitModelTotalBits = 11;
1498const quint32 kBitModelTotal = (1 << kNumBitModelTotalBits);
1500template<
int numMoveBits>
1505 void updateModel(quint32 symbol)
1508 prob += (kBitModelTotal - prob) >> numMoveBits;
1510 prob -= (prob) >> numMoveBits;
1516 prob = kBitModelTotal / 2;
1520template<
int numMoveBits>
1521class CBitDecoder :
public CBitModel<numMoveBits>
1524 quint32 decode(RangeDecoder *decoder)
1526 quint32 newBound = (decoder->range >> kNumBitModelTotalBits) * this->prob;
1527 if (decoder->code < newBound) {
1528 decoder->range = newBound;
1529 this->prob += (kBitModelTotal - this->prob) >> numMoveBits;
1530 if (decoder->range < kTopValue) {
1531 decoder->code = (decoder->code << 8) | decoder->readByte();
1532 decoder->range <<= 8;
1536 decoder->range -= newBound;
1537 decoder->code -= newBound;
1538 this->prob -= (this->prob) >> numMoveBits;
1539 if (decoder->range < kTopValue) {
1540 decoder->code = (decoder->code << 8) | decoder->readByte();
1541 decoder->range <<= 8;
1548inline bool isJcc(
unsigned char b0,
unsigned char b1)
1550 return (b0 == 0x0F && (b1 & 0xF0) == 0x80);
1552inline bool isJ(
unsigned char b0,
unsigned char b1)
1554 return ((b1 & 0xFE) == 0xE8 || isJcc(b0, b1));
1556inline unsigned getIndex(
unsigned char b0,
unsigned char b1)
1558 return ((b1 == 0xE8) ? b0 : ((b1 == 0xE9) ? 256 : 257));
1561const int kNumMoveBits = 5;
1563static QByteArray decodeBCJ2(
const QByteArray &mainStream,
const QByteArray &callStream,
const QByteArray &jumpStream,
const QByteArray &rangeBuffer)
1565 unsigned char prevByte = 0;
1566 QByteArray outStream;
1567 int mainStreamPos = 0;
1568 int callStreamPos = 0;
1569 int jumpStreamPos = 0;
1571 RangeDecoder rangeDecoder(rangeBuffer);
1573 QList<CBitDecoder<kNumMoveBits>> statusDecoder(256 + 2);
1575 for (
int i = 0; i < 256 + 2; i++) {
1576 statusDecoder[i].init();
1581 unsigned char b = 0;
1582 const quint32 kBurstSize = (1 << 18);
1583 for (i = 0; i < kBurstSize; i++) {
1584 if (mainStreamPos == mainStream.
size()) {
1588 b = mainStream[mainStreamPos++];
1591 if (isJ(prevByte, b)) {
1597 if (i == kBurstSize) {
1601 unsigned index = getIndex(prevByte, b);
1602 if (statusDecoder[index].decode(&rangeDecoder) == 1) {
1604 if (callStreamPos + 4 > callStream.
size()) {
1605 return QByteArray();
1608 if (jumpStreamPos + 4 > jumpStream.
size()) {
1609 return QByteArray();
1613 for (
int i = 0; i < 4; i++) {
1616 b0 = callStream[callStreamPos++];
1618 b0 = jumpStream[jumpStreamPos++];
1621 src |= ((quint32)b0);
1624 quint32 dest = src - (quint32(outStream.
size()) + 4);
1625 outStream.
append((
unsigned char)(dest));
1626 outStream.
append((
unsigned char)(dest >> 8));
1627 outStream.
append((
unsigned char)(dest >> 16));
1628 outStream.
append((
unsigned char)(dest >> 24));
1629 prevByte = (
unsigned char)(dest >> 24);
1636KFilterBase *K7Zip::K7ZipPrivate::getFilter(
const Folder *folder,
1637 const Folder::FolderInfo *coder,
1638 const int currentCoderIndex,
1639 QByteArray &deflatedData,
1640 QList<QByteArray> &inflatedDatas)
1642 KFilterBase *
filter =
nullptr;
1644 switch (coder->methodID) {
1648 qCDebug(KArchiveLog) <<
"filter not found";
1656 qCDebug(KArchiveLog) <<
"filter not found";
1672 if (coder->properties.
size() >= 2) {
1674 qCDebug(KArchiveLog) <<
"Password is required for AES decryption";
1678#if HAVE_OPENSSL_SUPPORT
1679 QByteArray decryptedData = decryptAES(coder->properties, password, deflatedData);
1680 if (decryptedData.
isEmpty()) {
1681 qCDebug(KArchiveLog) <<
"AES decryption failed";
1685 if (folder->folderInfos.
size() > 1 && currentCoderIndex < folder->folderInfos.size()) {
1686 deflatedData = decryptedData;
1687 int nextCoderIndex = currentCoderIndex + 1;
1688 filter = getFilter(folder, folder->folderInfos[nextCoderIndex], nextCoderIndex, decryptedData, inflatedDatas);
1690 inflatedDatas.
append(decryptedData);
1699 qCDebug(KArchiveLog) <<
"filter not found";
1705 QByteArray bcj2 = decodeBCJ2(inflatedDatas[0], inflatedDatas[1], inflatedDatas[2], deflatedData);
1706 inflatedDatas.
clear();
1707 inflatedDatas.
append(bcj2);
1713 qCDebug(KArchiveLog) <<
"filter not found";
1723QByteArray K7Zip::K7ZipPrivate::readAndDecodePackedStreams(
bool readMainStreamInfo)
1726 return QByteArray();
1729 if (readMainStreamInfo) {
1730 readMainStreamsInfo();
1733 QByteArray inflatedData;
1735 quint64 startPos = 32 + packPos;
1736 for (
int i = 0; i < folders.
size(); i++) {
1737 const Folder *folder = folders.
at(i);
1738 quint64 unpackSize64 = folder->getUnpackSize();
1739 size_t unpackSize = (size_t)unpackSize64;
1740 if (unpackSize != unpackSize64) {
1741 qCDebug(KArchiveLog) <<
"unsupported";
1742 return inflatedData;
1746 quint32 mainCoderIndex = 0;
1747 QList<int> outStreamIndexed;
1748 int outStreamIndex = 0;
1749 for (
int j = 0; j < folder->folderInfos.
size(); j++) {
1750 const Folder::FolderInfo *info = folder->folderInfos[j];
1751 for (
int k = 0; k < info->numOutStreams; k++, outStreamIndex++) {
1752 if (folder->findBindPairForOutStream(outStreamIndex) < 0) {
1753 outStreamIndexed.
append(outStreamIndex);
1760 if (!outStreamIndexed.
isEmpty()) {
1761 folder->findOutStream(outStreamIndexed[0], mainCoderIndex, temp);
1764 quint32 startInIndex = folder->getCoderInStreamIndex(mainCoderIndex);
1765 quint32 startOutIndex = folder->getCoderOutStreamIndex(mainCoderIndex);
1767 Folder::FolderInfo *mainCoder = folder->folderInfos[mainCoderIndex];
1769 QList<int> seqInStreams;
1770 QList<quint32> coderIndexes;
1771 seqInStreams.
reserve(mainCoder->numInStreams);
1772 coderIndexes.
reserve(mainCoder->numInStreams);
1773 for (
int j = 0; j < (int)mainCoder->numInStreams; j++) {
1776 getInStream(folder, startInIndex + j, seqInStream, coderIndex);
1777 seqInStreams.
append(seqInStream);
1778 coderIndexes.
append(coderIndex);
1781 QList<int> seqOutStreams;
1782 seqOutStreams.
reserve(mainCoder->numOutStreams);
1783 for (
int j = 0; j < (int)mainCoder->numOutStreams; j++) {
1785 getOutStream(folder, startOutIndex + j, seqOutStream);
1786 seqOutStreams.
append(seqOutStream);
1789 QList<QByteArray> datas;
1790 for (
int j = 0; j < (int)mainCoder->numInStreams; j++) {
1791 quint64 size = packSizes[j + i];
1792 std::unique_ptr<char[]> encodedBuffer(
new char[size]);
1793 QIODevice *dev = q->
device();
1794 dev->
seek(startPos);
1795 quint64 n = dev->
read(encodedBuffer.get(), size);
1797 qCDebug(KArchiveLog) <<
"Failed read next size, should read " << size <<
", read " << n;
1798 return inflatedData;
1800 QByteArray deflatedData(encodedBuffer.get(), size);
1801 datas.
append(deflatedData);
1807 QList<QByteArray> inflatedDatas;
1808 QByteArray deflatedData;
1809 for (
int j = 0; j < seqInStreams.
size(); ++j) {
1812 if ((quint32)j != mainCoderIndex) {
1813 coderIndex = coderIndexes[j];
1815 coderIndex = mainCoderIndex;
1818 Folder::FolderInfo *coder = folder->folderInfos[coderIndex];
1819 deflatedData = datas[seqInStreams[j]];
1821 KFilterBase *
filter = getFilter(folder, coder, coderIndex, deflatedData, inflatedDatas);
1822 if (coder->methodID == k_BCJ2) {
1827 if (coder->methodID == k_AES) {
1830 return QByteArray();
1835 QByteArray outBuffer;
1837 outBuffer.
resize(unpackSize);
1840 QByteArray inflatedDataTmp;
1841 while (result != KFilterBase::End && result != KFilterBase::Error && !
filter->inBufferEmpty()) {
1843 result =
filter->uncompress();
1844 if (result == KFilterBase::Error) {
1845 qCDebug(KArchiveLog) <<
" decode error";
1848 return QByteArray();
1850 int uncompressedBytes = outBuffer.
size() -
filter->outBufferAvailable();
1853 inflatedDataTmp.
append(outBuffer.
data(), uncompressedBytes);
1855 if (result == KFilterBase::End) {
1861 if (result != KFilterBase::End && !
filter->inBufferEmpty()) {
1862 qCDebug(KArchiveLog) <<
"decode failed result" << result;
1865 return QByteArray();
1871 inflatedDatas.
append(inflatedDataTmp);
1874 QByteArray inflated;
1875 for (
const QByteArray &data : std::as_const(inflatedDatas)) {
1879 inflatedDatas.
clear();
1881 if (folder->unpackCRCDefined) {
1882 if ((
size_t)inflated.
size() < unpackSize) {
1883 qCDebug(KArchiveLog) <<
"wrong crc size data";
1884 return QByteArray();
1886 quint32 crc = crc32(0, (Bytef *)(inflated.
data()), unpackSize);
1887 if (crc != folder->unpackCRC) {
1888 qCDebug(KArchiveLog) <<
"wrong crc";
1889 return QByteArray();
1893 inflatedData.
append(inflated);
1896 return inflatedData;
1901void K7Zip::K7ZipPrivate::createItemsFromEntities(
const KArchiveDirectory *dir,
const QString &path, QByteArray &data)
1903 const QStringList l =
dir->entries();
1905 for (; it != l.
end(); ++it) {
1906 const KArchiveEntry *entry =
dir->entry((*it));
1908 FileInfo *fileInfo =
new FileInfo;
1909 fileInfo->attribDefined =
true;
1911 fileInfo->path =
path + entry->
name();
1912 mTimesDefined.
append(
true);
1916 const K7ZipFileEntry *fileEntry =
static_cast<const K7ZipFileEntry *
>(entry);
1918 fileInfo->attributes = FILE_ATTRIBUTE_ARCHIVE;
1919 fileInfo->attributes |= FILE_ATTRIBUTE_UNIX_EXTENSION + ((entry->
permissions() & 0xFFFF) << 16);
1920 fileInfo->size = fileEntry->
size();
1922 if (fileInfo->size > 0) {
1923 fileInfo->hasStream =
true;
1925 unpackSizes.
append(fileInfo->size);
1926 }
else if (!symLink.
isEmpty()) {
1927 fileInfo->hasStream =
true;
1931 fileInfos.
append(fileInfo);
1933 fileInfo->attributes = FILE_ATTRIBUTE_DIRECTORY;
1934 fileInfo->attributes |= FILE_ATTRIBUTE_UNIX_EXTENSION + ((entry->
permissions() & 0xFFFF) << 16);
1935 fileInfo->isDir =
true;
1936 fileInfos.
append(fileInfo);
1937 createItemsFromEntities((KArchiveDirectory *)entry, path + (*it) + QLatin1Char(
'/'), data);
1942void K7Zip::K7ZipPrivate::writeByte(
unsigned char b)
1948void K7Zip::K7ZipPrivate::writeNumber(quint64 value)
1953 for (i = 0; i < 8; i++) {
1954 if (value < ((quint64(1) << (7 * (i + 1))))) {
1955 firstByte |= (int)(value >> (8 * i));
1961 writeByte(firstByte);
1962 for (; i > 0; i--) {
1963 writeByte((
int)value);
1968void K7Zip::K7ZipPrivate::writeBoolVector(
const QList<bool> &boolVector)
1972 for (
int i = 0; i < boolVector.
size(); i++) {
1973 if (boolVector[i]) {
1988void K7Zip::K7ZipPrivate::writeUInt32(quint32 value)
1990 for (
int i = 0; i < 4; i++) {
1991 writeByte((
unsigned char)value);
1996void K7Zip::K7ZipPrivate::writeUInt64(quint64 value)
1998 for (
int i = 0; i < 8; i++) {
1999 writeByte((
unsigned char)value);
2004void K7Zip::K7ZipPrivate::writeAlignedBoolHeader(
const QList<bool> &v,
int numDefined,
int type,
unsigned itemSize)
2006 const unsigned bvSize = (numDefined == v.
size()) ? 0 : ((unsigned)v.
size() + 7) / 8;
2007 const quint64 dataSize = (quint64)numDefined * itemSize + bvSize + 2;
2011 writeNumber(dataSize);
2012 if (numDefined == v.
size()) {
2021void K7Zip::K7ZipPrivate::writeUInt64DefVector(
const QList<quint64> &v,
const QList<bool> &defined,
int type)
2025 for (
int i = 0; i < defined.
size(); i++) {
2031 if (numDefined == 0) {
2035 writeAlignedBoolHeader(defined, numDefined, type, 8);
2037 for (
int i = 0; i < defined.
size(); i++) {
2044void K7Zip::K7ZipPrivate::writeHashDigests(
const QList<bool> &digestsDefined,
const QList<quint32> &digests)
2048 for (i = 0; i < digestsDefined.
size(); i++) {
2049 if (digestsDefined[i]) {
2054 if (numDefined == 0) {
2059 if (numDefined == digestsDefined.
size()) {
2063 writeBoolVector(digestsDefined);
2066 for (i = 0; i < digests.
size(); i++) {
2067 if (digestsDefined[i]) {
2068 writeUInt32(digests[i]);
2073void K7Zip::K7ZipPrivate::writePackInfo(quint64 dataOffset, QList<quint64> &packedSizes, QList<bool> &packedCRCsDefined, QList<quint32> &packedCRCs)
2078 writeByte(kPackInfo);
2079 writeNumber(dataOffset);
2080 writeNumber(packedSizes.
size());
2083 for (
int i = 0; i < packedSizes.
size(); i++) {
2084 writeNumber(packedSizes[i]);
2087 writeHashDigests(packedCRCsDefined, packedCRCs);
2092void K7Zip::K7ZipPrivate::writeFolder(
const Folder *folder)
2094 writeNumber(folder->folderInfos.
size());
2095 for (
int i = 0; i < folder->folderInfos.
size(); i++) {
2096 const Folder::FolderInfo *info = folder->folderInfos.
at(i);
2098 size_t propsSize = info->properties.
size();
2100 quint64
id = info->methodID;
2102 for (idSize = 1; idSize <
sizeof(id); idSize++) {
2103 if ((
id >> (8 * idSize)) == 0) {
2109 for (
int t = idSize - 1; t >= 0; t--,
id >>= 8) {
2110 longID[t] = (int)(
id & 0xFF);
2114 b = (int)(idSize & 0xF);
2115 bool isComplex = !info->isSimpleCoder();
2116 b |= (isComplex ? 0x10 : 0);
2117 b |= ((propsSize != 0) ? 0x20 : 0);
2120 for (
size_t j = 0; j < idSize; ++j) {
2121 writeByte(longID[j]);
2125 writeNumber(info->numInStreams);
2126 writeNumber(info->numOutStreams);
2129 if (propsSize == 0) {
2133 writeNumber(propsSize);
2134 for (
size_t j = 0; j < propsSize; ++j) {
2135 writeByte(info->properties[j]);
2140 for (
int i = 0; i < folder->inIndexes.
size(); i++) {
2141 writeNumber(folder->inIndexes[i]);
2142 writeNumber(folder->outIndexes[i]);
2145 if (folder->packedStreams.
size() > 1) {
2146 for (
int i = 0; i < folder->packedStreams.
size(); i++) {
2147 writeNumber(folder->packedStreams[i]);
2152void K7Zip::K7ZipPrivate::writeUnpackInfo(
const QList<Folder *> &folderItems)
2158 writeByte(kUnpackInfo);
2161 writeNumber(folderItems.
size());
2164 for (
int i = 0; i < folderItems.
size(); i++) {
2165 writeFolder(folderItems[i]);
2169 writeByte(kCodersUnpackSize);
2171 for (i = 0; i < folderItems.
size(); i++) {
2172 const Folder *folder = folderItems[i];
2173 for (
int j = 0; j < folder->unpackSizes.
size(); j++) {
2174 writeNumber(folder->unpackSizes.
at(j));
2178 QList<bool> unpackCRCsDefined;
2179 QList<quint32> unpackCRCs;
2182 for (i = 0; i < folderItems.
size(); i++) {
2183 const Folder *folder = folderItems[i];
2184 unpackCRCsDefined.
append(folder->unpackCRCDefined);
2185 unpackCRCs.
append(folder->unpackCRC);
2187 writeHashDigests(unpackCRCsDefined, unpackCRCs);
2192void K7Zip::K7ZipPrivate::writeSubStreamsInfo(
const QList<quint64> &unpackSizes,
const QList<bool> &digestsDefined,
const QList<quint32> &digests)
2194 writeByte(kSubStreamsInfo);
2196 for (
int i = 0; i < numUnpackStreamsInFolders.
size(); i++) {
2197 if (numUnpackStreamsInFolders.
at(i) != 1) {
2198 writeByte(kNumUnpackStream);
2199 for (
int j = 0; j < numUnpackStreamsInFolders.
size(); j++) {
2200 writeNumber(numUnpackStreamsInFolders.
at(j));
2206 bool needFlag =
true;
2208 for (
int i = 0; i < numUnpackStreamsInFolders.
size(); i++) {
2209 for (quint32 j = 0; j < numUnpackStreamsInFolders.
at(i); j++) {
2210 if (j + 1 != numUnpackStreamsInFolders.
at(i)) {
2215 writeNumber(unpackSizes[index]);
2221 QList<bool> digestsDefined2;
2222 QList<quint32> digests2;
2224 int digestIndex = 0;
2225 for (
int i = 0; i < folders.
size(); i++) {
2226 int numSubStreams = (int)numUnpackStreamsInFolders.
at(i);
2227 if (numSubStreams == 1 && folders.
at(i)->unpackCRCDefined) {
2230 for (
int j = 0; j < numSubStreams; j++, digestIndex++) {
2231 digestsDefined2.
append(digestsDefined[digestIndex]);
2232 digests2.
append(digests[digestIndex]);
2236 writeHashDigests(digestsDefined2, digests2);
2240QByteArray K7Zip::K7ZipPrivate::encodeStream(QList<quint64> &packSizes, QList<Folder *> &folds)
2243 folder->unpackCRCDefined =
true;
2244 folder->unpackCRC = crc32(0, (Bytef *)(header.
data()), header.
size());
2247 Folder::FolderInfo *info =
new Folder::FolderInfo();
2248 info->numInStreams = 1;
2249 info->numOutStreams = 1;
2250 info->methodID = k_LZMA2;
2252 quint32 dictSize = header.
size();
2253 const quint32 kMinReduceSize = (1 << 16);
2254 if (dictSize < kMinReduceSize) {
2255 dictSize = kMinReduceSize;
2259 for (dict = 0; dict < 40; dict++) {
2260 if (dictSize <= lzma2_dic_size_from_prop(dict)) {
2265 info->properties.
append(dict);
2266 folder->folderInfos.
append(info);
2271 QByteArray encodedData;
2274 QBuffer inBuffer(&enc);
2276 KCompressionDevice flt(&inBuffer,
false, KCompressionDevice::Xz);
2279 KFilterBase *
filter = flt.filterBase();
2283 const int ret = flt.write(header);
2284 if (ret != header.
size()) {
2285 qCDebug(KArchiveLog) <<
"write error write " << ret <<
"expected" << header.
size();
2290 encodedData = inBuffer.
data();
2297void K7Zip::K7ZipPrivate::writeHeader(quint64 &headerOffset)
2299 quint64 packedSize = 0;
2300 for (
int i = 0; i < packSizes.
size(); ++i) {
2301 packedSize += packSizes[i];
2304 headerOffset = packedSize;
2311 writeByte(kMainStreamsInfo);
2312 writePackInfo(0, packSizes, packCRCsDefined, packCRCs);
2314 writeUnpackInfo(folders);
2316 QList<quint64> unpackFileSizes;
2317 QList<bool> digestsDefined;
2318 QList<quint32> digests;
2319 for (
int i = 0; i < fileInfos.
size(); i++) {
2320 const FileInfo *file = fileInfos.
at(i);
2321 if (!file->hasStream) {
2324 unpackFileSizes.
append(file->size);
2325 digestsDefined.
append(file->crcDefined);
2326 digests.
append(file->crc);
2329 writeSubStreamsInfo(unpackSizes, digestsDefined, digests);
2338 writeByte(kFilesInfo);
2339 writeNumber(fileInfos.
size());
2343 QList<bool> emptyStreamVector;
2344 int numEmptyStreams = 0;
2345 for (
int i = 0; i < fileInfos.
size(); i++) {
2346 if (fileInfos.
at(i)->hasStream) {
2347 emptyStreamVector.
append(
false);
2349 emptyStreamVector.
append(
true);
2354 if (numEmptyStreams > 0) {
2355 writeByte(kEmptyStream);
2356 writeNumber(((
unsigned)emptyStreamVector.
size() + 7) / 8);
2357 writeBoolVector(emptyStreamVector);
2359 QList<bool> emptyFileVector;
2360 QList<bool> antiVector;
2361 int numEmptyFiles = 0;
2362 int numAntiItems = 0;
2363 for (
int i = 0; i < fileInfos.
size(); i++) {
2364 const FileInfo *file = fileInfos.
at(i);
2365 if (!file->hasStream) {
2366 emptyFileVector.
append(!file->isDir);
2369 bool isAnti = (i < this->isAnti.size() && this->isAnti[i]);
2370 antiVector.
append(isAnti);
2378 if (numEmptyFiles > 0) {
2379 writeByte(kEmptyFile);
2380 writeNumber(((
unsigned)emptyFileVector.
size() + 7) / 8);
2381 writeBoolVector(emptyFileVector);
2384 if (numAntiItems > 0) {
2386 writeNumber(((
unsigned)antiVector.
size() + 7) / 8);
2387 writeBoolVector(antiVector);
2396 size_t namesDataSize = 0;
2397 for (
int i = 0; i < fileInfos.
size(); i++) {
2398 const QString &
name = fileInfos.
at(i)->path;
2403 if (numDefined > 0) {
2408 writeNumber(namesDataSize);
2410 for (
int i = 0; i < fileInfos.
size(); i++) {
2411 const QString &
name = fileInfos.
at(i)->path;
2414 writeByte((
unsigned char)c);
2415 writeByte((
unsigned char)(c >> 8));
2424 writeUInt64DefVector(mTimes, mTimesDefined, kMTime);
2426 writeUInt64DefVector(startPositions, startPositionsDefined, kStartPos);
2430 QList<bool> boolVector;
2433 for (
int i = 0; i < fileInfos.
size(); i++) {
2434 bool defined = fileInfos.
at(i)->attribDefined;
2435 boolVector.
append(defined);
2441 if (numDefined > 0) {
2442 writeAlignedBoolHeader(boolVector, numDefined, kAttributes, 4);
2443 for (
int i = 0; i < fileInfos.
size(); i++) {
2444 const FileInfo *file = fileInfos.
at(i);
2445 if (file->attribDefined) {
2446 writeUInt32(file->attributes);
2456static void setUInt32(
unsigned char *p, quint32 d)
2458 for (
int i = 0; i < 4; i++, d >>= 8) {
2463static void setUInt64(
unsigned char *p, quint64 d)
2465 for (
int i = 0; i < 8; i++, d >>= 8) {
2466 p[i] = (
unsigned char)d;
2470void K7Zip::K7ZipPrivate::writeStartHeader(
const quint64 nextHeaderSize,
const quint32 nextHeaderCRC,
const quint64 nextHeaderOffset)
2472 unsigned char buf[24];
2473 setUInt64(buf + 4, nextHeaderOffset);
2474 setUInt64(buf + 12, nextHeaderSize);
2475 setUInt32(buf + 20, nextHeaderCRC);
2476 setUInt32(buf, crc32(0, (Bytef *)(buf + 4), 20));
2480void K7Zip::K7ZipPrivate::writeSignature()
2482 unsigned char buf[8];
2483 memcpy(buf, k7zip_signature, 6);
2504 qint64 n = dev->
read(header, 32);
2510 for (
int i = 0; i < 6; ++i) {
2511 if ((
unsigned char)header[i] != k7zip_signature[i]) {
2518 int major = header[6];
2519 int minor = header[7];
2527 quint32 startHeaderCRC = GetUi32(header + 8);
2528 quint64 nextHeaderOffset = GetUi64(header + 12);
2529 quint64 nextHeaderSize = GetUi64(header + 20);
2530 quint32 nextHeaderCRC = GetUi32(header + 28);
2532 quint32 crc = crc32(0, (Bytef *)(header + 0xC), 20);
2534 if (crc != startHeaderCRC) {
2539 if (nextHeaderSize == 0) {
2543 if (nextHeaderSize > (quint64)0xFFFFFFFF) {
2548 if ((qint64)nextHeaderOffset < 0) {
2553 dev->
seek(nextHeaderOffset + 32);
2556 inBuffer.
resize(nextHeaderSize);
2559 if (n != (qint64)nextHeaderSize) {
2560 setErrorString(tr(
"Failed read next header size; should read %1, read %2").arg(nextHeaderSize).arg(n));
2563 d->buffer = inBuffer.
data();
2565 d->end = nextHeaderSize;
2567 d->headerSize = 32 + nextHeaderSize;
2570 crc = crc32(0, (Bytef *)(d->buffer), (quint32)nextHeaderSize);
2572 if (crc != nextHeaderCRC) {
2577 int type = d->readByte();
2579 if (type != kHeader) {
2580 if (type != kEncodedHeader) {
2585 decodedData = d->readAndDecodePackedStreams();
2587 int external = d->readByte();
2588 if (external != 0) {
2589 int dataIndex = (int)d->readNumber();
2590 if (dataIndex < 0) {
2595 d->end = decodedData.
size();
2603 type = d->readByte();
2604 if (type != kHeader) {
2611 type = d->readByte();
2613 if (type == kArchiveProperties) {
2619 if (type == kAdditionalStreamsInfo) {
2625 if (type == kMainStreamsInfo) {
2626 if (!d->readMainStreamsInfo()) {
2627 setErrorString(tr(
"Error while reading main streams information"));
2636 type = d->readByte();
2638 for (
int i = 0; i < d->folders.size(); ++i) {
2639 Folder *folder = d->folders.at(i);
2640 d->unpackSizes.
append(folder->getUnpackSize());
2641 d->digestsDefined.append(folder->unpackCRCDefined);
2642 d->digests.append(folder->unpackCRC);
2650 if (type != kFilesInfo) {
2656 int numFiles = d->readNumber();
2657 for (
int i = 0; i < numFiles; ++i) {
2658 d->fileInfos.append(
new FileInfo);
2664 int numEmptyStreams = 0;
2667 quint64 type = d->readByte();
2672 quint64 size = d->readNumber();
2674 size_t ppp = d->pos;
2676 bool addPropIdToList =
true;
2677 bool isKnownType =
true;
2679 if (type > ((quint32)1 << 30)) {
2680 isKnownType =
false;
2683 case kEmptyStream: {
2684 d->readBoolVector(numFiles, emptyStreamVector);
2685 for (
int i = 0; i < emptyStreamVector.
size(); ++i) {
2686 if (emptyStreamVector[i]) {
2694 d->readBoolVector(numEmptyStreams, emptyFileVector);
2697 d->readBoolVector(numEmptyStreams, antiFileVector);
2700 if (!d->readUInt64DefVector(numFiles, d->cTimes, d->cTimesDefined)) {
2705 if (!d->readUInt64DefVector(numFiles, d->aTimes, d->aTimesDefined)) {
2710 if (!d->readUInt64DefVector(numFiles, d->mTimes, d->mTimesDefined)) {
2716 int external = d->readByte();
2717 if (external != 0) {
2718 int dataIndex = d->readNumber();
2719 if (dataIndex < 0 /*|| dataIndex >= dataVector->Size()*/) {
2720 qCDebug(KArchiveLog) <<
"wrong data index";
2727 for (
int i = 0; i < numFiles; i++) {
2728 name = d->readString();
2729 d->fileInfos.
at(i)->path = name;
2735 d->readBoolVector2(numFiles, attributesAreDefined);
2736 int external = d->readByte();
2737 if (external != 0) {
2738 int dataIndex = d->readNumber();
2739 if (dataIndex < 0) {
2740 qCDebug(KArchiveLog) <<
"wrong data index";
2746 for (
int i = 0; i < numFiles; i++) {
2747 FileInfo *fileInfo = d->fileInfos.at(i);
2748 fileInfo->attribDefined = attributesAreDefined[i];
2749 if (fileInfo->attribDefined) {
2750 fileInfo->attributes = d->readUInt32();
2756 if (!d->readUInt64DefVector(numFiles, d->startPositions, d->startPositionsDefined)) {
2762 for (quint64 i = 0; i < size; i++) {
2763 if (d->readByte() != 0) {
2768 addPropIdToList =
false;
2772 addPropIdToList = isKnownType =
false;
2777 if (addPropIdToList) {
2778 d->fileInfoPopIDs.append(type);
2781 d->skipData(d->readNumber());
2784 bool checkRecordsSize = (major > 0 || minor > 2);
2785 if (checkRecordsSize && d->pos - ppp != size) {
2787 "(checkRecordsSize: %1, d->pos - ppp: %2, size: %3)")
2788 .arg(checkRecordsSize)
2795 int emptyFileIndex = 0;
2798 int numAntiItems = 0;
2800 if (emptyStreamVector.
isEmpty()) {
2801 emptyStreamVector.
fill(
false, numFiles);
2804 if (antiFileVector.
isEmpty()) {
2805 antiFileVector.
fill(
false, numEmptyStreams);
2807 if (emptyFileVector.
isEmpty()) {
2808 emptyFileVector.
fill(
false, numEmptyStreams);
2811 for (
int i = 0; i < numEmptyStreams; i++) {
2812 if (antiFileVector[i]) {
2817 d->outData = d->readAndDecodePackedStreams(
false);
2820 int filesWithoutNames = 0;
2828 for (
int i = 0; i < numFiles; i++) {
2829 FileInfo *fileInfo = d->fileInfos.at(i);
2833 if (fileInfo->path.
isEmpty()) {
2835 filesWithoutNames++;
2836 fileInfo->path = QStringLiteral(
"%1_%2").
arg(defaultBaseName).
arg(filesWithoutNames);
2838 fileInfo->path = defaultBaseName;
2843 fileInfo->hasStream = !emptyStreamVector[i];
2844 if (fileInfo->hasStream) {
2845 fileInfo->isDir =
false;
2847 fileInfo->size = d->unpackSizes[sizeIndex];
2848 fileInfo->crc = d->digests[sizeIndex];
2849 fileInfo->crcDefined = d->digestsDefined[sizeIndex];
2852 fileInfo->isDir = !emptyFileVector[emptyFileIndex];
2853 isAnti = antiFileVector[emptyFileIndex];
2856 fileInfo->crcDefined =
false;
2858 if (numAntiItems != 0) {
2859 d->isAnti.append(isAnti);
2863 bool symlink =
false;
2864 if (fileInfo->attributes & FILE_ATTRIBUTE_UNIX_EXTENSION) {
2865 access = fileInfo->attributes >> 16;
2866 if ((access & QT_STAT_MASK) == QT_STAT_LNK) {
2870 if (fileInfo->isDir) {
2871 access = S_IFDIR | 0755;
2878 if (!fileInfo->isDir) {
2880 oldPos += fileInfo->size;
2887 entryName = fileInfo->path;
2889 entryName = fileInfo->path.
mid(index + 1);
2891 Q_ASSERT(!entryName.
isEmpty());
2894 if (d->mTimesDefined.size() > i && d->mTimesDefined[i]) {
2895 mTime = KArchivePrivate::time_tToDateTime(toTimeT(d->mTimes[i]));
2897 mTime = KArchivePrivate::time_tToDateTime(time(
nullptr));
2900 if (fileInfo->isDir) {
2910 e =
new K7ZipFileEntry(
this,
2922 e =
new K7ZipFileEntry(
this, entryName, access, mTime,
rootDir()->user(),
rootDir()->group(), target, 0, 0,
nullptr);
2933 (void)d->addEntryV2(e);
2955 Folder *folder =
new Folder();
2957 folder->unpackSizes.
clear();
2958 folder->unpackSizes.
append(d->outData.size());
2960 Folder::FolderInfo *info =
new Folder::FolderInfo();
2962 info->numInStreams = 1;
2963 info->numOutStreams = 1;
2964 info->methodID = k_LZMA2;
2966 quint32 dictSize = d->outData.size();
2968 const quint32 kMinReduceSize = (1 << 16);
2969 if (dictSize < kMinReduceSize) {
2970 dictSize = kMinReduceSize;
2975 for (dict = 0; dict < 40; dict++) {
2976 if (dictSize <= lzma2_dic_size_from_prop(dict)) {
2980 info->properties.
append(dict);
2982 folder->folderInfos.
append(info);
2983 d->folders.append(folder);
2987 d->createItemsFromEntities(dir,
QString(), data);
2990 folder->unpackCRCDefined =
true;
2991 folder->unpackCRC = crc32(0, (Bytef *)(d->outData.data()), d->outData.size());
2995 if (!d->outData.isEmpty()) {
3006 static_cast<KXzFilter *
>(filter)->init(
QIODevice::WriteOnly, KXzFilter::LZMA2, info->properties);
3008 const int ret = flt.
write(d->outData);
3009 if (ret != d->outData.size()) {
3015 encodedData = inBuffer.
data();
3020 int numUnpackStream = 0;
3021 for (
int i = 0; i < d->fileInfos.size(); ++i) {
3022 if (d->fileInfos.at(i)->hasStream) {
3026 d->numUnpackStreamsInFolders.
append(numUnpackStream);
3028 quint64 headerOffset;
3029 d->writeHeader(headerOffset);
3036 encodedStream = d->encodeStream(packSizes, folders);
3046 d->writeByte(kEncodedHeader);
3049 d->writePackInfo(headerOffset, packSizes, emptyDefined, emptyCrcs);
3050 d->writeUnpackInfo(folders);
3052 for (
int i = 0; i < packSizes.
size(); i++) {
3053 headerOffset += packSizes.
at(i);
3055 qDeleteAll(folders);
3059 quint64 nextHeaderSize = d->header.size();
3060 quint32 nextHeaderCRC = crc32(0, (Bytef *)(d->header.data()), d->header.size());
3061 quint64 nextHeaderOffset = headerOffset;
3064 d->writeSignature();
3065 d->writeStartHeader(nextHeaderSize, nextHeaderCRC, nextHeaderOffset);
3068 device()->
write(d->header.data(), d->header.size());
3075 d->m_currentFile->setSize(size);
3076 d->m_currentFile =
nullptr;
3083 if (!d->m_currentFile) {
3088 if (d->m_currentFile->position() == d->outData.size()) {
3089 d->outData.append(data, size);
3091 d->outData.remove(d->m_currentFile->position(), d->m_currentFile->size());
3092 d->outData.insert(d->m_currentFile->position(), data, size);
3108 setErrorString(tr(
"Application error: 7-Zip file must be open before being written into"));
3109 qCWarning(KArchiveLog) <<
"doPrepareWriting failed: !isOpen()";
3114 setErrorString(tr(
"Application error: attempted to write into non-writable 7-Zip file"));
3115 qCWarning(KArchiveLog) <<
"doPrepareWriting failed: !(mode() & QIODevice::WriteOnly)";
3135 new K7ZipFileEntry(
this,
fileName, perm, mtime, user, group,
QString() , d->outData.
size(), 0 , d->outData);
3139 d->m_entryList << e;
3140 d->m_currentFile = e;
3158 setErrorString(tr(
"Application error: 7-Zip file must be open before being written into"));
3159 qCWarning(KArchiveLog) <<
"doWriteDir failed: !isOpen()";
3180 dirName = name.
mid(i + 1);
3198 setErrorString(tr(
"Application error: 7-Zip file must be open before being written into"));
3199 qCWarning(KArchiveLog) <<
"doWriteSymLink failed: !isOpen()";
3204 setErrorString(tr(
"Application error: attempted to write into non-writable 7-Zip file"));
3205 qCWarning(KArchiveLog) <<
"doWriteSymLink failed: !(mode() & QIODevice::WriteOnly)";
3221 K7ZipFileEntry *e =
new K7ZipFileEntry(
this,
fileName, perm, mtime, user, group, target, 0, 0,
nullptr);
3222 d->outData.append(encodedTarget);
3228 d->m_entryList << e;
3233void K7Zip::virtual_hook(
int id,
void *data)
3235 KArchive::virtual_hook(
id, data);
A class for reading / writing p7zip archives.
bool doWriteDir(const QString &name, const QString &user, const QString &group, mode_t perm, const QDateTime &atime, const QDateTime &mtime, const QDateTime &ctime) override
Reimplemented from KArchive.
bool doWriteData(const char *data, qint64 size) override
Reimplemented from KArchive.
void setPassword(const QString &password)
Sets the password to use for encrypted archives.
bool doFinishWriting(qint64 size) override
Reimplemented from KArchive.
bool openArchive(QIODevice::OpenMode mode) override
Opens the archive for reading.
bool closeArchive() override
Closes the archive.
bool doWriteSymLink(const QString &name, const QString &target, const QString &user, const QString &group, mode_t perm, const QDateTime &atime, const QDateTime &mtime, const QDateTime &ctime) override
Reimplemented from KArchive.
~K7Zip() override
If the archive is still opened, then it will be closed automatically by the destructor.
K7Zip(const QString &filename)
Creates an instance that operates on the given filename using the compression filter associated to gi...
bool passwordNeeded() const
Whether the archive needs a password to be opened.
bool doPrepareWriting(const QString &name, const QString &user, const QString &group, qint64 size, mode_t perm, const QDateTime &atime, const QDateTime &mtime, const QDateTime &ctime) override
Reimplemented from KArchive.
Represents a directory entry in a KArchive.
bool addEntryV2(KArchiveEntry *)
const KArchiveEntry * entry(const QString &name) const
Returns the entry in the archive with the given name.
A base class for entries in an KArchive.
mode_t permissions() const
The permissions and mode flags as returned by the stat() function in st_mode.
QString user() const
User who created the file.
virtual bool isDirectory() const
Checks whether the entry is a directory.
QDateTime date() const
Creation date of the file.
QString group() const
Group of the user who created the file.
QString name() const
Name of the file without path.
QString symLinkTarget() const
Symlink if there is one.
virtual bool isFile() const
Checks whether the entry is a file.
Represents a file entry in a KArchive.
qint64 size() const
Size of the data.
qint64 position() const
Position of the data in the [uncompressed] archive.
QIODevice * device() const
The underlying device.
virtual bool close()
Closes the archive.
virtual KArchiveDirectory * rootDir()
Retrieves or create the root directory.
KArchive(const QString &fileName)
Base constructor (protected since this is a pure virtual class).
const KArchiveDirectory * directory() const
If an archive is opened for reading, then the contents of the archive can be accessed via this functi...
KArchiveDirectory * findOrCreate(const QString &path)
Ensures that path exists, create otherwise.
QIODevice::OpenMode mode() const
Returns the mode in which the archive was opened.
QString fileName() const
The name of the archive file, as passed to the constructor that takes a fileName, or an empty string ...
bool isOpen() const
Checks whether the archive is open.
void setErrorString(const QString &errorStr)
Sets error description.
A class for reading and writing compressed data onto a device (e.g.
static KFilterBase * filterForCompressionType(CompressionType type)
Call this to create the appropriate filter for the CompressionType named type.
void close() override
Close after reading or writing.
bool open(QIODevice::OpenMode mode) override
Open for reading or writing.
This is the base class for compression filters such as gzip and bzip2.
Q_SCRIPTABLE Q_NOREPLY void start()
Type type(const QSqlDatabase &db)
KIOCORE_EXPORT SimpleJob * symlink(const QString &target, const QUrl &dest, JobFlags flags=DefaultFlags)
QString normalize(QStringView str)
QString path(const QString &relativePath)
KIOCORE_EXPORT QString dir(const QString &fileClass)
QString name(StandardAction id)
QAction * clear(const QObject *recvr, const char *slot, QObject *parent)
const QList< QKeySequence > & end()
const QByteArray & data() const const
QByteArray & append(QByteArrayView data)
const char * constData() const const
bool isEmpty() const const
QByteArray mid(qsizetype pos, qsizetype len) const const
void reserve(qsizetype size)
void resize(qsizetype newSize, char c)
qsizetype size() const const
qint64 toSecsSinceEpoch() const const
QString cleanPath(const QString &path)
QString decodeName(const QByteArray &localFileName)
QByteArray encodeName(const QString &fileName)
QString completeBaseName() const const
QByteArray read(qint64 maxSize)
virtual bool seek(qint64 pos)
qint64 write(const QByteArray &data)
void append(QList< T > &&value)
const_reference at(qsizetype i) const const
QList< T > & fill(parameter_type value, qsizetype size)
bool isEmpty() const const
void reserve(qsizetype size)
qsizetype size() const const
QString arg(Args &&... args) const const
const QChar at(qsizetype position) const const
bool endsWith(QChar c, Qt::CaseSensitivity cs) const const
bool isEmpty() const const
qsizetype lastIndexOf(QChar ch, Qt::CaseSensitivity cs) const const
QString left(qsizetype n) const const
qsizetype length() const const
QString mid(qsizetype position, qsizetype n) const const
QString & remove(QChar ch, Qt::CaseSensitivity cs)
qsizetype size() const const
QByteArray toLatin1() const const
QByteArray toUtf8() const const
QFuture< void > filter(QThreadPool *pool, Sequence &sequence, KeepFunctor &&filterFunction)