2021-10-15 15:41:16 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Copyright (C) 2021 Maschell
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
****************************************************************************/
|
|
|
|
#include <utils/logger.h>
|
|
|
|
#include "WUXFileWriter.h"
|
|
|
|
#include "WUDDumperState.h"
|
|
|
|
|
2021-10-16 13:03:32 +02:00
|
|
|
WUXFileWriter::WUXFileWriter(const char *path, int32_t cacheSize, int32_t sectorSize, bool split) :
|
|
|
|
WUDFileWriter(path, cacheSize, sectorSize, split) {
|
2021-10-15 15:41:16 +02:00
|
|
|
wuxHeader_t wuxHeader = {0};
|
|
|
|
wuxHeader.magic0 = WUX_MAGIC_0;
|
|
|
|
wuxHeader.magic1 = WUX_MAGIC_1;
|
|
|
|
wuxHeader.sectorSize = swap_uint32(this->sectorSize);
|
|
|
|
wuxHeader.uncompressedSize = swap_uint64(WUD_FILE_SIZE);
|
|
|
|
wuxHeader.flags = 0;
|
|
|
|
|
|
|
|
this->write((uint8_t *) &wuxHeader, sizeof(wuxHeader_t));
|
|
|
|
this->sectorTableStart = this->tell();
|
|
|
|
this->totalSectorCount = WUD_FILE_SIZE / this->sectorSize;
|
|
|
|
|
|
|
|
this->sectorIndexTable = (void *) malloc(totalSectorCount * 4);
|
|
|
|
if (sectorIndexTable == nullptr) {
|
|
|
|
DEBUG_FUNCTION_LINE("Failed to alloc");
|
|
|
|
WUDFileWriter::close();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
memset(this->sectorIndexTable, 0, totalSectorCount * 4);
|
|
|
|
|
|
|
|
if (this->write((uint8_t *) this->sectorIndexTable, totalSectorCount * 4) < 0) {
|
|
|
|
WUDFileWriter::close();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->sectorTableEnd = this->tell();
|
|
|
|
uint64_t tableEnd = this->sectorTableEnd;
|
|
|
|
|
|
|
|
this->sectorTableEnd += this->sectorSize - 1;
|
|
|
|
this->sectorTableEnd -= (this->sectorTableEnd % this->sectorSize);
|
|
|
|
|
|
|
|
uint64_t padding = this->sectorTableEnd - tableEnd;
|
|
|
|
auto *paddingData = (uint8_t *) malloc(padding);
|
|
|
|
memset(paddingData, 0, padding);
|
|
|
|
this->write(reinterpret_cast<const uint8_t *>(paddingData), padding);
|
|
|
|
free(paddingData);
|
|
|
|
this->hashMap.clear();
|
2021-10-16 13:03:32 +02:00
|
|
|
flush();
|
2021-10-15 15:41:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int32_t WUXFileWriter::writeSector(const uint8_t *buffer, uint32_t numberOfSectors) {
|
|
|
|
char hashOut[32];
|
|
|
|
int32_t curWrittenSectors = 0;
|
|
|
|
for (uint32_t i = 0; i < numberOfSectors; i++) {
|
|
|
|
uint32_t addr = ((uint32_t) buffer) + (i * this->sectorSize);
|
|
|
|
calculateHash256(reinterpret_cast<unsigned char *>(addr), this->sectorSize, reinterpret_cast<unsigned char *>(hashOut));
|
|
|
|
char tmp[34];
|
|
|
|
auto *test = (uint32_t *) hashOut;
|
|
|
|
snprintf(tmp, 33, "%08X%08X%08X%08X", test[0], test[1], test[2], test[3]);
|
|
|
|
std::string hash(tmp);
|
|
|
|
|
|
|
|
auto *indexTable = (uint32_t *) this->sectorIndexTable;
|
|
|
|
|
|
|
|
auto it = hashMap.find(hash);
|
|
|
|
if (it != hashMap.end()) {
|
|
|
|
indexTable[this->currentSector] = swap_uint32(this->hashMap[hash]);
|
|
|
|
} else {
|
|
|
|
indexTable[this->currentSector] = swap_uint32(this->writtenSector);
|
|
|
|
hashMap[hash] = writtenSector;
|
|
|
|
if (isOpen()) {
|
|
|
|
if (!write((uint8_t *) addr, this->sectorSize)) {
|
|
|
|
DEBUG_FUNCTION_LINE("Write failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this->writtenSector++;
|
|
|
|
curWrittenSectors++;
|
|
|
|
}
|
|
|
|
this->currentSector++;
|
|
|
|
}
|
|
|
|
return curWrittenSectors;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WUXFileWriter::writeSectorIndexTable() {
|
|
|
|
if (this->isOpen()) {
|
2021-10-16 13:03:32 +02:00
|
|
|
flush();
|
2021-10-15 15:41:16 +02:00
|
|
|
// We need to make sure to call CFile::seek!
|
2021-10-16 13:03:32 +02:00
|
|
|
seek((int64_t) sectorTableStart, SEEK_SET_BASE_CLASS);
|
|
|
|
write((uint8_t *) sectorIndexTable, totalSectorCount * 4);
|
|
|
|
flush();
|
2021-10-15 15:41:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WUXFileWriter::~WUXFileWriter() {
|
|
|
|
WUXFileWriter::close();
|
|
|
|
free(sectorIndexTable);
|
|
|
|
}
|
2021-10-16 13:03:32 +02:00
|
|
|
|
|
|
|
void WUXFileWriter::finalize() {
|
|
|
|
WUDFileWriter::finalize();
|
|
|
|
writeSectorIndexTable();
|
|
|
|
WUXFileWriter::close();
|
|
|
|
|
|
|
|
}
|