Av1an/TargetQuality/target_quality.py
Luis Garcia 3fc04fe08d Move non target_quality specific functions to vmaf
Signed-off-by: Luis Garcia <luigi311.lg@gmail.com>
2020-11-26 15:04:15 -07:00

182 lines
5.9 KiB
Python

import subprocess
from subprocess import STDOUT, PIPE
import numpy as np
from scipy import interpolate
from Av1an.commandtypes import CommandPair, Command
from Projects import Project
from VMAF import call_vmaf, read_json, transform_vmaf
from Chunks.chunk import Chunk
from Av1an.bar import process_pipe
def adapt_probing_rate(rate, frames):
"""
Change probing rate depending on amount of frames in scene.
Ensure that low frame count scenes get decent amount of probes
:param rate: given rate of probing
:param frames: amount of frames in scene
:return: new probing rate
"""
if frames < 20:
return 1
elif frames < 40:
return min(rate, 2)
elif frames < 120:
return min(rate, 3)
elif frames < 240:
return min(rate, 4)
elif frames > 240:
return max(rate, 5)
elif frames > 480:
return 10
def get_target_q(scores, target_quality):
"""
Interpolating scores to get Q closest to target
Interpolation type for 2 probes changes to linear
"""
x = [x[1] for x in sorted(scores)]
y = [float(x[0]) for x in sorted(scores)]
if len(x) > 2:
interpolation = 'quadratic'
else:
interpolation = 'linear'
f = interpolate.interp1d(x, y, kind=interpolation)
xnew = np.linspace(min(x), max(x), max(x) - min(x))
tl = list(zip(xnew, f(xnew)))
q = min(tl, key=lambda l: abs(l[1] - target_quality))
return int(q[0]), round(q[1], 3)
def weighted_search(num1, vmaf1, num2, vmaf2, target):
"""
Returns weighted value closest to searched
:param num1: Q of first probe
:param vmaf1: VMAF of first probe
:param num2: Q of second probe
:param vmaf2: VMAF of first probe
:param target: VMAF target
:return: Q for new probe
"""
dif1 = abs(transform_vmaf(target) - transform_vmaf(vmaf2))
dif2 = abs(transform_vmaf(target) - transform_vmaf(vmaf1))
tot = dif1 + dif2
new_point = int(round(num1 * (dif1 / tot) + (num2 * (dif2 / tot))))
return new_point
def probe_cmd(chunk: Chunk, q, ffmpeg_pipe, encoder, probing_rate) -> CommandPair:
"""
Generate and return commands for probes at set Q values
These are specifically not the commands that are generated
by the user or encoder defaults, since these
should be faster than the actual encoding commands.
These should not be moved into encoder classes at this point.
"""
pipe = ['ffmpeg', '-y', '-hide_banner', '-loglevel', 'error', '-i', '-', '-vf',
f'select=not(mod(n\\,{probing_rate}))', *ffmpeg_pipe]
probe_name = gen_probes_names(chunk, q).with_suffix('.ivf').as_posix()
if encoder == 'aom':
params = ['aomenc', '--passes=1', '--threads=8',
'--end-usage=q', '--cpu-used=6', f'--cq-level={q}']
cmd = CommandPair(pipe, [*params, '-o', probe_name, '-'])
elif encoder == 'x265':
params = ['x265', '--log-level', '0', '--no-progress',
'--y4m', '--preset', 'medium', '--crf', f'{q}']
cmd = CommandPair(pipe, [*params, '-o', probe_name, '-'])
elif encoder == 'rav1e':
params = ['rav1e', '-y', '-s', '10', '--tiles', '8', '--quantizer', f'{q}']
cmd = CommandPair(pipe, [*params, '-o', probe_name, '-'])
elif encoder == 'vpx':
params = ['vpxenc', '-b', '10', '--profile=2','--passes=1', '--pass=1', '--codec=vp9',
'--threads=8', '--cpu-used=9', '--end-usage=q',
f'--cq-level={q}']
cmd = CommandPair(pipe, [*params, '-o', probe_name, '-'])
elif encoder == 'svt_av1':
params = ['SvtAv1EncApp', '-i', 'stdin',
'--preset', '8', '--rc', '0', '--qp', f'{q}']
cmd = CommandPair(pipe, [*params, '-b', probe_name, '-'])
elif encoder == 'svt_vp9':
params = ['SvtVp9EncApp', '-i', 'stdin',
'-enc-mode', '8', '-q', f'{q}']
# TODO: pipe needs to output rawvideo
cmd = CommandPair(pipe, [*params, '-b', probe_name, '-'])
elif encoder == 'x264':
params = ['x264', '--log-level', 'error', '--demuxer', 'y4m',
'-', '--no-progress', '--preset', 'slow', '--crf',
f'{q}']
cmd = CommandPair(pipe, [*params, '-o', probe_name, '-'])
return cmd
def gen_probes_names(chunk: Chunk, q):
"""Make name of vmaf probe
"""
return chunk.fake_input_path.with_name(f'v_{q}{chunk.name}').with_suffix('.ivf')
def make_pipes(ffmpeg_gen_cmd: Command, command: CommandPair):
ffmpeg_gen_pipe = subprocess.Popen(ffmpeg_gen_cmd, stdout=PIPE, stderr=STDOUT)
ffmpeg_pipe = subprocess.Popen(command[0], stdin=ffmpeg_gen_pipe.stdout, stdout=PIPE, stderr=STDOUT)
pipe = subprocess.Popen(command[1], stdin=ffmpeg_pipe.stdout, stdout=PIPE,
stderr=STDOUT,
universal_newlines=True)
return pipe
def vmaf_probe(chunk: Chunk, q, args: Project, probing_rate):
"""
Make encoding probe to get VMAF that Q returns
:param chunk: the Chunk
:param q: Value to make probe
:param args: the Project
:return : path to json file with vmaf scores
"""
cmd = probe_cmd(chunk, q, args.ffmpeg_pipe, args.encoder, probing_rate)
pipe = make_pipes(chunk.ffmpeg_gen_cmd, cmd)
process_pipe(pipe)
file = call_vmaf(chunk, gen_probes_names(chunk, q), args.n_threads, args.vmaf_path, args.vmaf_res, vmaf_filter=args.vmaf_filter,
vmaf_rate=probing_rate)
return file
def get_closest(q_list, q, positive=True):
"""
Returns closest value from the list, ascending or descending
:param q_list: list of q values that been already used
:param q:
:param positive: search direction, positive - only values bigger than q
:return: q value from list
"""
if positive:
q_list = [x for x in q_list if x > q]
else:
q_list = [x for x in q_list if x < q]
return min(q_list, key=lambda x: abs(x - q))