|
|
|
@ -31,10 +31,13 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
private const string VideotoolboxAlias = "vt";
|
|
|
|
|
private const string OpenclAlias = "ocl";
|
|
|
|
|
private const string CudaAlias = "cu";
|
|
|
|
|
private const string DrmAlias = "dr";
|
|
|
|
|
private const string VulkanAlias = "vk";
|
|
|
|
|
private readonly IApplicationPaths _appPaths;
|
|
|
|
|
private readonly IMediaEncoder _mediaEncoder;
|
|
|
|
|
private readonly ISubtitleEncoder _subtitleEncoder;
|
|
|
|
|
private readonly IConfiguration _config;
|
|
|
|
|
private readonly Version _minKernelVersionAmdVkFmtModifier = new Version(5, 15);
|
|
|
|
|
private readonly Version _minKernelVersioni915Hang = new Version(5, 18);
|
|
|
|
|
|
|
|
|
|
private static readonly string[] _videoProfilesH264 = new[]
|
|
|
|
@ -149,6 +152,14 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
&& _mediaEncoder.SupportsFilter("hwupload_cuda");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool IsVulkanFullSupported()
|
|
|
|
|
{
|
|
|
|
|
return _mediaEncoder.SupportsHwaccel("vulkan")
|
|
|
|
|
&& _mediaEncoder.SupportsFilter("libplacebo")
|
|
|
|
|
&& _mediaEncoder.SupportsFilter("scale_vulkan")
|
|
|
|
|
&& _mediaEncoder.SupportsFilterWithOption(FilterOptionType.OverlayVulkanFrameSync);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool IsHwTonemapAvailable(EncodingJobInfo state, EncodingOptions options)
|
|
|
|
|
{
|
|
|
|
|
if (state.VideoStream == null
|
|
|
|
@ -176,6 +187,19 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
|| string.Equals(state.VideoStream.VideoRangeType, "HLG", StringComparison.OrdinalIgnoreCase));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool IsVulkanHwTonemapAvailable(EncodingJobInfo state, EncodingOptions options)
|
|
|
|
|
{
|
|
|
|
|
if (state.VideoStream == null)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// libplacebo has partial Dolby Vision to SDR tonemapping support.
|
|
|
|
|
return options.EnableTonemapping
|
|
|
|
|
&& string.Equals(state.VideoStream.VideoRange, "HDR", StringComparison.OrdinalIgnoreCase)
|
|
|
|
|
&& GetVideoColorBitDepth(state) == 10;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool IsVaapiVppTonemapAvailable(EncodingJobInfo state, EncodingOptions options)
|
|
|
|
|
{
|
|
|
|
|
if (state.VideoStream == null
|
|
|
|
@ -756,8 +780,13 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
}
|
|
|
|
|
else if (_mediaEncoder.IsVaapiDeviceAmd)
|
|
|
|
|
{
|
|
|
|
|
args.Append(GetOpenclDeviceArgs(0, "Advanced Micro Devices", null, OpenclAlias));
|
|
|
|
|
filterDevArgs = GetFilterHwDeviceArgs(OpenclAlias);
|
|
|
|
|
if (!IsVulkanFullSupported()
|
|
|
|
|
|| !_mediaEncoder.IsVaapiDeviceSupportVulkanFmtModifier
|
|
|
|
|
|| Environment.OSVersion.Version < _minKernelVersionAmdVkFmtModifier)
|
|
|
|
|
{
|
|
|
|
|
args.Append(GetOpenclDeviceArgs(0, "Advanced Micro Devices", null, OpenclAlias));
|
|
|
|
|
filterDevArgs = GetFilterHwDeviceArgs(OpenclAlias);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
@ -2774,22 +2803,41 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
return string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var args = "tonemap_{0}=format={1}:p=bt709:t=bt709:m=bt709";
|
|
|
|
|
var args = string.Empty;
|
|
|
|
|
var algorithm = options.TonemappingAlgorithm;
|
|
|
|
|
|
|
|
|
|
if (hwTonemapSuffix.Contains("vaapi", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
if (string.Equals(hwTonemapSuffix, "vaapi", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
args += ",procamp_vaapi=b={2}:c={3}:extra_hw_frames=16";
|
|
|
|
|
args = "tonemap_vaapi=format={0}:p=bt709:t=bt709:m=bt709,procamp_vaapi=b={1}:c={2}:extra_hw_frames=16";
|
|
|
|
|
return string.Format(
|
|
|
|
|
CultureInfo.InvariantCulture,
|
|
|
|
|
args,
|
|
|
|
|
hwTonemapSuffix,
|
|
|
|
|
videoFormat ?? "nv12",
|
|
|
|
|
options.VppTonemappingBrightness,
|
|
|
|
|
options.VppTonemappingContrast);
|
|
|
|
|
}
|
|
|
|
|
else if (string.Equals(hwTonemapSuffix, "vulkan", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
args = "libplacebo=format={1}:tonemapping={2}:color_primaries=bt709:color_trc=bt709:colorspace=bt709:peak_detect=0:upscaler=none:downscaler=none";
|
|
|
|
|
|
|
|
|
|
if (!string.Equals(options.TonemappingRange, "auto", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
args += ":range={6}";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (string.Equals(options.TonemappingAlgorithm, "bt2390", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
algorithm = "bt.2390";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else if (string.Equals(options.TonemappingAlgorithm, "none", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
algorithm = "clip";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
args += ":tonemap={2}:peak={3}:desat={4}";
|
|
|
|
|
args = "tonemap_{0}=format={1}:p=bt709:t=bt709:m=bt709:tonemap={2}:peak={3}:desat={4}";
|
|
|
|
|
|
|
|
|
|
if (options.TonemappingParam != 0)
|
|
|
|
|
{
|
|
|
|
@ -2807,7 +2855,7 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
args,
|
|
|
|
|
hwTonemapSuffix,
|
|
|
|
|
videoFormat ?? "nv12",
|
|
|
|
|
options.TonemappingAlgorithm,
|
|
|
|
|
algorithm,
|
|
|
|
|
options.TonemappingPeak,
|
|
|
|
|
options.TonemappingDesat,
|
|
|
|
|
options.TonemappingParam,
|
|
|
|
@ -3770,7 +3818,9 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
var vidDecoder = GetHardwareVideoDecoder(state, options) ?? string.Empty;
|
|
|
|
|
var isSwDecoder = string.IsNullOrEmpty(vidDecoder);
|
|
|
|
|
var isSwEncoder = !vidEncoder.Contains("vaapi", StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
var isVaapiOclSupported = isLinux && IsVaapiSupported(state) && IsVaapiFullSupported() && IsOpenclFullSupported();
|
|
|
|
|
var isVaapiFullSupported = isLinux && IsVaapiSupported(state) && IsVaapiFullSupported();
|
|
|
|
|
var isVaapiOclSupported = isVaapiFullSupported && IsOpenclFullSupported();
|
|
|
|
|
var isVaapiVkSupported = isVaapiFullSupported && IsVulkanFullSupported();
|
|
|
|
|
|
|
|
|
|
// legacy vaapi pipeline(copy-back)
|
|
|
|
|
if ((isSwDecoder && isSwEncoder)
|
|
|
|
@ -3798,14 +3848,24 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
if (_mediaEncoder.IsVaapiDeviceInteliHD)
|
|
|
|
|
{
|
|
|
|
|
// Intel iHD path, with extra vpp tonemap and overlay support.
|
|
|
|
|
return GetVaapiFullVidFiltersPrefered(state, options, vidDecoder, vidEncoder);
|
|
|
|
|
return GetIntelVaapiFullVidFiltersPrefered(state, options, vidDecoder, vidEncoder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// prefered vaapi + vulkan filters pipeline
|
|
|
|
|
if (_mediaEncoder.IsVaapiDeviceAmd
|
|
|
|
|
&& isVaapiVkSupported
|
|
|
|
|
&& _mediaEncoder.IsVaapiDeviceSupportVulkanFmtModifier
|
|
|
|
|
&& Environment.OSVersion.Version >= _minKernelVersionAmdVkFmtModifier)
|
|
|
|
|
{
|
|
|
|
|
// AMD radeonsi path(Vega/gfx9+, kernel>=5.15), with extra vulkan tonemap and overlay support.
|
|
|
|
|
return GetAmdVaapiFullVidFiltersPrefered(state, options, vidDecoder, vidEncoder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Intel i965 and Amd radeonsi/r600 path, only featuring scale and deinterlace support.
|
|
|
|
|
// Intel i965 and Amd radeonsi/r600 path(Polaris/gfx8-), only featuring scale and deinterlace support.
|
|
|
|
|
return GetVaapiLimitedVidFiltersPrefered(state, options, vidDecoder, vidEncoder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public (List<string> MainFilters, List<string> SubFilters, List<string> OverlayFilters) GetVaapiFullVidFiltersPrefered(
|
|
|
|
|
public (List<string> MainFilters, List<string> SubFilters, List<string> OverlayFilters) GetIntelVaapiFullVidFiltersPrefered(
|
|
|
|
|
EncodingJobInfo state,
|
|
|
|
|
EncodingOptions options,
|
|
|
|
|
string vidDecoder,
|
|
|
|
@ -4003,6 +4063,203 @@ namespace MediaBrowser.Controller.MediaEncoding
|
|
|
|
|
return (mainFilters, subFilters, overlayFilters);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public (List<string> MainFilters, List<string> SubFilters, List<string> OverlayFilters) GetAmdVaapiFullVidFiltersPrefered(
|
|
|
|
|
EncodingJobInfo state,
|
|
|
|
|
EncodingOptions options,
|
|
|
|
|
string vidDecoder,
|
|
|
|
|
string vidEncoder)
|
|
|
|
|
{
|
|
|
|
|
var inW = state.VideoStream?.Width;
|
|
|
|
|
var inH = state.VideoStream?.Height;
|
|
|
|
|
var reqW = state.BaseRequest.Width;
|
|
|
|
|
var reqH = state.BaseRequest.Height;
|
|
|
|
|
var reqMaxW = state.BaseRequest.MaxWidth;
|
|
|
|
|
var reqMaxH = state.BaseRequest.MaxHeight;
|
|
|
|
|
var threeDFormat = state.MediaSource.Video3DFormat;
|
|
|
|
|
|
|
|
|
|
var isVaapiDecoder = vidDecoder.Contains("vaapi", StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
var isVaapiEncoder = vidEncoder.Contains("vaapi", StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
var isSwDecoder = string.IsNullOrEmpty(vidDecoder);
|
|
|
|
|
var isSwEncoder = !isVaapiEncoder;
|
|
|
|
|
var isVaInVaOut = isVaapiDecoder && isVaapiEncoder;
|
|
|
|
|
|
|
|
|
|
var doDeintH264 = state.DeInterlace("h264", true) || state.DeInterlace("avc", true);
|
|
|
|
|
var doDeintHevc = state.DeInterlace("h265", true) || state.DeInterlace("hevc", true);
|
|
|
|
|
var doVkTonemap = IsVulkanHwTonemapAvailable(state, options);
|
|
|
|
|
var doDeintH2645 = doDeintH264 || doDeintHevc;
|
|
|
|
|
|
|
|
|
|
var hasSubs = state.SubtitleStream != null && state.SubtitleDeliveryMethod == SubtitleDeliveryMethod.Encode;
|
|
|
|
|
var hasTextSubs = hasSubs && state.SubtitleStream.IsTextSubtitleStream;
|
|
|
|
|
var hasGraphicalSubs = hasSubs && !state.SubtitleStream.IsTextSubtitleStream;
|
|
|
|
|
var hasAssSubs = hasSubs
|
|
|
|
|
&& (string.Equals(state.SubtitleStream.Codec, "ass", StringComparison.OrdinalIgnoreCase)
|
|
|
|
|
|| string.Equals(state.SubtitleStream.Codec, "ssa", StringComparison.OrdinalIgnoreCase));
|
|
|
|
|
|
|
|
|
|
/* Make main filters for video stream */
|
|
|
|
|
var mainFilters = new List<string>();
|
|
|
|
|
|
|
|
|
|
mainFilters.Add(GetOverwriteColorPropertiesParam(state, doVkTonemap));
|
|
|
|
|
|
|
|
|
|
if (isSwDecoder)
|
|
|
|
|
{
|
|
|
|
|
// INPUT sw surface(memory)
|
|
|
|
|
// sw deint
|
|
|
|
|
if (doDeintH2645)
|
|
|
|
|
{
|
|
|
|
|
var swDeintFilter = GetSwDeinterlaceFilter(state, options);
|
|
|
|
|
mainFilters.Add(swDeintFilter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var outFormat = doVkTonemap ? "yuv420p10le" : "nv12";
|
|
|
|
|
var swScaleFilter = GetSwScaleFilter(state, options, vidEncoder, inW, inH, threeDFormat, reqW, reqH, reqMaxW, reqMaxH);
|
|
|
|
|
// sw scale
|
|
|
|
|
mainFilters.Add(swScaleFilter);
|
|
|
|
|
mainFilters.Add("format=" + outFormat);
|
|
|
|
|
|
|
|
|
|
// keep video at memory except vk tonemap,
|
|
|
|
|
// since the overhead caused by hwupload >>> using sw filter.
|
|
|
|
|
// sw => hw
|
|
|
|
|
if (doVkTonemap)
|
|
|
|
|
{
|
|
|
|
|
mainFilters.Add("hwupload=derive_device=vulkan:extra_hw_frames=16");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (isVaapiDecoder)
|
|
|
|
|
{
|
|
|
|
|
// INPUT vaapi surface(vram)
|
|
|
|
|
// hw deint
|
|
|
|
|
if (doDeintH2645)
|
|
|
|
|
{
|
|
|
|
|
var deintFilter = GetHwDeinterlaceFilter(state, options, "vaapi");
|
|
|
|
|
mainFilters.Add(deintFilter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var outFormat = doVkTonemap ? string.Empty : (hasSubs && isVaInVaOut ? "bgra" : "nv12");
|
|
|
|
|
var hwScaleFilter = GetHwScaleFilter("vaapi", outFormat, inW, inH, reqW, reqH, reqMaxW, reqMaxH);
|
|
|
|
|
|
|
|
|
|
// allocate extra pool sizes for overlay_vulkan
|
|
|
|
|
if (!string.IsNullOrEmpty(hwScaleFilter) && isVaInVaOut && hasSubs)
|
|
|
|
|
{
|
|
|
|
|
hwScaleFilter += ":extra_hw_frames=32";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// hw scale
|
|
|
|
|
mainFilters.Add(hwScaleFilter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((isVaapiDecoder && doVkTonemap) || (isVaInVaOut && (doVkTonemap || hasSubs)))
|
|
|
|
|
{
|
|
|
|
|
// map from vaapi to vulkan via vaapi-vulkan interop (Vega/gfx9+).
|
|
|
|
|
mainFilters.Add("hwmap=derive_device=vulkan");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// vk tonemap
|
|
|
|
|
if (doVkTonemap)
|
|
|
|
|
{
|
|
|
|
|
var outFormat = isVaInVaOut && hasSubs ? "bgra" : "nv12";
|
|
|
|
|
var tonemapFilter = GetHwTonemapFilter(options, "vulkan", outFormat);
|
|
|
|
|
mainFilters.Add(tonemapFilter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (doVkTonemap && isVaInVaOut && !hasSubs)
|
|
|
|
|
{
|
|
|
|
|
// OUTPUT vaapi(nv12/bgra) surface(vram)
|
|
|
|
|
// reverse-mapping via vaapi-vulkan interop.
|
|
|
|
|
mainFilters.Add("hwmap=derive_device=vaapi:reverse=1");
|
|
|
|
|
mainFilters.Add("format=vaapi");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var memoryOutput = false;
|
|
|
|
|
var isUploadForVkTonemap = isSwDecoder && doVkTonemap;
|
|
|
|
|
if ((isVaapiDecoder && isSwEncoder) || isUploadForVkTonemap)
|
|
|
|
|
{
|
|
|
|
|
memoryOutput = true;
|
|
|
|
|
|
|
|
|
|
// OUTPUT nv12 surface(memory)
|
|
|
|
|
mainFilters.Add("hwdownload");
|
|
|
|
|
mainFilters.Add("format=nv12");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// OUTPUT nv12 surface(memory)
|
|
|
|
|
if (isSwDecoder && isVaapiEncoder)
|
|
|
|
|
{
|
|
|
|
|
memoryOutput = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (memoryOutput)
|
|
|
|
|
{
|
|
|
|
|
// text subtitles
|
|
|
|
|
if (hasTextSubs)
|
|
|
|
|
{
|
|
|
|
|
var textSubtitlesFilter = GetTextSubtitlesFilter(state, false, false);
|
|
|
|
|
mainFilters.Add(textSubtitlesFilter);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (memoryOutput && isVaapiEncoder)
|
|
|
|
|
{
|
|
|
|
|
if (!hasGraphicalSubs)
|
|
|
|
|
{
|
|
|
|
|
mainFilters.Add("hwupload_vaapi");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Make sub and overlay filters for subtitle stream */
|
|
|
|
|
var subFilters = new List<string>();
|
|
|
|
|
var overlayFilters = new List<string>();
|
|
|
|
|
if (isVaInVaOut)
|
|
|
|
|
{
|
|
|
|
|
if (hasSubs)
|
|
|
|
|
{
|
|
|
|
|
if (hasGraphicalSubs)
|
|
|
|
|
{
|
|
|
|
|
// scale=s=1280x720,format=bgra,hwupload
|
|
|
|
|
var subSwScaleFilter = GetCustomSwScaleFilter(inW, inH, reqW, reqH, reqMaxW, reqMaxH);
|
|
|
|
|
subFilters.Add(subSwScaleFilter);
|
|
|
|
|
subFilters.Add("format=bgra");
|
|
|
|
|
}
|
|
|
|
|
else if (hasTextSubs)
|
|
|
|
|
{
|
|
|
|
|
var alphaSrcFilter = GetAlphaSrcFilter(state, inW, inH, reqW, reqH, reqMaxW, reqMaxH, hasAssSubs ? 10 : 5);
|
|
|
|
|
var subTextSubtitlesFilter = GetTextSubtitlesFilter(state, true, true);
|
|
|
|
|
subFilters.Add(alphaSrcFilter);
|
|
|
|
|
subFilters.Add("format=bgra");
|
|
|
|
|
subFilters.Add(subTextSubtitlesFilter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
subFilters.Add("hwupload=derive_device=vulkan:extra_hw_frames=16");
|
|
|
|
|
|
|
|
|
|
overlayFilters.Add("overlay_vulkan=eof_action=endall:shortest=1:repeatlast=0");
|
|
|
|
|
|
|
|
|
|
// explicitly sync using libplacebo.
|
|
|
|
|
overlayFilters.Add("libplacebo=format=nv12:upscaler=none:downscaler=none");
|
|
|
|
|
|
|
|
|
|
// OUTPUT vaapi(nv12/bgra) surface(vram)
|
|
|
|
|
// reverse-mapping via vaapi-vulkan interop.
|
|
|
|
|
overlayFilters.Add("hwmap=derive_device=vaapi:reverse=1");
|
|
|
|
|
overlayFilters.Add("format=vaapi");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (memoryOutput)
|
|
|
|
|
{
|
|
|
|
|
if (hasGraphicalSubs)
|
|
|
|
|
{
|
|
|
|
|
var subSwScaleFilter = isSwDecoder
|
|
|
|
|
? GetSwScaleFilter(state, options, vidEncoder, inW, inH, threeDFormat, reqW, reqH, reqMaxW, reqMaxH)
|
|
|
|
|
: GetCustomSwScaleFilter(inW, inH, reqW, reqH, reqMaxW, reqMaxH);
|
|
|
|
|
subFilters.Add(subSwScaleFilter);
|
|
|
|
|
overlayFilters.Add("overlay=eof_action=pass:shortest=1:repeatlast=0");
|
|
|
|
|
|
|
|
|
|
if (isVaapiEncoder)
|
|
|
|
|
{
|
|
|
|
|
overlayFilters.Add("hwupload_vaapi");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (mainFilters, subFilters, overlayFilters);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public (List<string> MainFilters, List<string> SubFilters, List<string> OverlayFilters) GetVaapiLimitedVidFiltersPrefered(
|
|
|
|
|
EncodingJobInfo state,
|
|
|
|
|
EncodingOptions options,
|
|
|
|
|