Files
ffmpeg-patched/tools/normalize.py
T
gx dcecd42de4 FFmpeg 7.1 + cuframes input demuxer
Snapshot FFmpeg n7.1 (release tag) с применённым patch'ем для cuframes
input format. Используется как FFMPEG_REPO_OVERRIDE в NickM-27/FFmpeg-Builds
fork для статической сборки patched binary под Frigate (Debian 12 / glibc 2.36).

Apply changes:
  + libavformat/cuframesdec.c (новый — реализация демуксера)
  M libavformat/Makefile (CONFIG_CUFRAMES_DEMUXER target)
  M libavformat/allformats.c (extern declaration)
  M configure (--enable-libcuframes option + dep check)

Patch source: https://git.goldix.org/gx/cuframes (filter/ffmpeg-7.1-cuframes-demuxer.patch)

History сброшена (snapshot вместо fork) потому что upstream shallow clone
не позволял push в gitea. Полная история FFmpeg — на github.com/FFmpeg/FFmpeg n7.1.
2026-05-17 11:43:10 +01:00

74 lines
2.2 KiB
Python
Executable File

#!/usr/bin/env python3
import argparse
import logging
import shlex
import subprocess
HELP = '''
Normalize audio input.
The command uses ffprobe to analyze an input file with the ebur128
filter, and finally run ffmpeg to normalize the input depending on the
computed adjustment.
ffmpeg encoding arguments can be passed through the extra arguments
after options, for example as in:
normalize.py --input input.mp3 --output output.mp3 -- -loglevel debug -y
'''
logging.basicConfig(format='normalize|%(levelname)s> %(message)s', level=logging.INFO)
log = logging.getLogger()
class Formatter(
argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter
):
pass
def normalize():
parser = argparse.ArgumentParser(description=HELP, formatter_class=Formatter)
parser.add_argument('--input', '-i', required=True, help='specify input file')
parser.add_argument('--output', '-o', required=True, help='specify output file')
parser.add_argument('--dry-run', '-n', help='simulate commands', action='store_true')
parser.add_argument('encode_arguments', nargs='*', help='specify encode options used for the actual encoding')
args = parser.parse_args()
analysis_cmd = [
'ffprobe', '-v', 'error', '-of', 'compact=p=0:nk=1',
'-show_entries', 'frame_tags=lavfi.r128.I', '-f', 'lavfi',
f"amovie='{args.input}',ebur128=metadata=1"
]
def _run_command(cmd, dry_run=False):
log.info(f"Running command:\n$ {shlex.join(cmd)}")
if not dry_run:
result = subprocess.run(cmd, check=True, stdout=subprocess.PIPE)
return result
result = _run_command(analysis_cmd)
loudness = ref = -23
for line in result.stdout.splitlines():
sline = line.rstrip()
if sline:
loudness = sline
adjust = ref - float(loudness)
if abs(adjust) < 0.0001:
logging.info(f"No normalization needed for '{args.input}'")
return
logging.info(f"Adjusting '{args.input}' by {adjust:.2f}dB...")
normalize_cmd = [
'ffmpeg', '-i', args.input, '-af', f'volume={adjust:.2f}dB'
] + args.encode_arguments + [args.output]
_run_command(normalize_cmd, args.dry_run)
if __name__ == '__main__':
normalize()