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
|
2008-08-28 21:53:15 +02:00
|
|
|
lzma_block_header_size(lzma_block *options)
|
2007-12-08 23:42:33 +01:00
|
|
|
{
|
2008-06-18 17:02:10 +02:00
|
|
|
// Block Header Size + Block Flags + CRC32.
|
|
|
|
size_t size = 1 + 1 + 4;
|
2007-12-08 23:42:33 +01:00
|
|
|
|
|
|
|
// Compressed Size
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->compressed_size != LZMA_VLI_VALUE_UNKNOWN) {
|
|
|
|
if (options->compressed_size > LZMA_VLI_VALUE_MAX / 4 - 1
|
|
|
|
|| options->compressed_size == 0
|
|
|
|
|| (options->compressed_size & 3))
|
2007-12-08 23:42:33 +01:00
|
|
|
return LZMA_PROG_ERROR;
|
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
size += lzma_vli_size(options->compressed_size / 4 - 1);
|
2007-12-08 23:42:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Uncompressed Size
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->uncompressed_size != LZMA_VLI_VALUE_UNKNOWN) {
|
|
|
|
const size_t add = lzma_vli_size(options->uncompressed_size);
|
|
|
|
if (add == 0)
|
2007-12-08 23:42:33 +01:00
|
|
|
return LZMA_PROG_ERROR;
|
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
size += add;
|
2007-12-08 23:42:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// List of Filter Flags
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->filters == NULL
|
|
|
|
|| options->filters[0].id == LZMA_VLI_VALUE_UNKNOWN)
|
|
|
|
return LZMA_PROG_ERROR;
|
|
|
|
|
2007-12-08 23:42:33 +01:00
|
|
|
for (size_t i = 0; options->filters[i].id != LZMA_VLI_VALUE_UNKNOWN;
|
|
|
|
++i) {
|
|
|
|
// Don't allow too many filters.
|
2008-06-18 17:02:10 +02:00
|
|
|
if (i == 4)
|
2007-12-08 23:42:33 +01:00
|
|
|
return LZMA_PROG_ERROR;
|
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
uint32_t add;
|
|
|
|
return_if_error(lzma_filter_flags_size(&add,
|
|
|
|
options->filters + i));
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
size += add;
|
2007-12-08 23:42:33 +01:00
|
|
|
}
|
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
// Pad to a multiple of four bytes.
|
|
|
|
options->header_size = (size + 3) & ~(size_t)(3);
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
// NOTE: We don't verify that Total Size of the Block stays within
|
|
|
|
// limits. This is because it is possible that we are called with
|
|
|
|
// exaggerated values 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
|
2008-08-28 21:53:15 +02:00
|
|
|
lzma_block_header_encode(const lzma_block *options, uint8_t *out)
|
2007-12-08 23:42:33 +01:00
|
|
|
{
|
2008-06-18 17:02:10 +02:00
|
|
|
if ((options->header_size & 3)
|
|
|
|
|| options->header_size < LZMA_BLOCK_HEADER_SIZE_MIN
|
|
|
|
|| options->header_size > LZMA_BLOCK_HEADER_SIZE_MAX)
|
2007-12-08 23:42:33 +01:00
|
|
|
return LZMA_PROG_ERROR;
|
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
// 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 a little later.
|
2007-12-08 23:42:33 +01:00
|
|
|
size_t out_pos = 2;
|
|
|
|
|
|
|
|
// Compressed Size
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->compressed_size != LZMA_VLI_VALUE_UNKNOWN) {
|
|
|
|
// Compressed Size must be non-zero, fit into a 63-bit
|
|
|
|
// integer and be a multiple of four. Also the Total Size
|
|
|
|
// of the Block must fit into 63-bit integer.
|
|
|
|
if (options->compressed_size == 0
|
|
|
|
|| (options->compressed_size & 3)
|
|
|
|
|| options->compressed_size
|
|
|
|
> LZMA_VLI_VALUE_MAX
|
|
|
|
|| lzma_block_total_size_get(options) == 0)
|
2007-12-08 23:42:33 +01:00
|
|
|
return LZMA_PROG_ERROR;
|
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
return_if_error(lzma_vli_encode(
|
|
|
|
options->compressed_size / 4 - 1, NULL,
|
|
|
|
out, &out_pos, out_size));
|
2007-12-08 23:42:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Uncompressed Size
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->uncompressed_size != LZMA_VLI_VALUE_UNKNOWN)
|
|
|
|
return_if_error(lzma_vli_encode(
|
|
|
|
options->uncompressed_size, NULL,
|
|
|
|
out, &out_pos, out_size));
|
2007-12-08 23:42:33 +01:00
|
|
|
|
|
|
|
// Filter Flags
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->filters == NULL
|
|
|
|
|| options->filters[0].id == LZMA_VLI_VALUE_UNKNOWN)
|
2007-12-08 23:42:33 +01:00
|
|
|
return LZMA_PROG_ERROR;
|
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
size_t filter_count = 0;
|
|
|
|
do {
|
|
|
|
// There can be at maximum of four filters.
|
|
|
|
if (filter_count == 4)
|
|
|
|
return LZMA_PROG_ERROR;
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-08-28 21:53:15 +02: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
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
} while (options->filters[++filter_count].id
|
|
|
|
!= LZMA_VLI_VALUE_UNKNOWN);
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
// Block Flags
|
|
|
|
out[1] = filter_count - 1;
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->compressed_size != LZMA_VLI_VALUE_UNKNOWN)
|
|
|
|
out[1] |= 0x40;
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
if (options->uncompressed_size != LZMA_VLI_VALUE_UNKNOWN)
|
|
|
|
out[1] |= 0x80;
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-06-18 17:02:10 +02:00
|
|
|
// Padding
|
|
|
|
memzero(out + out_pos, out_size - out_pos);
|
2007-12-08 23:42:33 +01:00
|
|
|
|
2008-06-18 17:02:10 +02: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;
|
|
|
|
}
|