12940b44dSPeter Avalos ///////////////////////////////////////////////////////////////////////////////
22940b44dSPeter Avalos //
32940b44dSPeter Avalos /// \file       filter_decoder.c
42940b44dSPeter Avalos /// \brief      Filter ID mapping to filter-specific functions
52940b44dSPeter Avalos //
62940b44dSPeter Avalos //  Author:     Lasse Collin
72940b44dSPeter Avalos //
82940b44dSPeter Avalos //  This file has been put into the public domain.
92940b44dSPeter Avalos //  You can do whatever you want with this file.
102940b44dSPeter Avalos //
112940b44dSPeter Avalos ///////////////////////////////////////////////////////////////////////////////
122940b44dSPeter Avalos 
132940b44dSPeter Avalos #include "filter_encoder.h"
142940b44dSPeter Avalos #include "filter_common.h"
152940b44dSPeter Avalos #include "lzma_encoder.h"
162940b44dSPeter Avalos #include "lzma2_encoder.h"
172940b44dSPeter Avalos #include "simple_encoder.h"
182940b44dSPeter Avalos #include "delta_encoder.h"
192940b44dSPeter Avalos 
202940b44dSPeter Avalos 
212940b44dSPeter Avalos typedef struct {
222940b44dSPeter Avalos 	/// Filter ID
232940b44dSPeter Avalos 	lzma_vli id;
242940b44dSPeter Avalos 
252940b44dSPeter Avalos 	/// Initializes the filter encoder and calls lzma_next_filter_init()
262940b44dSPeter Avalos 	/// for filters + 1.
272940b44dSPeter Avalos 	lzma_init_function init;
282940b44dSPeter Avalos 
292940b44dSPeter Avalos 	/// Calculates memory usage of the encoder. If the options are
302940b44dSPeter Avalos 	/// invalid, UINT64_MAX is returned.
312940b44dSPeter Avalos 	uint64_t (*memusage)(const void *options);
322940b44dSPeter Avalos 
33*15ab8c86SJohn Marino 	/// Calculates the recommended Uncompressed Size for .xz Blocks to
34*15ab8c86SJohn Marino 	/// which the input data can be split to make multithreaded
35*15ab8c86SJohn Marino 	/// encoding possible. If this is NULL, it is assumed that
36*15ab8c86SJohn Marino 	/// the encoder is fast enough with single thread.
37*15ab8c86SJohn Marino 	uint64_t (*block_size)(const void *options);
382940b44dSPeter Avalos 
392940b44dSPeter Avalos 	/// Tells the size of the Filter Properties field. If options are
402940b44dSPeter Avalos 	/// invalid, UINT32_MAX is returned. If this is NULL, props_size_fixed
412940b44dSPeter Avalos 	/// is used.
422940b44dSPeter Avalos 	lzma_ret (*props_size_get)(uint32_t *size, const void *options);
432940b44dSPeter Avalos 	uint32_t props_size_fixed;
442940b44dSPeter Avalos 
452940b44dSPeter Avalos 	/// Encodes Filter Properties.
462940b44dSPeter Avalos 	///
472940b44dSPeter Avalos 	/// \return     - LZMA_OK: Properties encoded successfully.
482940b44dSPeter Avalos 	///             - LZMA_OPTIONS_ERROR: Unsupported options
492940b44dSPeter Avalos 	///             - LZMA_PROG_ERROR: Invalid options or not enough
502940b44dSPeter Avalos 	///               output space
512940b44dSPeter Avalos 	lzma_ret (*props_encode)(const void *options, uint8_t *out);
522940b44dSPeter Avalos 
532940b44dSPeter Avalos } lzma_filter_encoder;
542940b44dSPeter Avalos 
552940b44dSPeter Avalos 
562940b44dSPeter Avalos static const lzma_filter_encoder encoders[] = {
572940b44dSPeter Avalos #ifdef HAVE_ENCODER_LZMA1
582940b44dSPeter Avalos 	{
592940b44dSPeter Avalos 		.id = LZMA_FILTER_LZMA1,
602940b44dSPeter Avalos 		.init = &lzma_lzma_encoder_init,
612940b44dSPeter Avalos 		.memusage = &lzma_lzma_encoder_memusage,
62*15ab8c86SJohn Marino 		.block_size = NULL, // FIXME
632940b44dSPeter Avalos 		.props_size_get = NULL,
642940b44dSPeter Avalos 		.props_size_fixed = 5,
652940b44dSPeter Avalos 		.props_encode = &lzma_lzma_props_encode,
662940b44dSPeter Avalos 	},
672940b44dSPeter Avalos #endif
682940b44dSPeter Avalos #ifdef HAVE_ENCODER_LZMA2
692940b44dSPeter Avalos 	{
702940b44dSPeter Avalos 		.id = LZMA_FILTER_LZMA2,
712940b44dSPeter Avalos 		.init = &lzma_lzma2_encoder_init,
722940b44dSPeter Avalos 		.memusage = &lzma_lzma2_encoder_memusage,
73*15ab8c86SJohn Marino 		.block_size = &lzma_lzma2_block_size, // FIXME
742940b44dSPeter Avalos 		.props_size_get = NULL,
752940b44dSPeter Avalos 		.props_size_fixed = 1,
762940b44dSPeter Avalos 		.props_encode = &lzma_lzma2_props_encode,
772940b44dSPeter Avalos 	},
782940b44dSPeter Avalos #endif
792940b44dSPeter Avalos #ifdef HAVE_ENCODER_X86
802940b44dSPeter Avalos 	{
812940b44dSPeter Avalos 		.id = LZMA_FILTER_X86,
822940b44dSPeter Avalos 		.init = &lzma_simple_x86_encoder_init,
832940b44dSPeter Avalos 		.memusage = NULL,
84*15ab8c86SJohn Marino 		.block_size = NULL,
852940b44dSPeter Avalos 		.props_size_get = &lzma_simple_props_size,
862940b44dSPeter Avalos 		.props_encode = &lzma_simple_props_encode,
872940b44dSPeter Avalos 	},
882940b44dSPeter Avalos #endif
892940b44dSPeter Avalos #ifdef HAVE_ENCODER_POWERPC
902940b44dSPeter Avalos 	{
912940b44dSPeter Avalos 		.id = LZMA_FILTER_POWERPC,
922940b44dSPeter Avalos 		.init = &lzma_simple_powerpc_encoder_init,
932940b44dSPeter Avalos 		.memusage = NULL,
94*15ab8c86SJohn Marino 		.block_size = NULL,
952940b44dSPeter Avalos 		.props_size_get = &lzma_simple_props_size,
962940b44dSPeter Avalos 		.props_encode = &lzma_simple_props_encode,
972940b44dSPeter Avalos 	},
982940b44dSPeter Avalos #endif
992940b44dSPeter Avalos #ifdef HAVE_ENCODER_IA64
1002940b44dSPeter Avalos 	{
1012940b44dSPeter Avalos 		.id = LZMA_FILTER_IA64,
1022940b44dSPeter Avalos 		.init = &lzma_simple_ia64_encoder_init,
1032940b44dSPeter Avalos 		.memusage = NULL,
104*15ab8c86SJohn Marino 		.block_size = NULL,
1052940b44dSPeter Avalos 		.props_size_get = &lzma_simple_props_size,
1062940b44dSPeter Avalos 		.props_encode = &lzma_simple_props_encode,
1072940b44dSPeter Avalos 	},
1082940b44dSPeter Avalos #endif
1092940b44dSPeter Avalos #ifdef HAVE_ENCODER_ARM
1102940b44dSPeter Avalos 	{
1112940b44dSPeter Avalos 		.id = LZMA_FILTER_ARM,
1122940b44dSPeter Avalos 		.init = &lzma_simple_arm_encoder_init,
1132940b44dSPeter Avalos 		.memusage = NULL,
114*15ab8c86SJohn Marino 		.block_size = NULL,
1152940b44dSPeter Avalos 		.props_size_get = &lzma_simple_props_size,
1162940b44dSPeter Avalos 		.props_encode = &lzma_simple_props_encode,
1172940b44dSPeter Avalos 	},
1182940b44dSPeter Avalos #endif
1192940b44dSPeter Avalos #ifdef HAVE_ENCODER_ARMTHUMB
1202940b44dSPeter Avalos 	{
1212940b44dSPeter Avalos 		.id = LZMA_FILTER_ARMTHUMB,
1222940b44dSPeter Avalos 		.init = &lzma_simple_armthumb_encoder_init,
1232940b44dSPeter Avalos 		.memusage = NULL,
124*15ab8c86SJohn Marino 		.block_size = NULL,
1252940b44dSPeter Avalos 		.props_size_get = &lzma_simple_props_size,
1262940b44dSPeter Avalos 		.props_encode = &lzma_simple_props_encode,
1272940b44dSPeter Avalos 	},
1282940b44dSPeter Avalos #endif
1292940b44dSPeter Avalos #ifdef HAVE_ENCODER_SPARC
1302940b44dSPeter Avalos 	{
1312940b44dSPeter Avalos 		.id = LZMA_FILTER_SPARC,
1322940b44dSPeter Avalos 		.init = &lzma_simple_sparc_encoder_init,
1332940b44dSPeter Avalos 		.memusage = NULL,
134*15ab8c86SJohn Marino 		.block_size = NULL,
1352940b44dSPeter Avalos 		.props_size_get = &lzma_simple_props_size,
1362940b44dSPeter Avalos 		.props_encode = &lzma_simple_props_encode,
1372940b44dSPeter Avalos 	},
1382940b44dSPeter Avalos #endif
1392940b44dSPeter Avalos #ifdef HAVE_ENCODER_DELTA
1402940b44dSPeter Avalos 	{
1412940b44dSPeter Avalos 		.id = LZMA_FILTER_DELTA,
1422940b44dSPeter Avalos 		.init = &lzma_delta_encoder_init,
1432940b44dSPeter Avalos 		.memusage = &lzma_delta_coder_memusage,
144*15ab8c86SJohn Marino 		.block_size = NULL,
1452940b44dSPeter Avalos 		.props_size_get = NULL,
1462940b44dSPeter Avalos 		.props_size_fixed = 1,
1472940b44dSPeter Avalos 		.props_encode = &lzma_delta_props_encode,
1482940b44dSPeter Avalos 	},
1492940b44dSPeter Avalos #endif
1502940b44dSPeter Avalos };
1512940b44dSPeter Avalos 
1522940b44dSPeter Avalos 
1532940b44dSPeter Avalos static const lzma_filter_encoder *
encoder_find(lzma_vli id)1542940b44dSPeter Avalos encoder_find(lzma_vli id)
1552940b44dSPeter Avalos {
1562940b44dSPeter Avalos 	for (size_t i = 0; i < ARRAY_SIZE(encoders); ++i)
1572940b44dSPeter Avalos 		if (encoders[i].id == id)
1582940b44dSPeter Avalos 			return encoders + i;
1592940b44dSPeter Avalos 
1602940b44dSPeter Avalos 	return NULL;
1612940b44dSPeter Avalos }
1622940b44dSPeter Avalos 
1632940b44dSPeter Avalos 
1642940b44dSPeter Avalos extern LZMA_API(lzma_bool)
lzma_filter_encoder_is_supported(lzma_vli id)1652940b44dSPeter Avalos lzma_filter_encoder_is_supported(lzma_vli id)
1662940b44dSPeter Avalos {
1672940b44dSPeter Avalos 	return encoder_find(id) != NULL;
1682940b44dSPeter Avalos }
1692940b44dSPeter Avalos 
1702940b44dSPeter Avalos 
1712940b44dSPeter Avalos extern LZMA_API(lzma_ret)
lzma_filters_update(lzma_stream * strm,const lzma_filter * filters)1722940b44dSPeter Avalos lzma_filters_update(lzma_stream *strm, const lzma_filter *filters)
1732940b44dSPeter Avalos {
1742940b44dSPeter Avalos 	if (strm->internal->next.update == NULL)
1752940b44dSPeter Avalos 		return LZMA_PROG_ERROR;
1762940b44dSPeter Avalos 
1772940b44dSPeter Avalos 	// Validate the filter chain.
1782940b44dSPeter Avalos 	if (lzma_raw_encoder_memusage(filters) == UINT64_MAX)
1792940b44dSPeter Avalos 		return LZMA_OPTIONS_ERROR;
1802940b44dSPeter Avalos 
1812940b44dSPeter Avalos 	// The actual filter chain in the encoder is reversed. Some things
1822940b44dSPeter Avalos 	// still want the normal order chain, so we provide both.
1832940b44dSPeter Avalos 	size_t count = 1;
1842940b44dSPeter Avalos 	while (filters[count].id != LZMA_VLI_UNKNOWN)
1852940b44dSPeter Avalos 		++count;
1862940b44dSPeter Avalos 
1872940b44dSPeter Avalos 	lzma_filter reversed_filters[LZMA_FILTERS_MAX + 1];
1882940b44dSPeter Avalos 	for (size_t i = 0; i < count; ++i)
1892940b44dSPeter Avalos 		reversed_filters[count - i - 1] = filters[i];
1902940b44dSPeter Avalos 
1912940b44dSPeter Avalos 	reversed_filters[count].id = LZMA_VLI_UNKNOWN;
1922940b44dSPeter Avalos 
1932940b44dSPeter Avalos 	return strm->internal->next.update(strm->internal->next.coder,
1942940b44dSPeter Avalos 			strm->allocator, filters, reversed_filters);
1952940b44dSPeter Avalos }
1962940b44dSPeter Avalos 
1972940b44dSPeter Avalos 
1982940b44dSPeter Avalos extern lzma_ret
lzma_raw_encoder_init(lzma_next_coder * next,const lzma_allocator * allocator,const lzma_filter * options)199*15ab8c86SJohn Marino lzma_raw_encoder_init(lzma_next_coder *next, const lzma_allocator *allocator,
2002940b44dSPeter Avalos 		const lzma_filter *options)
2012940b44dSPeter Avalos {
2022940b44dSPeter Avalos 	return lzma_raw_coder_init(next, allocator,
2032940b44dSPeter Avalos 			options, (lzma_filter_find)(&encoder_find), true);
2042940b44dSPeter Avalos }
2052940b44dSPeter Avalos 
2062940b44dSPeter Avalos 
2072940b44dSPeter Avalos extern LZMA_API(lzma_ret)
lzma_raw_encoder(lzma_stream * strm,const lzma_filter * options)2082940b44dSPeter Avalos lzma_raw_encoder(lzma_stream *strm, const lzma_filter *options)
2092940b44dSPeter Avalos {
2102940b44dSPeter Avalos 	lzma_next_strm_init(lzma_raw_coder_init, strm, options,
2112940b44dSPeter Avalos 			(lzma_filter_find)(&encoder_find), true);
2122940b44dSPeter Avalos 
2132940b44dSPeter Avalos 	strm->internal->supported_actions[LZMA_RUN] = true;
2142940b44dSPeter Avalos 	strm->internal->supported_actions[LZMA_SYNC_FLUSH] = true;
2152940b44dSPeter Avalos 	strm->internal->supported_actions[LZMA_FINISH] = true;
2162940b44dSPeter Avalos 
2172940b44dSPeter Avalos 	return LZMA_OK;
2182940b44dSPeter Avalos }
2192940b44dSPeter Avalos 
2202940b44dSPeter Avalos 
2212940b44dSPeter Avalos extern LZMA_API(uint64_t)
lzma_raw_encoder_memusage(const lzma_filter * filters)2222940b44dSPeter Avalos lzma_raw_encoder_memusage(const lzma_filter *filters)
2232940b44dSPeter Avalos {
2242940b44dSPeter Avalos 	return lzma_raw_coder_memusage(
2252940b44dSPeter Avalos 			(lzma_filter_find)(&encoder_find), filters);
2262940b44dSPeter Avalos }
2272940b44dSPeter Avalos 
2282940b44dSPeter Avalos 
229*15ab8c86SJohn Marino extern uint64_t
lzma_mt_block_size(const lzma_filter * filters)230*15ab8c86SJohn Marino lzma_mt_block_size(const lzma_filter *filters)
2312940b44dSPeter Avalos {
232*15ab8c86SJohn Marino 	uint64_t max = 0;
2332940b44dSPeter Avalos 
2342940b44dSPeter Avalos 	for (size_t i = 0; filters[i].id != LZMA_VLI_UNKNOWN; ++i) {
2352940b44dSPeter Avalos 		const lzma_filter_encoder *const fe
2362940b44dSPeter Avalos 				= encoder_find(filters[i].id);
237*15ab8c86SJohn Marino 		if (fe->block_size != NULL) {
238*15ab8c86SJohn Marino 			const uint64_t size
239*15ab8c86SJohn Marino 					= fe->block_size(filters[i].options);
240*15ab8c86SJohn Marino 			if (size == 0)
241*15ab8c86SJohn Marino 				return 0;
2422940b44dSPeter Avalos 
2432940b44dSPeter Avalos 			if (size > max)
2442940b44dSPeter Avalos 				max = size;
2452940b44dSPeter Avalos 		}
2462940b44dSPeter Avalos 	}
2472940b44dSPeter Avalos 
2482940b44dSPeter Avalos 	return max;
2492940b44dSPeter Avalos }
2502940b44dSPeter Avalos 
2512940b44dSPeter Avalos 
2522940b44dSPeter Avalos extern LZMA_API(lzma_ret)
lzma_properties_size(uint32_t * size,const lzma_filter * filter)2532940b44dSPeter Avalos lzma_properties_size(uint32_t *size, const lzma_filter *filter)
2542940b44dSPeter Avalos {
2552940b44dSPeter Avalos 	const lzma_filter_encoder *const fe = encoder_find(filter->id);
2562940b44dSPeter Avalos 	if (fe == NULL) {
2572940b44dSPeter Avalos 		// Unknown filter - if the Filter ID is a proper VLI,
2582940b44dSPeter Avalos 		// return LZMA_OPTIONS_ERROR instead of LZMA_PROG_ERROR,
2592940b44dSPeter Avalos 		// because it's possible that we just don't have support
2602940b44dSPeter Avalos 		// compiled in for the requested filter.
2612940b44dSPeter Avalos 		return filter->id <= LZMA_VLI_MAX
2622940b44dSPeter Avalos 				? LZMA_OPTIONS_ERROR : LZMA_PROG_ERROR;
2632940b44dSPeter Avalos 	}
2642940b44dSPeter Avalos 
2652940b44dSPeter Avalos 	if (fe->props_size_get == NULL) {
2662940b44dSPeter Avalos 		// No props_size_get() function, use props_size_fixed.
2672940b44dSPeter Avalos 		*size = fe->props_size_fixed;
2682940b44dSPeter Avalos 		return LZMA_OK;
2692940b44dSPeter Avalos 	}
2702940b44dSPeter Avalos 
2712940b44dSPeter Avalos 	return fe->props_size_get(size, filter->options);
2722940b44dSPeter Avalos }
2732940b44dSPeter Avalos 
2742940b44dSPeter Avalos 
2752940b44dSPeter Avalos extern LZMA_API(lzma_ret)
lzma_properties_encode(const lzma_filter * filter,uint8_t * props)2762940b44dSPeter Avalos lzma_properties_encode(const lzma_filter *filter, uint8_t *props)
2772940b44dSPeter Avalos {
2782940b44dSPeter Avalos 	const lzma_filter_encoder *const fe = encoder_find(filter->id);
2792940b44dSPeter Avalos 	if (fe == NULL)
2802940b44dSPeter Avalos 		return LZMA_PROG_ERROR;
2812940b44dSPeter Avalos 
2822940b44dSPeter Avalos 	if (fe->props_encode == NULL)
2832940b44dSPeter Avalos 		return LZMA_OK;
2842940b44dSPeter Avalos 
2852940b44dSPeter Avalos 	return fe->props_encode(filter->options, props);
2862940b44dSPeter Avalos }
287