blob: d2e3e4b45d5349e89ca5a74c27a39bfb55dcf63a [file] [log] [blame]
Stefan Reinauer14e22772010-04-27 06:56:47 +00001/*
Stefan Reinauerd6b4f1c2010-09-23 18:29:40 +00002 * coreboot interface to memory-saving variant of LZMA decoder
3 *
4 * Copyright (C) 2006 Carl-Daniel Hailfinger
5 * Released under the GNU GPL v2 or later
6 *
Lee Leahy73402172017-03-10 15:23:24 -08007 * Parts of this file are based on C/7zip/Compress/LZMA_C/LzmaTest.c from the
8 * LZMA SDK 4.42, which is written and distributed to public domain by Igor
9 * Pavlov.
Stefan Reinauerd6b4f1c2010-09-23 18:29:40 +000010 *
11 */
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000012
Patrick Georgi79941642009-04-24 16:44:34 +000013#include <console/console.h>
14#include <string.h>
Myles Watson58170782009-10-28 16:13:28 +000015#include <lib.h>
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000016
Edward O'Callaghancd31afd2014-11-11 12:27:06 +110017#include "lzmadecode.h"
18
Julius Wernera25b5d22016-02-08 11:46:22 -080019size_t ulzman(const void *src, size_t srcn, void *dst, size_t dstn)
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000020{
21 unsigned char properties[LZMA_PROPERTIES_SIZE];
Julius Wernera25b5d22016-02-08 11:46:22 -080022 const int data_offset = LZMA_PROPERTIES_SIZE + 8;
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000023 UInt32 outSize;
24 SizeT inProcessed;
25 SizeT outProcessed;
26 int res;
27 CLzmaDecoderState state;
28 SizeT mallocneeds;
Kyösti Mälkkifcbbb912020-04-20 10:21:39 +030029 static unsigned char scratchpad[15980];
Julius Wernera25b5d22016-02-08 11:46:22 -080030 const unsigned char *cp;
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000031
Alex Rebert41de2a02020-02-22 18:13:39 -050032 if (srcn < data_offset) {
33 printk(BIOS_WARNING, "lzma: Input too small.\n");
34 return 0;
35 }
36
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000037 memcpy(properties, src, LZMA_PROPERTIES_SIZE);
Hung-Te Lind51557a2013-01-31 12:14:46 +080038 /* The outSize in LZMA stream is a 64bit integer stored in little-endian
39 * (ref: lzma.cc@LZMACompress: put_64). To prevent accessing by
Martin Rothcbf2bd72013-07-09 21:51:14 -060040 * unaligned memory address and to load in correct endianness, read each
41 * byte and re-construct. */
Hung-Te Lind51557a2013-01-31 12:14:46 +080042 cp = src + LZMA_PROPERTIES_SIZE;
43 outSize = cp[3] << 24 | cp[2] << 16 | cp[1] << 8 | cp[0];
Patrick Rudolphf595ba22018-03-16 12:25:28 +010044 if (outSize > dstn)
45 outSize = dstn;
Julius Wernera25b5d22016-02-08 11:46:22 -080046 if (LzmaDecodeProperties(&state.Properties, properties,
47 LZMA_PROPERTIES_SIZE) != LZMA_RESULT_OK) {
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +000048 printk(BIOS_WARNING, "lzma: Incorrect stream properties.\n");
Ward Vandewegeec2bd532008-08-27 21:53:11 +000049 return 0;
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000050 }
51 mallocneeds = (LzmaGetNumProbs(&state.Properties) * sizeof(CProb));
52 if (mallocneeds > 15980) {
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +000053 printk(BIOS_WARNING, "lzma: Decoder scratchpad too small!\n");
Ward Vandewegeec2bd532008-08-27 21:53:11 +000054 return 0;
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000055 }
56 state.Probs = (CProb *)scratchpad;
Julius Wernera25b5d22016-02-08 11:46:22 -080057 res = LzmaDecode(&state, src + data_offset, srcn - data_offset,
58 &inProcessed, dst, outSize, &outProcessed);
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000059 if (res != 0) {
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +000060 printk(BIOS_WARNING, "lzma: Decoding error = %d\n", res);
Ward Vandewegeec2bd532008-08-27 21:53:11 +000061 return 0;
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000062 }
Julius Wernerd8086872015-06-25 23:22:36 -070063 return outProcessed;
Carl-Daniel Hailfingercba07dd2006-09-14 15:12:36 +000064}