| 1 | /* | 
 
 
 
 
 | 2 | Copyright (C) 2005-2014 Sergey A. Tachenov | 
 
 
 
 
 | 3 |  | 
 
 
 
 
 | 4 | This file is part of QuaZIP. | 
 
 
 
 
 | 5 |  | 
 
 
 
 
 | 6 | QuaZIP is free software: you can redistribute it and/or modify | 
 
 
 
 
 | 7 | it under the terms of the GNU Lesser General Public License as published by | 
 
 
 
 
 | 8 | the Free Software Foundation, either version 2.1 of the License, or | 
 
 
 
 
 | 9 | (at your option) any later version. | 
 
 
 
 
 | 10 |  | 
 
 
 
 
 | 11 | QuaZIP is distributed in the hope that it will be useful, | 
 
 
 
 
 | 12 | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 
 
 
 
 | 13 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 
 
 
 
 | 14 | GNU Lesser General Public License for more details. | 
 
 
 
 
 | 15 |  | 
 
 
 
 
 | 16 | You should have received a copy of the GNU Lesser General Public License | 
 
 
 
 
 | 17 | along with QuaZIP.  If not, see <http://www.gnu.org/licenses/>. | 
 
 
 
 
 | 18 |  | 
 
 
 
 
 | 19 | See COPYING file for the full LGPL text. | 
 
 
 
 
 | 20 |  | 
 
 
 
 
 | 21 | Original ZIP package is copyrighted by Gilles Vollant and contributors, | 
 
 
 
 
 | 22 | see quazip/(un)zip.h files for details. Basically it's the zlib license. | 
 
 
 
 
 | 23 | */ | 
 
 
 
 
 | 24 |  | 
 
 
 
 
 | 25 | #include <QFileInfo> | 
 
 
 
 
 | 26 |  | 
 
 
 
 
 | 27 | #include "quazipnewinfo.h" | 
 
 
 
 
 | 28 |  | 
 
 
 
 
 | 29 | #include <string.h> | 
 
 
 
 
 | 30 |  | 
 
 
 
 
 | 31 | static void QuaZipNewInfo_setPermissions(QuaZipNewInfo *info, | 
 
 
 
 
 | 32 | QFile::Permissions perm, bool isDir) | 
 
 
 
 
 | 33 | { | 
 
 
 
 
 | 34 | quint32 uPerm = isDir ? 0040000 : 0100000; | 
 
 
 
 
 | 35 | if ((perm & QFile::ReadOwner) != 0) | 
 
 
 
 
 | 36 | uPerm |= 0400; | 
 
 
 
 
 | 37 | if ((perm & QFile::WriteOwner) != 0) | 
 
 
 
 
 | 38 | uPerm |= 0200; | 
 
 
 
 
 | 39 | if ((perm & QFile::ExeOwner) != 0) | 
 
 
 
 
 | 40 | uPerm |= 0100; | 
 
 
 
 
 | 41 | if ((perm & QFile::ReadGroup) != 0) | 
 
 
 
 
 | 42 | uPerm |= 0040; | 
 
 
 
 
 | 43 | if ((perm & QFile::WriteGroup) != 0) | 
 
 
 
 
 | 44 | uPerm |= 0020; | 
 
 
 
 
 | 45 | if ((perm & QFile::ExeGroup) != 0) | 
 
 
 
 
 | 46 | uPerm |= 0010; | 
 
 
 
 
 | 47 | if ((perm & QFile::ReadOther) != 0) | 
 
 
 
 
 | 48 | uPerm |= 0004; | 
 
 
 
 
 | 49 | if ((perm & QFile::WriteOther) != 0) | 
 
 
 
 
 | 50 | uPerm |= 0002; | 
 
 
 
 
 | 51 | if ((perm & QFile::ExeOther) != 0) | 
 
 
 
 
 | 52 | uPerm |= 0001; | 
 
 
 
 
 | 53 | info->externalAttr = (info->externalAttr & ~0xFFFF0000u) | (uPerm << 16); | 
 
 
 
 
 | 54 | } | 
 
 
 
 
 | 55 |  | 
 
 
 
 
 | 56 | template<typename FileInfo> | 
 
 
 
 
 | 57 | void QuaZipNewInfo_init(QuaZipNewInfo &self, const FileInfo &existing) | 
 
 
 
 
 | 58 | { | 
 
 
 
 
 | 59 | self.name = existing.name; | 
 
 
 
 
 | 60 | self.dateTime = existing.dateTime; | 
 
 
 
 
 | 61 | self.internalAttr = existing.internalAttr; | 
 
 
 
 
 | 62 | self.externalAttr = existing.externalAttr; | 
 
 
 
 
 | 63 | self.comment = existing.comment; | 
 
 
 
 
 | 64 | self.extraLocal = existing.extra; | 
 
 
 
 
 | 65 | self.extraGlobal = existing.extra; | 
 
 
 
 
 | 66 | self.uncompressedSize = existing.uncompressedSize; | 
 
 
 
 
 | 67 | } | 
 
 
 
 
 | 68 |  | 
 
 
 
 
 | 69 | QuaZipNewInfo::QuaZipNewInfo(const QuaZipFileInfo &existing) | 
 
 
 
 
 | 70 | { | 
 
 
 
 
 | 71 | QuaZipNewInfo_init(*this, existing); | 
 
 
 
 
 | 72 | } | 
 
 
 
 
 | 73 |  | 
 
 
 
 
 | 74 | QuaZipNewInfo::QuaZipNewInfo(const QuaZipFileInfo64 &existing) | 
 
 
 
 
 | 75 | { | 
 
 
 
 
 | 76 | QuaZipNewInfo_init(*this, existing); | 
 
 
 
 
 | 77 | } | 
 
 
 
 
 | 78 |  | 
 
 
 
 
 | 79 | QuaZipNewInfo::QuaZipNewInfo(const QString& name): | 
 
 
 
 
 | 80 | name(name), dateTime(QDateTime::currentDateTime()), internalAttr(0), externalAttr(0), | 
 
 
 
 
 | 81 | uncompressedSize(0) | 
 
 
 
 
 | 82 | { | 
 
 
 
 
 | 83 | } | 
 
 
 
 
 | 84 |  | 
 
 
 
 
 | 85 | QuaZipNewInfo::QuaZipNewInfo(const QString& name, const QString& file): | 
 
 
 
 
 | 86 | name(name), internalAttr(0), externalAttr(0), uncompressedSize(0) | 
 
 
 
 
 | 87 | { | 
 
 
 
 
 | 88 | QFileInfo info(file); | 
 
 
 
 
 | 89 | QDateTime lm = info.lastModified(); | 
 
 
 
 
 | 90 | if (!info.exists()) { | 
 
 
 
 
 | 91 | dateTime = QDateTime::currentDateTime(); | 
 
 
 
 
 | 92 | } else { | 
 
 
 
 
 | 93 | dateTime = lm; | 
 
 
 
 
 | 94 | QuaZipNewInfo_setPermissions(this, info.permissions(), info.isDir()); | 
 
 
 
 
 | 95 | } | 
 
 
 
 
 | 96 | } | 
 
 
 
 
 | 97 |  | 
 
 
 
 
 | 98 | void QuaZipNewInfo::setFileDateTime(const QString& file) | 
 
 
 
 
 | 99 | { | 
 
 
 
 
 | 100 | QFileInfo info(file); | 
 
 
 
 
 | 101 | QDateTime lm = info.lastModified(); | 
 
 
 
 
 | 102 | if (info.exists()) | 
 
 
 
 
 | 103 | dateTime = lm; | 
 
 
 
 
 | 104 | } | 
 
 
 
 
 | 105 |  | 
 
 
 
 
 | 106 | void QuaZipNewInfo::setFilePermissions(const QString &file) | 
 
 
 
 
 | 107 | { | 
 
 
 
 
 | 108 | QFileInfo info = QFileInfo(file); | 
 
 
 
 
 | 109 | QFile::Permissions perm = info.permissions(); | 
 
 
 
 
 | 110 | QuaZipNewInfo_setPermissions(this, perm, info.isDir()); | 
 
 
 
 
 | 111 | } | 
 
 
 
 
 | 112 |  | 
 
 
 
 
 | 113 | void QuaZipNewInfo::setPermissions(QFile::Permissions permissions) | 
 
 
 
 
 | 114 | { | 
 
 
 
 
 | 115 | QuaZipNewInfo_setPermissions(this, permissions, name.endsWith('/')); | 
 
 
 
 
 | 116 | } | 
 
 
 
 
 | 117 |  | 
 
 
 
 
 | 118 | void QuaZipNewInfo::setFileNTFSTimes(const QString &fileName) | 
 
 
 
 
 | 119 | { | 
 
 
 
 
 | 120 | QFileInfo fi(fileName); | 
 
 
 
 
 | 121 | if (!fi.exists()) { | 
 
 
 
 
 | 122 | qWarning("QuaZipNewInfo::setFileNTFSTimes(): '%s' doesn't exist", | 
 
 
 
 
 | 123 | fileName.toUtf8().constData()); | 
 
 
 
 
 | 124 | return; | 
 
 
 
 
 | 125 | } | 
 
 
 
 
 | 126 | setFileNTFSmTime(fi.lastModified()); | 
 
 
 
 
 | 127 | setFileNTFSaTime(fi.lastRead()); | 
 
 
 
 
 | 128 | setFileNTFScTime(fi.created()); | 
 
 
 
 
 | 129 | } | 
 
 
 
 
 | 130 |  | 
 
 
 
 
 | 131 | static void setNTFSTime(QByteArray &extra, const QDateTime &time, int position, | 
 
 
 
 
 | 132 | int fineTicks) { | 
 
 
 
 
 | 133 | int ntfsPos = -1, timesPos = -1; | 
 
 
 
 
 | 134 | unsigned ntfsLength = 0, ntfsTimesLength = 0; | 
 
 
 
 
 | 135 | for (int i = 0; i <= extra.size() - 4; ) { | 
 
 
 
 
 | 136 | unsigned type = static_cast<unsigned>(static_cast<unsigned char>( | 
 
 
 
 
 | 137 | extra.at(i))) | 
 
 
 
 
 | 138 | | (static_cast<unsigned>(static_cast<unsigned char>( | 
 
 
 
 
 | 139 | extra.at(i + 1))) << 8); | 
 
 
 
 
 | 140 | i += 2; | 
 
 
 
 
 | 141 | unsigned length = static_cast<unsigned>(static_cast<unsigned char>( | 
 
 
 
 
 | 142 | extra.at(i))) | 
 
 
 
 
 | 143 | | (static_cast<unsigned>(static_cast<unsigned char>( | 
 
 
 
 
 | 144 | extra.at(i + 1))) << 8); | 
 
 
 
 
 | 145 | i += 2; | 
 
 
 
 
 | 146 | if (type == QUAZIP_EXTRA_NTFS_MAGIC) { | 
 
 
 
 
 | 147 | ntfsPos = i - 4; // the beginning of the NTFS record | 
 
 
 
 
 | 148 | ntfsLength = length; | 
 
 
 
 
 | 149 | if (length <= 4) { | 
 
 
 
 
 | 150 | break; // no times in the NTFS record | 
 
 
 
 
 | 151 | } | 
 
 
 
 
 | 152 | i += 4; // reserved | 
 
 
 
 
 | 153 | while (i <= extra.size() - 4) { | 
 
 
 
 
 | 154 | unsigned tag = static_cast<unsigned>( | 
 
 
 
 
 | 155 | static_cast<unsigned char>(extra.at(i))) | 
 
 
 
 
 | 156 | | (static_cast<unsigned>( | 
 
 
 
 
 | 157 | static_cast<unsigned char>(extra.at(i + 1))) | 
 
 
 
 
 | 158 | << 8); | 
 
 
 
 
 | 159 | i += 2; | 
 
 
 
 
 | 160 | unsigned tagsize = static_cast<unsigned>( | 
 
 
 
 
 | 161 | static_cast<unsigned char>(extra.at(i))) | 
 
 
 
 
 | 162 | | (static_cast<unsigned>( | 
 
 
 
 
 | 163 | static_cast<unsigned char>(extra.at(i + 1))) | 
 
 
 
 
 | 164 | << 8); | 
 
 
 
 
 | 165 | i += 2; | 
 
 
 
 
 | 166 | if (tag == QUAZIP_EXTRA_NTFS_TIME_MAGIC) { | 
 
 
 
 
 | 167 | timesPos = i - 4; // the beginning of the NTFS times tag | 
 
 
 
 
 | 168 | ntfsTimesLength = tagsize; | 
 
 
 
 
 | 169 | break; | 
 
 
 
 
 | 170 | } else { | 
 
 
 
 
 | 171 | i += tagsize; | 
 
 
 
 
 | 172 | } | 
 
 
 
 
 | 173 | } | 
 
 
 
 
 | 174 | break; // I ain't going to search for yet another NTFS record! | 
 
 
 
 
 | 175 | } else { | 
 
 
 
 
 | 176 | i += length; | 
 
 
 
 
 | 177 | } | 
 
 
 
 
 | 178 | } | 
 
 
 
 
 | 179 | if (ntfsPos == -1) { | 
 
 
 
 
 | 180 | // No NTFS record, need to create one. | 
 
 
 
 
 | 181 | ntfsPos = extra.size(); | 
 
 
 
 
 | 182 | ntfsLength = 32; | 
 
 
 
 
 | 183 | extra.resize(extra.size() + 4 + ntfsLength); | 
 
 
 
 
 | 184 | // the NTFS record header | 
 
 
 
 
 | 185 | extra[ntfsPos] = static_cast<char>(QUAZIP_EXTRA_NTFS_MAGIC); | 
 
 
 
 
 | 186 | extra[ntfsPos + 1] = static_cast<char>(QUAZIP_EXTRA_NTFS_MAGIC >> 8); | 
 
 
 
 
 | 187 | extra[ntfsPos + 2] = 32; // the 2-byte size in LittleEndian | 
 
 
 
 
 | 188 | extra[ntfsPos + 3] = 0; | 
 
 
 
 
 | 189 | // zero the record | 
 
 
 
 
 | 190 | memset(extra.data() + ntfsPos + 4, 0, 32); | 
 
 
 
 
 | 191 | timesPos = ntfsPos + 8; | 
 
 
 
 
 | 192 | // now set the tag data | 
 
 
 
 
 | 193 | extra[timesPos] = static_cast<char>(QUAZIP_EXTRA_NTFS_TIME_MAGIC); | 
 
 
 
 
 | 194 | extra[timesPos + 1] = static_cast<char>(QUAZIP_EXTRA_NTFS_TIME_MAGIC | 
 
 
 
 
 | 195 | >> 8); | 
 
 
 
 
 | 196 | // the size: | 
 
 
 
 
 | 197 | extra[timesPos + 2] = 24; | 
 
 
 
 
 | 198 | extra[timesPos + 3] = 0; | 
 
 
 
 
 | 199 | ntfsTimesLength = 24; | 
 
 
 
 
 | 200 | } | 
 
 
 
 
 | 201 | if (timesPos == -1) { | 
 
 
 
 
 | 202 | // No time tag in the NTFS record, need to add one. | 
 
 
 
 
 | 203 | timesPos = ntfsPos + 4 + ntfsLength; | 
 
 
 
 
 | 204 | extra.resize(extra.size() + 28); | 
 
 
 
 
 | 205 | // Now we need to move the rest of the field | 
 
 
 
 
 | 206 | // (possibly zero bytes, but memmove() is OK with that). | 
 
 
 
 
 | 207 | // 0 ......... ntfsPos .. ntfsPos + 4   ... timesPos | 
 
 
 
 
 | 208 | // <some data> <header>   <NTFS record>     <need-to-move data>    <end> | 
 
 
 
 
 | 209 | memmove(extra.data() + timesPos + 28, extra.data() + timesPos, | 
 
 
 
 
 | 210 | extra.size() - 28 - timesPos); | 
 
 
 
 
 | 211 | ntfsLength += 28; | 
 
 
 
 
 | 212 | // now set the tag data | 
 
 
 
 
 | 213 | extra[timesPos] = static_cast<char>(QUAZIP_EXTRA_NTFS_TIME_MAGIC); | 
 
 
 
 
 | 214 | extra[timesPos + 1] = static_cast<char>(QUAZIP_EXTRA_NTFS_TIME_MAGIC | 
 
 
 
 
 | 215 | >> 8); | 
 
 
 
 
 | 216 | // the size: | 
 
 
 
 
 | 217 | extra[timesPos + 2] = 24; | 
 
 
 
 
 | 218 | extra[timesPos + 3] = 0; | 
 
 
 
 
 | 219 | // zero the record | 
 
 
 
 
 | 220 | memset(extra.data() + timesPos + 4, 0, 24); | 
 
 
 
 
 | 221 | ntfsTimesLength = 24; | 
 
 
 
 
 | 222 | } | 
 
 
 
 
 | 223 | if (ntfsTimesLength < 24) { | 
 
 
 
 
 | 224 | // Broken times field. OK, this is really unlikely, but just in case... | 
 
 
 
 
 | 225 | size_t timesEnd = timesPos + 4 + ntfsTimesLength; | 
 
 
 
 
 | 226 | extra.resize(extra.size() + (24 - ntfsTimesLength)); | 
 
 
 
 
 | 227 | // Move it! | 
 
 
 
 
 | 228 | // 0 ......... timesPos .... timesPos + 4 .. timesEnd | 
 
 
 
 
 | 229 | // <some data> <time header> <broken times> <need-to-move data> <end> | 
 
 
 
 
 | 230 | memmove(extra.data() + timesEnd + (24 - ntfsTimesLength), | 
 
 
 
 
 | 231 | extra.data() + timesEnd, | 
 
 
 
 
 | 232 | extra.size() - (24 - ntfsTimesLength) - timesEnd); | 
 
 
 
 
 | 233 | // Now we have to increase the NTFS record and time tag lengths. | 
 
 
 
 
 | 234 | ntfsLength += (24 - ntfsTimesLength); | 
 
 
 
 
 | 235 | ntfsTimesLength = 24; | 
 
 
 
 
 | 236 | extra[ntfsPos + 2] = static_cast<char>(ntfsLength); | 
 
 
 
 
 | 237 | extra[ntfsPos + 3] = static_cast<char>(ntfsLength >> 8); | 
 
 
 
 
 | 238 | extra[timesPos + 2] = static_cast<char>(ntfsTimesLength); | 
 
 
 
 
 | 239 | extra[timesPos + 3] = static_cast<char>(ntfsTimesLength >> 8); | 
 
 
 
 
 | 240 | } | 
 
 
 
 
 | 241 | QDateTime base(QDate(1601, 1, 1), QTime(0, 0), Qt::UTC); | 
 
 
 
 
 | 242 | #if (QT_VERSION >= 0x040700) | 
 
 
 
 
 | 243 | quint64 ticks = base.msecsTo(time) * 10000 + fineTicks; | 
 
 
 
 
 | 244 | #else | 
 
 
 
 
 | 245 | QDateTime utc = time.toUTC(); | 
 
 
 
 
 | 246 | quint64 ticks = (static_cast<qint64>(base.date().daysTo(utc.date())) | 
 
 
 
 
 | 247 | * Q_INT64_C(86400000) | 
 
 
 
 
 | 248 | + static_cast<qint64>(base.time().msecsTo(utc.time()))) | 
 
 
 
 
 | 249 | * Q_INT64_C(10000) + fineTicks; | 
 
 
 
 
 | 250 | #endif | 
 
 
 
 
 | 251 | extra[timesPos + 4 + position] = static_cast<char>(ticks); | 
 
 
 
 
 | 252 | extra[timesPos + 5 + position] = static_cast<char>(ticks >> 8); | 
 
 
 
 
 | 253 | extra[timesPos + 6 + position] = static_cast<char>(ticks >> 16); | 
 
 
 
 
 | 254 | extra[timesPos + 7 + position] = static_cast<char>(ticks >> 24); | 
 
 
 
 
 | 255 | extra[timesPos + 8 + position] = static_cast<char>(ticks >> 32); | 
 
 
 
 
 | 256 | extra[timesPos + 9 + position] = static_cast<char>(ticks >> 40); | 
 
 
 
 
 | 257 | extra[timesPos + 10 + position] = static_cast<char>(ticks >> 48); | 
 
 
 
 
 | 258 | extra[timesPos + 11 + position] = static_cast<char>(ticks >> 56); | 
 
 
 
 
 | 259 | } | 
 
 
 
 
 | 260 |  | 
 
 
 
 
 | 261 | void QuaZipNewInfo::setFileNTFSmTime(const QDateTime &mTime, int fineTicks) | 
 
 
 
 
 | 262 | { | 
 
 
 
 
 | 263 | setNTFSTime(extraLocal, mTime, 0, fineTicks); | 
 
 
 
 
 | 264 | setNTFSTime(extraGlobal, mTime, 0, fineTicks); | 
 
 
 
 
 | 265 | } | 
 
 
 
 
 | 266 |  | 
 
 
 
 
 | 267 | void QuaZipNewInfo::setFileNTFSaTime(const QDateTime &aTime, int fineTicks) | 
 
 
 
 
 | 268 | { | 
 
 
 
 
 | 269 | setNTFSTime(extraLocal, aTime, 8, fineTicks); | 
 
 
 
 
 | 270 | setNTFSTime(extraGlobal, aTime, 8, fineTicks); | 
 
 
 
 
 | 271 | } | 
 
 
 
 
 | 272 |  | 
 
 
 
 
 | 273 | void QuaZipNewInfo::setFileNTFScTime(const QDateTime &cTime, int fineTicks) | 
 
 
 
 
 | 274 | { | 
 
 
 
 
 | 275 | setNTFSTime(extraLocal, cTime, 16, fineTicks); | 
 
 
 
 
 | 276 | setNTFSTime(extraGlobal, cTime, 16, fineTicks); | 
 
 
 
 
 | 277 | } |