Files
libaaruformat/src/options.c

146 lines
5.9 KiB
C

/*
* 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 <errno.h>
#include <limits.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "structs/options.h"
#include "log.h"
/**
* @brief Parses the options string for AaruFormat image creation/opening.
*
* Parses the options string and returns a struct with the parsed options for image creation or opening.
*
* @param options String with options to parse (may be NULL).
* @return Parsed options as an aaru_options struct.
*/
aaru_options parse_options(const char *options)
{
const char *options_str = options != NULL ? options : "(null)";
TRACE("Entering parse_options(%s)", options_str);
aaru_options parsed = {.compress = true,
.deduplicate = true,
.dictionary = 33554432,
.table_shift = -1,
.data_shift = 12,
.block_alignment = 9,
.md5 = false,
.sha1 = false,
.sha256 = false,
.blake3 = false,
.spamsum = false};
if(options == NULL)
{
TRACE("Exiting parse_options() = {compress: %d, deduplicate: %d, dictionary: %u, table_shift: %d, "
"data_shift: %u, block_alignment: %u, md5: %d, sha1: %d, sha256: %d, blake3: %d, spamsum: %d}",
parsed.compress, parsed.deduplicate, parsed.dictionary, parsed.table_shift, parsed.data_shift,
parsed.block_alignment, parsed.md5, parsed.sha1, parsed.sha256, parsed.blake3, parsed.spamsum);
return parsed;
}
char buffer[1024];
strncpy(buffer, options, sizeof(buffer));
buffer[sizeof(buffer) - 1] = '\0';
char *saveptr = NULL;
char *token = strtok_r(buffer, ";", &saveptr);
while(token != NULL)
{
char *equal = strchr(token, '=');
if(equal)
{
*equal = '\0';
const char *key = token;
const char *value = equal + 1;
const bool bval = strncmp(value, "true", 4) == 0;
if(strncmp(key, "compress", 8) == 0)
parsed.compress = bval;
else if(strncmp(key, "deduplicate", 11) == 0)
parsed.deduplicate = bval;
else if(strncmp(key, "dictionary", 10) == 0)
{
parsed.dictionary = (uint32_t)strtoul(value, NULL, 10);
if(parsed.dictionary == 0) parsed.dictionary = 33554432;
}
else if(strncmp(key, "table_shift", 11) == 0)
{
errno = 0;
char *endptr = NULL;
long parsed_value = strtol(value, &endptr, 10);
if(errno == 0 && endptr != value)
{
if(parsed_value < INT8_MIN) parsed_value = INT8_MIN;
if(parsed_value > INT8_MAX) parsed_value = INT8_MAX;
parsed.table_shift = (int8_t)parsed_value;
if(parsed.table_shift == 0) parsed.table_shift = 9;
}
}
else if(strncmp(key, "data_shift", 10) == 0)
{
errno = 0;
char *endptr = NULL;
long parsed_value = strtol(value, &endptr, 10);
if(errno == 0 && endptr != value && parsed_value >= 0)
{
if(parsed_value > UINT8_MAX) parsed_value = UINT8_MAX;
parsed.data_shift = (uint8_t)parsed_value;
if(parsed.data_shift == 0) parsed.data_shift = 12;
}
}
else if(strncmp(key, "block_alignment", 15) == 0)
{
errno = 0;
char *endptr = NULL;
long parsed_value = strtol(value, &endptr, 10);
if(errno == 0 && endptr != value && parsed_value >= 0)
{
if(parsed_value > UINT8_MAX) parsed_value = UINT8_MAX;
parsed.block_alignment = (uint8_t)parsed_value;
if(parsed.block_alignment == 0) parsed.block_alignment = 9;
}
}
else if(strncmp(key, "md5", 3) == 0)
parsed.md5 = bval;
else if(strncmp(key, "sha1", 4) == 0)
parsed.sha1 = bval;
else if(strncmp(key, "sha256", 6) == 0)
parsed.sha256 = bval;
else if(strncmp(key, "blake3", 6) == 0)
parsed.blake3 = bval;
else if(strncmp(key, "spamsum", 7) == 0)
parsed.spamsum = bval;
}
token = strtok_r(NULL, ";", &saveptr);
}
TRACE("Exiting parse_options() = {compress: %d, deduplicate: %d, dictionary: %u, table_shift: %d, "
"data_shift: %u, block_alignment: %u, md5: %d, sha1: %d, sha256: %d, blake3: %d, spamsum: %d}",
parsed.compress, parsed.deduplicate, parsed.dictionary, parsed.table_shift, parsed.data_shift,
parsed.block_alignment, parsed.md5, parsed.sha1, parsed.sha256, parsed.blake3, parsed.spamsum);
return parsed;
}