KArchive

k7zip.cpp
1/* This file is part of the KDE libraries
2 SPDX-FileCopyrightText: 2011 Mario Bensi <mbensi@ipsquad.net>
3
4 SPDX-License-Identifier: LGPL-2.0-or-later
5*/
6
7#include "k7zip.h"
8#include "karchive_p.h"
9#include "loggingcategory.h"
10
11#include <QBuffer>
12#include <QCryptographicHash>
13#include <QDebug>
14#include <QDir>
15#include <QFile>
16#include <QTimeZone>
17#include <qplatformdefs.h>
18
19#include "kcompressiondevice.h"
20#include "klimitediodevice_p.h"
21#include <kfilterbase.h>
22#include <kxzfilter.h>
23
24#include "zlib.h"
25#include <memory>
26#include <time.h> // time()
27
28#if HAVE_OPENSSL_SUPPORT
29#include <openssl/evp.h>
30#endif
31
32#ifndef QT_STAT_LNK
33#define QT_STAT_LNK 0120000
34#endif // QT_STAT_LNK
35
36////////////////////////////////////////////////////////////////////////
37/////////////////////////// K7Zip //////////////////////////////////////
38////////////////////////////////////////////////////////////////////////
39
40#define BUFFER_SIZE 8 * 1024
41
42static const unsigned char k7zip_signature[6] = {'7', 'z', 0xBC, 0xAF, 0x27, 0x1C};
43// static const unsigned char XZ_HEADER_MAGIC[6] = { 0xFD, '7', 'z', 'X', 'Z', 0x00 };
44
45/* clang-format off */
46static QChar GetUi16(const char *p)
47{
48 return QChar(static_cast<unsigned char>(p[0])
49 | (static_cast<unsigned char>(p[1]) << 8));
50}
51
52static quint32 GetUi32(const char *p)
53{
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));
58}
59
60static quint64 GetUi64(const char *p)
61{
62 return (GetUi32(p)
63 | (static_cast<quint64>(GetUi32(p + 4)) << 32));
64}
65
66static quint32 lzma2_dic_size_from_prop(int p)
67{
68 return ((static_cast<quint32>(2) | (p & 1)) << ((p / 2) + 11));
69}
70
71/* clang-format on*/
72
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 /* trick for Unix */
87
88enum HeaderType {
89 kEnd,
90
91 kHeader,
92
93 kArchiveProperties,
94
95 kAdditionalStreamsInfo,
96 kMainStreamsInfo,
97 kFilesInfo,
98
99 kPackInfo,
100 kUnpackInfo,
101 kSubStreamsInfo,
102
103 kSize,
104 kCRC,
105
106 kFolder,
107
108 kCodersUnpackSize,
109 kNumUnpackStream,
110
111 kEmptyStream,
112 kEmptyFile,
113 kAnti,
114
115 kName,
116 kCTime,
117 kATime,
118 kMTime,
119 kAttributes,
120 kComment,
121
122 kEncodedHeader,
123
124 kStartPos,
125 kDummy,
126};
127
128// Method ID
129// static const quint64 k_Copy = 0x00;
130// static const quint64 k_Delta = 0x03;
131// static const quint64 k_x86 = 0x04; //BCJ
132// static const quint64 k_PPC = 0x05; // BIG Endian
133// static const quint64 k_IA64 = 0x06;
134// static const quint64 k_ARM = 0x07; // little Endian
135// static const quint64 k_ARM_Thumb = 0x08; // little Endian
136// static const quint64 k_SPARC = 0x09;
137static const quint64 k_LZMA2 = 0x21;
138// static const quint64 k_Swap2 = 0x020302;
139// static const quint64 k_Swap4 = 0x020304;
140static const quint64 k_LZMA = 0x030101;
141static const quint64 k_BCJ = 0x03030103;
142static const quint64 k_BCJ2 = 0x0303011B;
143// static const quint64 k_7zPPC = 0x03030205;
144// static const quint64 k_Alpha = 0x03030301;
145// static const quint64 k_7zIA64 = 0x03030401;
146// static const quint64 k_7zARM = 0x03030501;
147// static const quint64 k_M68 = 0x03030605; //Big Endian
148// static const quint64 k_ARMT = 0x03030701;
149// static const quint64 k_7zSPARC = 0x03030805;
150static const quint64 k_PPMD = 0x030401;
151// static const quint64 k_Experimental = 0x037F01;
152// static const quint64 k_Shrink = 0x040101;
153// static const quint64 k_Implode = 0x040106;
154// static const quint64 k_Deflate = 0x040108;
155// static const quint64 k_Deflate64 = 0x040109;
156// static const quint64 k_Imploding = 0x040110;
157// static const quint64 k_Jpeg = 0x040160;
158// static const quint64 k_WavPack = 0x040161;
159// static const quint64 k_PPMd = 0x040162;
160// static const quint64 k_wzAES = 0x040163;
161static const quint64 k_BZip2 = 0x040202;
162// static const quint64 k_Rar15 = 0x040301;
163// static const quint64 k_Rar20 = 0x040302;
164// static const quint64 k_Rar29 = 0x040303;
165// static const quint64 k_Arj = 0x040401; //1 2 3
166// static const quint64 k_Arj4 = 0x040402;
167// static const quint64 k_Z = 0x0405;
168// static const quint64 k_Lzh = 0x0406;
169// static const quint64 k_Cab = 0x0408;
170// static const quint64 k_DeflateNSIS = 0x040901;
171// static const quint64 k_Bzip2NSIS = 0x040902;
172static const quint64 k_AES = 0x06F10701;
173
174/**
175 * A K7ZipFileEntry represents a file in a 7zip archive.
176 */
177class K7ZipFileEntry : public KArchiveFile
178{
179public:
180 K7ZipFileEntry(K7Zip *zip,
181 const QString &name,
182 int access,
183 const QDateTime &date,
184 const QString &user,
185 const QString &group,
186 const QString &symlink,
187 qint64 pos,
188 qint64 size,
189 const QByteArray &data);
190
191 ~K7ZipFileEntry() override;
192
193 /**
194 * @return the content of this file.
195 * Call data() with care (only once per file), this data isn't cached.
196 */
197 QByteArray data() const override;
198
199 /**
200 * This method returns QIODevice (internal class: KLimitedIODevice)
201 * on top of the underlying QIODevice. This is obviously for reading only.
202 *
203 * WARNING: Note that the ownership of the device is being transferred to the caller,
204 * who will have to delete it.
205 *
206 * The returned device auto-opens (in readonly mode), no need to open it.
207 * @return the QIODevice of the file
208 */
209 QIODevice *createDevice() const override;
210
211private:
212 const QByteArray m_data;
213 QBuffer *m_buffer;
214};
215
216K7ZipFileEntry::K7ZipFileEntry(K7Zip *zip,
217 const QString &name,
218 int access,
219 const QDateTime &date,
220 const QString &user,
221 const QString &group,
222 const QString &symlink,
223 qint64 pos,
224 qint64 size,
225 const QByteArray &data)
226 : KArchiveFile(zip, name, access, date, user, group, symlink, pos, size)
227 , m_data(data)
228 , m_buffer(new QBuffer)
229{
230 m_buffer->setData(m_data);
231 m_buffer->open(QIODevice::ReadOnly);
232}
233
234K7ZipFileEntry::~K7ZipFileEntry()
235{
236 delete m_buffer;
237}
238
239QByteArray K7ZipFileEntry::data() const
240{
241 return m_data.mid(position(), size());
242}
243
244QIODevice *K7ZipFileEntry::createDevice() const
245{
246 return new KLimitedIODevice(m_buffer, position(), size());
247}
248
249class FileInfo
250{
251public:
252 FileInfo()
253 : size(0)
254 , attributes(0)
255 , crc(0)
256 , attribDefined(false)
257 , crcDefined(false)
258 , hasStream(false)
259 , isDir(false)
260 {
261 }
262
263 QString path;
264 quint64 size;
265 quint32 attributes;
266 quint32 crc;
267 bool attribDefined;
268 bool crcDefined;
269 bool hasStream;
270 bool isDir;
271};
272
273class Folder
274{
275public:
276 class FolderInfo
277 {
278 public:
279 FolderInfo()
280 : numInStreams(0)
281 , numOutStreams(0)
282 , methodID(0)
283 {
284 }
285
286 bool isSimpleCoder() const
287 {
288 return (numInStreams == 1) && (numOutStreams == 1);
289 }
290
291 int numInStreams;
292 int numOutStreams;
293 QList<unsigned char> properties;
294 quint64 methodID;
295 };
296
297 Folder()
298 : unpackCRCDefined(false)
299 , unpackCRC(0)
300 {
301 }
302
303 ~Folder()
304 {
305 qDeleteAll(folderInfos);
306 }
307
308 Q_DISABLE_COPY(Folder)
309
310 quint64 getUnpackSize() const
311 {
312 if (unpackSizes.isEmpty()) {
313 return 0;
314 }
315 for (int i = unpackSizes.size() - 1; i >= 0; i--) {
316 if (findBindPairForOutStream(i) < 0) {
317 return unpackSizes.at(i);
318 }
319 }
320 return 0;
321 }
322
323 int getNumOutStreams() const
324 {
325 int result = 0;
326 for (int i = 0; i < folderInfos.size(); i++) {
327 result += folderInfos.at(i)->numOutStreams;
328 }
329 return result;
330 }
331
332 quint32 getCoderInStreamIndex(quint32 coderIndex) const
333 {
334 quint32 streamIndex = 0;
335 for (quint32 i = 0; i < coderIndex; i++) {
336 streamIndex += folderInfos.at(i)->numInStreams;
337 }
338 return streamIndex;
339 }
340
341 quint32 getCoderOutStreamIndex(quint32 coderIndex) const
342 {
343 quint32 streamIndex = 0;
344 for (quint32 i = 0; i < coderIndex; i++) {
345 streamIndex += folderInfos.at(i)->numOutStreams;
346 }
347 return streamIndex;
348 }
349
350 int findBindPairForInStream(size_t inStreamIndex) const
351 {
352 for (int i = 0; i < inIndexes.size(); i++) {
353 if (inIndexes[i] == inStreamIndex) {
354 return i;
355 }
356 }
357 return -1;
358 }
359
360 int findBindPairForOutStream(size_t outStreamIndex) const
361 {
362 for (int i = 0; i < outIndexes.size(); i++) {
363 if (outIndexes[i] == outStreamIndex) {
364 return i;
365 }
366 }
367 return -1;
368 }
369
370 int findPackStreamArrayIndex(size_t inStreamIndex) const
371 {
372 for (int i = 0; i < packedStreams.size(); i++) {
373 if (packedStreams[i] == inStreamIndex) {
374 return i;
375 }
376 }
377 return -1;
378 }
379
380 void findInStream(quint32 streamIndex, quint32 &coderIndex, quint32 &coderStreamIndex) const
381 {
382 for (coderIndex = 0; coderIndex < (quint32)folderInfos.size(); coderIndex++) {
383 quint32 curSize = folderInfos[coderIndex]->numInStreams;
384 if (streamIndex < curSize) {
385 coderStreamIndex = streamIndex;
386 return;
387 }
388 streamIndex -= curSize;
389 }
390 }
391
392 void findOutStream(quint32 streamIndex, quint32 &coderIndex, quint32 &coderStreamIndex) const
393 {
394 for (coderIndex = 0; coderIndex < (quint32)folderInfos.size(); coderIndex++) {
395 quint32 curSize = folderInfos[coderIndex]->numOutStreams;
396 if (streamIndex < curSize) {
397 coderStreamIndex = streamIndex;
398 return;
399 }
400 streamIndex -= curSize;
401 }
402 }
403
404 bool isEncrypted() const
405 {
406 for (int i = folderInfos.size() - 1; i >= 0; i--) {
407 if (folderInfos.at(i)->methodID == k_AES) {
408 return true;
409 }
410 }
411 return false;
412 }
413
414 // bool CheckStructure() const;
415
416 bool unpackCRCDefined;
417 quint32 unpackCRC;
418 QList<FolderInfo *> folderInfos;
419 QList<quint64> inIndexes;
420 QList<quint64> outIndexes;
421 QList<quint64> packedStreams;
422 QList<quint64> unpackSizes;
423};
424
425class Q_DECL_HIDDEN K7Zip::K7ZipPrivate
426{
427public:
428 K7ZipPrivate(K7Zip *parent)
429 : q(parent)
430 , packPos(0)
431 , numPackStreams(0)
432 , buffer(nullptr)
433 , pos(0)
434 , end(0)
435 , headerSize(0)
436 , countSize(0)
437 , m_currentFile(nullptr)
438 {
439 }
440
441 ~K7ZipPrivate()
442 {
443 qDeleteAll(folders);
444 qDeleteAll(fileInfos);
445 }
446
447 K7Zip *q;
448
449 QList<bool> packCRCsDefined;
450 QList<quint32> packCRCs;
451 QList<quint64> numUnpackStreamsInFolders;
452
453 QList<Folder *> folders;
454 QList<FileInfo *> fileInfos;
455 // File information
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;
465
466 quint64 packPos;
467 quint64 numPackStreams;
468 QList<quint64> packSizes;
469 QList<quint64> unpackSizes;
470 QList<bool> digestsDefined;
471 QList<quint32> digests;
472
473 QList<bool> isAnti;
474
475 QString password;
476
477 const char *buffer;
478 quint64 pos;
479 quint64 end;
480 quint64 headerSize;
481 quint64 countSize;
482
483 // Write
484 QByteArray header;
485 QByteArray outData; // Store data in this buffer before compress and write in archive.
486 K7ZipFileEntry *m_currentFile;
487 QList<KArchiveEntry *> m_entryList;
488
489 void clear()
490 {
491 packCRCsDefined.clear();
492 packCRCs.clear();
493 numUnpackStreamsInFolders.clear();
494 qDeleteAll(folders);
495 folders.clear();
496 qDeleteAll(fileInfos);
497 fileInfos.clear();
498 cTimesDefined.clear();
499 cTimes.clear();
500 aTimesDefined.clear();
501 aTimes.clear();
502 mTimesDefined.clear();
503 mTimes.clear();
504 startPositionsDefined.clear();
505 startPositions.clear();
506 fileInfoPopIDs.clear();
507 packSizes.clear();
508 unpackSizes.clear();
509 digestsDefined.clear();
510 digests.clear();
511 isAnti.clear();
512
513 buffer = nullptr;
514 pos = 0;
515 end = 0;
516 headerSize = 0;
517 countSize = 0;
518 }
519
520 // Read
521 int readByte();
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);
532
533 Folder *folderItem();
534 bool readMainStreamsInfo();
535 bool readPackInfo();
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);
540
541 // Write
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);
559};
560
563 , d(new K7ZipPrivate(this))
564{
565}
566
568 : KArchive(dev)
569 , d(new K7ZipPrivate(this))
570{
571 Q_ASSERT(dev);
572}
573
575{
576 if (isOpen()) {
577 close();
578 }
579
580 delete d;
581}
582
583void K7Zip::setPassword(const QString &password) {
584 d->password = password;
585}
586
588{
589 for (int i = 0; i < d->folders.size(); i++) {
590 if (d->folders.at(i)->isEncrypted()) {
591 return true;
592 }
593 }
594
595 return false;
596}
597
598int K7Zip::K7ZipPrivate::readByte()
599{
600 if (!buffer || pos + 1 > end) {
601 return -1;
602 }
603 return buffer[pos++];
604}
605
606quint32 K7Zip::K7ZipPrivate::readUInt32()
607{
608 if (!buffer || (quint64)(pos + 4) > end) {
609 qCDebug(KArchiveLog) << "error size";
610 return 0;
611 }
612
613 quint32 res = GetUi32(buffer + pos);
614 pos += 4;
615 return res;
616}
617
618quint64 K7Zip::K7ZipPrivate::readUInt64()
619{
620 if (!buffer || (quint64)(pos + 8) > end) {
621 qCDebug(KArchiveLog) << "error size";
622 return 0;
623 }
624
625 quint64 res = GetUi64(buffer + pos);
626 pos += 8;
627 return res;
628}
629
630quint64 K7Zip::K7ZipPrivate::readNumber()
631{
632 if (!buffer || pos >= end) {
633 return 0;
634 }
635
636 unsigned char firstByte = buffer[pos++];
637 unsigned char mask = 0x80;
638 quint64 value = 0;
639 for (int i = 0; i < 8; i++) {
640 if ((firstByte & mask) == 0) {
641 quint64 highPart = firstByte & (mask - 1);
642 value += (highPart << (i * 8));
643 return value;
644 }
645
646 if (pos >= end) {
647 return 0;
648 }
649
650 value |= ((unsigned char)buffer[pos++] << (8 * i));
651 mask >>= 1;
652 }
653 return value;
654}
655
656QString K7Zip::K7ZipPrivate::readString()
657{
658 if (!buffer) {
659 return QString();
660 }
661
662 const char *buf = buffer + pos;
663 size_t rem = (end - pos) / 2 * 2;
664 {
665 size_t i;
666 for (i = 0; i < rem; i += 2) {
667 if (buf[i] == 0 && buf[i + 1] == 0) {
668 break;
669 }
670 }
671 if (i == rem) {
672 qCDebug(KArchiveLog) << "read string error";
673 return QString();
674 }
675 rem = i;
676 }
677
678 int len = (int)(rem / 2);
679 if (len < 0 || (size_t)len * 2 != rem) {
680 qCDebug(KArchiveLog) << "read string unsupported";
681 return QString();
682 }
683
684 QString p;
685 for (int i = 0; i < len; i++, buf += 2) {
686 p += GetUi16(buf);
687 }
688
689 pos += rem + 2;
690 return p;
691}
692
693void K7Zip::K7ZipPrivate::skipData(int size)
694{
695 if (!buffer || pos + size > end) {
696 return;
697 }
698 pos += size;
699}
700
701bool K7Zip::K7ZipPrivate::findAttribute(int attribute)
702{
703 if (!buffer) {
704 return false;
705 }
706
707 for (;;) {
708 int type = readByte();
709 if (type == attribute) {
710 return true;
711 }
712 if (type == kEnd) {
713 return false;
714 }
715 skipData(readNumber());
716 }
717}
718
719void K7Zip::K7ZipPrivate::readBoolVector(int numItems, QList<bool> &v)
720{
721 if (!buffer) {
722 return;
723 }
724
725 unsigned char b = 0;
726 unsigned char mask = 0;
727 for (int i = 0; i < numItems; i++) {
728 if (mask == 0) {
729 b = readByte();
730 mask = 0x80;
731 }
732 v.append((b & mask) != 0);
733 mask >>= 1;
734 }
735}
736
737void K7Zip::K7ZipPrivate::readBoolVector2(int numItems, QList<bool> &v)
738{
739 if (!buffer) {
740 return;
741 }
742
743 int allAreDefined = readByte();
744 if (allAreDefined == 0) {
745 readBoolVector(numItems, v);
746 return;
747 }
748
749 for (int i = 0; i < numItems; i++) {
750 v.append(true);
751 }
752}
753
754void K7Zip::K7ZipPrivate::readHashDigests(int numItems, QList<bool> &digestsDefined, QList<quint32> &digests)
755{
756 if (!buffer) {
757 return;
758 }
759
760 readBoolVector2(numItems, digestsDefined);
761 for (int i = 0; i < numItems; i++) {
762 quint32 crc = 0;
763 if (digestsDefined[i]) {
764 crc = GetUi32(buffer + pos);
765 pos += 4;
766 }
767 digests.append(crc);
768 }
769}
770
771Folder *K7Zip::K7ZipPrivate::folderItem()
772{
773 if (!buffer) {
774 return nullptr;
775 }
776
777 Folder *folder = new Folder;
778 int numCoders = readNumber();
779
780 quint64 numInStreamsTotal = 0;
781 quint64 numOutStreamsTotal = 0;
782 for (int i = 0; i < numCoders; ++i) {
783 // BYTE
784 // {
785 // 0:3 CodecIdSize
786 // 4: Is Complex Coder
787 // 5: There Are Attributes
788 // 6: Reserved
789 // 7: There are more alternative methods. (Not used
790 // anymore, must be 0).
791 // }
792 unsigned char coderInfo = readByte();
793 int codecIdSize = (coderInfo & 0xF);
794 if (codecIdSize > 8) {
795 qCDebug(KArchiveLog) << "unsupported codec id size";
796 delete folder;
797 return nullptr;
798 }
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();
803 }
804
805 int id = 0;
806 for (int j = 0; j < codecIdSize; j++) {
807 id |= codecID[codecIdSize - 1 - j] << (8 * j);
808 }
809 info->methodID = id;
810
811 // if (Is Complex Coder)
812 if ((coderInfo & 0x10) != 0) {
813 info->numInStreams = readNumber();
814 info->numOutStreams = readNumber();
815 } else {
816 info->numInStreams = 1;
817 info->numOutStreams = 1;
818 }
819
820 // if (There Are Attributes)
821 if ((coderInfo & 0x20) != 0) {
822 int propertiesSize = readNumber();
823 for (int i = 0; i < propertiesSize; ++i) {
824 info->properties.append(readByte());
825 }
826 }
827
828 if ((coderInfo & 0x80) != 0) {
829 qCDebug(KArchiveLog) << "unsupported";
830 delete info;
831 delete folder;
832 return nullptr;
833 }
834
835 numInStreamsTotal += info->numInStreams;
836 numOutStreamsTotal += info->numOutStreams;
837 folder->folderInfos.append(info);
838 }
839
840 int numBindPairs = numOutStreamsTotal - 1;
841 for (int i = 0; i < numBindPairs; i++) {
842 folder->inIndexes.append(readNumber());
843 folder->outIndexes.append(readNumber());
844 }
845
846 int numPackedStreams = numInStreamsTotal - numBindPairs;
847 if (numPackedStreams > 1) {
848 for (int i = 0; i < numPackedStreams; ++i) {
849 folder->packedStreams.append(readNumber());
850 }
851 } else {
852 if (numPackedStreams == 1) {
853 for (quint64 i = 0; i < numInStreamsTotal; i++) {
854 if (folder->findBindPairForInStream(i) < 0) {
855 folder->packedStreams.append(i);
856 break;
857 }
858 }
859 if (folder->packedStreams.size() != 1) {
860 delete folder;
861 return nullptr;
862 }
863 }
864 }
865 return folder;
866}
867
868bool K7Zip::K7ZipPrivate::readUInt64DefVector(int numFiles, QList<quint64> &values, QList<bool> &defined)
869{
870 if (!buffer) {
871 return false;
872 }
873
874 readBoolVector2(numFiles, defined);
875
876 int external = readByte();
877 if (external != 0) {
878 int dataIndex = readNumber();
879 if (dataIndex < 0 /*|| dataIndex >= dataVector->Size()*/) {
880 qCDebug(KArchiveLog) << "wrong data index";
881 return false;
882 }
883
884 // TODO : go to the new index
885 }
886
887 for (int i = 0; i < numFiles; i++) {
888 quint64 t = 0;
889 if (defined[i]) {
890 t = readUInt64();
891 }
892 values.append(t);
893 }
894 return true;
895}
896
897bool K7Zip::K7ZipPrivate::readPackInfo()
898{
899 if (!buffer) {
900 return false;
901 }
902
903 packPos = readNumber();
904 numPackStreams = readNumber();
905 packSizes.clear();
906
907 packCRCsDefined.clear();
908 packCRCs.clear();
909
910 if (!findAttribute(kSize)) {
911 qCDebug(KArchiveLog) << "kSize not found";
912 return false;
913 }
914
915 for (quint64 i = 0; i < numPackStreams; ++i) {
916 packSizes.append(readNumber());
917 }
918
919 for (;;) {
920 int type = readByte();
921 if (type == kEnd) {
922 break;
923 }
924 if (type == kCRC) {
925 readHashDigests(numPackStreams, packCRCsDefined, packCRCs);
926 continue;
927 }
928 skipData(readNumber());
929 }
930
931 if (packCRCs.isEmpty()) {
932 for (quint64 i = 0; i < numPackStreams; ++i) {
933 packCRCsDefined.append(false);
934 packCRCs.append(0);
935 }
936 }
937 return true;
938}
939
940bool K7Zip::K7ZipPrivate::readUnpackInfo()
941{
942 if (!buffer) {
943 return false;
944 }
945
946 if (!findAttribute(kFolder)) {
947 qCDebug(KArchiveLog) << "kFolder not found";
948 return false;
949 }
950
951 int numFolders = readNumber();
952 qDeleteAll(folders);
953 folders.clear();
954 int external = readByte();
955 switch (external) {
956 case 0: {
957 for (int i = 0; i < numFolders; ++i) {
958 folders.append(folderItem());
959 }
960 break;
961 }
962 case 1: {
963 int dataIndex = readNumber();
964 if (dataIndex < 0 /*|| dataIndex >= dataVector->Size()*/) {
965 qCDebug(KArchiveLog) << "wrong data index";
966 }
967 // TODO : go to the new index
968 break;
969 }
970 default:
971 qCDebug(KArchiveLog) << "external error";
972 return false;
973 }
974
975 if (!findAttribute(kCodersUnpackSize)) {
976 qCDebug(KArchiveLog) << "kCodersUnpackSize not found";
977 return false;
978 }
979
980 for (int i = 0; i < numFolders; ++i) {
981 Folder *folder = folders.at(i);
982 int numOutStreams = folder->getNumOutStreams();
983 for (int j = 0; j < numOutStreams; ++j) {
984 folder->unpackSizes.append(readNumber());
985 }
986 }
987
988 for (;;) {
989 int type = readByte();
990 if (type == kEnd) {
991 break;
992 }
993 if (type == kCRC) {
994 QList<bool> crcsDefined;
995 QList<quint32> crcs;
996 readHashDigests(numFolders, crcsDefined, crcs);
997 for (int i = 0; i < numFolders; i++) {
998 Folder *folder = folders.at(i);
999 folder->unpackCRCDefined = crcsDefined[i];
1000 folder->unpackCRC = crcs[i];
1001 }
1002 continue;
1003 }
1004 skipData(readNumber());
1005 }
1006 return true;
1007}
1008
1009bool K7Zip::K7ZipPrivate::readSubStreamsInfo()
1010{
1011 if (!buffer) {
1012 return false;
1013 }
1014
1015 numUnpackStreamsInFolders.clear();
1016
1017 int type;
1018 for (;;) {
1019 type = readByte();
1020 if (type == kNumUnpackStream) {
1021 for (int i = 0; i < folders.size(); i++) {
1022 numUnpackStreamsInFolders.append(readNumber());
1023 }
1024 continue;
1025 }
1026 if (type == kCRC || type == kSize) {
1027 break;
1028 }
1029 if (type == kEnd) {
1030 break;
1031 }
1032 skipData(readNumber());
1033 }
1034
1035 if (numUnpackStreamsInFolders.isEmpty()) {
1036 for (int i = 0; i < folders.size(); i++) {
1037 numUnpackStreamsInFolders.append(1);
1038 }
1039 }
1040
1041 for (int i = 0; i < numUnpackStreamsInFolders.size(); i++) {
1042 quint64 numSubstreams = numUnpackStreamsInFolders.at(i);
1043 if (numSubstreams == 0) {
1044 continue;
1045 }
1046 quint64 sum = 0;
1047 for (quint64 j = 1; j < numSubstreams; j++) {
1048 if (type == kSize) {
1049 int size = readNumber();
1050 unpackSizes.append(size);
1051 sum += size;
1052 }
1053 }
1054 unpackSizes.append(folders.at(i)->getUnpackSize() - sum);
1055 }
1056
1057 if (type == kSize) {
1058 type = readByte();
1059 }
1060
1061 int numDigests = 0;
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;
1067 }
1068 numDigestsTotal += numSubstreams;
1069 }
1070
1071 for (;;) {
1072 if (type == kCRC) {
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);
1083 } else {
1084 for (quint64 j = 0; j < numSubstreams; j++, digestIndex++) {
1085 digestsDefined.append(digestsDefined2[digestIndex]);
1086 digests.append(digests2[digestIndex]);
1087 }
1088 }
1089 }
1090 } else if (type == kEnd) {
1091 if (digestsDefined.isEmpty()) {
1092 for (int i = 0; i < numDigestsTotal; i++) {
1093 digestsDefined.append(false);
1094 digests.append(0);
1095 }
1096 }
1097
1098 break;
1099 } else {
1100 skipData(readNumber());
1101 }
1102
1103 type = readByte();
1104 }
1105 return true;
1106}
1107
1108#define TICKSPERSEC 10000000
1109#define TICKSPERMSEC 10000
1110#define SECSPERDAY 86400
1111#define SECSPERHOUR 3600
1112#define SECSPERMIN 60
1113#define EPOCHWEEKDAY 1 /* Jan 1, 1601 was Monday */
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)
1119
1120static uint toTimeT(const long long liTime)
1121{
1122 long long time = liTime / TICKSPERSEC;
1123
1124 /* The native version of RtlTimeToTimeFields does not take leap seconds
1125 * into account */
1126
1127 /* Split the time into days and seconds within the day */
1128 long int days = time / SECSPERDAY;
1129 int secondsInDay = time % SECSPERDAY;
1130
1131 /* compute time of day */
1132 short hour = (short)(secondsInDay / SECSPERHOUR);
1133 secondsInDay = secondsInDay % SECSPERHOUR;
1134 short minute = (short)(secondsInDay / SECSPERMIN);
1135 short second = (short)(secondsInDay % SECSPERMIN);
1136
1137 /* compute year, month and day of month. */
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;
1143 /* the result is based on a year starting on March.
1144 * To convert take 12 from January and February and
1145 * increase the year by one. */
1146
1147 short month;
1148 short year;
1149 if (months < 14) {
1150 month = (short)(months - 1);
1151 year = (short)(years + 1524);
1152 } else {
1153 month = (short)(months - 13);
1154 year = (short)(years + 1525);
1155 }
1156 /* calculation of day of month is based on the wonderful
1157 * sequence of INT( n * 30.6): it reproduces the·
1158 * 31-30-31-30-31-31 month lengths exactly for small n's */
1159 short day = (short)(yearday - (1959 * months) / 64);
1160
1161 QDateTime t(QDate(year, month, day), QTime(hour, minute, second));
1162 t.setTimeZone(QTimeZone::utc());
1163 return t.toSecsSinceEpoch();
1164}
1165
1166long long rtlSecondsSince1970ToSpecTime(quint32 seconds)
1167{
1168 long long secs = seconds * (long long)TICKSPERSEC + TICKS_1601_TO_1970;
1169 return secs;
1170}
1171
1172bool K7Zip::K7ZipPrivate::readMainStreamsInfo()
1173{
1174 if (!buffer) {
1175 return false;
1176 }
1177
1178 quint32 type;
1179 for (;;) {
1180 type = readByte();
1181 if (type > ((quint32)1 << 30)) {
1182 qCDebug(KArchiveLog) << "type error";
1183 return false;
1184 }
1185 switch (type) {
1186 case kEnd:
1187 return true;
1188 case kPackInfo: {
1189 if (!readPackInfo()) {
1190 qCDebug(KArchiveLog) << "error during read pack information";
1191 return false;
1192 }
1193 break;
1194 }
1195 case kUnpackInfo: {
1196 if (!readUnpackInfo()) {
1197 qCDebug(KArchiveLog) << "error during read pack information";
1198 return false;
1199 }
1200 break;
1201 }
1202 case kSubStreamsInfo: {
1203 if (!readSubStreamsInfo()) {
1204 qCDebug(KArchiveLog) << "error during read substreams information";
1205 return false;
1206 }
1207 break;
1208 }
1209 default:
1210 qCDebug(KArchiveLog) << "Wrong type";
1211 return false;
1212 }
1213 }
1214
1215 qCDebug(KArchiveLog) << "should not reach";
1216 return false;
1217}
1218
1219static bool getInStream(const Folder *folder, quint32 streamIndex, int &seqInStream, quint32 &coderIndex)
1220{
1221 for (int i = 0; i < folder->packedStreams.size(); i++) {
1222 if (folder->packedStreams[i] == streamIndex) {
1223 seqInStream = i;
1224 return true;
1225 }
1226 }
1227
1228 int binderIndex = folder->findBindPairForInStream(streamIndex);
1229 if (binderIndex < 0) {
1230 return false;
1231 }
1232
1233 quint32 coderStreamIndex;
1234 folder->findOutStream(folder->outIndexes[binderIndex], coderIndex, coderStreamIndex);
1235
1236 quint32 startIndex = folder->getCoderInStreamIndex(coderIndex);
1237
1238 if (folder->folderInfos[coderIndex]->numInStreams > 1) {
1239 return false;
1240 }
1241
1242 for (int i = 0; i < (int)folder->folderInfos[coderIndex]->numInStreams; i++) {
1243 getInStream(folder, startIndex + i, seqInStream, coderIndex);
1244 }
1245
1246 return true;
1247}
1248
1249static bool getOutStream(const Folder *folder, quint32 streamIndex, int &seqOutStream)
1250{
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);
1255
1256 for (int j = 0; j < coderInfo->numOutStreams; j++, outStreamIndex++) {
1257 if (folder->findBindPairForOutStream(outStreamIndex) < 0) {
1258 outStreams.append(outStreamIndex);
1259 }
1260 }
1261 }
1262
1263 for (int i = 0; i < outStreams.size(); i++) {
1264 if (outStreams[i] == streamIndex) {
1265 seqOutStream = i;
1266 return true;
1267 }
1268 }
1269
1270 int binderIndex = folder->findBindPairForOutStream(streamIndex);
1271 if (binderIndex < 0) {
1272 return false;
1273 }
1274
1275 quint32 coderIndex;
1276 quint32 coderStreamIndex;
1277 folder->findInStream(folder->inIndexes[binderIndex], coderIndex, coderStreamIndex);
1278
1279 quint32 startIndex = folder->getCoderOutStreamIndex(coderIndex);
1280
1281 if (folder->folderInfos[coderIndex]->numOutStreams > 1) {
1282 return false;
1283 }
1284
1285 for (int i = 0; i < (int)folder->folderInfos[coderIndex]->numOutStreams; i++) {
1286 getOutStream(folder, startIndex + i, seqOutStream);
1287 }
1288
1289 return true;
1290}
1291
1292static const int catCycle = 6;
1293
1294static QByteArray calculateKey(const QByteArray &password, quint32 numCyclesPower, const QByteArray &salt)
1295{
1296 quint32 rounds, stages;
1297
1298 if (numCyclesPower > catCycle) {
1299 rounds = 1 << catCycle;
1300 stages = 1 << (numCyclesPower - catCycle);
1301 } else {
1302 rounds = 1 << numCyclesPower;
1303 stages = 1;
1304 }
1305
1306 QByteArray saltPassword = salt + password;
1307 quint64 s = 0;
1308
1309 QCryptographicHash hash(QCryptographicHash::Sha256);
1310
1311 for (quint32 i = 0; i < stages; i++) {
1312 QByteArray result;
1313 result.reserve(saltPassword.size() + rounds * 8);
1314
1315 for (quint32 j = 0; j < rounds; j++) {
1316 result += saltPassword;
1317
1318 quint64 value = s + j;
1319 for (int k = 0; k < 8; k++) {
1320 result.append(value >> (k * 8));
1321 }
1322 }
1323
1324 hash.addData(result);
1325 s += rounds;
1326 }
1327
1328 return hash.result();
1329}
1330
1331#if HAVE_OPENSSL_SUPPORT
1332
1333static QByteArray decryptAES(const QList<quint8> &coderProperties, const QString &password, QByteArray &encryptedData)
1334{
1335 QStringEncoder toUtf16LE(QStringEncoder::Utf16LE);
1336 const QByteArray passwordBytes = toUtf16LE(password);
1337
1338 quint8 firstByte = coderProperties[0];
1339 quint32 numCyclesPower = firstByte & 0x3F;
1340
1341 if ((firstByte & 0xC0) == 0) {
1342 qCDebug(KArchiveLog) << "Unsupported AES properties";
1343 return QByteArray();
1344 }
1345
1346 int saltSize = ((firstByte >> 7) & 1) + (coderProperties[1] >> 4);
1347 int ivSize = ((firstByte >> 6) & 1) + (coderProperties[1] & 0x0F);
1348
1349 QByteArray salt((const char *)coderProperties.data() + 2, saltSize);
1350 QByteArray iv((const char *)coderProperties.data() + 2 + saltSize, ivSize);
1351
1352 if (ivSize < 16) {
1353 iv.append(16 - ivSize, '\x00');
1354 }
1355
1356 const QByteArray key = calculateKey(passwordBytes, numCyclesPower, salt);
1357 if (key.size() != 32) {
1358 qCDebug(KArchiveLog) << "Failed to calculate key";
1359 return QByteArray();
1360 }
1361
1362 EVP_CIPHER_CTX *ctx = EVP_CIPHER_CTX_new();
1363 if (!ctx) {
1364 qCDebug(KArchiveLog) << "Failed to create OpenSSL cipher context";
1365 return QByteArray();
1366 }
1367
1368 const auto ctxCleanupGuard = qScopeGuard([&ctx] {
1369 EVP_CIPHER_CTX_free(ctx);
1370 });
1371
1372
1373 int padLen = encryptedData.size() % 16;
1374 if (padLen > 0) {
1375 encryptedData.append(16 - padLen, '\x00');
1376 }
1377
1378 QByteArray decryptedData;
1379 int len, plainTextLen = 0;
1380 decryptedData.resize(encryptedData.size());
1381
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();
1385 }
1386
1387 // Disable automatic padding
1388 EVP_CIPHER_CTX_set_padding(ctx, 0);
1389
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();
1393 }
1394 plainTextLen += len;
1395
1396 if (EVP_DecryptFinal_ex(ctx, (unsigned char *)decryptedData.data() + len, &len) != 1) {
1397 qCDebug(KArchiveLog) << "EVP_DecryptFinal_ex failed";
1398 return QByteArray();
1399 }
1400 plainTextLen += len;
1401
1402 decryptedData.resize(plainTextLen);
1403 return decryptedData;
1404}
1405
1406#endif
1407
1408const int kNumTopBits = 24;
1409const quint32 kTopValue = (1 << kNumTopBits);
1410
1411class RangeDecoder
1412{
1413 int pos;
1414
1415public:
1416 QByteArray stream;
1417 quint32 range;
1418 quint32 code;
1419
1420 RangeDecoder(const QByteArray &s)
1421 : pos(0)
1422 , stream(s)
1423 , range(0xFFFFFFFF)
1424 , code(0)
1425 {
1426 for (int i = 0; i < 5; i++) {
1427 code = (code << 8) | readByte();
1428 }
1429 }
1430
1431 unsigned char readByte()
1432 {
1433 if (pos >= stream.size()) {
1434 return 0;
1435 }
1436 return stream[pos++];
1437 }
1438
1439 void normalize()
1440 {
1441 while (range < kTopValue) {
1442 code = (code << 8) | readByte();
1443 range <<= 8;
1444 }
1445 }
1446
1447 quint32 getThreshold(quint32 total)
1448 {
1449 return (code) / (range /= total);
1450 }
1451
1452 void decode(quint32 start, quint32 size)
1453 {
1454 code -= start * range;
1455 range *= size;
1456 normalize();
1457 }
1458
1459 quint32 decodeDirectBits(int numTotalBits)
1460 {
1461 quint32 r = range;
1462 quint32 c = code;
1463 quint32 result = 0;
1464 for (int i = numTotalBits; i != 0; i--) {
1465 r >>= 1;
1466 quint32 t = (c - r) >> 31;
1467 c -= r & (t - 1);
1468 result = (result << 1) | (1 - t);
1469
1470 if (r < kTopValue) {
1471 c = (c << 8) | readByte();
1472 r <<= 8;
1473 }
1474 }
1475 range = r;
1476 code = c;
1477 return result;
1478 }
1479
1480 quint32 DecodeBit(quint32 size0, quint32 numTotalBits)
1481 {
1482 quint32 newBound = (range >> numTotalBits) * size0;
1483 quint32 symbol;
1484 if (code < newBound) {
1485 symbol = 0;
1486 range = newBound;
1487 } else {
1488 symbol = 1;
1489 code -= newBound;
1490 range -= newBound;
1491 }
1492 normalize();
1493 return symbol;
1494 }
1495};
1496
1497const int kNumBitModelTotalBits = 11;
1498const quint32 kBitModelTotal = (1 << kNumBitModelTotalBits);
1499
1500template<int numMoveBits>
1501class CBitModel
1502{
1503public:
1504 quint32 prob;
1505 void updateModel(quint32 symbol)
1506 {
1507 if (symbol == 0) {
1508 prob += (kBitModelTotal - prob) >> numMoveBits;
1509 } else {
1510 prob -= (prob) >> numMoveBits;
1511 }
1512 }
1513
1514 void init()
1515 {
1516 prob = kBitModelTotal / 2;
1517 }
1518};
1519
1520template<int numMoveBits>
1521class CBitDecoder : public CBitModel<numMoveBits>
1522{
1523public:
1524 quint32 decode(RangeDecoder *decoder)
1525 {
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;
1533 }
1534 return 0;
1535 } else {
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;
1542 }
1543 return 1;
1544 }
1545 }
1546};
1547
1548inline bool isJcc(unsigned char b0, unsigned char b1)
1549{
1550 return (b0 == 0x0F && (b1 & 0xF0) == 0x80);
1551}
1552inline bool isJ(unsigned char b0, unsigned char b1)
1553{
1554 return ((b1 & 0xFE) == 0xE8 || isJcc(b0, b1));
1555}
1556inline unsigned getIndex(unsigned char b0, unsigned char b1)
1557{
1558 return ((b1 == 0xE8) ? b0 : ((b1 == 0xE9) ? 256 : 257));
1559}
1560
1561const int kNumMoveBits = 5;
1562
1563static QByteArray decodeBCJ2(const QByteArray &mainStream, const QByteArray &callStream, const QByteArray &jumpStream, const QByteArray &rangeBuffer)
1564{
1565 unsigned char prevByte = 0;
1566 QByteArray outStream;
1567 int mainStreamPos = 0;
1568 int callStreamPos = 0;
1569 int jumpStreamPos = 0;
1570
1571 RangeDecoder rangeDecoder(rangeBuffer);
1572
1573 QList<CBitDecoder<kNumMoveBits>> statusDecoder(256 + 2);
1574
1575 for (int i = 0; i < 256 + 2; i++) {
1576 statusDecoder[i].init();
1577 }
1578
1579 for (;;) {
1580 quint32 i;
1581 unsigned char b = 0;
1582 const quint32 kBurstSize = (1 << 18);
1583 for (i = 0; i < kBurstSize; i++) {
1584 if (mainStreamPos == mainStream.size()) {
1585 return outStream;
1586 }
1587
1588 b = mainStream[mainStreamPos++];
1589 outStream.append(b);
1590
1591 if (isJ(prevByte, b)) {
1592 break;
1593 }
1594 prevByte = b;
1595 }
1596
1597 if (i == kBurstSize) {
1598 continue;
1599 }
1600
1601 unsigned index = getIndex(prevByte, b);
1602 if (statusDecoder[index].decode(&rangeDecoder) == 1) {
1603 if (b == 0xE8) {
1604 if (callStreamPos + 4 > callStream.size()) {
1605 return QByteArray();
1606 }
1607 } else {
1608 if (jumpStreamPos + 4 > jumpStream.size()) {
1609 return QByteArray();
1610 }
1611 }
1612 quint32 src = 0;
1613 for (int i = 0; i < 4; i++) {
1614 unsigned char b0;
1615 if (b == 0xE8) {
1616 b0 = callStream[callStreamPos++];
1617 } else {
1618 b0 = jumpStream[jumpStreamPos++];
1619 }
1620 src <<= 8;
1621 src |= ((quint32)b0);
1622 }
1623
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);
1630 } else {
1631 prevByte = b;
1632 }
1633 }
1634}
1635
1636KFilterBase *K7Zip::K7ZipPrivate::getFilter(const Folder *folder,
1637 const Folder::FolderInfo *coder,
1638 const int currentCoderIndex,
1639 QByteArray &deflatedData,
1640 QList<QByteArray> &inflatedDatas)
1641{
1642 KFilterBase *filter = nullptr;
1643
1644 switch (coder->methodID) {
1645 case k_LZMA:
1646 filter = KCompressionDevice::filterForCompressionType(KCompressionDevice::Xz);
1647 if (!filter) {
1648 qCDebug(KArchiveLog) << "filter not found";
1649 return nullptr;
1650 }
1651 static_cast<KXzFilter *>(filter)->init(QIODevice::ReadOnly, KXzFilter::LZMA, coder->properties);
1652 break;
1653 case k_LZMA2:
1654 filter = KCompressionDevice::filterForCompressionType(KCompressionDevice::Xz);
1655 if (!filter) {
1656 qCDebug(KArchiveLog) << "filter not found";
1657 return nullptr;
1658 }
1659 static_cast<KXzFilter *>(filter)->init(QIODevice::ReadOnly, KXzFilter::LZMA2, coder->properties);
1660 break;
1661 case k_PPMD: {
1662 /*if (coder->properties.size() == 5) {
1663 //Byte order = *(const Byte *)coder.Props;
1664 qint32 dicSize = ((unsigned char)coder->properties[1] |
1665 (((unsigned char)coder->properties[2]) << 8) |
1666 (((unsigned char)coder->properties[3]) << 16) |
1667 (((unsigned char)coder->properties[4]) << 24));
1668 }*/
1669 break;
1670 }
1671 case k_AES: {
1672 if (coder->properties.size() >= 2) {
1673 if (password.isEmpty()) {
1674 qCDebug(KArchiveLog) << "Password is required for AES decryption";
1675 return nullptr;
1676 }
1677
1678#if HAVE_OPENSSL_SUPPORT
1679 QByteArray decryptedData = decryptAES(coder->properties, password, deflatedData);
1680 if (decryptedData.isEmpty()) {
1681 qCDebug(KArchiveLog) << "AES decryption failed";
1682 return nullptr;
1683 }
1684
1685 if (folder->folderInfos.size() > 1 && currentCoderIndex < folder->folderInfos.size()) {
1686 deflatedData = decryptedData; // set the data for the filter to the decrypted data
1687 int nextCoderIndex = currentCoderIndex + 1;
1688 filter = getFilter(folder, folder->folderInfos[nextCoderIndex], nextCoderIndex, decryptedData, inflatedDatas);
1689 } else {
1690 inflatedDatas.append(decryptedData);
1691 }
1692#endif
1693 }
1694 break;
1695 }
1696 case k_BCJ:
1697 filter = KCompressionDevice::filterForCompressionType(KCompressionDevice::Xz);
1698 if (!filter) {
1699 qCDebug(KArchiveLog) << "filter not found";
1700 return nullptr;
1701 }
1702 static_cast<KXzFilter *>(filter)->init(QIODevice::ReadOnly, KXzFilter::BCJ, coder->properties);
1703 break;
1704 case k_BCJ2: {
1705 QByteArray bcj2 = decodeBCJ2(inflatedDatas[0], inflatedDatas[1], inflatedDatas[2], deflatedData);
1706 inflatedDatas.clear();
1707 inflatedDatas.append(bcj2);
1708 break;
1709 }
1710 case k_BZip2:
1711 filter = KCompressionDevice::filterForCompressionType(KCompressionDevice::BZip2);
1712 if (!filter) {
1713 qCDebug(KArchiveLog) << "filter not found";
1714 return nullptr;
1715 }
1717 break;
1718 }
1719
1720 return filter;
1721}
1722
1723QByteArray K7Zip::K7ZipPrivate::readAndDecodePackedStreams(bool readMainStreamInfo)
1724{
1725 if (!buffer) {
1726 return QByteArray();
1727 }
1728
1729 if (readMainStreamInfo) {
1730 readMainStreamsInfo();
1731 }
1732
1733 QByteArray inflatedData;
1734
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;
1743 }
1744
1745 // Find main coder
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);
1754 break;
1755 }
1756 }
1757 }
1758
1759 quint32 temp = 0;
1760 if (!outStreamIndexed.isEmpty()) {
1761 folder->findOutStream(outStreamIndexed[0], mainCoderIndex, temp);
1762 }
1763
1764 quint32 startInIndex = folder->getCoderInStreamIndex(mainCoderIndex);
1765 quint32 startOutIndex = folder->getCoderOutStreamIndex(mainCoderIndex);
1766
1767 Folder::FolderInfo *mainCoder = folder->folderInfos[mainCoderIndex];
1768
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++) {
1774 int seqInStream;
1775 quint32 coderIndex;
1776 getInStream(folder, startInIndex + j, seqInStream, coderIndex);
1777 seqInStreams.append(seqInStream);
1778 coderIndexes.append(coderIndex);
1779 }
1780
1781 QList<int> seqOutStreams;
1782 seqOutStreams.reserve(mainCoder->numOutStreams);
1783 for (int j = 0; j < (int)mainCoder->numOutStreams; j++) {
1784 int seqOutStream;
1785 getOutStream(folder, startOutIndex + j, seqOutStream);
1786 seqOutStreams.append(seqOutStream);
1787 }
1788
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);
1796 if (n != size) {
1797 qCDebug(KArchiveLog) << "Failed read next size, should read " << size << ", read " << n;
1798 return inflatedData;
1799 }
1800 QByteArray deflatedData(encodedBuffer.get(), size);
1801 datas.append(deflatedData);
1802 startPos += size;
1803 pos += size;
1804 headerSize += size;
1805 }
1806
1807 QList<QByteArray> inflatedDatas;
1808 QByteArray deflatedData;
1809 for (int j = 0; j < seqInStreams.size(); ++j) {
1810 int coderIndex = 0;
1811
1812 if ((quint32)j != mainCoderIndex) {
1813 coderIndex = coderIndexes[j];
1814 } else {
1815 coderIndex = mainCoderIndex;
1816 }
1817
1818 Folder::FolderInfo *coder = folder->folderInfos[coderIndex];
1819 deflatedData = datas[seqInStreams[j]];
1820
1821 KFilterBase *filter = getFilter(folder, coder, coderIndex, deflatedData, inflatedDatas);
1822 if (coder->methodID == k_BCJ2) {
1823 continue;
1824 }
1825
1826 if (!filter) {
1827 if (coder->methodID == k_AES) {
1828 continue;
1829 }
1830 return QByteArray();
1831 }
1832
1833 filter->setInBuffer(deflatedData.data(), deflatedData.size());
1834
1835 QByteArray outBuffer;
1836 // reserve memory
1837 outBuffer.resize(unpackSize);
1838
1839 KFilterBase::Result result = KFilterBase::Ok;
1840 QByteArray inflatedDataTmp;
1841 while (result != KFilterBase::End && result != KFilterBase::Error && !filter->inBufferEmpty()) {
1842 filter->setOutBuffer(outBuffer.data(), outBuffer.size());
1843 result = filter->uncompress();
1844 if (result == KFilterBase::Error) {
1845 qCDebug(KArchiveLog) << " decode error";
1846 filter->terminate();
1847 delete filter;
1848 return QByteArray();
1849 }
1850 int uncompressedBytes = outBuffer.size() - filter->outBufferAvailable();
1851
1852 // append the uncompressed data to inflate buffer
1853 inflatedDataTmp.append(outBuffer.data(), uncompressedBytes);
1854
1855 if (result == KFilterBase::End) {
1856 // qCDebug(KArchiveLog) << "Finished unpacking";
1857 break; // Finished.
1858 }
1859 }
1860
1861 if (result != KFilterBase::End && !filter->inBufferEmpty()) {
1862 qCDebug(KArchiveLog) << "decode failed result" << result;
1863 filter->terminate();
1864 delete filter;
1865 return QByteArray();
1866 }
1867
1868 filter->terminate();
1869 delete filter;
1870
1871 inflatedDatas.append(inflatedDataTmp);
1872 }
1873
1874 QByteArray inflated;
1875 for (const QByteArray &data : std::as_const(inflatedDatas)) {
1876 inflated.append(data);
1877 }
1878
1879 inflatedDatas.clear();
1880
1881 if (folder->unpackCRCDefined) {
1882 if ((size_t)inflated.size() < unpackSize) {
1883 qCDebug(KArchiveLog) << "wrong crc size data";
1884 return QByteArray();
1885 }
1886 quint32 crc = crc32(0, (Bytef *)(inflated.data()), unpackSize);
1887 if (crc != folder->unpackCRC) {
1888 qCDebug(KArchiveLog) << "wrong crc";
1889 return QByteArray();
1890 }
1891 }
1892
1893 inflatedData.append(inflated);
1894 }
1895
1896 return inflatedData;
1897}
1898
1899///////////////// Write ////////////////////
1900
1901void K7Zip::K7ZipPrivate::createItemsFromEntities(const KArchiveDirectory *dir, const QString &path, QByteArray &data)
1902{
1903 const QStringList l = dir->entries();
1905 for (; it != l.end(); ++it) {
1906 const KArchiveEntry *entry = dir->entry((*it));
1907
1908 FileInfo *fileInfo = new FileInfo;
1909 fileInfo->attribDefined = true;
1910
1911 fileInfo->path = path + entry->name();
1912 mTimesDefined.append(true);
1913 mTimes.append(rtlSecondsSince1970ToSpecTime(entry->date().toSecsSinceEpoch()));
1914
1915 if (entry->isFile()) {
1916 const K7ZipFileEntry *fileEntry = static_cast<const K7ZipFileEntry *>(entry);
1917
1918 fileInfo->attributes = FILE_ATTRIBUTE_ARCHIVE;
1919 fileInfo->attributes |= FILE_ATTRIBUTE_UNIX_EXTENSION + ((entry->permissions() & 0xFFFF) << 16);
1920 fileInfo->size = fileEntry->size();
1921 QString symLink = fileEntry->symLinkTarget();
1922 if (fileInfo->size > 0) {
1923 fileInfo->hasStream = true;
1924 data.append(outData.mid(fileEntry->position(), fileEntry->size()));
1925 unpackSizes.append(fileInfo->size);
1926 } else if (!symLink.isEmpty()) {
1927 fileInfo->hasStream = true;
1928 data.append(symLink.toUtf8());
1929 unpackSizes.append(symLink.size());
1930 }
1931 fileInfos.append(fileInfo);
1932 } else if (entry->isDirectory()) {
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);
1938 }
1939 }
1940}
1941
1942void K7Zip::K7ZipPrivate::writeByte(unsigned char b)
1943{
1944 header.append(b);
1945 countSize++;
1946}
1947
1948void K7Zip::K7ZipPrivate::writeNumber(quint64 value)
1949{
1950 int firstByte = 0;
1951 short mask = 0x80;
1952 int i;
1953 for (i = 0; i < 8; i++) {
1954 if (value < ((quint64(1) << (7 * (i + 1))))) {
1955 firstByte |= (int)(value >> (8 * i));
1956 break;
1957 }
1958 firstByte |= mask;
1959 mask >>= 1;
1960 }
1961 writeByte(firstByte);
1962 for (; i > 0; i--) {
1963 writeByte((int)value);
1964 value >>= 8;
1965 }
1966}
1967
1968void K7Zip::K7ZipPrivate::writeBoolVector(const QList<bool> &boolVector)
1969{
1970 int b = 0;
1971 short mask = 0x80;
1972 for (int i = 0; i < boolVector.size(); i++) {
1973 if (boolVector[i]) {
1974 b |= mask;
1975 }
1976 mask >>= 1;
1977 if (mask == 0) {
1978 writeByte(b);
1979 mask = 0x80;
1980 b = 0;
1981 }
1982 }
1983 if (mask != 0x80) {
1984 writeByte(b);
1985 }
1986}
1987
1988void K7Zip::K7ZipPrivate::writeUInt32(quint32 value)
1989{
1990 for (int i = 0; i < 4; i++) {
1991 writeByte((unsigned char)value);
1992 value >>= 8;
1993 }
1994}
1995
1996void K7Zip::K7ZipPrivate::writeUInt64(quint64 value)
1997{
1998 for (int i = 0; i < 8; i++) {
1999 writeByte((unsigned char)value);
2000 value >>= 8;
2001 }
2002}
2003
2004void K7Zip::K7ZipPrivate::writeAlignedBoolHeader(const QList<bool> &v, int numDefined, int type, unsigned itemSize)
2005{
2006 const unsigned bvSize = (numDefined == v.size()) ? 0 : ((unsigned)v.size() + 7) / 8;
2007 const quint64 dataSize = (quint64)numDefined * itemSize + bvSize + 2;
2008 // SkipAlign(3 + (unsigned)bvSize + (unsigned)GetBigNumberSize(dataSize), itemSize);
2009
2010 writeByte(type);
2011 writeNumber(dataSize);
2012 if (numDefined == v.size()) {
2013 writeByte(1);
2014 } else {
2015 writeByte(0);
2016 writeBoolVector(v);
2017 }
2018 writeByte(0);
2019}
2020
2021void K7Zip::K7ZipPrivate::writeUInt64DefVector(const QList<quint64> &v, const QList<bool> &defined, int type)
2022{
2023 int numDefined = 0;
2024
2025 for (int i = 0; i < defined.size(); i++) {
2026 if (defined[i]) {
2027 numDefined++;
2028 }
2029 }
2030
2031 if (numDefined == 0) {
2032 return;
2033 }
2034
2035 writeAlignedBoolHeader(defined, numDefined, type, 8);
2036
2037 for (int i = 0; i < defined.size(); i++) {
2038 if (defined[i]) {
2039 writeUInt64(v[i]);
2040 }
2041 }
2042}
2043
2044void K7Zip::K7ZipPrivate::writeHashDigests(const QList<bool> &digestsDefined, const QList<quint32> &digests)
2045{
2046 int numDefined = 0;
2047 int i;
2048 for (i = 0; i < digestsDefined.size(); i++) {
2049 if (digestsDefined[i]) {
2050 numDefined++;
2051 }
2052 }
2053
2054 if (numDefined == 0) {
2055 return;
2056 }
2057
2058 writeByte(kCRC);
2059 if (numDefined == digestsDefined.size()) {
2060 writeByte(1);
2061 } else {
2062 writeByte(0);
2063 writeBoolVector(digestsDefined);
2064 }
2065
2066 for (i = 0; i < digests.size(); i++) {
2067 if (digestsDefined[i]) {
2068 writeUInt32(digests[i]);
2069 }
2070 }
2071}
2072
2073void K7Zip::K7ZipPrivate::writePackInfo(quint64 dataOffset, QList<quint64> &packedSizes, QList<bool> &packedCRCsDefined, QList<quint32> &packedCRCs)
2074{
2075 if (packedSizes.isEmpty()) {
2076 return;
2077 }
2078 writeByte(kPackInfo);
2079 writeNumber(dataOffset);
2080 writeNumber(packedSizes.size());
2081 writeByte(kSize);
2082
2083 for (int i = 0; i < packedSizes.size(); i++) {
2084 writeNumber(packedSizes[i]);
2085 }
2086
2087 writeHashDigests(packedCRCsDefined, packedCRCs);
2088
2089 writeByte(kEnd);
2090}
2091
2092void K7Zip::K7ZipPrivate::writeFolder(const Folder *folder)
2093{
2094 writeNumber(folder->folderInfos.size());
2095 for (int i = 0; i < folder->folderInfos.size(); i++) {
2096 const Folder::FolderInfo *info = folder->folderInfos.at(i);
2097 {
2098 size_t propsSize = info->properties.size();
2099
2100 quint64 id = info->methodID;
2101 size_t idSize;
2102 for (idSize = 1; idSize < sizeof(id); idSize++) {
2103 if ((id >> (8 * idSize)) == 0) {
2104 break;
2105 }
2106 }
2107
2108 int longID[15];
2109 for (int t = idSize - 1; t >= 0; t--, id >>= 8) {
2110 longID[t] = (int)(id & 0xFF);
2111 }
2112
2113 int b;
2114 b = (int)(idSize & 0xF);
2115 bool isComplex = !info->isSimpleCoder();
2116 b |= (isComplex ? 0x10 : 0);
2117 b |= ((propsSize != 0) ? 0x20 : 0);
2118
2119 writeByte(b);
2120 for (size_t j = 0; j < idSize; ++j) {
2121 writeByte(longID[j]);
2122 }
2123
2124 if (isComplex) {
2125 writeNumber(info->numInStreams);
2126 writeNumber(info->numOutStreams);
2127 }
2128
2129 if (propsSize == 0) {
2130 continue;
2131 }
2132
2133 writeNumber(propsSize);
2134 for (size_t j = 0; j < propsSize; ++j) {
2135 writeByte(info->properties[j]);
2136 }
2137 }
2138 }
2139
2140 for (int i = 0; i < folder->inIndexes.size(); i++) {
2141 writeNumber(folder->inIndexes[i]);
2142 writeNumber(folder->outIndexes[i]);
2143 }
2144
2145 if (folder->packedStreams.size() > 1) {
2146 for (int i = 0; i < folder->packedStreams.size(); i++) {
2147 writeNumber(folder->packedStreams[i]);
2148 }
2149 }
2150}
2151
2152void K7Zip::K7ZipPrivate::writeUnpackInfo(const QList<Folder *> &folderItems)
2153{
2154 if (folderItems.isEmpty()) {
2155 return;
2156 }
2157
2158 writeByte(kUnpackInfo);
2159
2160 writeByte(kFolder);
2161 writeNumber(folderItems.size());
2162 {
2163 writeByte(0);
2164 for (int i = 0; i < folderItems.size(); i++) {
2165 writeFolder(folderItems[i]);
2166 }
2167 }
2168
2169 writeByte(kCodersUnpackSize);
2170 int i;
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));
2175 }
2176 }
2177
2178 QList<bool> unpackCRCsDefined;
2179 QList<quint32> unpackCRCs;
2180 unpackCRCsDefined.reserve(folderItems.size());
2181 unpackCRCs.reserve(folderItems.size());
2182 for (i = 0; i < folderItems.size(); i++) {
2183 const Folder *folder = folderItems[i];
2184 unpackCRCsDefined.append(folder->unpackCRCDefined);
2185 unpackCRCs.append(folder->unpackCRC);
2186 }
2187 writeHashDigests(unpackCRCsDefined, unpackCRCs);
2188
2189 writeByte(kEnd);
2190}
2191
2192void K7Zip::K7ZipPrivate::writeSubStreamsInfo(const QList<quint64> &unpackSizes, const QList<bool> &digestsDefined, const QList<quint32> &digests)
2193{
2194 writeByte(kSubStreamsInfo);
2195
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));
2201 }
2202 break;
2203 }
2204 }
2205
2206 bool needFlag = true;
2207 int index = 0;
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)) {
2211 if (needFlag) {
2212 writeByte(kSize);
2213 }
2214 needFlag = false;
2215 writeNumber(unpackSizes[index]);
2216 }
2217 index++;
2218 }
2219 }
2220
2221 QList<bool> digestsDefined2;
2222 QList<quint32> digests2;
2223
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) {
2228 digestIndex++;
2229 } else {
2230 for (int j = 0; j < numSubStreams; j++, digestIndex++) {
2231 digestsDefined2.append(digestsDefined[digestIndex]);
2232 digests2.append(digests[digestIndex]);
2233 }
2234 }
2235 }
2236 writeHashDigests(digestsDefined2, digests2);
2237 writeByte(kEnd);
2238}
2239
2240QByteArray K7Zip::K7ZipPrivate::encodeStream(QList<quint64> &packSizes, QList<Folder *> &folds)
2241{
2242 Folder *folder = new Folder;
2243 folder->unpackCRCDefined = true;
2244 folder->unpackCRC = crc32(0, (Bytef *)(header.data()), header.size());
2245 folder->unpackSizes.append(header.size());
2246
2247 Folder::FolderInfo *info = new Folder::FolderInfo();
2248 info->numInStreams = 1;
2249 info->numOutStreams = 1;
2250 info->methodID = k_LZMA2;
2251
2252 quint32 dictSize = header.size();
2253 const quint32 kMinReduceSize = (1 << 16);
2254 if (dictSize < kMinReduceSize) {
2255 dictSize = kMinReduceSize;
2256 }
2257
2258 int dict;
2259 for (dict = 0; dict < 40; dict++) {
2260 if (dictSize <= lzma2_dic_size_from_prop(dict)) {
2261 break;
2262 }
2263 }
2264
2265 info->properties.append(dict);
2266 folder->folderInfos.append(info);
2267
2268 folds.append(folder);
2269
2270 // compress data
2271 QByteArray encodedData;
2272 if (!header.isEmpty()) {
2273 QByteArray enc;
2274 QBuffer inBuffer(&enc);
2275
2276 KCompressionDevice flt(&inBuffer, false, KCompressionDevice::Xz);
2277 flt.open(QIODevice::WriteOnly);
2278
2279 KFilterBase *filter = flt.filterBase();
2280
2281 static_cast<KXzFilter *>(filter)->init(QIODevice::WriteOnly, KXzFilter::LZMA2, info->properties);
2282
2283 const int ret = flt.write(header);
2284 if (ret != header.size()) {
2285 qCDebug(KArchiveLog) << "write error write " << ret << "expected" << header.size();
2286 return encodedData;
2287 }
2288
2289 flt.close();
2290 encodedData = inBuffer.data();
2291 }
2292
2293 packSizes.append(encodedData.size());
2294 return encodedData;
2295}
2296
2297void K7Zip::K7ZipPrivate::writeHeader(quint64 &headerOffset)
2298{
2299 quint64 packedSize = 0;
2300 for (int i = 0; i < packSizes.size(); ++i) {
2301 packedSize += packSizes[i];
2302 }
2303
2304 headerOffset = packedSize;
2305
2306 writeByte(kHeader);
2307
2308 // Archive Properties
2309
2310 if (!folders.isEmpty()) {
2311 writeByte(kMainStreamsInfo);
2312 writePackInfo(0, packSizes, packCRCsDefined, packCRCs);
2313
2314 writeUnpackInfo(folders);
2315
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) {
2322 continue;
2323 }
2324 unpackFileSizes.append(file->size);
2325 digestsDefined.append(file->crcDefined);
2326 digests.append(file->crc);
2327 }
2328
2329 writeSubStreamsInfo(unpackSizes, digestsDefined, digests);
2330 writeByte(kEnd);
2331 }
2332
2333 if (fileInfos.isEmpty()) {
2334 writeByte(kEnd);
2335 return;
2336 }
2337
2338 writeByte(kFilesInfo);
2339 writeNumber(fileInfos.size());
2340
2341 {
2342 /* ---------- Empty Streams ---------- */
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);
2348 } else {
2349 emptyStreamVector.append(true);
2350 numEmptyStreams++;
2351 }
2352 }
2353
2354 if (numEmptyStreams > 0) {
2355 writeByte(kEmptyStream);
2356 writeNumber(((unsigned)emptyStreamVector.size() + 7) / 8);
2357 writeBoolVector(emptyStreamVector);
2358
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);
2367 if (!file->isDir) {
2368 numEmptyFiles++;
2369 bool isAnti = (i < this->isAnti.size() && this->isAnti[i]);
2370 antiVector.append(isAnti);
2371 if (isAnti) {
2372 numAntiItems++;
2373 }
2374 }
2375 }
2376 }
2377
2378 if (numEmptyFiles > 0) {
2379 writeByte(kEmptyFile);
2380 writeNumber(((unsigned)emptyFileVector.size() + 7) / 8);
2381 writeBoolVector(emptyFileVector);
2382 }
2383
2384 if (numAntiItems > 0) {
2385 writeByte(kAnti);
2386 writeNumber(((unsigned)antiVector.size() + 7) / 8);
2387 writeBoolVector(antiVector);
2388 }
2389 }
2390 }
2391
2392 {
2393 /* ---------- Names ---------- */
2394
2395 int numDefined = 0;
2396 size_t namesDataSize = 0;
2397 for (int i = 0; i < fileInfos.size(); i++) {
2398 const QString &name = fileInfos.at(i)->path;
2399 numDefined++;
2400 namesDataSize += (name.length() + 1) * 2;
2401 }
2402
2403 if (numDefined > 0) {
2404 namesDataSize++;
2405 // SkipAlign(2 + GetBigNumberSize(namesDataSize), 2);
2406
2407 writeByte(kName);
2408 writeNumber(namesDataSize);
2409 writeByte(0);
2410 for (int i = 0; i < fileInfos.size(); i++) {
2411 const QString &name = fileInfos.at(i)->path;
2412 for (int t = 0; t < name.length(); t++) {
2413 wchar_t c = name[t].toLatin1();
2414 writeByte((unsigned char)c);
2415 writeByte((unsigned char)(c >> 8));
2416 }
2417 // End of string
2418 writeByte(0);
2419 writeByte(0);
2420 }
2421 }
2422 }
2423
2424 writeUInt64DefVector(mTimes, mTimesDefined, kMTime);
2425
2426 writeUInt64DefVector(startPositions, startPositionsDefined, kStartPos);
2427
2428 {
2429 /* ---------- Write Attrib ---------- */
2430 QList<bool> boolVector;
2431 int numDefined = 0;
2432 boolVector.reserve(fileInfos.size());
2433 for (int i = 0; i < fileInfos.size(); i++) {
2434 bool defined = fileInfos.at(i)->attribDefined;
2435 boolVector.append(defined);
2436 if (defined) {
2437 numDefined++;
2438 }
2439 }
2440
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);
2447 }
2448 }
2449 }
2450 }
2451
2452 writeByte(kEnd); // for files
2453 writeByte(kEnd); // for headers*/
2454}
2455
2456static void setUInt32(unsigned char *p, quint32 d)
2457{
2458 for (int i = 0; i < 4; i++, d >>= 8) {
2459 p[i] = (unsigned)d;
2460 }
2461}
2462
2463static void setUInt64(unsigned char *p, quint64 d)
2464{
2465 for (int i = 0; i < 8; i++, d >>= 8) {
2466 p[i] = (unsigned char)d;
2467 }
2468}
2469
2470void K7Zip::K7ZipPrivate::writeStartHeader(const quint64 nextHeaderSize, const quint32 nextHeaderCRC, const quint64 nextHeaderOffset)
2471{
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));
2477 q->device()->write((char *)buf, 24);
2478}
2479
2480void K7Zip::K7ZipPrivate::writeSignature()
2481{
2482 unsigned char buf[8];
2483 memcpy(buf, k7zip_signature, 6);
2484 buf[6] = 0 /*kMajorVersion*/;
2485 buf[7] = 3;
2486 q->device()->write((char *)buf, 8);
2487}
2488
2490{
2491 if (!(mode & QIODevice::ReadOnly)) {
2492 return true;
2493 }
2494
2495 QIODevice *dev = device();
2496
2497 if (!dev) {
2498 setErrorString(tr("Could not get underlying device"));
2499 return false;
2500 }
2501
2502 char header[32];
2503 // check signature
2504 qint64 n = dev->read(header, 32);
2505 if (n != 32) {
2506 setErrorString(tr("Read header failed"));
2507 return false;
2508 }
2509
2510 for (int i = 0; i < 6; ++i) {
2511 if ((unsigned char)header[i] != k7zip_signature[i]) {
2512 setErrorString(tr("Check signature failed"));
2513 return false;
2514 }
2515 }
2516
2517 // get Archive Version
2518 int major = header[6];
2519 int minor = header[7];
2520
2521 /*if (major > 0 || minor > 2) {
2522 qCDebug(KArchiveLog) << "wrong archive version";
2523 return false;
2524 }*/
2525
2526 // get Start Header CRC
2527 quint32 startHeaderCRC = GetUi32(header + 8);
2528 quint64 nextHeaderOffset = GetUi64(header + 12);
2529 quint64 nextHeaderSize = GetUi64(header + 20);
2530 quint32 nextHeaderCRC = GetUi32(header + 28);
2531
2532 quint32 crc = crc32(0, (Bytef *)(header + 0xC), 20);
2533
2534 if (crc != startHeaderCRC) {
2535 setErrorString(tr("Bad CRC"));
2536 return false;
2537 }
2538
2539 if (nextHeaderSize == 0) {
2540 return true;
2541 }
2542
2543 if (nextHeaderSize > (quint64)0xFFFFFFFF) {
2544 setErrorString(tr("Next header size is too big"));
2545 return false;
2546 }
2547
2548 if ((qint64)nextHeaderOffset < 0) {
2549 setErrorString(tr("Next header size is less than zero"));
2550 return false;
2551 }
2552
2553 dev->seek(nextHeaderOffset + 32);
2554
2555 QByteArray inBuffer;
2556 inBuffer.resize(nextHeaderSize);
2557
2558 n = dev->read(inBuffer.data(), inBuffer.size());
2559 if (n != (qint64)nextHeaderSize) {
2560 setErrorString(tr("Failed read next header size; should read %1, read %2").arg(nextHeaderSize).arg(n));
2561 return false;
2562 }
2563 d->buffer = inBuffer.data();
2564 d->pos = 0;
2565 d->end = nextHeaderSize;
2566
2567 d->headerSize = 32 + nextHeaderSize;
2568 // int physSize = 32 + nextHeaderSize + nextHeaderOffset;
2569
2570 crc = crc32(0, (Bytef *)(d->buffer), (quint32)nextHeaderSize);
2571
2572 if (crc != nextHeaderCRC) {
2573 setErrorString(tr("Bad next header CRC"));
2574 return false;
2575 }
2576
2577 int type = d->readByte();
2578 QByteArray decodedData;
2579 if (type != kHeader) {
2580 if (type != kEncodedHeader) {
2581 setErrorString(tr("Error in header"));
2582 return false;
2583 }
2584
2585 decodedData = d->readAndDecodePackedStreams();
2586
2587 int external = d->readByte();
2588 if (external != 0) {
2589 int dataIndex = (int)d->readNumber();
2590 if (dataIndex < 0) {
2591 // qCDebug(KArchiveLog) << "dataIndex error";
2592 }
2593 d->buffer = decodedData.constData();
2594 d->pos = 0;
2595 d->end = decodedData.size();
2596 }
2597
2598 if (passwordNeeded() && d->password.isEmpty()) {
2599 setErrorString(tr("Password needed for this archive"));
2600 return false;
2601 }
2602
2603 type = d->readByte();
2604 if (type != kHeader) {
2605 setErrorString(tr("Wrong header type"));
2606 return false;
2607 }
2608 }
2609 // read header
2610
2611 type = d->readByte();
2612
2613 if (type == kArchiveProperties) {
2614 // TODO : implement this part
2615 setErrorString(tr("Not implemented"));
2616 return false;
2617 }
2618
2619 if (type == kAdditionalStreamsInfo) {
2620 // TODO : implement this part
2621 setErrorString(tr("Not implemented"));
2622 return false;
2623 }
2624
2625 if (type == kMainStreamsInfo) {
2626 if (!d->readMainStreamsInfo()) {
2627 setErrorString(tr("Error while reading main streams information"));
2628 return false;
2629 }
2630
2631 if (passwordNeeded() && d->password.isEmpty()) {
2632 setErrorString(tr("Password needed for this archive"));
2633 return false;
2634 }
2635
2636 type = d->readByte();
2637 } else {
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);
2643 }
2644 }
2645
2646 if (type == kEnd) {
2647 return true;
2648 }
2649
2650 if (type != kFilesInfo) {
2651 setErrorString(tr("Error while reading header"));
2652 return false;
2653 }
2654
2655 // read files info
2656 int numFiles = d->readNumber();
2657 for (int i = 0; i < numFiles; ++i) {
2658 d->fileInfos.append(new FileInfo);
2659 }
2660
2661 QList<bool> emptyStreamVector;
2662 QList<bool> emptyFileVector;
2663 QList<bool> antiFileVector;
2664 int numEmptyStreams = 0;
2665
2666 for (;;) {
2667 quint64 type = d->readByte();
2668 if (type == kEnd) {
2669 break;
2670 }
2671
2672 quint64 size = d->readNumber();
2673
2674 size_t ppp = d->pos;
2675
2676 bool addPropIdToList = true;
2677 bool isKnownType = true;
2678
2679 if (type > ((quint32)1 << 30)) {
2680 isKnownType = false;
2681 } else {
2682 switch (type) {
2683 case kEmptyStream: {
2684 d->readBoolVector(numFiles, emptyStreamVector);
2685 for (int i = 0; i < emptyStreamVector.size(); ++i) {
2686 if (emptyStreamVector[i]) {
2687 numEmptyStreams++;
2688 }
2689 }
2690
2691 break;
2692 }
2693 case kEmptyFile:
2694 d->readBoolVector(numEmptyStreams, emptyFileVector);
2695 break;
2696 case kAnti:
2697 d->readBoolVector(numEmptyStreams, antiFileVector);
2698 break;
2699 case kCTime:
2700 if (!d->readUInt64DefVector(numFiles, d->cTimes, d->cTimesDefined)) {
2701 return false;
2702 }
2703 break;
2704 case kATime:
2705 if (!d->readUInt64DefVector(numFiles, d->aTimes, d->aTimesDefined)) {
2706 return false;
2707 }
2708 break;
2709 case kMTime:
2710 if (!d->readUInt64DefVector(numFiles, d->mTimes, d->mTimesDefined)) {
2711 setErrorString(tr("Error reading modification time"));
2712 return false;
2713 }
2714 break;
2715 case kName: {
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";
2721 }
2722
2723 // TODO : go to the new index
2724 }
2725
2726 QString name;
2727 for (int i = 0; i < numFiles; i++) {
2728 name = d->readString();
2729 d->fileInfos.at(i)->path = name;
2730 }
2731 break;
2732 }
2733 case kAttributes: {
2734 QList<bool> attributesAreDefined;
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";
2741 }
2742
2743 // TODO : go to the new index
2744 }
2745
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();
2751 }
2752 }
2753 break;
2754 }
2755 case kStartPos:
2756 if (!d->readUInt64DefVector(numFiles, d->startPositions, d->startPositionsDefined)) {
2757 setErrorString(tr("Error reading MTime"));
2758 return false;
2759 }
2760 break;
2761 case kDummy: {
2762 for (quint64 i = 0; i < size; i++) {
2763 if (d->readByte() != 0) {
2764 setErrorString(tr("Invalid"));
2765 return false;
2766 }
2767 }
2768 addPropIdToList = false;
2769 break;
2770 }
2771 default:
2772 addPropIdToList = isKnownType = false;
2773 }
2774 }
2775
2776 if (isKnownType) {
2777 if (addPropIdToList) {
2778 d->fileInfoPopIDs.append(type);
2779 }
2780 } else {
2781 d->skipData(d->readNumber());
2782 }
2783
2784 bool checkRecordsSize = (major > 0 || minor > 2);
2785 if (checkRecordsSize && d->pos - ppp != size) {
2786 setErrorString(tr("Read size failed "
2787 "(checkRecordsSize: %1, d->pos - ppp: %2, size: %3)")
2788 .arg(checkRecordsSize)
2789 .arg(d->pos - ppp)
2790 .arg(size));
2791 return false;
2792 }
2793 }
2794
2795 int emptyFileIndex = 0;
2796 int sizeIndex = 0;
2797
2798 int numAntiItems = 0;
2799
2800 if (emptyStreamVector.isEmpty()) {
2801 emptyStreamVector.fill(false, numFiles);
2802 }
2803
2804 if (antiFileVector.isEmpty()) {
2805 antiFileVector.fill(false, numEmptyStreams);
2806 }
2807 if (emptyFileVector.isEmpty()) {
2808 emptyFileVector.fill(false, numEmptyStreams);
2809 }
2810
2811 for (int i = 0; i < numEmptyStreams; i++) {
2812 if (antiFileVector[i]) {
2813 numAntiItems++;
2814 }
2815 }
2816
2817 d->outData = d->readAndDecodePackedStreams(false);
2818
2819 int oldPos = 0;
2820 int filesWithoutNames = 0;
2821
2822 // "contents" is used as the default name when the archive was opened from a QIODevice
2823 // instead of a file, meaning there is no actual file name available.
2824 const QString defaultBaseName = d->q->fileName().isEmpty()
2825 ? tr("contents")
2826 : QFileInfo(d->q->fileName()).completeBaseName();
2827
2828 for (int i = 0; i < numFiles; i++) {
2829 FileInfo *fileInfo = d->fileInfos.at(i);
2830
2831 // If the kName property is not present or doesn't contain all the file names,
2832 // then the file name is the name of the archive
2833 if (fileInfo->path.isEmpty()) {
2834 if (numFiles > 1) {
2835 filesWithoutNames++;
2836 fileInfo->path = QStringLiteral("%1_%2").arg(defaultBaseName).arg(filesWithoutNames);
2837 } else {
2838 fileInfo->path = defaultBaseName;
2839 }
2840 }
2841
2842 bool isAnti;
2843 fileInfo->hasStream = !emptyStreamVector[i];
2844 if (fileInfo->hasStream) {
2845 fileInfo->isDir = false;
2846 isAnti = false;
2847 fileInfo->size = d->unpackSizes[sizeIndex];
2848 fileInfo->crc = d->digests[sizeIndex];
2849 fileInfo->crcDefined = d->digestsDefined[sizeIndex];
2850 sizeIndex++;
2851 } else {
2852 fileInfo->isDir = !emptyFileVector[emptyFileIndex];
2853 isAnti = antiFileVector[emptyFileIndex];
2854 emptyFileIndex++;
2855 fileInfo->size = 0;
2856 fileInfo->crcDefined = false;
2857 }
2858 if (numAntiItems != 0) {
2859 d->isAnti.append(isAnti);
2860 }
2861
2862 int access;
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) {
2867 symlink = true;
2868 }
2869 } else {
2870 if (fileInfo->isDir) {
2871 access = S_IFDIR | 0755;
2872 } else {
2873 access = 0100644;
2874 }
2875 }
2876
2877 qint64 pos = 0;
2878 if (!fileInfo->isDir) {
2879 pos = oldPos;
2880 oldPos += fileInfo->size;
2881 }
2882
2883 KArchiveEntry *e;
2884 QString entryName;
2885 int index = fileInfo->path.lastIndexOf(QLatin1Char('/'));
2886 if (index == -1) {
2887 entryName = fileInfo->path;
2888 } else {
2889 entryName = fileInfo->path.mid(index + 1);
2890 }
2891 Q_ASSERT(!entryName.isEmpty());
2892
2893 QDateTime mTime;
2894 if (d->mTimesDefined.size() > i && d->mTimesDefined[i]) {
2895 mTime = KArchivePrivate::time_tToDateTime(toTimeT(d->mTimes[i]));
2896 } else {
2897 mTime = KArchivePrivate::time_tToDateTime(time(nullptr));
2898 }
2899
2900 if (fileInfo->isDir) {
2901 QString path = QDir::cleanPath(fileInfo->path);
2902 const KArchiveEntry *ent = rootDir()->entry(path);
2903 if (ent && ent->isDirectory()) {
2904 e = nullptr;
2905 } else {
2906 e = new KArchiveDirectory(this, entryName, access, mTime, rootDir()->user(), rootDir()->group(), QString() /*symlink*/);
2907 }
2908 } else {
2909 if (!symlink) {
2910 e = new K7ZipFileEntry(this,
2911 entryName,
2912 access,
2913 mTime,
2914 rootDir()->user(),
2915 rootDir()->group(),
2916 QString() /*symlink*/,
2917 pos,
2918 fileInfo->size,
2919 d->outData);
2920 } else {
2921 QString target = QFile::decodeName(d->outData.mid(pos, fileInfo->size));
2922 e = new K7ZipFileEntry(this, entryName, access, mTime, rootDir()->user(), rootDir()->group(), target, 0, 0, nullptr);
2923 }
2924 }
2925
2926 if (e) {
2927 if (index == -1) {
2928 // We don't want to fail opening potentially malformed files, so void the return value
2929 (void)rootDir()->addEntryV2(e);
2930 } else {
2931 QString path = QDir::cleanPath(fileInfo->path.left(index));
2933 (void)d->addEntryV2(e);
2934 }
2935 }
2936 }
2937
2938 return true;
2939}
2940
2942{
2943 // Unnecessary check (already checked by KArchive::close())
2944 if (!isOpen()) {
2945 // qCWarning(KArchiveLog) << "You must open the file before close it\n";
2946 return false;
2947 }
2948
2949 if ((mode() == QIODevice::ReadOnly)) {
2950 return true;
2951 }
2952
2953 d->clear();
2954
2955 Folder *folder = new Folder();
2956
2957 folder->unpackSizes.clear();
2958 folder->unpackSizes.append(d->outData.size());
2959
2960 Folder::FolderInfo *info = new Folder::FolderInfo();
2961
2962 info->numInStreams = 1;
2963 info->numOutStreams = 1;
2964 info->methodID = k_LZMA2;
2965
2966 quint32 dictSize = d->outData.size();
2967
2968 const quint32 kMinReduceSize = (1 << 16);
2969 if (dictSize < kMinReduceSize) {
2970 dictSize = kMinReduceSize;
2971 }
2972
2973 // k_LZMA2 method
2974 int dict;
2975 for (dict = 0; dict < 40; dict++) {
2976 if (dictSize <= lzma2_dic_size_from_prop(dict)) {
2977 break;
2978 }
2979 }
2980 info->properties.append(dict);
2981
2982 folder->folderInfos.append(info);
2983 d->folders.append(folder);
2984
2985 const KArchiveDirectory *dir = directory();
2986 QByteArray data;
2987 d->createItemsFromEntities(dir, QString(), data);
2988 d->outData = data;
2989
2990 folder->unpackCRCDefined = true;
2991 folder->unpackCRC = crc32(0, (Bytef *)(d->outData.data()), d->outData.size());
2992
2993 // compress data
2994 QByteArray encodedData;
2995 if (!d->outData.isEmpty()) {
2996 QByteArray enc;
2997 QBuffer inBuffer(&enc);
2998
2999 KCompressionDevice flt(&inBuffer, false, KCompressionDevice::Xz);
3000 if(!flt.open(QIODevice::WriteOnly)) {
3001 return false;
3002 }
3003
3004 KFilterBase *filter = flt.filterBase();
3005
3006 static_cast<KXzFilter *>(filter)->init(QIODevice::WriteOnly, KXzFilter::LZMA2, info->properties);
3007
3008 const int ret = flt.write(d->outData);
3009 if (ret != d->outData.size()) {
3010 setErrorString(tr("Write error"));
3011 return false;
3012 }
3013
3014 flt.close();
3015 encodedData = inBuffer.data();
3016 }
3017
3018 d->packSizes.append(encodedData.size());
3019
3020 int numUnpackStream = 0;
3021 for (int i = 0; i < d->fileInfos.size(); ++i) {
3022 if (d->fileInfos.at(i)->hasStream) {
3023 numUnpackStream++;
3024 }
3025 }
3026 d->numUnpackStreamsInFolders.append(numUnpackStream);
3027
3028 quint64 headerOffset;
3029 d->writeHeader(headerOffset);
3030
3031 // Encode Header
3032 QByteArray encodedStream;
3033 {
3034 QList<quint64> packSizes;
3035 QList<Folder *> folders;
3036 encodedStream = d->encodeStream(packSizes, folders);
3037
3038 if (folders.isEmpty()) {
3039 // FIXME Not sure why this is an error. Come up with a better message
3040 setErrorString(tr("Failed while encoding header"));
3041 return false;
3042 }
3043
3044 d->header.clear();
3045
3046 d->writeByte(kEncodedHeader);
3047 QList<bool> emptyDefined;
3048 QList<quint32> emptyCrcs;
3049 d->writePackInfo(headerOffset, packSizes, emptyDefined, emptyCrcs);
3050 d->writeUnpackInfo(folders);
3051 d->writeByte(kEnd);
3052 for (int i = 0; i < packSizes.size(); i++) {
3053 headerOffset += packSizes.at(i);
3054 }
3055 qDeleteAll(folders);
3056 }
3057 // end encode header
3058
3059 quint64 nextHeaderSize = d->header.size();
3060 quint32 nextHeaderCRC = crc32(0, (Bytef *)(d->header.data()), d->header.size());
3061 quint64 nextHeaderOffset = headerOffset;
3062
3063 device()->seek(0);
3064 d->writeSignature();
3065 d->writeStartHeader(nextHeaderSize, nextHeaderCRC, nextHeaderOffset);
3066 device()->write(encodedData.data(), encodedData.size());
3067 device()->write(encodedStream.data(), encodedStream.size());
3068 device()->write(d->header.data(), d->header.size());
3069
3070 return true;
3071}
3072
3073bool K7Zip::doFinishWriting(qint64 size)
3074{
3075 d->m_currentFile->setSize(size);
3076 d->m_currentFile = nullptr;
3077
3078 return true;
3079}
3080
3081bool K7Zip::doWriteData(const char *data, qint64 size)
3082{
3083 if (!d->m_currentFile) {
3084 setErrorString(tr("No file currently selected"));
3085 return false;
3086 }
3087
3088 if (d->m_currentFile->position() == d->outData.size()) {
3089 d->outData.append(data, size);
3090 } else {
3091 d->outData.remove(d->m_currentFile->position(), d->m_currentFile->size());
3092 d->outData.insert(d->m_currentFile->position(), data, size);
3093 }
3094
3095 return true;
3096}
3097
3099 const QString &user,
3100 const QString &group,
3101 qint64 /*size*/,
3102 mode_t perm,
3103 const QDateTime & /*atime*/,
3104 const QDateTime &mtime,
3105 const QDateTime & /*ctime*/)
3106{
3107 if (!isOpen()) {
3108 setErrorString(tr("Application error: 7-Zip file must be open before being written into"));
3109 qCWarning(KArchiveLog) << "doPrepareWriting failed: !isOpen()";
3110 return false;
3111 }
3112
3113 if (!(mode() & QIODevice::WriteOnly)) {
3114 setErrorString(tr("Application error: attempted to write into non-writable 7-Zip file"));
3115 qCWarning(KArchiveLog) << "doPrepareWriting failed: !(mode() & QIODevice::WriteOnly)";
3116 return false;
3117 }
3118
3119 // Find or create parent dir
3120 KArchiveDirectory *parentDir = rootDir();
3121 // QString fileName( name );
3122 // In some files we can find dir/./file => call cleanPath
3124 int i = name.lastIndexOf(QLatin1Char('/'));
3125 if (i != -1) {
3126 QString dir = name.left(i);
3127 fileName = name.mid(i + 1);
3128 parentDir = findOrCreate(dir);
3129 }
3130
3131 // test if the entry already exist
3132 const KArchiveEntry *entry = parentDir->entry(fileName);
3133 if (!entry) {
3134 K7ZipFileEntry *e =
3135 new K7ZipFileEntry(this, fileName, perm, mtime, user, group, QString() /*symlink*/, d->outData.size(), 0 /*unknown yet*/, d->outData);
3136 if (!parentDir->addEntryV2(e)) {
3137 return false;
3138 }
3139 d->m_entryList << e;
3140 d->m_currentFile = e;
3141 } else {
3142 // TODO : find and replace in m_entryList
3143 // d->m_currentFile = static_cast<K7ZipFileEntry*>(entry);
3144 }
3145
3146 return true;
3147}
3148
3150 const QString &user,
3151 const QString &group,
3152 mode_t perm,
3153 const QDateTime & /*atime*/,
3154 const QDateTime &mtime,
3155 const QDateTime & /*ctime*/)
3156{
3157 if (!isOpen()) {
3158 setErrorString(tr("Application error: 7-Zip file must be open before being written into"));
3159 qCWarning(KArchiveLog) << "doWriteDir failed: !isOpen()";
3160 return false;
3161 }
3162
3163 if (!(mode() & QIODevice::WriteOnly)) {
3164 // qCWarning(KArchiveLog) << "You must open the tar file for writing\n";
3165 return false;
3166 }
3167
3168 // In some tar files we can find dir/./ => call cleanPath
3169 QString dirName(QDir::cleanPath(name));
3170
3171 // Remove trailing '/'
3172 if (dirName.endsWith(QLatin1Char('/'))) {
3173 dirName.remove(dirName.size() - 1, 1);
3174 }
3175
3176 KArchiveDirectory *parentDir = rootDir();
3177 int i = dirName.lastIndexOf(QLatin1Char('/'));
3178 if (i != -1) {
3179 QString dir = name.left(i);
3180 dirName = name.mid(i + 1);
3181 parentDir = findOrCreate(dir);
3182 }
3183
3184 KArchiveDirectory *e = new KArchiveDirectory(this, dirName, perm, mtime, user, group, QString() /*symlink*/);
3185 return parentDir->addEntryV2(e);
3186}
3187
3189 const QString &target,
3190 const QString &user,
3191 const QString &group,
3192 mode_t perm,
3193 const QDateTime & /*atime*/,
3194 const QDateTime &mtime,
3195 const QDateTime & /*ctime*/)
3196{
3197 if (!isOpen()) {
3198 setErrorString(tr("Application error: 7-Zip file must be open before being written into"));
3199 qCWarning(KArchiveLog) << "doWriteSymLink failed: !isOpen()";
3200 return false;
3201 }
3202
3203 if (!(mode() & QIODevice::WriteOnly)) {
3204 setErrorString(tr("Application error: attempted to write into non-writable 7-Zip file"));
3205 qCWarning(KArchiveLog) << "doWriteSymLink failed: !(mode() & QIODevice::WriteOnly)";
3206 return false;
3207 }
3208
3209 // Find or create parent dir
3210 KArchiveDirectory *parentDir = rootDir();
3211 // In some files we can find dir/./file => call cleanPath
3213 int i = name.lastIndexOf(QLatin1Char('/'));
3214 if (i != -1) {
3215 QString dir = name.left(i);
3216 fileName = name.mid(i + 1);
3217 parentDir = findOrCreate(dir);
3218 }
3219 QByteArray encodedTarget = QFile::encodeName(target);
3220
3221 K7ZipFileEntry *e = new K7ZipFileEntry(this, fileName, perm, mtime, user, group, target, 0, 0, nullptr);
3222 d->outData.append(encodedTarget);
3223
3224 if (!parentDir->addEntryV2(e)) {
3225 return false;
3226 }
3227
3228 d->m_entryList << e;
3229
3230 return true;
3231}
3232
3233void K7Zip::virtual_hook(int id, void *data)
3234{
3235 KArchive::virtual_hook(id, data);
3236}
A class for reading / writing p7zip archives.
Definition k7zip.h:19
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.
Definition k7zip.cpp:3149
bool doWriteData(const char *data, qint64 size) override
Reimplemented from KArchive.
Definition k7zip.cpp:3081
void setPassword(const QString &password)
Sets the password to use for encrypted archives.
Definition k7zip.cpp:583
bool doFinishWriting(qint64 size) override
Reimplemented from KArchive.
Definition k7zip.cpp:3073
bool openArchive(QIODevice::OpenMode mode) override
Opens the archive for reading.
Definition k7zip.cpp:2489
bool closeArchive() override
Closes the archive.
Definition k7zip.cpp:2941
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.
Definition k7zip.cpp:3188
~K7Zip() override
If the archive is still opened, then it will be closed automatically by the destructor.
Definition k7zip.cpp:574
K7Zip(const QString &filename)
Creates an instance that operates on the given filename using the compression filter associated to gi...
Definition k7zip.cpp:561
bool passwordNeeded() const
Whether the archive needs a password to be opened.
Definition k7zip.cpp:587
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.
Definition k7zip.cpp:3098
Represents a directory entry in a KArchive.
bool addEntryV2(KArchiveEntry *)
Definition karchive.cpp:928
const KArchiveEntry * entry(const QString &name) const
Returns the entry in the archive with the given name.
Definition karchive.cpp:908
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.
Definition karchive.cpp:721
QString user() const
User who created the file.
Definition karchive.cpp:726
virtual bool isDirectory() const
Checks whether the entry is a directory.
Definition karchive.cpp:746
QDateTime date() const
Creation date of the file.
Definition karchive.cpp:711
QString group() const
Group of the user who created the file.
Definition karchive.cpp:731
QString name() const
Name of the file without path.
Definition karchive.cpp:716
QString symLinkTarget() const
Symlink if there is one.
Definition karchive.cpp:736
virtual bool isFile() const
Checks whether the entry is a file.
Definition karchive.cpp:741
Represents a file entry in a KArchive.
qint64 size() const
Size of the data.
Definition karchive.cpp:796
qint64 position() const
Position of the data in the [uncompressed] archive.
Definition karchive.cpp:791
QIODevice * device() const
The underlying device.
Definition karchive.cpp:630
virtual bool close()
Closes the archive.
Definition karchive.cpp:214
virtual KArchiveDirectory * rootDir()
Retrieves or create the root directory.
Definition karchive.cpp:517
KArchive(const QString &fileName)
Base constructor (protected since this is a pure virtual class).
Definition karchive.cpp:115
const KArchiveDirectory * directory() const
If an archive is opened for reading, then the contents of the archive can be accessed via this functi...
Definition karchive.cpp:256
KArchiveDirectory * findOrCreate(const QString &path)
Ensures that path exists, create otherwise.
Definition karchive.cpp:529
QIODevice::OpenMode mode() const
Returns the mode in which the archive was opened.
Definition karchive.cpp:625
QString fileName() const
The name of the archive file, as passed to the constructor that takes a fileName, or an empty string ...
Definition karchive.cpp:640
bool isOpen() const
Checks whether the archive is open.
Definition karchive.cpp:635
void setErrorString(const QString &errorStr)
Sets error description.
Definition karchive.cpp:482
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.
Definition kfilterbase.h:27
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()
QCA_EXPORT void init()
const QByteArray & data() const const
QByteArray & append(QByteArrayView data)
void clear()
const char * constData() const const
char * data()
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)
typedef ConstIterator
void append(QList< T > &&value)
const_reference at(qsizetype i) const const
iterator begin()
void clear()
pointer data()
iterator end()
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)
QTimeZone utc()
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 12:10:52 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.