1
0
Fork 0
mirror of https://git.tukaani.org/xz.git synced 2024-04-04 12:36:23 +02:00
xz-archive/src/liblzma/common/block_header_encoder.c

141 lines
3.7 KiB
C
Raw Normal View History

2007-12-08 23:42:33 +01:00
///////////////////////////////////////////////////////////////////////////////
//
/// \file block_header_encoder.c
/// \brief Encodes Block Header for .lzma files
//
// Copyright (C) 2007 Lasse Collin
//
// 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.
//
///////////////////////////////////////////////////////////////////////////////
#include "common.h"
#include "check.h"
extern LZMA_API lzma_ret
lzma_block_header_size(lzma_block *options)
2007-12-08 23:42:33 +01:00
{
// Block Header Size + Block Flags + CRC32.
uint32_t size = 1 + 1 + 4;
2007-12-08 23:42:33 +01:00
// Compressed Size
if (options->compressed_size != LZMA_VLI_UNKNOWN) {
const uint32_t add = lzma_vli_size(options->compressed_size);
if (add == 0 || options->compressed_size == 0)
2007-12-08 23:42:33 +01:00
return LZMA_PROG_ERROR;
size += add;
2007-12-08 23:42:33 +01:00
}
// Uncompressed Size
if (options->uncompressed_size != LZMA_VLI_UNKNOWN) {
const uint32_t add = lzma_vli_size(options->uncompressed_size);
if (add == 0)
2007-12-08 23:42:33 +01:00
return LZMA_PROG_ERROR;
size += add;
2007-12-08 23:42:33 +01:00
}
// List of Filter Flags
if (options->filters == NULL
|| options->filters[0].id == LZMA_VLI_UNKNOWN)
return LZMA_PROG_ERROR;
for (size_t i = 0; options->filters[i].id != LZMA_VLI_UNKNOWN; ++i) {
2007-12-08 23:42:33 +01:00
// Don't allow too many filters.
if (i == LZMA_FILTERS_MAX)
2007-12-08 23:42:33 +01:00
return LZMA_PROG_ERROR;
uint32_t add;
return_if_error(lzma_filter_flags_size(&add,
options->filters + i));
2007-12-08 23:42:33 +01:00
size += add;
2007-12-08 23:42:33 +01:00
}
// Pad to a multiple of four bytes.
options->header_size = (size + 3) & ~UINT32_C(3);
2007-12-08 23:42:33 +01:00
// NOTE: We don't verify that the encoded size of the Block stays
// within limits. This is because it is possible that we are called
// with exaggerated Compressed Size (e.g. LZMA_VLI_MAX) to reserve
// space for Block Header, and later called again with lower,
// real values.
2007-12-08 23:42:33 +01:00
return LZMA_OK;
}
extern LZMA_API lzma_ret
lzma_block_header_encode(const lzma_block *options, uint8_t *out)
2007-12-08 23:42:33 +01:00
{
// Valdidate everything but filters.
if (lzma_block_unpadded_size(options) == 0
|| !lzma_vli_is_valid(options->uncompressed_size))
2007-12-08 23:42:33 +01:00
return LZMA_PROG_ERROR;
// Indicate the size of the buffer _excluding_ the CRC32 field.
const size_t out_size = options->header_size - 4;
// Store the Block Header Size.
out[0] = out_size / 4;
// We write Block Flags in pieces.
out[1] = 0x00;
2007-12-08 23:42:33 +01:00
size_t out_pos = 2;
// Compressed Size
if (options->compressed_size != LZMA_VLI_UNKNOWN) {
return_if_error(lzma_vli_encode(
options->compressed_size, NULL,
out, &out_pos, out_size));
out[1] |= 0x40;
2007-12-08 23:42:33 +01:00
}
// Uncompressed Size
if (options->uncompressed_size != LZMA_VLI_UNKNOWN) {
return_if_error(lzma_vli_encode(
options->uncompressed_size, NULL,
out, &out_pos, out_size));
2007-12-08 23:42:33 +01:00
out[1] |= 0x80;
}
2007-12-08 23:42:33 +01:00
// Filter Flags
if (options->filters == NULL
|| options->filters[0].id == LZMA_VLI_UNKNOWN)
2007-12-08 23:42:33 +01:00
return LZMA_PROG_ERROR;
size_t filter_count = 0;
do {
// There can be at maximum of four filters.
if (filter_count == LZMA_FILTERS_MAX)
return LZMA_PROG_ERROR;
2007-12-08 23:42:33 +01:00
return_if_error(lzma_filter_flags_encode(
options->filters + filter_count,
out, &out_pos, out_size));
2007-12-08 23:42:33 +01:00
} while (options->filters[++filter_count].id != LZMA_VLI_UNKNOWN);
2007-12-08 23:42:33 +01:00
out[1] |= filter_count - 1;
2007-12-08 23:42:33 +01:00
// Padding
memzero(out + out_pos, out_size - out_pos);
2007-12-08 23:42:33 +01:00
// CRC32
integer_write_32(out + out_size, lzma_crc32(out, out_size, 0));
2007-12-08 23:42:33 +01:00
return LZMA_OK;
}