2025-08-04 16:31:29 +01:00
|
|
|
/*
|
|
|
|
|
* This file is part of the Aaru Data Preservation Suite.
|
|
|
|
|
* Copyright (c) 2019-2025 Natalia Portillo.
|
|
|
|
|
*
|
|
|
|
|
* This library is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU Lesser General Public License as
|
|
|
|
|
* published by the Free Software Foundation; either version 2.1 of the
|
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This library 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
|
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
#include "aaruformat.h"
|
2025-08-05 23:10:54 +01:00
|
|
|
#include "internal.h"
|
2025-08-13 20:16:42 +01:00
|
|
|
#include "log.h"
|
2025-08-04 16:31:29 +01:00
|
|
|
|
|
|
|
|
int32_t process_ddt_v2(aaruformatContext *ctx, IndexEntry *entry, bool *foundUserDataDdt)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Entering process_ddt_v2(%p, %p, %d)", ctx, entry, *foundUserDataDdt);
|
|
|
|
|
|
2025-08-04 16:31:29 +01:00
|
|
|
int pos = 0;
|
|
|
|
|
size_t readBytes = 0;
|
|
|
|
|
DdtHeader2 ddtHeader;
|
2025-08-04 19:44:56 +01:00
|
|
|
uint8_t *cmpData = NULL;
|
2025-08-04 16:31:29 +01:00
|
|
|
uint8_t lzmaProperties[LZMA_PROPERTIES_LENGTH];
|
|
|
|
|
size_t lzmaSize = 0;
|
|
|
|
|
int errorNo = 0;
|
|
|
|
|
crc64_ctx *crc64_context = NULL;
|
|
|
|
|
uint64_t crc64 = 0;
|
2025-08-04 19:44:56 +01:00
|
|
|
uint8_t *buffer = NULL;
|
2025-08-04 16:31:29 +01:00
|
|
|
|
|
|
|
|
// Check if the context and image stream are valid
|
|
|
|
|
if(ctx == NULL || ctx->imageStream == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Invalid context or image stream.");
|
|
|
|
|
|
|
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_NOT_AARUFORMAT");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_NOT_AARUFORMAT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Seek to block
|
|
|
|
|
pos = fseek(ctx->imageStream, entry->offset, SEEK_SET);
|
|
|
|
|
if(pos < 0 || ftell(ctx->imageStream) != entry->offset)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not seek to %" PRIu64 " as indicated by index entry...", entry->offset);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Even if those two checks shall have been done before
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Reading DDT block header at position %" PRIu64, entry->offset);
|
2025-08-04 16:31:29 +01:00
|
|
|
readBytes = fread(&ddtHeader, 1, sizeof(DdtHeader2), ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(readBytes != sizeof(DdtHeader2))
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not read block header at %" PRIu64 "", entry->offset);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*foundUserDataDdt = false;
|
|
|
|
|
|
|
|
|
|
ctx->imageInfo.ImageSize += ddtHeader.cmpLength;
|
|
|
|
|
|
|
|
|
|
if(entry->dataType == UserData)
|
|
|
|
|
{
|
|
|
|
|
// User area sectors is blocks stored in DDT minus the negative and overflow displacement blocks
|
|
|
|
|
ctx->imageInfo.Sectors = ddtHeader.blocks - ddtHeader.negative - ddtHeader.overflow;
|
|
|
|
|
// We need the header later for the shift calculations
|
|
|
|
|
ctx->userDataDdtHeader = ddtHeader;
|
|
|
|
|
ctx->ddtVersion = 2;
|
2025-09-28 15:15:05 +01:00
|
|
|
// Store the primary DDT table's file offset for secondary table references
|
|
|
|
|
ctx->primaryDdtOffset = entry->offset;
|
2025-08-04 16:31:29 +01:00
|
|
|
|
|
|
|
|
// Check for DDT compression
|
|
|
|
|
switch(ddtHeader.compression)
|
|
|
|
|
{
|
|
|
|
|
case Lzma:
|
|
|
|
|
lzmaSize = ddtHeader.cmpLength - LZMA_PROPERTIES_LENGTH;
|
|
|
|
|
|
|
|
|
|
cmpData = (uint8_t *)malloc(lzmaSize);
|
|
|
|
|
if(cmpData == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Cannot allocate memory for DDT, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
buffer = malloc(ddtHeader.length);
|
|
|
|
|
if(buffer == NULL)
|
2025-08-04 16:31:29 +01:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Cannot allocate memory for DDT, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(lzmaProperties, 1, LZMA_PROPERTIES_LENGTH, ctx->imageStream);
|
|
|
|
|
if(readBytes != LZMA_PROPERTIES_LENGTH)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Could not read LZMA properties, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(cmpData, 1, lzmaSize, ctx->imageStream);
|
|
|
|
|
if(readBytes != lzmaSize)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Could not read compressed block, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = ddtHeader.length;
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Decompressing block of size %zu bytes", ddtHeader.length);
|
|
|
|
|
errorNo = aaruf_lzma_decode_buffer(buffer, &readBytes, cmpData, &lzmaSize, lzmaProperties,
|
|
|
|
|
LZMA_PROPERTIES_LENGTH);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
|
|
|
|
if(errorNo != 0)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Got error %d from LZMA, stopping...", errorNo);
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(readBytes != ddtHeader.length)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Error decompressing block, should be {0} bytes but got {1} bytes., stopping...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free(cmpData);
|
|
|
|
|
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not initialize CRC64.");
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
|
|
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
aaruf_crc64_update(crc64_context, buffer, readBytes);
|
2025-08-04 16:31:29 +01:00
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
if(crc64 != ddtHeader.crc64)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Expected DDT CRC 0x%16lX but got 0x%16lX.", ddtHeader.crc64, crc64);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_INVALID_BLOCK_CRC");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_INVALID_BLOCK_CRC;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->userDataDdtMini = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->userDataDdtBig = (uint32_t *)buffer;
|
|
|
|
|
|
2025-08-04 16:31:29 +01:00
|
|
|
ctx->inMemoryDdt = true;
|
|
|
|
|
*foundUserDataDdt = true;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
case None:
|
2025-08-04 19:44:56 +01:00
|
|
|
buffer = malloc(ddtHeader.length);
|
|
|
|
|
|
|
|
|
|
if(buffer == NULL)
|
2025-08-04 16:31:29 +01:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Cannot allocate memory for DDT, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Reading DDT of length %zu bytes", ddtHeader.length);
|
2025-08-04 19:44:56 +01:00
|
|
|
readBytes = fread(buffer, 1, ddtHeader.length, ctx->imageStream);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
if(readBytes != ddtHeader.length)
|
2025-08-04 16:31:29 +01:00
|
|
|
{
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not read deduplication table, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not initialize CRC64.");
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
aaruf_crc64_update(crc64_context, buffer, readBytes);
|
2025-08-04 16:31:29 +01:00
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
if(crc64 != ddtHeader.crc64)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Expected DDT CRC 0x%16lX but got 0x%16lX.", ddtHeader.crc64, crc64);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_INVALID_BLOCK_CRC");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_INVALID_BLOCK_CRC;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->userDataDdtMini = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->userDataDdtBig = (uint32_t *)buffer;
|
|
|
|
|
|
2025-08-04 16:31:29 +01:00
|
|
|
ctx->inMemoryDdt = true;
|
|
|
|
|
*foundUserDataDdt = true;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Found unknown compression type %d, continuing...", ddtHeader.compression);
|
2025-08-04 16:31:29 +01:00
|
|
|
*foundUserDataDdt = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if(entry->dataType == CdSectorPrefixCorrected || entry->dataType == CdSectorSuffixCorrected)
|
|
|
|
|
{
|
|
|
|
|
switch(ddtHeader.compression)
|
|
|
|
|
{
|
|
|
|
|
case Lzma:
|
|
|
|
|
lzmaSize = ddtHeader.cmpLength - LZMA_PROPERTIES_LENGTH;
|
|
|
|
|
|
|
|
|
|
cmpData = (uint8_t *)malloc(lzmaSize);
|
|
|
|
|
if(cmpData == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Cannot allocate memory for DDT, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
buffer = malloc(ddtHeader.length);
|
|
|
|
|
if(buffer == NULL)
|
2025-08-04 16:31:29 +01:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Cannot allocate memory for DDT, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(lzmaProperties, 1, LZMA_PROPERTIES_LENGTH, ctx->imageStream);
|
|
|
|
|
if(readBytes != LZMA_PROPERTIES_LENGTH)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Could not read LZMA properties, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(cmpData, 1, lzmaSize, ctx->imageStream);
|
|
|
|
|
if(readBytes != lzmaSize)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Could not read compressed block, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = ddtHeader.length;
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Decompressing block of size %zu bytes", ddtHeader.length);
|
|
|
|
|
errorNo = aaruf_lzma_decode_buffer(buffer, &readBytes, cmpData, &lzmaSize, lzmaProperties,
|
|
|
|
|
LZMA_PROPERTIES_LENGTH);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
|
|
|
|
if(errorNo != 0)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Got error %d from LZMA, stopping...", errorNo);
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(readBytes != ddtHeader.length)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Error decompressing block, should be {0} bytes but got {1} bytes., stopping...");
|
2025-08-04 16:31:29 +01:00
|
|
|
free(cmpData);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not initialize CRC64.");
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
aaruf_crc64_update(crc64_context, buffer, readBytes);
|
2025-08-04 16:31:29 +01:00
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
if(crc64 != ddtHeader.crc64)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Expected DDT CRC 0x%16lX but got 0x%16lX.", ddtHeader.crc64, crc64);
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_INVALID_BLOCK_CRC");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_INVALID_BLOCK_CRC;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
if(entry->dataType == CdSectorPrefixCorrected)
|
|
|
|
|
{
|
|
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->sectorPrefixDdtMini = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->sectorPrefixDdt = (uint32_t *)buffer;
|
|
|
|
|
}
|
|
|
|
|
else if(entry->dataType == CdSectorSuffixCorrected)
|
|
|
|
|
{
|
|
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->sectorSuffixDdtMini = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->sectorSuffixDdt = (uint32_t *)buffer;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
free(buffer);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case None:
|
2025-08-04 19:44:56 +01:00
|
|
|
buffer = malloc(ddtHeader.length);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
if(buffer == NULL)
|
2025-08-04 16:31:29 +01:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Cannot allocate memory for deduplication table.");
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
readBytes = fread(buffer, 1, ddtHeader.length, ctx->imageStream);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
if(readBytes != ddtHeader.length)
|
2025-08-04 16:31:29 +01:00
|
|
|
{
|
2025-08-04 19:44:56 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not read deduplication table, continuing...");
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
2025-09-28 15:15:05 +01:00
|
|
|
FATAL("Could not initialize CRC64.");
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
aaruf_crc64_update(crc64_context, buffer, readBytes);
|
2025-08-04 16:31:29 +01:00
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
if(crc64 != ddtHeader.crc64)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Expected DDT CRC 0x%16lX but got 0x%16lX.", ddtHeader.crc64, crc64);
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-09-28 15:15:05 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_ERROR_INVALID_BLOCK_CRC");
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_ERROR_INVALID_BLOCK_CRC;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 19:44:56 +01:00
|
|
|
if(entry->dataType == CdSectorPrefixCorrected)
|
|
|
|
|
{
|
|
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->sectorPrefixDdtMini = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->sectorPrefixDdt = (uint32_t *)buffer;
|
|
|
|
|
}
|
|
|
|
|
else if(entry->dataType == CdSectorSuffixCorrected)
|
|
|
|
|
{
|
|
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->sectorSuffixDdtMini = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->sectorSuffixDdt = (uint32_t *)buffer;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
free(buffer);
|
2025-08-04 16:31:29 +01:00
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Found unknown compression type %d, continuing...", ddtHeader.compression);
|
2025-08-04 16:31:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting process_ddt_v2() = AARUF_STATUS_OK");
|
2025-08-05 01:29:50 +01:00
|
|
|
return AARUF_STATUS_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int32_t decode_ddt_entry_v2(aaruformatContext *ctx, uint64_t sectorAddress, uint64_t *offset, uint64_t *blockOffset,
|
|
|
|
|
uint8_t *sectorStatus)
|
2025-08-05 23:10:54 +01:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Entering decode_ddt_entry_v2(%p, %" PRIu64 ", %llu, %llu, %d)", ctx, sectorAddress, *offset, *blockOffset,
|
|
|
|
|
*sectorStatus);
|
2025-08-05 23:10:54 +01:00
|
|
|
// Check if the context and image stream are valid
|
|
|
|
|
if(ctx == NULL || ctx->imageStream == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Invalid context or image stream.");
|
|
|
|
|
|
|
|
|
|
TRACE("Exiting decode_ddt_entry_v2() = AARUF_ERROR_NOT_AARUFORMAT");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_NOT_AARUFORMAT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.tableShift > 0)
|
|
|
|
|
return decode_ddt_multi_level_v2(ctx, sectorAddress, offset, blockOffset, sectorStatus);
|
|
|
|
|
|
|
|
|
|
return decode_ddt_single_level_v2(ctx, sectorAddress, offset, blockOffset, sectorStatus);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int32_t decode_ddt_single_level_v2(aaruformatContext *ctx, uint64_t sectorAddress, uint64_t *offset,
|
|
|
|
|
uint64_t *blockOffset, uint8_t *sectorStatus)
|
2025-08-05 01:29:50 +01:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Entering decode_ddt_single_level_v2(%p, %" PRIu64 ", %llu, %llu, %d)", ctx, sectorAddress, *offset,
|
|
|
|
|
*blockOffset, *sectorStatus);
|
|
|
|
|
|
2025-08-05 01:29:50 +01:00
|
|
|
uint64_t ddtEntry = 0;
|
|
|
|
|
|
|
|
|
|
// Check if the context and image stream are valid
|
|
|
|
|
if(ctx == NULL || ctx->imageStream == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Invalid context or image stream.");
|
|
|
|
|
|
|
|
|
|
TRACE("Exiting decode_ddt_single_level_v2() = AARUF_ERROR_NOT_AARUFORMAT");
|
2025-08-05 01:29:50 +01:00
|
|
|
return AARUF_ERROR_NOT_AARUFORMAT;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-05 23:10:54 +01:00
|
|
|
// Should not really be here
|
2025-08-14 00:38:28 +01:00
|
|
|
if(ctx->userDataDdtHeader.tableShift != 0)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT table shift is not zero, but we are in single-level DDT decoding.");
|
|
|
|
|
TRACE("Exiting decode_ddt_single_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
|
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
2025-08-05 01:29:50 +01:00
|
|
|
|
|
|
|
|
// TODO: Take into account the negative and overflow blocks, library-wide
|
|
|
|
|
sectorAddress += ctx->userDataDdtHeader.negative;
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ddtEntry = ctx->userDataDdtMini[sectorAddress];
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ddtEntry = ctx->userDataDdtBig[sectorAddress];
|
|
|
|
|
else
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Unknown DDT size type %d.", ctx->userDataDdtHeader.sizeType);
|
|
|
|
|
TRACE("Exiting decode_ddt_single_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 01:29:50 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ddtEntry == 0)
|
|
|
|
|
{
|
|
|
|
|
*sectorStatus = SectorStatusNotDumped;
|
|
|
|
|
*offset = 0;
|
|
|
|
|
*blockOffset = 0;
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_single_level_v2(%p, %" PRIu64 ", %llu, %llu, %d) = AARUF_STATUS_OK", ctx,
|
|
|
|
|
sectorAddress, *offset, *blockOffset, *sectorStatus);
|
2025-08-05 01:29:50 +01:00
|
|
|
return AARUF_STATUS_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
{
|
|
|
|
|
*sectorStatus = ddtEntry >> 12;
|
|
|
|
|
ddtEntry &= 0xfff;
|
|
|
|
|
}
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
{
|
|
|
|
|
*sectorStatus = ddtEntry >> 28;
|
|
|
|
|
ddtEntry &= 0x0fffffff;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const uint64_t offsetMask = (uint64_t)((1 << ctx->userDataDdtHeader.dataShift) - 1);
|
2025-08-05 02:48:59 +01:00
|
|
|
*offset = ddtEntry & offsetMask;
|
2025-08-05 23:10:54 +01:00
|
|
|
*blockOffset = (ddtEntry >> ctx->userDataDdtHeader.dataShift) * (1 << ctx->userDataDdtHeader.blockAlignmentShift);
|
|
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_single_level_v2(%p, %" PRIu64 ", %llu, %llu, %d) = AARUF_STATUS_OK", ctx, sectorAddress,
|
|
|
|
|
*offset, *blockOffset, *sectorStatus);
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_STATUS_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int32_t decode_ddt_multi_level_v2(aaruformatContext *ctx, uint64_t sectorAddress, uint64_t *offset,
|
|
|
|
|
uint64_t *blockOffset, uint8_t *sectorStatus)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Entering decode_ddt_multi_level_v2(%p, %" PRIu64 ", %llu, %llu, %d)", ctx, sectorAddress, *offset,
|
|
|
|
|
*blockOffset, *sectorStatus);
|
|
|
|
|
|
2025-08-05 23:10:54 +01:00
|
|
|
uint64_t ddtEntry = 0;
|
|
|
|
|
uint8_t lzmaProperties[LZMA_PROPERTIES_LENGTH];
|
|
|
|
|
size_t lzmaSize = 0;
|
|
|
|
|
uint8_t *cmpData = NULL;
|
|
|
|
|
uint8_t *buffer = NULL;
|
|
|
|
|
int32_t errorNo = 0;
|
|
|
|
|
crc64_ctx *crc64_context = NULL;
|
|
|
|
|
uint64_t crc64 = 0;
|
|
|
|
|
int itemsPerDdtEntry = 0;
|
|
|
|
|
uint64_t ddtPosition = 0;
|
|
|
|
|
uint64_t secondaryDdtOffset = 0;
|
|
|
|
|
|
|
|
|
|
// Check if the context and image stream are valid
|
|
|
|
|
if(ctx == NULL || ctx->imageStream == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Invalid context or image stream.");
|
|
|
|
|
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_NOT_AARUFORMAT");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_NOT_AARUFORMAT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Should not really be here
|
2025-08-14 00:38:28 +01:00
|
|
|
if(ctx->userDataDdtHeader.tableShift == 0)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT table shift is zero, but we are in multi-level DDT decoding.");
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
|
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
2025-08-05 23:10:54 +01:00
|
|
|
|
|
|
|
|
// TODO: Take into account the negative and overflow blocks, library-wide
|
|
|
|
|
sectorAddress += ctx->userDataDdtHeader.negative;
|
|
|
|
|
|
|
|
|
|
itemsPerDdtEntry = 1 << ctx->userDataDdtHeader.tableShift;
|
|
|
|
|
ddtPosition = sectorAddress / itemsPerDdtEntry;
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
secondaryDdtOffset = ctx->userDataDdtMini[ddtPosition];
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
secondaryDdtOffset = ctx->userDataDdtBig[ddtPosition];
|
|
|
|
|
else
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Unknown DDT size type %d.", ctx->userDataDdtHeader.sizeType);
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Position in file of the child DDT table
|
2025-09-28 15:15:05 +01:00
|
|
|
secondaryDdtOffset *= 1 << ctx->userDataDdtHeader.blockAlignmentShift;
|
2025-08-05 23:10:54 +01:00
|
|
|
|
|
|
|
|
// Is the one we have cached the same as the one we need to read?
|
|
|
|
|
if(ctx->cachedDdtOffset != secondaryDdtOffset)
|
|
|
|
|
{
|
|
|
|
|
fseek(ctx->imageStream, secondaryDdtOffset, SEEK_SET);
|
|
|
|
|
DdtHeader2 ddtHeader;
|
|
|
|
|
size_t readBytes = fread(&ddtHeader, 1, sizeof(DdtHeader2), ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(readBytes != sizeof(DdtHeader2))
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not read block header at %" PRIu64 "", secondaryDdtOffset);
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ddtHeader.identifier != DeDuplicationTable2 || ddtHeader.type != UserData)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Invalid block header at %" PRIu64 "", secondaryDdtOffset);
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for DDT compression
|
|
|
|
|
switch(ddtHeader.compression)
|
|
|
|
|
{
|
|
|
|
|
case Lzma:
|
|
|
|
|
lzmaSize = ddtHeader.cmpLength - LZMA_PROPERTIES_LENGTH;
|
|
|
|
|
|
|
|
|
|
cmpData = (uint8_t *)malloc(lzmaSize);
|
|
|
|
|
if(cmpData == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Cannot allocate memory for DDT, stopping...");
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
buffer = malloc(ddtHeader.length);
|
|
|
|
|
if(buffer == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Cannot allocate memory for DDT, stopping...");
|
2025-08-05 23:10:54 +01:00
|
|
|
free(cmpData);
|
|
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(lzmaProperties, 1, LZMA_PROPERTIES_LENGTH, ctx->imageStream);
|
|
|
|
|
if(readBytes != LZMA_PROPERTIES_LENGTH)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not read LZMA properties, stopping...");
|
2025-08-05 23:10:54 +01:00
|
|
|
free(cmpData);
|
|
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(cmpData, 1, lzmaSize, ctx->imageStream);
|
|
|
|
|
if(readBytes != lzmaSize)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not read compressed block, stopping...");
|
2025-08-05 23:10:54 +01:00
|
|
|
free(cmpData);
|
|
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Decompressing block of size %zu bytes", ddtHeader.length);
|
2025-08-05 23:10:54 +01:00
|
|
|
readBytes = ddtHeader.length;
|
|
|
|
|
errorNo = aaruf_lzma_decode_buffer(buffer, &readBytes, cmpData, &lzmaSize, lzmaProperties,
|
|
|
|
|
LZMA_PROPERTIES_LENGTH);
|
|
|
|
|
|
|
|
|
|
if(errorNo != 0)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Got error %d from LZMA, stopping...", errorNo);
|
2025-08-05 23:10:54 +01:00
|
|
|
free(cmpData);
|
|
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(readBytes != ddtHeader.length)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Error decompressing block, should be {0} bytes but got {1} bytes., stopping...");
|
2025-08-05 23:10:54 +01:00
|
|
|
free(cmpData);
|
|
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_DECOMPRESS_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free(cmpData);
|
|
|
|
|
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not initialize CRC64.");
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
aaruf_crc64_update(crc64_context, buffer, readBytes);
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
if(crc64 != ddtHeader.crc64)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Expected DDT CRC 0x%16lX but got 0x%16lX.", ddtHeader.crc64, crc64);
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_INVALID_BLOCK_CRC");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_INVALID_BLOCK_CRC;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->cachedSecondaryDdtSmall = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->cachedSecondaryDdtBig = (uint32_t *)buffer;
|
|
|
|
|
|
|
|
|
|
ctx->cachedDdtOffset = secondaryDdtOffset;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
case None:
|
|
|
|
|
buffer = malloc(ddtHeader.length);
|
|
|
|
|
|
|
|
|
|
if(buffer == NULL)
|
|
|
|
|
{
|
2025-09-28 17:08:33 +01:00
|
|
|
FATAL("Cannot allocate memory for DDT, stopping...");
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(buffer, 1, ddtHeader.length, ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(readBytes != ddtHeader.length)
|
|
|
|
|
{
|
|
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not read deduplication table, stopping...");
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Could not initialize CRC64.");
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
aaruf_crc64_update(crc64_context, buffer, readBytes);
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
if(crc64 != ddtHeader.crc64)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Expected DDT CRC 0x%16lX but got 0x%16lX.", ddtHeader.crc64, crc64);
|
2025-08-05 23:10:54 +01:00
|
|
|
free(buffer);
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_INVALID_BLOCK_CRC");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_INVALID_BLOCK_CRC;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ddtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->cachedSecondaryDdtSmall = (uint16_t *)buffer;
|
|
|
|
|
else if(ddtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
ctx->cachedSecondaryDdtBig = (uint32_t *)buffer;
|
|
|
|
|
|
|
|
|
|
ctx->cachedDdtOffset = secondaryDdtOffset;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Found unknown compression type %d, stopping...", ddtHeader.compression);
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2() = AARUF_ERROR_CANNOT_READ_BLOCK");
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_ERROR_CANNOT_READ_BLOCK;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
2025-09-30 01:50:45 +01:00
|
|
|
ddtEntry = ctx->cachedSecondaryDdtSmall[sectorAddress % itemsPerDdtEntry];
|
2025-08-05 23:10:54 +01:00
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
2025-09-30 01:50:45 +01:00
|
|
|
ddtEntry = ctx->cachedSecondaryDdtBig[sectorAddress % itemsPerDdtEntry];
|
2025-08-05 23:10:54 +01:00
|
|
|
|
|
|
|
|
if(ddtEntry == 0)
|
|
|
|
|
{
|
|
|
|
|
*sectorStatus = SectorStatusNotDumped;
|
|
|
|
|
*offset = 0;
|
|
|
|
|
*blockOffset = 0;
|
2025-08-14 00:38:28 +01:00
|
|
|
|
|
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2(%p, %" PRIu64 ", %llu, %llu, %d) = AARUF_STATUS_OK", ctx,
|
|
|
|
|
sectorAddress, *offset, *blockOffset, *sectorStatus);
|
2025-08-05 23:10:54 +01:00
|
|
|
return AARUF_STATUS_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
{
|
|
|
|
|
*sectorStatus = ddtEntry >> 12;
|
|
|
|
|
ddtEntry &= 0xfff;
|
|
|
|
|
}
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
{
|
|
|
|
|
*sectorStatus = ddtEntry >> 28;
|
|
|
|
|
ddtEntry &= 0x0fffffff;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const uint64_t offsetMask = (uint64_t)((1 << ctx->userDataDdtHeader.dataShift) - 1);
|
|
|
|
|
*offset = ddtEntry & offsetMask;
|
|
|
|
|
*blockOffset = (ddtEntry >> ctx->userDataDdtHeader.dataShift) * (1 << ctx->userDataDdtHeader.blockAlignmentShift);
|
2025-08-05 01:29:50 +01:00
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting decode_ddt_multi_level_v2(%p, %" PRIu64 ", %llu, %llu, %d) = AARUF_STATUS_OK", ctx, sectorAddress,
|
|
|
|
|
*offset, *blockOffset, *sectorStatus);
|
2025-08-04 16:31:29 +01:00
|
|
|
return AARUF_STATUS_OK;
|
2025-09-28 15:15:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void set_ddt_entry_v2(aaruformatContext *ctx, uint64_t sectorAddress, uint64_t offset, uint64_t blockOffset,
|
|
|
|
|
uint8_t sectorStatus)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Entering set_ddt_entry_v2(%p, %" PRIu64 ", %llu, %llu, %d)", ctx, sectorAddress, offset, blockOffset,
|
|
|
|
|
sectorStatus);
|
|
|
|
|
|
|
|
|
|
// Check if the context and image stream are valid
|
|
|
|
|
if(ctx == NULL || ctx->imageStream == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Invalid context or image stream.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.tableShift > 0)
|
|
|
|
|
set_ddt_multi_level_v2(ctx, sectorAddress, false, offset, blockOffset, sectorStatus);
|
|
|
|
|
else
|
|
|
|
|
set_ddt_single_level_v2(ctx, sectorAddress, false, offset, blockOffset, sectorStatus);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void set_ddt_single_level_v2(aaruformatContext *ctx, uint64_t sectorAddress, bool negative, uint64_t offset,
|
|
|
|
|
uint64_t blockOffset, uint8_t sectorStatus)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Entering set_ddt_single_level_v2(%p, %" PRIu64 ", %llu, %llu, %d)", ctx, sectorAddress, offset, blockOffset,
|
|
|
|
|
sectorStatus);
|
|
|
|
|
|
|
|
|
|
// Check if the context and image stream are valid
|
|
|
|
|
if(ctx == NULL || ctx->imageStream == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Invalid context or image stream.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Should not really be here
|
|
|
|
|
if(ctx->userDataDdtHeader.tableShift != 0)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT table shift is not zero, but we are in single-level DDT setting.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Calculate positive or negative sector
|
|
|
|
|
if(negative)
|
|
|
|
|
sectorAddress -= ctx->userDataDdtHeader.negative;
|
|
|
|
|
else
|
|
|
|
|
sectorAddress += ctx->userDataDdtHeader.negative;
|
|
|
|
|
|
|
|
|
|
uint64_t ddtEntry = 0;
|
|
|
|
|
|
|
|
|
|
uint64_t blockIndex = blockOffset >> ctx->userDataDdtHeader.blockAlignmentShift;
|
2025-09-30 12:35:36 +01:00
|
|
|
ddtEntry = offset & ((1ULL << ctx->userDataDdtHeader.dataShift) - 1) | blockIndex
|
|
|
|
|
<< ctx->userDataDdtHeader.dataShift;
|
2025-09-28 15:15:05 +01:00
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
{
|
2025-09-30 12:35:36 +01:00
|
|
|
// Overflow detection for DDT entry
|
|
|
|
|
if(ddtEntry > 0xFFF)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT overflow: media does not fit in small DDT");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
ddtEntry |= (uint64_t)sectorStatus << 12;
|
|
|
|
|
ctx->cachedSecondaryDdtSmall[sectorAddress] = (uint16_t)ddtEntry;
|
|
|
|
|
}
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
{
|
2025-09-30 12:35:36 +01:00
|
|
|
// Overflow detection for DDT entry
|
|
|
|
|
if(ddtEntry > 0xFFFFFFF)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT overflow: media does not fit in big DDT");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
ddtEntry |= (uint64_t)sectorStatus << 28;
|
|
|
|
|
ctx->cachedSecondaryDdtBig[sectorAddress] = (uint32_t)ddtEntry;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void set_ddt_multi_level_v2(aaruformatContext *ctx, uint64_t sectorAddress, bool negative, uint64_t offset,
|
|
|
|
|
uint64_t blockOffset, uint8_t sectorStatus)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Entering set_ddt_multi_level_v2(%p, %" PRIu64 ", %d, %" PRIu64 ", %" PRIu64 ", %d)", ctx, sectorAddress,
|
|
|
|
|
negative, offset, blockOffset, sectorStatus);
|
|
|
|
|
|
|
|
|
|
uint64_t itemsPerDdtEntry = 0;
|
|
|
|
|
uint64_t ddtPosition = 0;
|
|
|
|
|
uint64_t secondaryDdtOffset = 0;
|
|
|
|
|
uint64_t ddtEntry = 0;
|
|
|
|
|
uint64_t blockIndex = 0;
|
|
|
|
|
uint8_t *buffer = NULL;
|
|
|
|
|
crc64_ctx *crc64_context = NULL;
|
|
|
|
|
uint64_t crc64 = 0;
|
|
|
|
|
DdtHeader2 ddtHeader;
|
|
|
|
|
size_t writtenBytes = 0;
|
|
|
|
|
long currentPos = 0;
|
|
|
|
|
long endOfFile = 0;
|
|
|
|
|
bool createNewTable = false;
|
|
|
|
|
|
|
|
|
|
// Check if the context and image stream are valid
|
|
|
|
|
if(ctx == NULL || ctx->imageStream == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Invalid context or image stream.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Should not really be here
|
|
|
|
|
if(ctx->userDataDdtHeader.tableShift == 0)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT table shift is zero, but we are in multi-level DDT setting.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Calculate positive or negative sector
|
|
|
|
|
if(negative)
|
|
|
|
|
sectorAddress -= ctx->userDataDdtHeader.negative;
|
|
|
|
|
else
|
|
|
|
|
sectorAddress += ctx->userDataDdtHeader.negative;
|
|
|
|
|
|
|
|
|
|
// Step 1: Calculate the corresponding secondary level table
|
|
|
|
|
itemsPerDdtEntry = 1 << ctx->userDataDdtHeader.tableShift;
|
|
|
|
|
ddtPosition = sectorAddress / itemsPerDdtEntry;
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
secondaryDdtOffset = ctx->userDataDdtMini[ddtPosition];
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
secondaryDdtOffset = ctx->userDataDdtBig[ddtPosition];
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
FATAL("Unknown DDT size type %d.", ctx->userDataDdtHeader.sizeType);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Position in file of the child DDT table
|
|
|
|
|
secondaryDdtOffset *= 1 << ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
|
|
|
|
|
// Step 2: Check if it corresponds to the currently in-memory cached secondary level table
|
|
|
|
|
if(ctx->cachedDdtOffset == secondaryDdtOffset && secondaryDdtOffset != 0)
|
|
|
|
|
{
|
|
|
|
|
// Update the corresponding DDT entry directly in the cached table
|
|
|
|
|
blockIndex = blockOffset >> ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
ddtEntry = offset & (1ULL << ctx->userDataDdtHeader.dataShift) - 1 | blockIndex
|
|
|
|
|
<< ctx->userDataDdtHeader.dataShift;
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
{
|
2025-09-30 12:35:36 +01:00
|
|
|
// Overflow detection for DDT entry
|
|
|
|
|
if(ddtEntry > 0xFFF)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT overflow: media does not fit in small DDT");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
ddtEntry |= (uint64_t)sectorStatus << 12;
|
2025-09-28 21:25:55 +01:00
|
|
|
TRACE("Setting small secondary DDT entry %d to %u", sectorAddress % itemsPerDdtEntry, (uint16_t)ddtEntry);
|
2025-09-28 15:15:05 +01:00
|
|
|
ctx->cachedSecondaryDdtSmall[sectorAddress % itemsPerDdtEntry] = (uint16_t)ddtEntry;
|
|
|
|
|
}
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
{
|
2025-09-30 12:35:36 +01:00
|
|
|
// Overflow detection for DDT entry
|
|
|
|
|
if(ddtEntry > 0xFFFFFFF)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT overflow: media does not fit in big DDT");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
ddtEntry |= (uint64_t)sectorStatus << 28;
|
2025-09-28 21:25:55 +01:00
|
|
|
TRACE("Setting small secondary DDT entry %d to %u", sectorAddress % itemsPerDdtEntry, (uint16_t)ddtEntry);
|
2025-09-28 15:15:05 +01:00
|
|
|
ctx->cachedSecondaryDdtBig[sectorAddress % itemsPerDdtEntry] = (uint32_t)ddtEntry;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TRACE("Updated cached secondary DDT entry at position %" PRIu64, sectorAddress % itemsPerDdtEntry);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 19:44:28 +01:00
|
|
|
// Step 2.5: Handle case where we have a cached secondary DDT that has never been written to disk
|
|
|
|
|
// but does not contain the requested block
|
|
|
|
|
if(ctx->cachedDdtOffset == 0 && (ctx->cachedSecondaryDdtSmall != NULL || ctx->cachedSecondaryDdtBig != NULL))
|
|
|
|
|
{
|
|
|
|
|
// Only write the cached table to disk if the requested block belongs to a different DDT position
|
|
|
|
|
if(ddtPosition != ctx->cachedDdtPosition)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Current secondary DDT in memory belongs to position %" PRIu64
|
|
|
|
|
" but requested block needs position %" PRIu64,
|
|
|
|
|
ctx->cachedDdtPosition, ddtPosition);
|
|
|
|
|
|
|
|
|
|
// Write the cached DDT to disk before proceeding with the new one
|
|
|
|
|
|
|
|
|
|
// Close the current data block first
|
|
|
|
|
if(ctx->writingBuffer != NULL) aaruf_close_current_block(ctx);
|
|
|
|
|
|
|
|
|
|
// Get current position and seek to end of file
|
|
|
|
|
currentPos = ftell(ctx->imageStream);
|
|
|
|
|
fseek(ctx->imageStream, 0, SEEK_END);
|
|
|
|
|
endOfFile = ftell(ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
// Align to block boundary
|
|
|
|
|
uint64_t alignmentMask = (1ULL << ctx->userDataDdtHeader.blockAlignmentShift) - 1;
|
|
|
|
|
endOfFile = (endOfFile + alignmentMask) & ~alignmentMask;
|
|
|
|
|
fseek(ctx->imageStream, endOfFile, SEEK_SET);
|
|
|
|
|
|
|
|
|
|
// Prepare DDT header for the never-written cached table
|
|
|
|
|
memset(&ddtHeader, 0, sizeof(DdtHeader2));
|
|
|
|
|
ddtHeader.identifier = DeDuplicationTable2;
|
|
|
|
|
ddtHeader.type = UserData;
|
|
|
|
|
ddtHeader.compression = None; // Use no compression for simplicity
|
|
|
|
|
ddtHeader.levels = ctx->userDataDdtHeader.levels;
|
|
|
|
|
ddtHeader.tableLevel = ctx->userDataDdtHeader.tableLevel + 1;
|
|
|
|
|
ddtHeader.previousLevelOffset = ctx->primaryDdtOffset;
|
|
|
|
|
ddtHeader.negative = ctx->userDataDdtHeader.negative;
|
|
|
|
|
ddtHeader.blocks = itemsPerDdtEntry;
|
|
|
|
|
ddtHeader.overflow = ctx->userDataDdtHeader.overflow;
|
|
|
|
|
ddtHeader.start = ctx->cachedDdtPosition * itemsPerDdtEntry; // Use cached position with table shift
|
|
|
|
|
ddtHeader.blockAlignmentShift = ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
ddtHeader.dataShift = ctx->userDataDdtHeader.dataShift;
|
|
|
|
|
ddtHeader.tableShift = 0; // Secondary tables are single level
|
|
|
|
|
ddtHeader.sizeType = ctx->userDataDdtHeader.sizeType;
|
|
|
|
|
ddtHeader.entries = itemsPerDdtEntry;
|
|
|
|
|
|
|
|
|
|
// Calculate data size
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ddtHeader.length = itemsPerDdtEntry * sizeof(uint16_t);
|
|
|
|
|
else
|
|
|
|
|
ddtHeader.length = itemsPerDdtEntry * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
ddtHeader.cmpLength = ddtHeader.length;
|
|
|
|
|
|
|
|
|
|
// Calculate CRC64 of the data
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not initialize CRC64.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->cachedSecondaryDdtSmall, ddtHeader.length);
|
|
|
|
|
else
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->cachedSecondaryDdtBig, ddtHeader.length);
|
|
|
|
|
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
ddtHeader.crc64 = crc64;
|
|
|
|
|
ddtHeader.cmpCrc64 = crc64;
|
|
|
|
|
|
|
|
|
|
// Write header
|
|
|
|
|
writtenBytes = fwrite(&ddtHeader, sizeof(DdtHeader2), 1, ctx->imageStream);
|
|
|
|
|
if(writtenBytes != 1)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not write never-written DDT header to file.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write data
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
writtenBytes = fwrite(ctx->cachedSecondaryDdtSmall, ddtHeader.length, 1, ctx->imageStream);
|
|
|
|
|
else
|
|
|
|
|
writtenBytes = fwrite(ctx->cachedSecondaryDdtBig, ddtHeader.length, 1, ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(writtenBytes != 1)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not write never-written DDT data to file.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add index entry for the newly written secondary DDT
|
|
|
|
|
IndexEntry newDdtEntry;
|
|
|
|
|
newDdtEntry.blockType = DeDuplicationTable2;
|
|
|
|
|
newDdtEntry.dataType = UserData;
|
|
|
|
|
newDdtEntry.offset = endOfFile;
|
|
|
|
|
|
|
|
|
|
utarray_push_back(ctx->indexEntries, &newDdtEntry);
|
|
|
|
|
TRACE("Added new DDT index entry for never-written table at offset %" PRIu64, endOfFile);
|
|
|
|
|
|
|
|
|
|
// Update the primary level table entry to point to the new location of the secondary table
|
|
|
|
|
uint64_t newSecondaryTableBlockOffset = endOfFile >> ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->userDataDdtMini[ctx->cachedDdtPosition] = (uint16_t)newSecondaryTableBlockOffset;
|
|
|
|
|
else
|
|
|
|
|
ctx->userDataDdtBig[ctx->cachedDdtPosition] = (uint32_t)newSecondaryTableBlockOffset;
|
|
|
|
|
|
|
|
|
|
// Write the updated primary table back to its original position in the file
|
|
|
|
|
long savedPos = ftell(ctx->imageStream);
|
|
|
|
|
fseek(ctx->imageStream, ctx->primaryDdtOffset + sizeof(DdtHeader2), SEEK_SET);
|
|
|
|
|
|
|
|
|
|
size_t primaryTableSize = ctx->userDataDdtHeader.sizeType == SmallDdtSizeType
|
|
|
|
|
? ctx->userDataDdtHeader.entries * sizeof(uint16_t)
|
|
|
|
|
: ctx->userDataDdtHeader.entries * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
writtenBytes = fwrite(ctx->userDataDdtMini, primaryTableSize, 1, ctx->imageStream);
|
|
|
|
|
else
|
|
|
|
|
writtenBytes = fwrite(ctx->userDataDdtBig, primaryTableSize, 1, ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(writtenBytes != 1)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not flush primary DDT table to file after writing never-written secondary table.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update nextBlockPosition to ensure future blocks don't overwrite the DDT
|
|
|
|
|
uint64_t ddtTotalSize = sizeof(DdtHeader2) + ddtHeader.length;
|
|
|
|
|
ctx->nextBlockPosition = (endOfFile + ddtTotalSize + alignmentMask) & ~alignmentMask;
|
2025-09-30 01:51:18 +01:00
|
|
|
blockOffset = ctx->nextBlockPosition;
|
|
|
|
|
offset = 0;
|
2025-09-28 19:44:28 +01:00
|
|
|
TRACE("Updated nextBlockPosition after never-written DDT write to %" PRIu64, ctx->nextBlockPosition);
|
|
|
|
|
|
|
|
|
|
// Free the cached table
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType && ctx->cachedSecondaryDdtSmall)
|
|
|
|
|
{
|
|
|
|
|
free(ctx->cachedSecondaryDdtSmall);
|
|
|
|
|
ctx->cachedSecondaryDdtSmall = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType && ctx->cachedSecondaryDdtBig)
|
|
|
|
|
{
|
|
|
|
|
free(ctx->cachedSecondaryDdtBig);
|
|
|
|
|
ctx->cachedSecondaryDdtBig = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Reset cached values since we've written and freed the table
|
|
|
|
|
ctx->cachedDdtOffset = 0;
|
|
|
|
|
ctx->cachedDdtPosition = 0;
|
|
|
|
|
|
|
|
|
|
// Restore file position
|
|
|
|
|
fseek(ctx->imageStream, savedPos, SEEK_SET);
|
|
|
|
|
|
|
|
|
|
TRACE("Successfully wrote never-written cached secondary DDT to disk");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// The cached DDT is actually for the requested block range, so we can use it directly
|
|
|
|
|
TRACE("Cached DDT is for the correct block range, using it directly");
|
|
|
|
|
// No need to write to disk, just continue with the cached table
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
// Step 3: Write the currently in-memory cached secondary level table to the end of the file
|
|
|
|
|
if(ctx->cachedDdtOffset != 0)
|
|
|
|
|
{
|
2025-09-28 16:13:56 +01:00
|
|
|
// Close the current data block first
|
|
|
|
|
if(ctx->writingBuffer != NULL) aaruf_close_current_block(ctx);
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
// Get current position and seek to end of file
|
|
|
|
|
currentPos = ftell(ctx->imageStream);
|
|
|
|
|
fseek(ctx->imageStream, 0, SEEK_END);
|
|
|
|
|
endOfFile = ftell(ctx->imageStream);
|
2025-09-28 18:42:33 +01:00
|
|
|
|
|
|
|
|
// Align to block boundary
|
|
|
|
|
uint64_t alignmentMask = (1ULL << ctx->userDataDdtHeader.blockAlignmentShift) - 1;
|
|
|
|
|
endOfFile = (endOfFile + alignmentMask) & ~alignmentMask;
|
2025-09-28 16:13:56 +01:00
|
|
|
fseek(ctx->imageStream, endOfFile, SEEK_SET);
|
2025-09-28 15:15:05 +01:00
|
|
|
|
|
|
|
|
// Prepare DDT header for the cached table
|
|
|
|
|
memset(&ddtHeader, 0, sizeof(DdtHeader2));
|
|
|
|
|
ddtHeader.identifier = DeDuplicationTable2;
|
|
|
|
|
ddtHeader.type = UserData;
|
|
|
|
|
ddtHeader.compression = None; // Use no compression for simplicity
|
|
|
|
|
ddtHeader.levels = ctx->userDataDdtHeader.levels;
|
|
|
|
|
ddtHeader.tableLevel = ctx->userDataDdtHeader.tableLevel + 1;
|
|
|
|
|
ddtHeader.previousLevelOffset = ctx->primaryDdtOffset; // Set to primary DDT table location
|
|
|
|
|
ddtHeader.negative = ctx->userDataDdtHeader.negative;
|
|
|
|
|
ddtHeader.blocks = itemsPerDdtEntry;
|
|
|
|
|
ddtHeader.overflow = ctx->userDataDdtHeader.overflow;
|
|
|
|
|
ddtHeader.start = ddtPosition * itemsPerDdtEntry; // First block this DDT table references
|
|
|
|
|
ddtHeader.blockAlignmentShift = ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
ddtHeader.dataShift = ctx->userDataDdtHeader.dataShift;
|
|
|
|
|
ddtHeader.tableShift = 0; // Secondary tables are single level
|
|
|
|
|
ddtHeader.sizeType = ctx->userDataDdtHeader.sizeType;
|
|
|
|
|
ddtHeader.entries = itemsPerDdtEntry;
|
|
|
|
|
|
|
|
|
|
// Calculate data size
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ddtHeader.length = itemsPerDdtEntry * sizeof(uint16_t);
|
|
|
|
|
else
|
|
|
|
|
ddtHeader.length = itemsPerDdtEntry * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
ddtHeader.cmpLength = ddtHeader.length;
|
|
|
|
|
|
|
|
|
|
// Calculate CRC64 of the data
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not initialize CRC64.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->cachedSecondaryDdtSmall, ddtHeader.length);
|
|
|
|
|
else
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->cachedSecondaryDdtBig, ddtHeader.length);
|
|
|
|
|
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
ddtHeader.crc64 = crc64;
|
|
|
|
|
ddtHeader.cmpCrc64 = crc64;
|
|
|
|
|
|
|
|
|
|
// Write header
|
|
|
|
|
writtenBytes = fwrite(&ddtHeader, sizeof(DdtHeader2), 1, ctx->imageStream);
|
|
|
|
|
if(writtenBytes != 1)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not write DDT header to file.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write data
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
writtenBytes = fwrite(ctx->cachedSecondaryDdtSmall, ddtHeader.length, 1, ctx->imageStream);
|
|
|
|
|
else
|
|
|
|
|
writtenBytes = fwrite(ctx->cachedSecondaryDdtBig, ddtHeader.length, 1, ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(writtenBytes != 1)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not write DDT data to file.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 17:08:33 +01:00
|
|
|
// Update index: remove old entry and add new one for the evicted secondary DDT
|
|
|
|
|
TRACE("Updating index for evicted secondary DDT");
|
|
|
|
|
|
|
|
|
|
// Remove old index entry for the cached DDT
|
|
|
|
|
if(ctx->cachedDdtOffset != 0)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Removing old index entry for DDT at offset %" PRIu64, ctx->cachedDdtOffset);
|
|
|
|
|
IndexEntry *entry = NULL;
|
|
|
|
|
|
|
|
|
|
// Find and remove the old index entry
|
|
|
|
|
for(unsigned int i = 0; i < utarray_len(ctx->indexEntries); i++)
|
|
|
|
|
{
|
|
|
|
|
entry = (IndexEntry *)utarray_eltptr(ctx->indexEntries, i);
|
|
|
|
|
if(entry && entry->offset == ctx->cachedDdtOffset && entry->blockType == DeDuplicationTable2)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Found old DDT index entry at position %u, removing", i);
|
|
|
|
|
utarray_erase(ctx->indexEntries, i, 1);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add new index entry for the newly written secondary DDT
|
|
|
|
|
IndexEntry newDdtEntry;
|
|
|
|
|
newDdtEntry.blockType = DeDuplicationTable2;
|
|
|
|
|
newDdtEntry.dataType = UserData;
|
|
|
|
|
newDdtEntry.offset = endOfFile;
|
|
|
|
|
|
|
|
|
|
utarray_push_back(ctx->indexEntries, &newDdtEntry);
|
|
|
|
|
TRACE("Added new DDT index entry at offset %" PRIu64, endOfFile);
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
// Step 4: Update the primary level table entry and flush it back to file
|
|
|
|
|
uint64_t newSecondaryTableBlockOffset = endOfFile >> ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
|
|
|
|
|
// Update the primary table entry to point to the new location of the secondary table
|
2025-09-28 18:43:08 +01:00
|
|
|
// Use ddtPosition which was calculated from sectorAddress, not cachedDdtOffset
|
2025-09-28 15:15:05 +01:00
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
2025-09-28 18:43:08 +01:00
|
|
|
ctx->userDataDdtMini[ddtPosition] = (uint16_t)newSecondaryTableBlockOffset;
|
2025-09-28 15:15:05 +01:00
|
|
|
else
|
2025-09-28 18:43:08 +01:00
|
|
|
ctx->userDataDdtBig[ddtPosition] = (uint32_t)newSecondaryTableBlockOffset;
|
2025-09-28 15:15:05 +01:00
|
|
|
|
|
|
|
|
// Write the updated primary table back to its original position in the file
|
|
|
|
|
long savedPos = ftell(ctx->imageStream);
|
|
|
|
|
fseek(ctx->imageStream, ctx->primaryDdtOffset + sizeof(DdtHeader2), SEEK_SET);
|
|
|
|
|
|
|
|
|
|
size_t primaryTableSize = ctx->userDataDdtHeader.sizeType == SmallDdtSizeType
|
|
|
|
|
? ctx->userDataDdtHeader.entries * sizeof(uint16_t)
|
|
|
|
|
: ctx->userDataDdtHeader.entries * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
writtenBytes = fwrite(ctx->userDataDdtMini, primaryTableSize, 1, ctx->imageStream);
|
|
|
|
|
else
|
|
|
|
|
writtenBytes = fwrite(ctx->userDataDdtBig, primaryTableSize, 1, ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(writtenBytes != 1)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not flush primary DDT table to file.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 18:43:44 +01:00
|
|
|
// Update nextBlockPosition to ensure future blocks don't overwrite the DDT
|
|
|
|
|
uint64_t ddtTotalSize = sizeof(DdtHeader2) + ddtHeader.length;
|
|
|
|
|
ctx->nextBlockPosition = (endOfFile + ddtTotalSize + alignmentMask) & ~alignmentMask;
|
2025-09-30 01:51:18 +01:00
|
|
|
blockOffset = ctx->nextBlockPosition;
|
|
|
|
|
offset = 0;
|
2025-09-28 18:43:44 +01:00
|
|
|
TRACE("Updated nextBlockPosition after DDT write to %" PRIu64, ctx->nextBlockPosition);
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
fseek(ctx->imageStream, savedPos, SEEK_SET);
|
|
|
|
|
|
|
|
|
|
// Free the cached table
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType && ctx->cachedSecondaryDdtSmall)
|
|
|
|
|
{
|
|
|
|
|
free(ctx->cachedSecondaryDdtSmall);
|
|
|
|
|
ctx->cachedSecondaryDdtSmall = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType && ctx->cachedSecondaryDdtBig)
|
|
|
|
|
{
|
|
|
|
|
free(ctx->cachedSecondaryDdtBig);
|
|
|
|
|
ctx->cachedSecondaryDdtBig = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Restore file position
|
|
|
|
|
fseek(ctx->imageStream, currentPos, SEEK_SET);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Step 5: Check if the specified block already has an existing secondary level table
|
2025-09-28 21:26:26 +01:00
|
|
|
createNewTable = ctx->cachedSecondaryDdtSmall == NULL && ctx->cachedSecondaryDdtBig == NULL;
|
2025-09-28 15:15:05 +01:00
|
|
|
|
2025-09-28 21:26:26 +01:00
|
|
|
if(!createNewTable && secondaryDdtOffset != 0)
|
2025-09-28 15:15:05 +01:00
|
|
|
{
|
|
|
|
|
// Load existing table
|
|
|
|
|
fseek(ctx->imageStream, secondaryDdtOffset, SEEK_SET);
|
|
|
|
|
size_t readBytes = fread(&ddtHeader, 1, sizeof(DdtHeader2), ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
if(readBytes != sizeof(DdtHeader2) || ddtHeader.identifier != DeDuplicationTable2 || ddtHeader.type != UserData)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Invalid secondary DDT header at %" PRIu64, secondaryDdtOffset);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Read the table data (assuming no compression for now)
|
|
|
|
|
buffer = malloc(ddtHeader.length);
|
|
|
|
|
if(buffer == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Cannot allocate memory for secondary DDT.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
readBytes = fread(buffer, 1, ddtHeader.length, ctx->imageStream);
|
|
|
|
|
if(readBytes != ddtHeader.length)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not read secondary DDT data.");
|
|
|
|
|
free(buffer);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Verify CRC
|
|
|
|
|
crc64_context = aaruf_crc64_init();
|
|
|
|
|
if(crc64_context == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Could not initialize CRC64.");
|
|
|
|
|
free(buffer);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
aaruf_crc64_update(crc64_context, buffer, readBytes);
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
if(crc64 != ddtHeader.crc64)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Secondary DDT CRC mismatch. Expected 0x%16lX but got 0x%16lX.", ddtHeader.crc64, crc64);
|
|
|
|
|
free(buffer);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Cache the loaded table
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->cachedSecondaryDdtSmall = (uint16_t *)buffer;
|
|
|
|
|
else
|
|
|
|
|
ctx->cachedSecondaryDdtBig = (uint32_t *)buffer;
|
|
|
|
|
|
|
|
|
|
ctx->cachedDdtOffset = secondaryDdtOffset;
|
|
|
|
|
}
|
2025-09-28 21:26:26 +01:00
|
|
|
|
|
|
|
|
if(createNewTable)
|
2025-09-28 15:15:05 +01:00
|
|
|
{
|
|
|
|
|
// Create a new empty table
|
|
|
|
|
size_t tableSize = ctx->userDataDdtHeader.sizeType == SmallDdtSizeType ? itemsPerDdtEntry * sizeof(uint16_t)
|
|
|
|
|
: itemsPerDdtEntry * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
buffer = calloc(1, tableSize);
|
|
|
|
|
if(buffer == NULL)
|
|
|
|
|
{
|
|
|
|
|
FATAL("Cannot allocate memory for new secondary DDT.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->cachedSecondaryDdtSmall = (uint16_t *)buffer;
|
|
|
|
|
else
|
|
|
|
|
ctx->cachedSecondaryDdtBig = (uint32_t *)buffer;
|
|
|
|
|
|
2025-09-28 19:44:15 +01:00
|
|
|
ctx->cachedDdtOffset = 0; // Will be set when written to file
|
|
|
|
|
ctx->cachedDdtPosition = ddtPosition; // Track which primary DDT position this new table belongs to
|
|
|
|
|
TRACE("Created new secondary DDT for position %" PRIu64, ddtPosition);
|
2025-09-28 15:15:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Step 6: Update the corresponding DDT entry
|
|
|
|
|
blockIndex = blockOffset >> ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
ddtEntry = offset & (1ULL << ctx->userDataDdtHeader.dataShift) - 1 | blockIndex << ctx->userDataDdtHeader.dataShift;
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
{
|
2025-09-30 12:35:36 +01:00
|
|
|
// Overflow detection for DDT entry
|
|
|
|
|
if(ddtEntry > 0xFFF)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT overflow: media does not fit in small DDT");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
ddtEntry |= (uint64_t)sectorStatus << 12;
|
2025-09-28 21:25:55 +01:00
|
|
|
TRACE("Setting small secondary DDT entry %d to %u", sectorAddress % itemsPerDdtEntry, (uint16_t)ddtEntry);
|
2025-09-28 15:15:05 +01:00
|
|
|
ctx->cachedSecondaryDdtSmall[sectorAddress % itemsPerDdtEntry] = (uint16_t)ddtEntry;
|
|
|
|
|
}
|
|
|
|
|
else if(ctx->userDataDdtHeader.sizeType == BigDdtSizeType)
|
|
|
|
|
{
|
2025-09-30 12:35:36 +01:00
|
|
|
// Overflow detection for DDT entry
|
|
|
|
|
if(ddtEntry > 0xFFFFFFF)
|
|
|
|
|
{
|
|
|
|
|
FATAL("DDT overflow: media does not fit in big DDT");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-28 15:15:05 +01:00
|
|
|
ddtEntry |= (uint64_t)sectorStatus << 28;
|
2025-09-28 21:25:55 +01:00
|
|
|
TRACE("Setting big secondary DDT entry %d to %u", sectorAddress % itemsPerDdtEntry, (uint32_t)ddtEntry);
|
2025-09-28 15:15:05 +01:00
|
|
|
ctx->cachedSecondaryDdtBig[sectorAddress % itemsPerDdtEntry] = (uint32_t)ddtEntry;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TRACE("Updated secondary DDT entry at position %" PRIu64, sectorAddress % itemsPerDdtEntry);
|
|
|
|
|
TRACE("Exiting set_ddt_multi_level_v2()");
|
2025-09-28 19:44:28 +01:00
|
|
|
}
|