3.1.0 switched to multi-processing and added KeyboardInterrupt support

This commit is contained in:
k4yt3x 2020-02-26 05:28:48 -05:00
parent 6a100b1526
commit 09db8bedd0

View File

@ -13,7 +13,7 @@ __ __ _ _ ___ __ __
Name: Video2X Controller Name: Video2X Controller
Creator: K4YT3X Creator: K4YT3X
Date Created: Feb 24, 2018 Date Created: Feb 24, 2018
Last Modified: January 4, 2020 Last Modified: February 26, 2020
Editor: BrianPetkovsek Editor: BrianPetkovsek
Editor: SAT3LL Editor: SAT3LL
@ -34,7 +34,7 @@ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details. GNU General Public License for more details.
You should have received a copy of the GNU General Public License You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. along with this program. If not, see <https://www.gnu.org/licenses/>.
Description: Video2X is an automation software based on waifu2x image Description: Video2X is an automation software based on waifu2x image
enlarging engine. It extracts frames from a video, enlarge it by a enlarging engine. It extracts frames from a video, enlarge it by a
@ -53,6 +53,7 @@ import contextlib
import pathlib import pathlib
import re import re
import shutil import shutil
import subprocess
import sys import sys
import tempfile import tempfile
import time import time
@ -65,7 +66,7 @@ import GPUtil
import psutil import psutil
VERSION = '3.0.0' VERSION = '3.1.0'
LEGAL_INFO = f'''Video2X Version: {VERSION} LEGAL_INFO = f'''Video2X Version: {VERSION}
Author: K4YT3X Author: K4YT3X
@ -82,10 +83,10 @@ LOGO = r'''
\/ |_| \__,_| \___| \___/ |____| /_/ \_\ \/ |_| \__,_| \___| \___/ |____| /_/ \_\
''' '''
# each thread might take up to 2.5 GB during initialization. # each process might take up to 2.5 GB during initialization.
# (system memory, not to be confused with GPU memory) # (system memory, not to be confused with GPU memory)
SYS_MEM_PER_THREAD = 2.5 SYS_MEM_PER_PROCESS = 2.5
GPU_MEM_PER_THREAD = 3.5 GPU_MEM_PER_PROCESS = 3.5
def parse_arguments(): def parse_arguments():
@ -107,7 +108,7 @@ def parse_arguments():
upscaler_options.add_argument('-m', '--method', help='upscaling method', action='store', default='gpu', choices=['cpu', 'gpu', 'cudnn']) upscaler_options.add_argument('-m', '--method', help='upscaling method', action='store', default='gpu', choices=['cpu', 'gpu', 'cudnn'])
upscaler_options.add_argument('-d', '--driver', help='upscaling driver', action='store', default='waifu2x_caffe', choices=AVAILABLE_DRIVERS) upscaler_options.add_argument('-d', '--driver', help='upscaling driver', action='store', default='waifu2x_caffe', choices=AVAILABLE_DRIVERS)
upscaler_options.add_argument('-y', '--model_dir', type=pathlib.Path, help='directory containing model JSON files', action='store') upscaler_options.add_argument('-y', '--model_dir', type=pathlib.Path, help='directory containing model JSON files', action='store')
upscaler_options.add_argument('-t', '--threads', help='number of threads to use for upscaling', action='store', type=int, default=1) upscaler_options.add_argument('-p', '--processes', help='number of processes to use for upscaling', action='store', type=int, default=1)
upscaler_options.add_argument('-c', '--config', type=pathlib.Path, help='video2x config file location', action='store', default=pathlib.Path(sys.argv[0]).parent.absolute() / 'video2x.yaml') upscaler_options.add_argument('-c', '--config', type=pathlib.Path, help='video2x config file location', action='store', default=pathlib.Path(sys.argv[0]).parent.absolute() / 'video2x.yaml')
upscaler_options.add_argument('-b', '--batch', help='enable batch mode (select all default values to questions)', action='store_true') upscaler_options.add_argument('-b', '--batch', help='enable batch mode (select all default values to questions)', action='store_true')
@ -135,7 +136,7 @@ def print_logo():
def check_memory(): def check_memory():
""" Check usable system memory """ Check usable system memory
Warn the user if insufficient memory is available for Warn the user if insufficient memory is available for
the number of threads that the user have chosen. the number of processes that the user have chosen.
""" """
memory_status = [] memory_status = []
@ -150,7 +151,7 @@ def check_memory():
pathlib.Path(r'C:\Program Files\NVIDIA Corporation\NVSMI\nvidia-smi.exe').is_file()): pathlib.Path(r'C:\Program Files\NVIDIA Corporation\NVSMI\nvidia-smi.exe').is_file()):
# Nvidia System Management Interface not available # Nvidia System Management Interface not available
Avalon.warning('Nvidia-smi not available, skipping available memory check') Avalon.warning('Nvidia-smi not available, skipping available memory check')
Avalon.warning('If you experience error \"cudaSuccess out of memory\", try reducing number of threads you\'re using') Avalon.warning('If you experience error \"cudaSuccess out of memory\", try reducing number of processes you\'re using')
else: else:
with contextlib.suppress(ValueError): with contextlib.suppress(ValueError):
# "0" is GPU ID. Both waifu2x drivers use the first GPU available, therefore only 0 makes sense # "0" is GPU ID. Both waifu2x drivers use the first GPU available, therefore only 0 makes sense
@ -161,28 +162,28 @@ def check_memory():
for memory_type, memory_available in memory_status: for memory_type, memory_available in memory_status:
if memory_type == 'system': if memory_type == 'system':
mem_per_thread = SYS_MEM_PER_THREAD mem_per_process = SYS_MEM_PER_PROCESS
else: else:
mem_per_thread = GPU_MEM_PER_THREAD mem_per_process = GPU_MEM_PER_PROCESS
# if user doesn't even have enough memory to run even one thread # if user doesn't even have enough memory to run even one process
if memory_available < mem_per_thread: if memory_available < mem_per_process:
Avalon.warning(f'You might have insufficient amount of {memory_type} memory available to run this program ({memory_available} GB)') Avalon.warning(f'You might have insufficient amount of {memory_type} memory available to run this program ({memory_available} GB)')
Avalon.warning('Proceed with caution') Avalon.warning('Proceed with caution')
if args.threads > 1: if args.processes > 1:
if Avalon.ask('Reduce number of threads to avoid crashing?', default=True, batch=args.batch): if Avalon.ask('Reduce number of processes to avoid crashing?', default=True, batch=args.batch):
args.threads = 1 args.processes = 1
# if memory available is less than needed, warn the user # if memory available is less than needed, warn the user
elif memory_available < (mem_per_thread * args.threads): elif memory_available < (mem_per_process * args.processes):
Avalon.warning(f'Each waifu2x-caffe thread will require up to {SYS_MEM_PER_THREAD} GB of system memory') Avalon.warning(f'Each waifu2x-caffe process will require up to {SYS_MEM_PER_PROCESS} GB of system memory')
Avalon.warning(f'You demanded {args.threads} threads to be created, but you only have {round(memory_available, 4)} GB {memory_type} memory available') Avalon.warning(f'You demanded {args.processes} processes to be created, but you only have {round(memory_available, 4)} GB {memory_type} memory available')
Avalon.warning(f'{mem_per_thread * args.threads} GB of {memory_type} memory is recommended for {args.threads} threads') Avalon.warning(f'{mem_per_process * args.processes} GB of {memory_type} memory is recommended for {args.processes} processes')
Avalon.warning(f'With your current amount of {memory_type} memory available, {int(memory_available // mem_per_thread)} threads is recommended') Avalon.warning(f'With your current amount of {memory_type} memory available, {int(memory_available // mem_per_process)} processes is recommended')
# ask the user if he / she wants to change to the recommended # ask the user if he / she wants to change to the recommended
# number of threads # number of processes
if Avalon.ask('Change to the recommended value?', default=True, batch=args.batch): if Avalon.ask('Change to the recommended value?', default=True, batch=args.batch):
args.threads = int(memory_available // mem_per_thread) args.processes = int(memory_available // mem_per_process)
else: else:
Avalon.warning('Proceed with caution') Avalon.warning('Proceed with caution')
@ -285,20 +286,20 @@ if (args.width and not args.height) or (not args.width and args.height):
if args.driver in ['waifu2x_caffe', 'waifu2x_converter', 'waifu2x_ncnn_vulkan']: if args.driver in ['waifu2x_caffe', 'waifu2x_converter', 'waifu2x_ncnn_vulkan']:
check_memory() check_memory()
# anime4k runs significantly faster with more threads # anime4k runs significantly faster with more processes
if args.driver == 'anime4k' and args.threads <= 1: if args.driver == 'anime4k' and args.processes <= 1:
Avalon.warning('Anime4K runs significantly faster with more threads') Avalon.warning('Anime4K runs significantly faster with more processes')
if Avalon.ask('Use more threads of Anime4K?', default=True, batch=args.batch): if Avalon.ask('Use more processes of Anime4K?', default=True, batch=args.batch):
while True: while True:
try: try:
threads = Avalon.gets('Amount of threads to use [5]: ', default=5, batch=args.batch) processes = Avalon.gets('Amount of processes to use [5]: ', default=5, batch=args.batch)
args.threads = int(threads) args.processes = int(processes)
break break
except ValueError: except ValueError:
if threads == '': if processes == '':
args.threads = 5 args.processes = 5
break break
Avalon.error(f'{threads} is not a valid integer') Avalon.error(f'{processes} is not a valid integer')
# read configurations from configuration file # read configurations from configuration file
config = read_config(args.config) config = read_config(args.config)
@ -309,17 +310,39 @@ config = read_config(args.config)
driver_settings = config[args.driver] driver_settings = config[args.driver]
# check if driver path exists # check if driver path exists
if not pathlib.Path(driver_settings['path']).is_file(): if not pathlib.Path(driver_settings['path']).exists():
if not pathlib.Path(f'{driver_settings["path"]}.exe').is_file(): if not pathlib.Path(f'{driver_settings["path"]}.exe').exists():
Avalon.error('Specified driver executable directory doesn\'t exist') Avalon.error('Specified driver executable directory doesn\'t exist')
Avalon.error('Please check the configuration file settings') Avalon.error('Please check the configuration file settings')
raise FileNotFoundError(driver_settings['path']) raise FileNotFoundError(driver_settings['path'])
# if the driver is Anime4K, check if JDK 12 is installed # if the driver is Anime4K, check if JDK 12 is installed
jdk_available = True
if args.driver == 'anime4k': if args.driver == 'anime4k':
if not pathlib.Path('C:/Program Files/Java/jdk-12.0.2/bin/java.exe').is_file():
Avalon.warning('Cannot find JDK 12 at its default installation location') # if specified JDK path doesn't exist
Avalon.warning('Please ensure you have JDK 12 installed and configured') if not pathlib.Path(driver_settings['java_path']).is_file():
# try to find JDK on system
if shutil.which('java') is not None:
# check if JDK has master version 12
java_version_output = subprocess.run(['java', '-version'], capture_output=True).stderr
if re.search(r'java version "12\.\d\.\d"', java_version_output.decode().split('\n')[0]) is not None:
driver_settings['java_path'] = shutil.which('java')
else:
jdk_available = False
# if java is not found in PATH
else:
jdk_available = False
# if JDK 12 is not found
# warn the user and exit
if jdk_available is False:
Avalon.error('Cannot find JDK 12 on this system')
Avalon.error('Please ensure you have JDK 12 installed and configured')
sys.exit(1)
# read FFmpeg configuration # read FFmpeg configuration
ffmpeg_settings = config['ffmpeg'] ffmpeg_settings = config['ffmpeg']
@ -388,7 +411,7 @@ try:
upscaler.scale_height = args.height upscaler.scale_height = args.height
upscaler.scale_ratio = args.ratio upscaler.scale_ratio = args.ratio
upscaler.model_dir = args.model_dir upscaler.model_dir = args.model_dir
upscaler.threads = args.threads upscaler.processes = args.processes
upscaler.video2x_cache_directory = video2x_cache_directory upscaler.video2x_cache_directory = video2x_cache_directory
upscaler.image_format = image_format upscaler.image_format = image_format
upscaler.preserve_frames = preserve_frames upscaler.preserve_frames = preserve_frames
@ -416,7 +439,7 @@ try:
upscaler.scale_height = args.height upscaler.scale_height = args.height
upscaler.scale_ratio = args.ratio upscaler.scale_ratio = args.ratio
upscaler.model_dir = args.model_dir upscaler.model_dir = args.model_dir
upscaler.threads = args.threads upscaler.processes = args.processes
upscaler.video2x_cache_directory = video2x_cache_directory upscaler.video2x_cache_directory = video2x_cache_directory
upscaler.image_format = image_format upscaler.image_format = image_format
upscaler.preserve_frames = preserve_frames upscaler.preserve_frames = preserve_frames