removed multi-threading in favor of muti-processing

This commit is contained in:
k4yt3x 2020-02-26 05:27:57 -05:00
parent a2150a8dbc
commit 6a100b1526
5 changed files with 254 additions and 274 deletions

View File

@ -4,13 +4,16 @@
Name: Anime4K Driver Name: Anime4K Driver
Author: K4YT3X Author: K4YT3X
Date Created: August 15, 2019 Date Created: August 15, 2019
Last Modified: November 15, 2019 Last Modified: February 26, 2020
Description: This class is a high-level wrapper Description: This class is a high-level wrapper
for Anime4k. for Anime4k.
""" """
# built-in imports # built-in imports
import os
import queue
import shlex
import subprocess import subprocess
import threading import threading
@ -31,7 +34,7 @@ class Anime4k:
self.driver_settings = driver_settings self.driver_settings = driver_settings
self.print_lock = threading.Lock() self.print_lock = threading.Lock()
def upscale(self, input_directory, output_directory, scale_ratio, upscaler_exceptions, push_strength=None, push_grad_strength=None): def upscale(self, input_directory, output_directory, scale_ratio, processes, push_strength=None, push_grad_strength=None):
""" Anime4K wrapper """ Anime4K wrapper
Arguments: Arguments:
@ -46,9 +49,9 @@ class Anime4k:
Returns: Returns:
subprocess.Popen.returncode -- command line return value of execution subprocess.Popen.returncode -- command line return value of execution
""" """
try:
# return value is the sum of all execution return codes # a list of all commands to be executed
return_value = 0 commands = queue.Queue()
# get a list lof all image files in input_directory # get a list lof all image files in input_directory
extracted_frame_files = [f for f in input_directory.iterdir() if str(f).lower().endswith('.png') or str(f).lower().endswith('.jpg')] extracted_frame_files = [f for f in input_directory.iterdir() if str(f).lower().endswith('.png') or str(f).lower().endswith('.jpg')]
@ -76,17 +79,33 @@ class Anime4k:
if locals()[arg] is not None: if locals()[arg] is not None:
execute.extend([locals([arg])]) execute.extend([locals([arg])])
self.print_lock.acquire() commands.put(execute)
Avalon.debug_info(f'Executing: {execute}', )
self.print_lock.release() # initialize two lists to hold running and finished processes
return_value += subprocess.run(execute, check=True).returncode anime4k_running_processes = []
anime4k_finished_processes = []
# run all commands in queue
while not commands.empty():
# if any commands have completed
# remove the subprocess.Popen project and move it into finished processes
for process in anime4k_running_processes:
if process.poll() is not None:
Avalon.debug_info(f'Subprocess {process.pid} exited with code {process.poll()}')
anime4k_finished_processes.append(process)
anime4k_running_processes.remove(process)
# when number running processes is less than what's specified
# create new processes and add to running process pool
while len(anime4k_running_processes) < processes:
next_in_queue = commands.get()
new_process = subprocess.Popen(next_in_queue)
anime4k_running_processes.append(new_process)
# print thread exiting message
self.print_lock.acquire() self.print_lock.acquire()
Avalon.debug_info(f'[upscaler] Thread {threading.current_thread().name} exiting') Avalon.debug_info(f'[upscaler] Subprocess {new_process.pid} executing: {shlex.join(next_in_queue)}')
self.print_lock.release() self.print_lock.release()
# return command execution return code # return command execution return code
return return_value return anime4k_finished_processes
except Exception as e:
upscaler_exceptions.append(e)

View File

@ -30,6 +30,7 @@ import copy
import pathlib import pathlib
import re import re
import shutil import shutil
import sys
import tempfile import tempfile
import threading import threading
import time import time
@ -65,7 +66,7 @@ class Upscaler:
self.scale_height = None self.scale_height = None
self.scale_ratio = None self.scale_ratio = None
self.model_dir = None self.model_dir = None
self.threads = 1 self.processes = 1
self.video2x_cache_directory = pathlib.Path(tempfile.gettempdir()) / 'video2x' self.video2x_cache_directory = pathlib.Path(tempfile.gettempdir()) / 'video2x'
self.image_format = 'png' self.image_format = 'png'
self.preserve_frames = False self.preserve_frames = False
@ -150,150 +151,136 @@ class Upscaler:
w2 {Waifu2x Object} -- initialized waifu2x object w2 {Waifu2x Object} -- initialized waifu2x object
""" """
# progress bar thread exit signal # progress bar process exit signal
self.progress_bar_exit_signal = False self.progress_bar_exit_signal = False
# create a container for exceptions in threads
# if this thread is not empty, then an exception has occured
self.upscaler_exceptions = []
# initialize waifu2x driver # initialize waifu2x driver
drivers = AVAILABLE_DRIVERS if self.waifu2x_driver not in AVAILABLE_DRIVERS:
if self.waifu2x_driver not in drivers: raise UnrecognizedDriverError(f'Unrecognized driver: {self.waifu2x_driver}')
raise UnrecognizedDriverError(f'Unrecognized waifu2x driver: {self.waifu2x_driver}')
# it's easier to do multi-threading with waifu2x_converter # create a container for all upscaler processes
# the number of threads can be passed directly to waifu2x_converter upscaler_processes = []
if self.waifu2x_driver == 'waifu2x_converter':
w2 = Waifu2xConverter(self.driver_settings, self.model_dir)
progress_bar = threading.Thread(target=self._progress_bar, args=([self.extracted_frames],))
progress_bar.start()
w2.upscale(self.extracted_frames, self.upscaled_frames, self.scale_ratio, self.threads, self.image_format, self.upscaler_exceptions)
for image in [f for f in self.upscaled_frames.iterdir() if f.is_file()]:
renamed = re.sub(f'_\[.*-.*\]\[x(\d+(\.\d+)?)\]\.{self.image_format}', f'.{self.image_format}', str(image))
(self.upscaled_frames / image).rename(self.upscaled_frames / renamed)
self.progress_bar_exit_signal = True
progress_bar.join()
return
# drivers that are to be multi-threaded by video2x
else:
# create a container for all upscaler threads
upscaler_threads = []
# list all images in the extracted frames # list all images in the extracted frames
frames = [(self.extracted_frames / f) for f in self.extracted_frames.iterdir() if f.is_file] frames = [(self.extracted_frames / f) for f in self.extracted_frames.iterdir() if f.is_file]
# if we have less images than threads, # if we have less images than processes,
# create only the threads necessary # create only the processes necessary
if len(frames) < self.threads: if len(frames) < self.processes:
self.threads = len(frames) self.processes = len(frames)
# create a directory for each thread and append directory # create a directory for each process and append directory
# name into a list # name into a list
process_directories = []
thread_pool = [] for process_id in range(self.processes):
thread_directories = [] process_directory = self.extracted_frames / str(process_id)
for thread_id in range(self.threads): process_directories.append(process_directory)
thread_directory = self.extracted_frames / str(thread_id)
thread_directories.append(thread_directory)
# delete old directories and create new directories # delete old directories and create new directories
if thread_directory.is_dir(): if process_directory.is_dir():
shutil.rmtree(thread_directory) shutil.rmtree(process_directory)
thread_directory.mkdir(parents=True, exist_ok=True) process_directory.mkdir(parents=True, exist_ok=True)
# append directory path into list # waifu2x-converter-cpp will perform multi-threading within its own process
thread_pool.append((thread_directory, thread_id)) if self.waifu2x_driver in ['waifu2x_converter', 'anime4k'] :
process_directories = [self.extracted_frames]
else:
# evenly distribute images into each directory # evenly distribute images into each directory
# until there is none left in the directory # until there is none left in the directory
for image in frames: for image in frames:
# move image # move image
image.rename(thread_pool[0][0] / image.name) image.rename(process_directories[0] / image.name)
# rotate list # rotate list
thread_pool = thread_pool[-1:] + thread_pool[:-1] process_directories = process_directories[-1:] + process_directories[:-1]
# create threads and start them # create threads and start them
for thread_info in thread_pool: for process_directory in process_directories:
# create a separate w2 instance for each thread # if the driver being used is waifu2x-caffe
if self.waifu2x_driver == 'waifu2x_caffe': if self.waifu2x_driver == 'waifu2x_caffe':
w2 = Waifu2xCaffe(copy.deepcopy(self.driver_settings), self.method, self.model_dir, self.bit_depth) driver = Waifu2xCaffe(copy.deepcopy(self.driver_settings), self.method, self.model_dir, self.bit_depth)
if self.scale_ratio: if self.scale_ratio:
thread = threading.Thread(target=w2.upscale, upscaler_processes.append(driver.upscale(process_directory,
args=(thread_info[0],
self.upscaled_frames, self.upscaled_frames,
self.scale_ratio, self.scale_ratio,
False, False,
False, False,
self.image_format, self.image_format))
self.upscaler_exceptions))
else: else:
thread = threading.Thread(target=w2.upscale, upscaler_processes.append(driver.upscale(process_directory,
args=(thread_info[0],
self.upscaled_frames, self.upscaled_frames,
False, False,
self.scale_width, self.scale_width,
self.scale_height, self.scale_height,
self.image_format, self.image_format))
self.upscaler_exceptions))
# if the driver being used is waifu2x_ncnn_vulkan # if the driver being used is waifu2x-converter-cpp
elif self.waifu2x_driver == 'waifu2x_ncnn_vulkan': elif self.waifu2x_driver == 'waifu2x_converter':
w2 = Waifu2xNcnnVulkan(copy.deepcopy(self.driver_settings)) driver = Waifu2xConverter(self.driver_settings, self.model_dir)
thread = threading.Thread(target=w2.upscale, upscaler_processes.append(driver.upscale(process_directory,
args=(thread_info[0],
self.upscaled_frames, self.upscaled_frames,
self.scale_ratio, self.scale_ratio,
self.upscaler_exceptions)) self.processes,
self.image_format))
# if the driver being used is waifu2x-ncnn-vulkan
elif self.waifu2x_driver == 'waifu2x_ncnn_vulkan':
driver = Waifu2xNcnnVulkan(copy.deepcopy(self.driver_settings))
upscaler_processes.append(driver.upscale(process_directory,
self.upscaled_frames,
self.scale_ratio))
# if the driver being used is anime4k # if the driver being used is anime4k
elif self.waifu2x_driver == 'anime4k': elif self.waifu2x_driver == 'anime4k':
w2 = Anime4k(copy.deepcopy(self.driver_settings)) driver = Anime4k(copy.deepcopy(self.driver_settings))
thread = threading.Thread(target=w2.upscale, upscaler_processes += driver.upscale(process_directory,
args=(thread_info[0],
self.upscaled_frames, self.upscaled_frames,
self.scale_ratio, self.scale_ratio,
self.upscaler_exceptions)) self.processes)
# create thread
thread.name = thread_info[1]
# add threads into the pool
upscaler_threads.append(thread)
# start progress bar in a different thread # start progress bar in a different thread
progress_bar = threading.Thread(target=self._progress_bar, args=(thread_directories,)) progress_bar = threading.Thread(target=self._progress_bar, args=(process_directories,))
progress_bar.start() progress_bar.start()
# create the clearer and start it # create the clearer and start it
Avalon.debug_info('Starting upscaled image cleaner') Avalon.debug_info('Starting upscaled image cleaner')
image_cleaner = ImageCleaner(self.extracted_frames, self.upscaled_frames, len(upscaler_threads)) image_cleaner = ImageCleaner(self.extracted_frames, self.upscaled_frames, len(upscaler_processes))
image_cleaner.start() image_cleaner.start()
# start all threads # wait for all process to exit
for thread in upscaler_threads: try:
thread.start() Avalon.debug_info('Main process waiting for subprocesses to exit')
for process in upscaler_processes:
Avalon.debug_info(f'Subprocess {process.pid} exited with code {process.wait()}')
except (KeyboardInterrupt, SystemExit):
Avalon.warning('Exit signal received')
Avalon.warning('Killing processes')
for process in upscaler_processes:
process.terminate()
# wait for threads to finish # cleanup and exit with exit code 1
for thread in upscaler_threads: Avalon.debug_info('Killing upscaled image cleaner')
thread.join() image_cleaner.stop()
self.progress_bar_exit_signal = True
sys.exit(1)
# if the driver is waifu2x-converter-cpp
# images need to be renamed to be recognizable for FFmpeg
if self.waifu2x_driver == 'waifu2x_converter':
for image in [f for f in self.upscaled_frames.iterdir() if f.is_file()]:
renamed = re.sub(f'_\\[.*\\]\\[x(\\d+(\\.\\d+)?)\\]\\.{self.image_format}', f'.{self.image_format}', str(image.name))
(self.upscaled_frames / image).rename(self.upscaled_frames / renamed)
# upscaling done, kill the clearer # upscaling done, kill the clearer
Avalon.debug_info('Killing upscaled image cleaner') Avalon.debug_info('Killing upscaled image cleaner')
image_cleaner.stop() image_cleaner.stop()
# pass exit signal to progress bar thread
self.progress_bar_exit_signal = True self.progress_bar_exit_signal = True
if len(self.upscaler_exceptions) != 0:
raise(self.upscaler_exceptions[0])
def run(self): def run(self):
"""Main controller for Video2X """ Main controller for Video2X
This function controls the flow of video conversion This function controls the flow of video conversion
and handles all necessary functions. and handles all necessary functions.
@ -337,6 +324,7 @@ class Upscaler:
# get a dict of all pixel formats and corresponding bit depth # get a dict of all pixel formats and corresponding bit depth
pixel_formats = fm.get_pixel_formats() pixel_formats = fm.get_pixel_formats()
# try getting pixel format's corresponding bti depth
try: try:
self.bit_depth = pixel_formats[fm.pixel_format] self.bit_depth = pixel_formats[fm.pixel_format]
except KeyError: except KeyError:

View File

@ -4,13 +4,15 @@
Name: Waifu2x Caffe Driver Name: Waifu2x Caffe Driver
Author: K4YT3X Author: K4YT3X
Date Created: Feb 24, 2018 Date Created: Feb 24, 2018
Last Modified: October 6, 2019 Last Modified: February 22, 2020
Description: This class is a high-level wrapper Description: This class is a high-level wrapper
for waifu2x-caffe. for waifu2x-caffe.
""" """
# built-in imports # built-in imports
import os
import shlex
import subprocess import subprocess
import threading import threading
@ -38,7 +40,7 @@ class Waifu2xCaffe:
self.model_dir = model_dir self.model_dir = model_dir
self.print_lock = threading.Lock() self.print_lock = threading.Lock()
def upscale(self, input_directory, output_directory, scale_ratio, scale_width, scale_height, image_format, upscaler_exceptions): def upscale(self, input_directory, output_directory, scale_ratio, scale_width, scale_height, image_format):
"""This is the core function for WAIFU2X class """This is the core function for WAIFU2X class
Arguments: Arguments:
@ -48,7 +50,6 @@ class Waifu2xCaffe:
height {int} -- output video height height {int} -- output video height
""" """
try:
# overwrite config file settings # overwrite config file settings
self.driver_settings['input_path'] = input_directory self.driver_settings['input_path'] = input_directory
self.driver_settings['output_path'] = output_directory self.driver_settings['output_path'] = output_directory
@ -61,11 +62,6 @@ class Waifu2xCaffe:
self.driver_settings['output_extention'] = image_format self.driver_settings['output_extention'] = image_format
# print thread start message
self.print_lock.acquire()
Avalon.debug_info(f'[upscaler] Thread {threading.current_thread().name} started')
self.print_lock.release()
# list to be executed # list to be executed
# initialize the list with waifu2x binary path as the first element # initialize the list with waifu2x binary path as the first element
execute = [str(self.driver_settings['path'])] execute = [str(self.driver_settings['path'])]
@ -84,15 +80,8 @@ class Waifu2xCaffe:
execute.append(f'--{key}') execute.append(f'--{key}')
execute.append(str(value)) execute.append(str(value))
Avalon.debug_info(f'Executing: {execute}') # return the Popen object of the new process created
completed_command = subprocess.run(execute, check=True)
# print thread exiting message
self.print_lock.acquire() self.print_lock.acquire()
Avalon.debug_info(f'[upscaler] Thread {threading.current_thread().name} exiting') Avalon.debug_info(f'[upscaler] Subprocess {os.getpid()} executing: {shlex.join(execute)}')
self.print_lock.release() self.print_lock.release()
return subprocess.Popen(execute)
# return command execution return code
return completed_command.returncode
except Exception as e:
upscaler_exceptions.append(e)

View File

@ -4,14 +4,16 @@
Name: Waifu2x Converter CPP Driver Name: Waifu2x Converter CPP Driver
Author: K4YT3X Author: K4YT3X
Date Created: February 8, 2019 Date Created: February 8, 2019
Last Modified: October 6, 2019 Last Modified: February 22, 2020
Description: This class is a high-level wrapper Description: This class is a high-level wrapper
for waifu2x-converter-cpp. for waifu2x-converter-cpp.
""" """
# built-in imports # built-in imports
import os
import pathlib import pathlib
import shlex
import subprocess import subprocess
import threading import threading
@ -33,7 +35,7 @@ class Waifu2xConverter:
self.driver_settings['model_dir'] = model_dir self.driver_settings['model_dir'] = model_dir
self.print_lock = threading.Lock() self.print_lock = threading.Lock()
def upscale(self, input_directory, output_directory, scale_ratio, jobs, image_format, upscaler_exceptions): def upscale(self, input_directory, output_directory, scale_ratio, jobs, image_format):
""" Waifu2x Converter Driver Upscaler """ Waifu2x Converter Driver Upscaler
This method executes the upscaling of extracted frames. This method executes the upscaling of extracted frames.
@ -44,7 +46,6 @@ class Waifu2xConverter:
threads {int} -- number of threads threads {int} -- number of threads
""" """
try:
# overwrite config file settings # overwrite config file settings
self.driver_settings['input'] = input_directory self.driver_settings['input'] = input_directory
self.driver_settings['output'] = output_directory self.driver_settings['output'] = output_directory
@ -55,12 +56,7 @@ class Waifu2xConverter:
# models_rgb must be specified manually for waifu2x-converter-cpp # models_rgb must be specified manually for waifu2x-converter-cpp
# if it's not specified in the arguments, create automatically # if it's not specified in the arguments, create automatically
if self.driver_settings['model-dir'] is None: if self.driver_settings['model-dir'] is None:
self.driver_settings['model-dir'] = pathlib.Path(self.driver_settings['waifu2x_converter_path']) / 'models_rgb' self.driver_settings['model-dir'] = pathlib.Path(self.driver_settings['path']) / 'models_rgb'
# print thread start message
self.print_lock.acquire()
Avalon.debug_info(f'[upscaler] Thread {threading.current_thread().name} started')
self.print_lock.release()
# list to be executed # list to be executed
# initialize the list with waifu2x binary path as the first element # initialize the list with waifu2x binary path as the first element
@ -89,8 +85,8 @@ class Waifu2xConverter:
execute.append(str(value)) execute.append(str(value))
Avalon.debug_info(f'Executing: {execute}') # return the Popen object of the new process created
return subprocess.run(execute, check=True).returncode self.print_lock.acquire()
Avalon.debug_info(f'[upscaler] Subprocess {os.getpid()} executing: {shlex.join(execute)}')
except Exception as e: self.print_lock.release()
upscaler_exceptions.append(e) return subprocess.Popen(execute)

View File

@ -7,7 +7,7 @@ Date Created: June 26, 2019
Last Modified: November 15, 2019 Last Modified: November 15, 2019
Editor: K4YT3X Editor: K4YT3X
Last Modified: January 4, 2020 Last Modified: February 22, 2020
Description: This class is a high-level wrapper Description: This class is a high-level wrapper
for waifu2x_ncnn_vulkan. for waifu2x_ncnn_vulkan.
@ -15,6 +15,7 @@ for waifu2x_ncnn_vulkan.
# built-in imports # built-in imports
import os import os
import shlex
import subprocess import subprocess
import threading import threading
@ -42,7 +43,7 @@ class Waifu2xNcnnVulkan:
self.print_lock = threading.Lock() self.print_lock = threading.Lock()
def upscale(self, input_directory, output_directory, scale_ratio, upscaler_exceptions): def upscale(self, input_directory, output_directory, scale_ratio):
"""This is the core function for WAIFU2X class """This is the core function for WAIFU2X class
Arguments: Arguments:
@ -51,17 +52,11 @@ class Waifu2xNcnnVulkan:
ratio {int} -- output video ratio ratio {int} -- output video ratio
""" """
try:
# overwrite config file settings # overwrite config file settings
self.driver_settings['i'] = input_directory self.driver_settings['i'] = input_directory
self.driver_settings['o'] = output_directory self.driver_settings['o'] = output_directory
self.driver_settings['s'] = int(scale_ratio) self.driver_settings['s'] = int(scale_ratio)
# print thread start message
self.print_lock.acquire()
Avalon.debug_info(f'[upscaler] Thread {threading.current_thread().name} started')
self.print_lock.release()
# list to be executed # list to be executed
# initialize the list with waifu2x binary path as the first element # initialize the list with waifu2x binary path as the first element
execute = [str(self.driver_settings['path'])] execute = [str(self.driver_settings['path'])]
@ -80,15 +75,8 @@ class Waifu2xNcnnVulkan:
execute.append(f'--{key}') execute.append(f'--{key}')
execute.append(str(value)) execute.append(str(value))
Avalon.debug_info(f'Executing: {execute}') # return the Popen object of the new process created
completed_command = subprocess.run(execute, check=True)
# print thread exiting message
self.print_lock.acquire() self.print_lock.acquire()
Avalon.debug_info(f'[upscaler] Thread {threading.current_thread().name} exiting') Avalon.debug_info(f'[upscaler] Subprocess {os.getpid()} executing: {shlex.join(execute)}')
self.print_lock.release() self.print_lock.release()
return subprocess.Popen(execute)
# return command execution return code
return completed_command.returncode
except Exception as e:
upscaler_exceptions.append(e)