1
0
Fork 0
mirror of https://github.com/Orama-Interactive/Pixelorama.git synced 2025-01-18 17:19:50 +00:00

GIF export speedup (#696)

* Simplify code table to a single Dictionary

* Speedup local color table creation

* Remove blep

* Fix format

* Rewrite add image data block function

* Speedup get_bits_number

* Format and remove commented method

Co-authored-by: Emmanouil Papadeas <35376950+OverloadedOrama@users.noreply.github.com>
This commit is contained in:
Martin Novák 2022-05-23 16:56:05 +02:00 committed by GitHub
parent 5e5925b8c6
commit 86beee6aab
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 54 additions and 159 deletions

View file

@ -258,34 +258,38 @@ func color_table_bit_size(color_table: Array) -> int:
func add_local_color_table(color_table: Array) -> void: func add_local_color_table(color_table: Array) -> void:
for color in color_table: for color in color_table:
data.append(color[0]) data.append_array([color[0], color[1], color[2]])
data.append(color[1])
data.append(color[2])
var size := color_table_bit_size(color_table) var size := color_table_bit_size(color_table)
var proper_size := int(pow(2, size + 1)) var proper_size := int(pow(2, size + 1))
if color_table.size() != proper_size: if color_table.size() != proper_size:
for i in range(proper_size - color_table.size()): for i in range(proper_size - color_table.size()):
data += PoolByteArray([0, 0, 0]) data.append_array([0, 0, 0])
func add_image_data_block(lzw_min_code_size: int, _data: PoolByteArray) -> void: func add_image_data_block(lzw_min_code_size: int, frame_data: PoolByteArray) -> void:
var max_block_size = 254
data.append(lzw_min_code_size) data.append(lzw_min_code_size)
# the amount of blocks which will be stored
# ceiled because the last block doesn't have to be exactly max_block_size
var block_count = ceil(frame_data.size() / float(max_block_size))
var block_size_index: int = 0 for i in range(block_count):
var i: int = 0 var start_block_index = i * max_block_size
var data_index: int = 0 var end_block_index = (i * max_block_size) + max_block_size - 1
while data_index < _data.size(): # final block can be smaller than max block size
if i == 0: end_block_index = (
data.append(0) end_block_index
block_size_index = data.size() - 1 if end_block_index < frame_data.size() - 1
data.append(_data[data_index]) else frame_data.size() - 1
data[block_size_index] += 1 )
data_index += 1 var block_size = end_block_index - start_block_index + 1
i += 1 var block = frame_data.subarray(start_block_index, end_block_index)
if i == 254:
i = 0
if not _data.empty(): # store block size and it's data
data.append(block_size)
data.append_array(block)
if not frame_data.empty():
data.append(0) data.append(0)

View file

@ -3,82 +3,36 @@ extends Reference
var lsbbitpacker = preload("./lsbbitpacker.gd") var lsbbitpacker = preload("./lsbbitpacker.gd")
var lsbbitunpacker = preload("./lsbbitunpacker.gd") var lsbbitunpacker = preload("./lsbbitunpacker.gd")
var code_table := {}
class CodeEntry: var entries_counter := 0
var sequence: PoolByteArray
var raw_array: Array
func _init(_sequence):
raw_array = _sequence
sequence = _sequence
func add(other):
return CodeEntry.new(self.raw_array + other.raw_array)
func to_string():
var result: String = ""
for element in self.sequence:
result += str(element) + ", "
return result.substr(0, result.length() - 2)
class CodeTable: func get_bit_length(value: int):
var entries: Dictionary = {} # bitwise or on value does ensure that the function works with value 0
var counter: int = 0 # long number at the end is log(2.0)
var lookup: Dictionary = {} return ceil(log(value | 0x1 + 1) / 0.6931471805599453)
func add(entry) -> int:
self.entries[self.counter] = entry
self.lookup[entry.raw_array] = self.counter
counter += 1
return counter
func find(entry) -> int:
return self.lookup.get(entry.raw_array, -1)
func has(entry) -> bool:
return self.find(entry) != -1
func get(index) -> CodeEntry:
return self.entries.get(index, null)
func to_string() -> String:
var result: String = "CodeTable:\n"
for id in self.entries:
result += str(id) + ": " + self.entries[id].to_string() + "\n"
result += "Counter: " + str(self.counter) + "\n"
return result
func log2(value: float) -> float: func initialize_color_code_table(colors: PoolByteArray) -> void:
return log(value) / log(2.0) code_table.clear()
entries_counter = 0
func get_bits_number_for(value: int) -> int:
if value == 0:
return 1
return int(ceil(log2(value + 1)))
func initialize_color_code_table(colors: PoolByteArray) -> CodeTable:
var result_code_table: CodeTable = CodeTable.new()
for color_id in colors: for color_id in colors:
# warning-ignore:return_value_discarded # warning-ignore:return_value_discarded
result_code_table.add(CodeEntry.new([color_id])) code_table[PoolByteArray([color_id])] = entries_counter
entries_counter += 1
# move counter to the first available compression code index # move counter to the first available compression code index
var last_color_index: int = colors.size() - 1 var last_color_index: int = colors.size() - 1
var clear_code_index: int = pow(2, get_bits_number_for(last_color_index)) var clear_code_index: int = pow(2, get_bit_length(last_color_index))
result_code_table.counter = clear_code_index + 2 entries_counter = clear_code_index + 2
return result_code_table
# compression and decompression done with source: # compression and decompression done with source:
# http://www.matthewflickinger.com/lab/whatsinagif/lzw_image_data.asp # http://www.matthewflickinger.com/lab/whatsinagif/lzw_image_data.asp
func compress_lzw(image: PoolByteArray, colors: PoolByteArray) -> Array: func compress_lzw(index_stream: PoolByteArray, colors: PoolByteArray) -> Array:
# Initialize code table # Initialize code table
var code_table: CodeTable = initialize_color_code_table(colors) initialize_color_code_table(colors)
# Clear Code index is 2**<code size> # Clear Code index is 2**<code size>
# <code size> is the amount of bits needed to write down all colors # <code size> is the amount of bits needed to write down all colors
# from color table. We use last color index because we can write # from color table. We use last color index because we can write
@ -86,126 +40,63 @@ func compress_lzw(image: PoolByteArray, colors: PoolByteArray) -> Array:
# Number 15 is in binary 0b1111, so we'll need 4 bits to write all # Number 15 is in binary 0b1111, so we'll need 4 bits to write all
# colors down. # colors down.
var last_color_index: int = colors.size() - 1 var last_color_index: int = colors.size() - 1
var clear_code_index: int = pow(2, get_bits_number_for(last_color_index)) var clear_code_index: int = pow(2, get_bit_length(last_color_index))
var index_stream: PoolByteArray = image var current_code_size: int = get_bit_length(clear_code_index)
var current_code_size: int = get_bits_number_for(clear_code_index)
var binary_code_stream = lsbbitpacker.LSBLZWBitPacker.new() var binary_code_stream = lsbbitpacker.LSBLZWBitPacker.new()
# initialize with Clear Code # initialize with Clear Code
binary_code_stream.write_bits(clear_code_index, current_code_size) binary_code_stream.write_bits(clear_code_index, current_code_size)
# Read first index from index stream. # Read first index from index stream.
var index_buffer: CodeEntry = CodeEntry.new([index_stream[0]]) var index_buffer := PoolByteArray([index_stream[0]])
var data_index: int = 1 var data_index: int = 1
# <LOOP POINT> # <LOOP POINT>
while data_index < index_stream.size(): while data_index < index_stream.size():
# Get the next index from the index stream. # Get the next index from the index stream.
var k: CodeEntry = CodeEntry.new([index_stream[data_index]]) var k := index_stream[data_index]
data_index += 1 data_index += 1
# Is index buffer + k in our code table? # Is index buffer + k in our code table?
var new_index_buffer: CodeEntry = index_buffer.add(k) var new_index_buffer := PoolByteArray(index_buffer)
new_index_buffer.push_back(k)
if code_table.has(new_index_buffer): # if YES if code_table.has(new_index_buffer): # if YES
# Add k to the end of the index buffer # Add k to the end of the index buffer
index_buffer = new_index_buffer index_buffer = new_index_buffer
else: # if NO else: # if NO
# Add a row for index buffer + k into our code table # Add a row for index buffer + k into our code table
binary_code_stream.write_bits(code_table.find(index_buffer), current_code_size) binary_code_stream.write_bits(code_table.get(index_buffer, -1), current_code_size)
# We don't want to add new code to code table if we've exceeded 4095 # We don't want to add new code to code table if we've exceeded 4095
# index. # index.
var last_entry_index: int = code_table.counter - 1 var last_entry_index: int = entries_counter - 1
if last_entry_index != 4095: if last_entry_index != 4095:
# Output the code for just the index buffer to our code stream # Output the code for just the index buffer to our code stream
# warning-ignore:return_value_discarded # warning-ignore:return_value_discarded
code_table.add(new_index_buffer) code_table[new_index_buffer] = entries_counter
entries_counter += 1
else: else:
# if we exceeded 4095 index (code table is full), we should # if we exceeded 4095 index (code table is full), we should
# output Clear Code and reset everything. # output Clear Code and reset everything.
binary_code_stream.write_bits(clear_code_index, current_code_size) binary_code_stream.write_bits(clear_code_index, current_code_size)
code_table = initialize_color_code_table(colors) initialize_color_code_table(colors)
# get_bits_number_for(clear_code_index) is the same as # get_bits_number_for(clear_code_index) is the same as
# LZW code size + 1 # LZW code size + 1
current_code_size = get_bits_number_for(clear_code_index) current_code_size = get_bit_length(clear_code_index)
# Detect when you have to save new codes in bigger bits boxes # Detect when you have to save new codes in bigger bits boxes
# change current code size when it happens because we want to save # change current code size when it happens because we want to save
# flexible code sized codes # flexible code sized codes
var new_code_size_candidate: int = get_bits_number_for(code_table.counter - 1) var new_code_size_candidate: int = get_bit_length(entries_counter - 1)
if new_code_size_candidate > current_code_size: if new_code_size_candidate > current_code_size:
current_code_size = new_code_size_candidate current_code_size = new_code_size_candidate
# Index buffer is set to k # Index buffer is set to k
index_buffer = k index_buffer = PoolByteArray([k])
# Output code for contents of index buffer # Output code for contents of index buffer
binary_code_stream.write_bits(code_table.find(index_buffer), current_code_size) binary_code_stream.write_bits(code_table.get(index_buffer, -1), current_code_size)
# output end with End Of Information Code # output end with End Of Information Code
binary_code_stream.write_bits(clear_code_index + 1, current_code_size) binary_code_stream.write_bits(clear_code_index + 1, current_code_size)
var min_code_size: int = get_bits_number_for(clear_code_index) - 1 var min_code_size: int = get_bit_length(clear_code_index) - 1
return [binary_code_stream.pack(), min_code_size] return [binary_code_stream.pack(), min_code_size]
func decompress_lzw(
code_stream_data: PoolByteArray, min_code_size: int, colors: PoolByteArray
) -> PoolByteArray:
var code_table: CodeTable = initialize_color_code_table(colors)
var index_stream: PoolByteArray = PoolByteArray([])
var binary_code_stream = lsbbitunpacker.LSBLZWBitUnpacker.new(code_stream_data)
var current_code_size: int = min_code_size + 1
var clear_code_index: int = pow(2, min_code_size)
# CODE is an index of code table, {CODE} is sequence inside
# code table with index CODE. The same goes for PREVCODE.
# Remove first Clear Code from stream. We don't need it.
binary_code_stream.remove_bits(current_code_size)
# let CODE be the first code in the code stream
var code: int = binary_code_stream.read_bits(current_code_size)
# output {CODE} to index stream
index_stream.append_array(code_table.get(code).sequence)
# set PREVCODE = CODE
var prevcode: int = code
# <LOOP POINT>
while true:
# let CODE be the next code in the code stream
code = binary_code_stream.read_bits(current_code_size)
# Detect Clear Code. When detected reset everything and get next code.
if code == clear_code_index:
code_table = initialize_color_code_table(colors)
current_code_size = min_code_size + 1
code = binary_code_stream.read_bits(current_code_size)
elif code == clear_code_index + 1: # Stop when detected EOI Code.
break
# is CODE in the code table?
var code_entry: CodeEntry = code_table.get(code)
if code_entry != null: # if YES
# output {CODE} to index stream
index_stream.append_array(code_entry.sequence)
# let k be the first index in {CODE}
var k: CodeEntry = CodeEntry.new([code_entry.sequence[0]])
# warning-ignore:return_value_discarded
# add {PREVCODE} + k to the code table
code_table.add(code_table.get(prevcode).add(k))
# set PREVCODE = CODE
prevcode = code
else: # if NO
# let k be the first index of {PREVCODE}
var prevcode_entry: CodeEntry = code_table.get(prevcode)
var k: CodeEntry = CodeEntry.new([prevcode_entry.sequence[0]])
# output {PREVCODE} + k to index stream
index_stream.append_array(prevcode_entry.add(k).sequence)
# add {PREVCODE} + k to code table
# warning-ignore:return_value_discarded
code_table.add(prevcode_entry.add(k))
# set PREVCODE = CODE
prevcode = code
# Detect when we should increase current code size and increase it.
var new_code_size_candidate: int = get_bits_number_for(code_table.counter)
if new_code_size_candidate > current_code_size:
current_code_size = new_code_size_candidate
return index_stream