123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- /*
- * Wrapper around the kernel's pre-boot decompression library.
- *
- * Copyright (C) IBM Corporation 2016.
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version
- * 2 of the License, or (at your option) any later version.
- */
- #include "elf.h"
- #include "page.h"
- #include "string.h"
- #include "stdio.h"
- #include "ops.h"
- #include "reg.h"
- #include "types.h"
- /*
- * The decompressor_*.c files play #ifdef games so they can be used in both
- * pre-boot and regular kernel code. We need these definitions to make the
- * includes work.
- */
- #define STATIC static
- #define INIT
- #define __always_inline inline
- /*
- * The build process will copy the required zlib source files and headers
- * out of lib/ and "fix" the includes so they do not pull in other kernel
- * headers.
- */
- #ifdef CONFIG_KERNEL_GZIP
- # include "decompress_inflate.c"
- #endif
- #ifdef CONFIG_KERNEL_XZ
- # include "xz_config.h"
- # include "../../../lib/decompress_unxz.c"
- #endif
- /* globals for tracking the state of the decompression */
- static unsigned long decompressed_bytes;
- static unsigned long limit;
- static unsigned long skip;
- static char *output_buffer;
- /*
- * flush() is called by __decompress() when the decompressor's scratch buffer is
- * full.
- */
- static long flush(void *v, unsigned long buffer_size)
- {
- unsigned long end = decompressed_bytes + buffer_size;
- unsigned long size = buffer_size;
- unsigned long offset = 0;
- char *in = v;
- char *out;
- /*
- * if we hit our decompression limit, we need to fake an error to abort
- * the in-progress decompression.
- */
- if (decompressed_bytes >= limit)
- return -1;
- /* skip this entire block */
- if (end <= skip) {
- decompressed_bytes += buffer_size;
- return buffer_size;
- }
- /* skip some data at the start, but keep the rest of the block */
- if (decompressed_bytes < skip && end > skip) {
- offset = skip - decompressed_bytes;
- in += offset;
- size -= offset;
- decompressed_bytes += offset;
- }
- out = &output_buffer[decompressed_bytes - skip];
- size = min(decompressed_bytes + size, limit) - decompressed_bytes;
- memcpy(out, in, size);
- decompressed_bytes += size;
- return buffer_size;
- }
- static void print_err(char *s)
- {
- /* suppress the "error" when we terminate the decompressor */
- if (decompressed_bytes >= limit)
- return;
- printf("Decompression error: '%s'\n\r", s);
- }
- /**
- * partial_decompress - decompresses part or all of a compressed buffer
- * @inbuf: input buffer
- * @input_size: length of the input buffer
- * @outbuf: input buffer
- * @output_size: length of the input buffer
- * @skip number of output bytes to ignore
- *
- * This function takes compressed data from inbuf, decompresses and write it to
- * outbuf. Once output_size bytes are written to the output buffer, or the
- * stream is exhausted the function will return the number of bytes that were
- * decompressed. Otherwise it will return whatever error code the decompressor
- * reported (NB: This is specific to each decompressor type).
- *
- * The skip functionality is mainly there so the program and discover
- * the size of the compressed image so that it can ask firmware (if present)
- * for an appropriately sized buffer.
- */
- long partial_decompress(void *inbuf, unsigned long input_size,
- void *outbuf, unsigned long output_size, unsigned long _skip)
- {
- int ret;
- /*
- * The skipped bytes needs to be included in the size of data we want
- * to decompress.
- */
- output_size += _skip;
- decompressed_bytes = 0;
- output_buffer = outbuf;
- limit = output_size;
- skip = _skip;
- ret = __decompress(inbuf, input_size, NULL, flush, outbuf,
- output_size, NULL, print_err);
- /*
- * If decompression was aborted due to an actual error rather than
- * a fake error that we used to abort, then we should report it.
- */
- if (decompressed_bytes < limit)
- return ret;
- return decompressed_bytes - skip;
- }
|