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,148 +151,134 @@ 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
@ -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)