|
- /*
- * Copyright (c) 2016 Umair Khan <omerjerk@gmail.com>
- *
- * This file is part of FFmpeg.
- *
- * FFmpeg 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.
- *
- * FFmpeg 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.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with FFmpeg; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- */
-
- #include "mlz.h"
-
- av_cold void ff_mlz_init_dict(void* context, MLZ *mlz) {
- mlz->dict = av_mallocz_array(TABLE_SIZE, sizeof(*mlz->dict));
-
- mlz->flush_code = FLUSH_CODE;
- mlz->current_dic_index_max = DIC_INDEX_INIT;
- mlz->dic_code_bit = CODE_BIT_INIT;
- mlz->bump_code = (DIC_INDEX_INIT - 1);
- mlz->next_code = FIRST_CODE;
- mlz->freeze_flag = 0;
- mlz->context = context;
- }
-
- av_cold void ff_mlz_flush_dict(MLZ *mlz) {
- MLZDict *dict = mlz->dict;
- int i;
- for ( i = 0; i < TABLE_SIZE; i++ ) {
- dict[i].string_code = CODE_UNSET;
- dict[i].parent_code = CODE_UNSET;
- dict[i].match_len = 0;
- }
- mlz->current_dic_index_max = DIC_INDEX_INIT;
- mlz->dic_code_bit = CODE_BIT_INIT; // DicCodeBitInit;
- mlz->bump_code = mlz->current_dic_index_max - 1;
- mlz->next_code = FIRST_CODE;
- mlz->freeze_flag = 0;
- }
-
- static void set_new_entry_dict(MLZDict* dict, int string_code, int parent_code, int char_code) {
- dict[string_code].parent_code = parent_code;
- dict[string_code].string_code = string_code;
- dict[string_code].char_code = char_code;
- if (parent_code < FIRST_CODE) {
- dict[string_code].match_len = 2;
- } else {
- dict[string_code].match_len = (dict[parent_code].match_len) + 1;
- }
- }
-
- static int decode_string(MLZ* mlz, unsigned char *buff, int string_code, int *first_char_code, unsigned long bufsize) {
- MLZDict* dict = mlz->dict;
- unsigned long count, offset;
- int current_code, parent_code, tmp_code;
-
- count = 0;
- current_code = string_code;
- *first_char_code = CODE_UNSET;
-
- while (count < bufsize) {
- switch (current_code) {
- case CODE_UNSET:
- return count;
- break;
- default:
- if (current_code < FIRST_CODE) {
- *first_char_code = current_code;
- buff[0] = current_code;
- count++;
- return count;
- } else {
- offset = dict[current_code].match_len - 1;
- tmp_code = dict[current_code].char_code;
- if (offset >= bufsize) {
- av_log(mlz->context, AV_LOG_ERROR, "MLZ offset error.\n");
- return count;
- }
- buff[offset] = tmp_code;
- count++;
- }
- current_code = dict[current_code].parent_code;
- if ((current_code < 0) || (current_code > (DIC_INDEX_MAX - 1))) {
- av_log(mlz->context, AV_LOG_ERROR, "MLZ dic index error.\n");
- return count;
- }
- if (current_code > FIRST_CODE) {
- parent_code = dict[current_code].parent_code;
- offset = (dict[current_code].match_len) - 1;
- if (parent_code < 0 || parent_code > DIC_INDEX_MAX-1) {
- av_log(mlz->context, AV_LOG_ERROR, "MLZ dic index error.\n");
- return count;
- }
- if (( offset > (DIC_INDEX_MAX - 1))) {
- av_log(mlz->context, AV_LOG_ERROR, "MLZ dic offset error.\n");
- return count;
- }
- }
- break;
- }
- }
- return count;
- }
-
- static int input_code(GetBitContext* gb, int len) {
- int tmp_code = 0;
- int i;
- for (i = 0; i < len; ++i) {
- tmp_code |= get_bits1(gb) << i;
- }
- return tmp_code;
- }
-
- int ff_mlz_decompression(MLZ* mlz, GetBitContext* gb, int size, unsigned char *buff) {
- MLZDict *dict = mlz->dict;
- unsigned long output_chars;
- int string_code, last_string_code, char_code;
-
- string_code = 0;
- char_code = -1;
- last_string_code = -1;
- output_chars = 0;
-
- while (output_chars < size) {
- string_code = input_code(gb, mlz->dic_code_bit);
- switch (string_code) {
- case FLUSH_CODE:
- case MAX_CODE:
- ff_mlz_flush_dict(mlz);
- char_code = -1;
- last_string_code = -1;
- break;
- case FREEZE_CODE:
- mlz->freeze_flag = 1;
- break;
- default:
- if (string_code > mlz->current_dic_index_max) {
- av_log(mlz->context, AV_LOG_ERROR, "String code %d exceeds maximum value of %d.\n", string_code, mlz->current_dic_index_max);
- return output_chars;
- }
- if (string_code == (int) mlz->bump_code) {
- ++mlz->dic_code_bit;
- mlz->current_dic_index_max *= 2;
- mlz->bump_code = mlz->current_dic_index_max - 1;
- } else {
- if (string_code >= mlz->next_code) {
- int ret = decode_string(mlz, &buff[output_chars], last_string_code, &char_code, size - output_chars);
- if (ret < 0 || ret > size - output_chars) {
- av_log(mlz->context, AV_LOG_ERROR, "output chars overflow\n");
- return output_chars;
- }
- output_chars += ret;
- ret = decode_string(mlz, &buff[output_chars], char_code, &char_code, size - output_chars);
- if (ret < 0 || ret > size - output_chars) {
- av_log(mlz->context, AV_LOG_ERROR, "output chars overflow\n");
- return output_chars;
- }
- output_chars += ret;
- set_new_entry_dict(dict, mlz->next_code, last_string_code, char_code);
- if (mlz->next_code >= TABLE_SIZE - 1) {
- av_log(mlz->context, AV_LOG_ERROR, "Too many MLZ codes\n");
- return output_chars;
- }
- mlz->next_code++;
- } else {
- int ret = decode_string(mlz, &buff[output_chars], string_code, &char_code, size - output_chars);
- if (ret < 0 || ret > size - output_chars) {
- av_log(mlz->context, AV_LOG_ERROR, "output chars overflow\n");
- return output_chars;
- }
- output_chars += ret;
- if (output_chars <= size && !mlz->freeze_flag) {
- if (last_string_code != -1) {
- set_new_entry_dict(dict, mlz->next_code, last_string_code, char_code);
- if (mlz->next_code >= TABLE_SIZE - 1) {
- av_log(mlz->context, AV_LOG_ERROR, "Too many MLZ codes\n");
- return output_chars;
- }
- mlz->next_code++;
- }
- } else {
- break;
- }
- }
- last_string_code = string_code;
- }
- break;
- }
- }
- return output_chars;
- }
|