Files
libaaruformat/src/index/index_v2.c

147 lines
4.8 KiB
C
Raw Normal View History

2025-08-01 21:10:36 +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 <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include "aaruformat.h"
#include "log.h"
2025-08-01 21:10:36 +01:00
#include "utarray.h"
2025-08-01 21:18:48 +01:00
UT_array *process_index_v2(aaruformatContext *ctx)
2025-08-01 21:10:36 +01:00
{
TRACE("Entering process_index_v2(%p)", ctx);
2025-08-01 21:10:36 +01:00
UT_array *index_entries = NULL;
IndexEntry entry;
if(ctx == NULL || ctx->imageStream == NULL) {return NULL;}
2025-08-01 21:10:36 +01:00
// Initialize the index entries array
UT_icd index_entry_icd = {sizeof(IndexEntry), NULL, NULL, NULL};
utarray_new(index_entries, &index_entry_icd);
// Read the index header
TRACE("Reading index header at position %llu", ctx->header.indexOffset);
2025-08-01 21:10:36 +01:00
fseek(ctx->imageStream, ctx->header.indexOffset, SEEK_SET);
2025-08-01 21:18:48 +01:00
IndexHeader2 idx_header;
fread(&idx_header, sizeof(IndexHeader2), 1, ctx->imageStream);
2025-08-01 21:10:36 +01:00
// Check if the index header is valid
2025-08-01 21:18:48 +01:00
if(idx_header.identifier != IndexBlock2)
2025-08-01 21:10:36 +01:00
{
FATAL("Incorrect index identifier.");
2025-08-01 21:10:36 +01:00
utarray_free(index_entries);
TRACE("Exiting process_index_v2() = NULL");
2025-08-01 21:10:36 +01:00
return NULL;
}
for(int i = 0; i < idx_header.entries; i++)
{
fread(&entry, sizeof(IndexEntry), 1, ctx->imageStream);
utarray_push_back(index_entries, &entry);
}
TRACE("Read %d index entries from index block at position %llu", idx_header.entries, ctx->header.indexOffset);
2025-08-01 21:10:36 +01:00
return index_entries;
}
2025-08-01 21:18:48 +01:00
int32_t verify_index_v2(aaruformatContext *ctx)
2025-08-01 21:10:36 +01:00
{
TRACE("Entering verify_index_v2(%p)", ctx);
2025-08-02 20:37:30 +01:00
size_t read_bytes = 0;
IndexHeader2 index_header;
uint64_t crc64 = 0;
IndexEntry *index_entries = NULL;
2025-08-01 21:10:36 +01:00
if(ctx == NULL || ctx->imageStream == NULL)
{
FATAL("Invalid context or image stream.");
TRACE("Exiting verify_index_v2() = AARUF_ERROR_NOT_AARUFORMAT");
return AARUF_ERROR_NOT_AARUFORMAT;
}
2025-08-01 21:10:36 +01:00
// This will traverse all blocks and check their CRC64 without uncompressing them
TRACE("Checking index integrity at %llu.", ctx->header.indexOffset);
2025-08-01 21:10:36 +01:00
fseek(ctx->imageStream, ctx->header.indexOffset, SEEK_SET);
// Read the index header
TRACE("Reading index header at position %llu", ctx->header.indexOffset);
2025-08-01 21:18:48 +01:00
read_bytes = fread(&index_header, 1, sizeof(IndexHeader2), ctx->imageStream);
2025-08-01 21:10:36 +01:00
2025-08-01 21:18:48 +01:00
if(read_bytes != sizeof(IndexHeader2))
2025-08-01 21:10:36 +01:00
{
FATAL("Could not read index header.");
TRACE("Exiting verify_index_v2() = AARUF_ERROR_CANNOT_READ_HEADER");
2025-08-01 21:10:36 +01:00
return AARUF_ERROR_CANNOT_READ_HEADER;
}
2025-08-02 20:37:30 +01:00
if(index_header.identifier != IndexBlock2)
2025-08-01 21:10:36 +01:00
{
FATAL("Incorrect index identifier.");
TRACE("Exiting verify_index_v2() = AARUF_ERROR_CANNOT_READ_INDEX");
2025-08-01 21:10:36 +01:00
return AARUF_ERROR_CANNOT_READ_INDEX;
}
TRACE("Index at %llu contains %d entries.", ctx->header.indexOffset, index_header.entries);
2025-08-01 21:10:36 +01:00
index_entries = malloc(sizeof(IndexEntry) * index_header.entries);
if(index_entries == NULL)
{
FATAL("Cannot allocate memory for index entries.");
TRACE("Exiting verify_index_v2() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2025-08-01 21:10:36 +01:00
return AARUF_ERROR_NOT_ENOUGH_MEMORY;
}
read_bytes = fread(index_entries, 1, sizeof(IndexEntry) * index_header.entries, ctx->imageStream);
if(read_bytes != sizeof(IndexEntry) * index_header.entries)
{
FATAL("Could not read index entries.");
2025-08-01 21:10:36 +01:00
free(index_entries);
TRACE("Exiting verify_index_v2() = AARUF_ERROR_CANNOT_READ_INDEX");
2025-08-01 21:10:36 +01:00
return AARUF_ERROR_CANNOT_READ_INDEX;
}
crc64 = aaruf_crc64_data((const uint8_t *)index_entries, sizeof(IndexEntry) * index_header.entries);
// Due to how C# wrote it, it is effectively reversed
if(ctx->header.imageMajorVersion <= AARUF_VERSION_V1) crc64 = bswap_64(crc64);
2025-08-01 21:10:36 +01:00
if(crc64 != index_header.crc64)
{
FATAL("Expected index CRC 0x%16llX but got 0x%16llX.", index_header.crc64, crc64);
2025-08-01 21:10:36 +01:00
free(index_entries);
TRACE("Exiting verify_index_v2() = AARUF_ERROR_INVALID_BLOCK_CRC");
2025-08-01 21:10:36 +01:00
return AARUF_ERROR_INVALID_BLOCK_CRC;
}
TRACE("Exiting verify_index_v2() = AARUF_OK");
2025-08-01 21:10:36 +01:00
return AARUF_STATUS_OK;
}