2022-05-28 12:57:21 +01:00
|
|
|
/*
|
|
|
|
|
* This file is part of the Aaru Data Preservation Suite.
|
2025-08-01 21:19:45 +01:00
|
|
|
* Copyright (c) 2019-2025 Natalia Portillo.
|
2022-05-28 12:57:21 +01:00
|
|
|
*
|
|
|
|
|
* 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/>.
|
|
|
|
|
*/
|
2019-03-17 20:39:40 +00:00
|
|
|
|
|
|
|
|
#include <errno.h>
|
2019-03-31 20:52:06 +01:00
|
|
|
#include <stdio.h>
|
2022-10-03 18:15:13 +01:00
|
|
|
#include <stdlib.h>
|
2022-06-21 21:08:19 +01:00
|
|
|
|
|
|
|
|
#ifdef __linux__
|
2019-03-17 23:19:13 +00:00
|
|
|
#include <sys/mman.h>
|
2022-06-21 21:08:19 +01:00
|
|
|
#endif
|
2019-03-17 20:39:40 +00:00
|
|
|
|
2022-05-28 12:10:04 +01:00
|
|
|
#include <aaruformat.h>
|
|
|
|
|
|
2025-08-13 16:17:45 +01:00
|
|
|
#include "internal.h"
|
2025-08-14 00:38:28 +01:00
|
|
|
#include "log.h"
|
2025-08-13 16:17:45 +01:00
|
|
|
|
2024-04-30 15:51:32 +01:00
|
|
|
int aaruf_close(void *context)
|
2019-03-17 20:39:40 +00:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Entering aaruf_close(%p)", context);
|
|
|
|
|
|
2025-08-13 16:17:45 +01:00
|
|
|
int i = 0;
|
|
|
|
|
mediaTagEntry *mediaTag = NULL;
|
2025-08-01 15:34:36 +01:00
|
|
|
mediaTagEntry *tmpMediaTag = NULL;
|
2019-08-03 02:11:36 +01:00
|
|
|
|
2019-03-17 20:39:40 +00:00
|
|
|
if(context == NULL)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Invalid context");
|
2019-03-17 20:39:40 +00:00
|
|
|
errno = EINVAL;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-30 15:51:32 +01:00
|
|
|
aaruformatContext *ctx = context;
|
2019-03-17 20:39:40 +00:00
|
|
|
|
2020-03-01 19:51:13 +00:00
|
|
|
// Not a libaaruformat context
|
2020-03-01 19:58:09 +00:00
|
|
|
if(ctx->magic != AARU_MAGIC)
|
2019-03-17 20:39:40 +00:00
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
FATAL("Invalid context");
|
2019-03-17 20:39:40 +00:00
|
|
|
errno = EINVAL;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-07 15:43:35 +01:00
|
|
|
if(ctx->isWriting)
|
|
|
|
|
{
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("File is writing");
|
|
|
|
|
|
|
|
|
|
TRACE("Seeking to start of image");
|
2025-08-07 15:43:35 +01:00
|
|
|
// Write the header at the beginning of the file
|
|
|
|
|
fseek(ctx->imageStream, 0, SEEK_SET);
|
2025-08-14 00:38:28 +01:00
|
|
|
|
|
|
|
|
TRACE("Writing header at position 0");
|
2025-08-07 15:43:35 +01:00
|
|
|
if(fwrite(&ctx->header, sizeof(AaruHeaderV2), 1, ctx->imageStream) != 1)
|
|
|
|
|
{
|
|
|
|
|
fclose(ctx->imageStream);
|
|
|
|
|
ctx->imageStream = NULL;
|
2025-08-13 16:17:45 +01:00
|
|
|
errno = AARUF_ERROR_CANNOT_WRITE_HEADER;
|
2025-08-07 15:43:35 +01:00
|
|
|
return -1;
|
|
|
|
|
}
|
2025-08-13 16:17:45 +01:00
|
|
|
|
|
|
|
|
// Close current block first
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Closing current block if any");
|
2025-08-13 16:17:45 +01:00
|
|
|
if(ctx->writingBuffer != NULL)
|
|
|
|
|
{
|
|
|
|
|
int error = aaruf_close_current_block(ctx);
|
|
|
|
|
|
|
|
|
|
if(error != AARUF_STATUS_OK) return error;
|
|
|
|
|
}
|
2025-09-28 16:36:23 +01:00
|
|
|
|
|
|
|
|
// Write cached secondary table to file end and update primary table entry with its position
|
|
|
|
|
if(ctx->userDataDdtHeader.tableShift > 0 && ctx->cachedDdtOffset != 0 &&
|
|
|
|
|
(ctx->cachedSecondaryDdtSmall != NULL || ctx->cachedSecondaryDdtBig != NULL))
|
|
|
|
|
{
|
|
|
|
|
TRACE("Writing cached secondary DDT table to file");
|
|
|
|
|
|
|
|
|
|
fseek(ctx->imageStream, 0, SEEK_END);
|
|
|
|
|
long endOfFile = ftell(ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
// Align the position according to block alignment shift
|
|
|
|
|
uint64_t alignmentMask = (1ULL << ctx->userDataDdtHeader.blockAlignmentShift) - 1;
|
|
|
|
|
if(endOfFile & alignmentMask)
|
|
|
|
|
{
|
|
|
|
|
// Calculate the next aligned position
|
|
|
|
|
uint64_t alignedPosition = (endOfFile + alignmentMask) & ~alignmentMask;
|
|
|
|
|
|
|
|
|
|
// Seek to the aligned position and pad with zeros if necessary
|
|
|
|
|
fseek(ctx->imageStream, alignedPosition, SEEK_SET);
|
|
|
|
|
endOfFile = alignedPosition;
|
|
|
|
|
|
|
|
|
|
TRACE("Aligned DDT write position from %ld to %" PRIu64 " (alignment shift: %d)",
|
|
|
|
|
ftell(ctx->imageStream) - (alignedPosition - endOfFile), alignedPosition,
|
|
|
|
|
ctx->userDataDdtHeader.blockAlignmentShift);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Prepare DDT header for the cached table
|
|
|
|
|
DdtHeader2 ddtHeader = {0};
|
|
|
|
|
ddtHeader.identifier = DeDuplicationTable2;
|
|
|
|
|
ddtHeader.type = UserData;
|
|
|
|
|
ddtHeader.compression = None;
|
|
|
|
|
ddtHeader.levels = ctx->userDataDdtHeader.levels;
|
|
|
|
|
ddtHeader.tableLevel = ctx->userDataDdtHeader.tableLevel + 1;
|
|
|
|
|
ddtHeader.previousLevelOffset = ctx->primaryDdtOffset;
|
|
|
|
|
ddtHeader.negative = ctx->userDataDdtHeader.negative;
|
|
|
|
|
ddtHeader.overflow = ctx->userDataDdtHeader.overflow;
|
|
|
|
|
ddtHeader.blockAlignmentShift = ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
ddtHeader.dataShift = ctx->userDataDdtHeader.dataShift;
|
|
|
|
|
ddtHeader.tableShift = 0; // Secondary tables are single level
|
|
|
|
|
ddtHeader.sizeType = ctx->userDataDdtHeader.sizeType;
|
|
|
|
|
|
|
|
|
|
uint64_t itemsPerDdtEntry = 1 << ctx->userDataDdtHeader.tableShift;
|
|
|
|
|
ddtHeader.blocks = itemsPerDdtEntry;
|
|
|
|
|
ddtHeader.entries = itemsPerDdtEntry;
|
|
|
|
|
|
|
|
|
|
// Calculate which DDT position this cached table represents
|
|
|
|
|
uint64_t cachedDdtPosition = ctx->cachedDdtOffset >> ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
ddtHeader.start = cachedDdtPosition * 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_ctx *crc64_context = aaruf_crc64_init();
|
|
|
|
|
if(crc64_context != NULL)
|
|
|
|
|
{
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
|
|
uint64_t crc64;
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
ddtHeader.crc64 = crc64;
|
|
|
|
|
ddtHeader.cmpCrc64 = crc64;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write header
|
|
|
|
|
if(fwrite(&ddtHeader, sizeof(DdtHeader2), 1, ctx->imageStream) == 1)
|
|
|
|
|
{
|
|
|
|
|
// Write data
|
|
|
|
|
size_t writtenBytes = 0;
|
|
|
|
|
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)
|
|
|
|
|
{
|
|
|
|
|
// Update primary table entry to point to new location
|
|
|
|
|
uint64_t newSecondaryTableBlockOffset = endOfFile >> ctx->userDataDdtHeader.blockAlignmentShift;
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
ctx->userDataDdtMini[cachedDdtPosition] = (uint16_t)newSecondaryTableBlockOffset;
|
|
|
|
|
else
|
|
|
|
|
ctx->userDataDdtBig[cachedDdtPosition] = (uint32_t)newSecondaryTableBlockOffset;
|
|
|
|
|
|
|
|
|
|
// Write updated primary table back to its original position
|
|
|
|
|
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)
|
|
|
|
|
fwrite(ctx->userDataDdtMini, primaryTableSize, 1, ctx->imageStream);
|
|
|
|
|
else
|
|
|
|
|
fwrite(ctx->userDataDdtBig, primaryTableSize, 1, ctx->imageStream);
|
|
|
|
|
|
|
|
|
|
TRACE("Successfully wrote cached secondary DDT table and updated primary table");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
TRACE("Failed to write cached secondary DDT data");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
TRACE("Failed to write cached secondary DDT header");
|
|
|
|
|
|
|
|
|
|
// Free the cached table
|
|
|
|
|
if(ctx->cachedSecondaryDdtSmall != NULL)
|
|
|
|
|
{
|
|
|
|
|
free(ctx->cachedSecondaryDdtSmall);
|
|
|
|
|
ctx->cachedSecondaryDdtSmall = NULL;
|
|
|
|
|
}
|
|
|
|
|
if(ctx->cachedSecondaryDdtBig != NULL)
|
|
|
|
|
{
|
|
|
|
|
free(ctx->cachedSecondaryDdtBig);
|
|
|
|
|
ctx->cachedSecondaryDdtBig = NULL;
|
|
|
|
|
}
|
|
|
|
|
ctx->cachedDdtOffset = 0;
|
|
|
|
|
|
|
|
|
|
// Set position
|
|
|
|
|
fseek(ctx->imageStream, 0, SEEK_END);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write the cached primary DDT table back to its position in the file
|
|
|
|
|
if(ctx->userDataDdtHeader.tableShift > 0 && (ctx->userDataDdtMini != NULL || ctx->userDataDdtBig != NULL))
|
|
|
|
|
{
|
|
|
|
|
TRACE("Writing cached primary DDT table back to file");
|
|
|
|
|
|
|
|
|
|
// Calculate CRC64 of the primary DDT table data first
|
|
|
|
|
crc64_ctx *crc64_context = aaruf_crc64_init();
|
|
|
|
|
if(crc64_context != NULL)
|
|
|
|
|
{
|
|
|
|
|
size_t primaryTableSize = ctx->userDataDdtHeader.sizeType == SmallDdtSizeType
|
|
|
|
|
? ctx->userDataDdtHeader.entries * sizeof(uint16_t)
|
|
|
|
|
: ctx->userDataDdtHeader.entries * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->userDataDdtMini, primaryTableSize);
|
|
|
|
|
else
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->userDataDdtBig, primaryTableSize);
|
|
|
|
|
|
|
|
|
|
uint64_t crc64;
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
// Properly populate all header fields for multi-level DDT primary table
|
|
|
|
|
ctx->userDataDdtHeader.identifier = DeDuplicationTable2;
|
|
|
|
|
ctx->userDataDdtHeader.type = UserData;
|
|
|
|
|
ctx->userDataDdtHeader.compression = None;
|
|
|
|
|
// levels, tableLevel, previousLevelOffset, negative, overflow, blockAlignmentShift,
|
|
|
|
|
// dataShift, tableShift, sizeType, entries, blocks, start are already set during creation
|
|
|
|
|
ctx->userDataDdtHeader.crc64 = crc64;
|
|
|
|
|
ctx->userDataDdtHeader.cmpCrc64 = crc64;
|
|
|
|
|
ctx->userDataDdtHeader.length = primaryTableSize;
|
|
|
|
|
ctx->userDataDdtHeader.cmpLength = primaryTableSize;
|
|
|
|
|
|
|
|
|
|
TRACE("Calculated CRC64 for primary DDT: 0x%16lX", crc64);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// First write the DDT header
|
|
|
|
|
fseek(ctx->imageStream, ctx->primaryDdtOffset, SEEK_SET);
|
|
|
|
|
|
|
|
|
|
size_t headerWritten = fwrite(&ctx->userDataDdtHeader, sizeof(DdtHeader2), 1, ctx->imageStream);
|
|
|
|
|
if(headerWritten != 1)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Failed to write primary DDT header to file");
|
|
|
|
|
return AARUF_ERROR_CANNOT_WRITE_HEADER;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Then write the table data (position is already after the header)
|
|
|
|
|
size_t primaryTableSize = ctx->userDataDdtHeader.sizeType == SmallDdtSizeType
|
|
|
|
|
? ctx->userDataDdtHeader.entries * sizeof(uint16_t)
|
|
|
|
|
: ctx->userDataDdtHeader.entries * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
// Write the primary table data
|
|
|
|
|
size_t writtenBytes = 0;
|
|
|
|
|
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)
|
|
|
|
|
TRACE("Successfully wrote primary DDT header and table to file (%" PRIu64 " entries, %zu bytes)",
|
|
|
|
|
ctx->userDataDdtHeader.entries, primaryTableSize);
|
|
|
|
|
else
|
|
|
|
|
TRACE("Failed to write primary DDT table to file");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write the single level DDT table block aligned just after the header
|
|
|
|
|
if(ctx->userDataDdtHeader.tableShift == 0 && (ctx->userDataDdtMini != NULL || ctx->userDataDdtBig != NULL))
|
|
|
|
|
{
|
|
|
|
|
TRACE("Writing single-level DDT table to file");
|
|
|
|
|
|
|
|
|
|
// Calculate CRC64 of the primary DDT table data
|
|
|
|
|
crc64_ctx *crc64_context = aaruf_crc64_init();
|
|
|
|
|
if(crc64_context != NULL)
|
|
|
|
|
{
|
|
|
|
|
size_t primaryTableSize = ctx->userDataDdtHeader.sizeType == SmallDdtSizeType
|
|
|
|
|
? ctx->userDataDdtHeader.entries * sizeof(uint16_t)
|
|
|
|
|
: ctx->userDataDdtHeader.entries * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
if(ctx->userDataDdtHeader.sizeType == SmallDdtSizeType)
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->userDataDdtMini, primaryTableSize);
|
|
|
|
|
else
|
|
|
|
|
aaruf_crc64_update(crc64_context, (uint8_t *)ctx->userDataDdtBig, primaryTableSize);
|
|
|
|
|
|
|
|
|
|
uint64_t crc64;
|
|
|
|
|
aaruf_crc64_final(crc64_context, &crc64);
|
|
|
|
|
|
|
|
|
|
// Properly populate all header fields
|
|
|
|
|
ctx->userDataDdtHeader.identifier = DeDuplicationTable2;
|
|
|
|
|
ctx->userDataDdtHeader.type = UserData;
|
|
|
|
|
ctx->userDataDdtHeader.compression = None;
|
|
|
|
|
ctx->userDataDdtHeader.levels = 1; // Single level
|
|
|
|
|
ctx->userDataDdtHeader.tableLevel = 0; // Top level
|
|
|
|
|
ctx->userDataDdtHeader.previousLevelOffset = 0; // No previous level for single-level DDT
|
|
|
|
|
// negative and overflow are already set during creation
|
|
|
|
|
// blockAlignmentShift, dataShift, tableShift, sizeType, entries, blocks, start are already set
|
|
|
|
|
ctx->userDataDdtHeader.crc64 = crc64;
|
|
|
|
|
ctx->userDataDdtHeader.cmpCrc64 = crc64;
|
|
|
|
|
ctx->userDataDdtHeader.length = primaryTableSize;
|
|
|
|
|
ctx->userDataDdtHeader.cmpLength = primaryTableSize;
|
|
|
|
|
|
|
|
|
|
TRACE("Calculated CRC64 for single-level DDT: 0x%16lX", crc64);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write the DDT header first
|
|
|
|
|
fseek(ctx->imageStream, ctx->primaryDdtOffset, SEEK_SET);
|
|
|
|
|
|
|
|
|
|
size_t headerWritten = fwrite(&ctx->userDataDdtHeader, sizeof(DdtHeader2), 1, ctx->imageStream);
|
|
|
|
|
if(headerWritten != 1)
|
|
|
|
|
{
|
|
|
|
|
TRACE("Failed to write single-level DDT header to file");
|
|
|
|
|
return AARUF_ERROR_CANNOT_WRITE_HEADER;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Then write the table data (position is already after the header)
|
|
|
|
|
size_t primaryTableSize = ctx->userDataDdtHeader.sizeType == SmallDdtSizeType
|
|
|
|
|
? ctx->userDataDdtHeader.entries * sizeof(uint16_t)
|
|
|
|
|
: ctx->userDataDdtHeader.entries * sizeof(uint32_t);
|
|
|
|
|
|
|
|
|
|
// Write the primary table data
|
|
|
|
|
size_t writtenBytes = 0;
|
|
|
|
|
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)
|
|
|
|
|
TRACE("Successfully wrote single-level DDT header and table to file (%" PRIu64 " entries, %zu bytes)",
|
|
|
|
|
ctx->userDataDdtHeader.entries, primaryTableSize);
|
|
|
|
|
else
|
|
|
|
|
TRACE("Failed to write single-level DDT table data to file");
|
|
|
|
|
}
|
2025-08-07 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Freeing memory pointers");
|
2019-03-17 22:41:04 +00:00
|
|
|
// This may do nothing if imageStream is NULL, but as the behaviour is undefined, better sure than sorry
|
2022-10-03 19:31:39 +01:00
|
|
|
if(ctx->imageStream != NULL)
|
|
|
|
|
{
|
|
|
|
|
fclose(ctx->imageStream);
|
|
|
|
|
ctx->imageStream = NULL;
|
|
|
|
|
}
|
2019-03-17 20:39:40 +00:00
|
|
|
|
2019-03-17 22:41:04 +00:00
|
|
|
free(ctx->sectorPrefix);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->sectorPrefix = NULL;
|
2019-03-17 22:41:04 +00:00
|
|
|
free(ctx->sectorPrefixCorrected);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->sectorPrefixCorrected = NULL;
|
2019-03-17 22:41:04 +00:00
|
|
|
free(ctx->sectorSuffix);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->sectorSuffix = NULL;
|
2019-03-17 22:41:04 +00:00
|
|
|
free(ctx->sectorSuffixCorrected);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->sectorSuffixCorrected = NULL;
|
2019-03-17 22:41:04 +00:00
|
|
|
free(ctx->sectorSubchannel);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->sectorSubchannel = NULL;
|
2019-03-17 22:41:04 +00:00
|
|
|
free(ctx->mode2Subheaders);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->mode2Subheaders = NULL;
|
2019-03-17 22:41:04 +00:00
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Freeing media tags");
|
|
|
|
|
if(ctx->mediaTags != NULL) HASH_ITER(hh, ctx->mediaTags, mediaTag, tmpMediaTag)
|
2019-03-17 23:01:54 +00:00
|
|
|
{
|
2022-10-04 20:32:26 +01:00
|
|
|
HASH_DEL(ctx->mediaTags, mediaTag);
|
2019-03-17 23:01:54 +00:00
|
|
|
free(mediaTag->data);
|
2022-10-04 20:32:26 +01:00
|
|
|
free(mediaTag);
|
2019-03-17 23:01:54 +00:00
|
|
|
}
|
|
|
|
|
|
2024-04-30 15:51:32 +01:00
|
|
|
#ifdef __linux__ // TODO: Implement
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Unmapping user data DDT if it is not in memory");
|
2022-10-03 19:31:39 +01:00
|
|
|
if(!ctx->inMemoryDdt)
|
|
|
|
|
{
|
|
|
|
|
munmap(ctx->userDataDdt, ctx->mappedMemoryDdtSize);
|
|
|
|
|
ctx->userDataDdt = NULL;
|
|
|
|
|
}
|
2022-06-21 21:08:19 +01:00
|
|
|
#endif
|
2019-03-17 23:25:45 +00:00
|
|
|
|
|
|
|
|
free(ctx->sectorPrefixDdt);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->sectorPrefixDdt = NULL;
|
2019-03-17 23:25:45 +00:00
|
|
|
free(ctx->sectorSuffixDdt);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->sectorSuffixDdt = NULL;
|
2019-03-17 23:25:45 +00:00
|
|
|
|
2019-03-17 23:41:07 +00:00
|
|
|
free(ctx->metadataBlock);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->metadataBlock = NULL;
|
2019-03-18 00:10:24 +00:00
|
|
|
free(ctx->trackEntries);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->trackEntries = NULL;
|
2019-03-18 22:06:10 +00:00
|
|
|
free(ctx->cicmBlock);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->cicmBlock = NULL;
|
2019-03-17 23:41:07 +00:00
|
|
|
|
2019-03-20 00:23:30 +00:00
|
|
|
if(ctx->dumpHardwareEntriesWithData != NULL)
|
|
|
|
|
{
|
2019-08-03 02:11:36 +01:00
|
|
|
for(i = 0; i < ctx->dumpHardwareHeader.entries; i++)
|
2019-03-20 00:23:30 +00:00
|
|
|
{
|
|
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].extents);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].extents = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].manufacturer);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].manufacturer = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].model);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].model = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].revision);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].revision = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].firmware);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].firmware = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].serial);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].serial = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].softwareName);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].softwareName = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].softwareVersion);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].softwareVersion = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
free(ctx->dumpHardwareEntriesWithData[i].softwareOperatingSystem);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData[i].softwareOperatingSystem = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
}
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->dumpHardwareEntriesWithData = NULL;
|
2019-03-20 00:23:30 +00:00
|
|
|
}
|
|
|
|
|
|
2019-03-31 14:56:03 +01:00
|
|
|
free(ctx->readableSectorTags);
|
2022-10-03 19:31:39 +01:00
|
|
|
ctx->readableSectorTags = NULL;
|
|
|
|
|
|
2022-10-03 19:32:25 +01:00
|
|
|
free(ctx->eccCdContext);
|
|
|
|
|
ctx->eccCdContext = NULL;
|
|
|
|
|
|
2022-10-04 19:44:34 +01:00
|
|
|
free(ctx->checksums.spamsum);
|
|
|
|
|
ctx->checksums.spamsum = NULL;
|
|
|
|
|
|
2022-10-03 19:31:39 +01:00
|
|
|
// TODO: Free caches
|
2019-03-31 14:56:03 +01:00
|
|
|
|
2019-03-17 20:39:40 +00:00
|
|
|
free(context);
|
|
|
|
|
|
2025-08-14 00:38:28 +01:00
|
|
|
TRACE("Exiting aaruf_close() = 0");
|
2019-03-17 20:39:40 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|