/* SPDX-License-Identifier: GPL-3.0-or-later */ /* * autodetect.c * * Copyright (C) 2019 David Oberhollenzer <goliath@infraroot.at> */ #include "../internal.h" static const struct { int id; const sqfs_u8 *value; size_t len; } magic[] = { { FSTREAM_COMPRESSOR_GZIP, (const sqfs_u8 *)"\x1F\x8B\x08", 3 }, { FSTREAM_COMPRESSOR_XZ, (const sqfs_u8 *)("\xFD" "7zXZ"), 6 }, { FSTREAM_COMPRESSOR_ZSTD, (const sqfs_u8 *)"\x28\xB5\x2F\xFD", 4 }, { FSTREAM_COMPRESSOR_BZIP2, (const sqfs_u8 *)"BZh", 3 }, }; int istream_detect_compressor(istream_t *strm, int (*probe)(const sqfs_u8 *data, size_t size)) { size_t i; int ret; ret = istream_precache(strm); if (ret != 0) return ret; if (probe != NULL) { ret = probe(strm->buffer + strm->buffer_offset, strm->buffer_used - strm->buffer_offset); if (ret < 0) return ret; /* XXX: this means the data is uncompressed. We do this check first since it might be perfectly OK for the uncompressed data to contain a magic number from the table. */ if (ret > 0) return 0; } for (i = 0; i < sizeof(magic) / sizeof(magic[0]); ++i) { if ((strm->buffer_used - strm->buffer_offset) < magic[i].len) continue; ret = memcmp(strm->buffer + strm->buffer_offset, magic[i].value, magic[i].len); if (ret == 0) return magic[i].id; } return 0; }