#!/bin/bash usage() { echo "estimate_fg.sh -i input_file [-o output_file] [-l NUM] [-s NUM] [-h NUM] [-I] [-U]" echo -e "\t-o file to output results to" echo -e "\t-l low value to use as minimum film-grain" echo -e "\t-s step value to use increment from low to high film-grain" echo -e "\t-h high value to use as maximum film-grain" echo -e "\t-I Install this as /usr/local/bin/estimate-film-grain" echo -e "\t-U Uninstall this from /usr/local/bin/estimate-film-grain" return 0 } check_not_negative_optarg() { OPTARG="$1" if [[ ${OPTARG} != ?(-)+([[:digit:]]) || ${OPTARG} -lt 0 ]]; then echo "${OPTARG} is not a positive integer" usage exit 1 fi } echoerr() { echo -e "$@" 1>&2; } OPTS='o:l:s:h:i:IU' NUM_OPTS="${#OPTS}" # only using -I or -U MIN_OPT=1 # using all MAX_OPT=$NUM_OPTS CALLING_DIR="$(pwd)" test "$#" -lt "$MIN_OPT" && echo "not enough arguments" && usage && exit 1 test "$#" -gt "$MAX_OPT" && echo "too many arguments" && usage && exit 1 while getopts "$OPTS" flag; do case "${flag}" in I) echo "attempting install" sudo ln -sf "$(pwd)/scripts/estimate_fg.sh" \ /usr/local/bin/estimate-film-grain || exit 1 echo "succesfull install" exit 0 ;; U) echo "attempting uninstall" sudo rm /usr/local/bin/estimate-film-grain || exit 1 echo "succesfull uninstall" exit 0 ;; i) if [[ ! -f "${OPTARG}" ]]; then echo "${OPTARG} does not exist" usage exit 1 fi INPUT="${OPTARG}" ;; o) OUTPUT_FILE="${OPTARG}" ;; l) check_not_negative_optarg "${OPTARG}" LOW_GRAIN="${OPTARG}" ;; s) check_not_negative_optarg "${OPTARG}" STEP_GRAIN="${OPTARG}" ;; h) check_not_negative_optarg "${OPTARG}" HIGH_GRAIN="${OPTARG}" ;; *) echo "wrong flags given" usage exit 1 ;; esac done # set default values test ! -n "$LOW_GRAIN" && LOW_GRAIN=0 test ! -n "$STEP_GRAIN" && STEP_GRAIN=5 test ! -n "$HIGH_GRAIN" && HIGH_GRAIN=30 echo "Estimating film grain for $INPUT" echo -e "\tTesting grain from $LOW_GRAIN-$HIGH_GRAIN with $STEP_GRAIN step increments" && sleep 2 # get time in seconds get_duration() { ffmpeg -i "$1" 2>&1 | grep "Duration" | awk '{print $2}' | tr -d , \ | awk -F: '{ print ($1 * 3600) + ($2 * 60) + $3 }' } get_avg_bitrate() { ffprobe -select_streams v:0 "$1" 2>&1 | grep " bitrate: " | cut -d' ' -f8 } # check if test bitrate is within 12% of target bitrate check_bitrate_bounds() { TEST_BITRATE="$1" TARGET_BITRATE="$2" TARGET_DELTA="$(echo "$TARGET_BITRATE * .60" | bc)" DIFF_BITRATE=$((TEST_BITRATE - TARGET_BITRATE)) DIFF_BITRATE="$(echo ${DIFF_BITRATE#-})" echoerr "TEST_BITRATE:\t$TEST_BITRATE" echoerr "TARGET_BITRATE:\t$TARGET_BITRATE" echoerr "TARGET_DELTA:\t$TARGET_DELTA" echoerr "DIFF_BITRATE:\t$DIFF_BITRATE" if [[ "$DIFF_BITRATE" < "$TARGET_DELTA" ]]; then echo "pass" else echo "fail" fi } # global variables SEGMENTS=15 SEGMENT_TIME=4 MAX_SEGMENTS=6 TOTAL_SECONDS="$(get_duration "$INPUT")" INPUT_BITRATE="$(get_avg_bitrate "$INPUT")" SEGMENT_DIR='/tmp/fg_segments' SEGMENTS_LIST="$SEGMENT_DIR/segments_list.txt" OUTPUT_CONCAT="$SEGMENT_DIR/concatenated.mkv" GRAIN_LOG="grain_log.txt" segment_video() { # set number of segments and start times SEGMENT_PERCENTAGE=$((100 / SEGMENTS)) SEGMENT=$SEGMENT_PERCENTAGE START_TIMES=() while [[ $SEGMENT -lt 100 ]] do START_TIME="$(echo "$SEGMENT * $TOTAL_SECONDS / 100" | bc)" START_TIMES+=("$START_TIME") SEGMENT=$((SEGMENT + SEGMENT_PERCENTAGE)) done # split up video into segments based on start times rm -rf "$SEGMENT_DIR" mkdir -p "$SEGMENT_DIR" NUM_SEGMENTS=0 for INDEX in "${!START_TIMES[@]}" do # don't concatenate the last segment if [[ $((INDEX + 1)) == "${#START_TIMES[@]}" ]]; then break fi # only encode the max number of segments if [[ $NUM_SEGMENTS == "$MAX_SEGMENTS" ]]; then return 0 fi START_TIME="${START_TIMES[$INDEX]}" OUTPUT_SEGMENT="$SEGMENT_DIR/segment_$INDEX.mkv" echo "START_TIME: $START_TIME" ffmpeg -ss "$START_TIME" -i "$INPUT" \ -hide_banner -loglevel error -t "$SEGMENT_TIME" \ -map 0:0 -reset_timestamps 1 -c copy "$OUTPUT_SEGMENT" OUTPUT_SEGMENT_BITRATE="$(get_avg_bitrate "$OUTPUT_SEGMENT")" echo "comparing: $OUTPUT_SEGMENT_BITRATE vs $INPUT_BITRATE" CHECK_BOUNDS="$(check_bitrate_bounds "$OUTPUT_SEGMENT_BITRATE" "$INPUT_BITRATE")" if [[ "$CHECK_BOUNDS" == "pass" ]]; then echo "$OUTPUT_SEGMENT is within bitrate bounds" echo "file '$(basename "$OUTPUT_SEGMENT")'" >> "$SEGMENTS_LIST" NUM_SEGMENTS=$((NUM_SEGMENTS + 1)) else echo "$OUTPUT_SEGMENT is not within bitrate bounds" rm "$OUTPUT_SEGMENT" fi done # ffmpeg -f concat -safe 0 -i "$SEGMENTS_LIST" -hide_banner -loglevel error -c copy "$OUTPUT_CONCAT" } get_output_bitrate() { INPUT="$1" BPS="$(ffprobe "$INPUT" 2>&1 | grep BPS | grep -v 'TAGS' | tr -d ' ' | cut -d':' -f2)" echo "scale=3;$BPS / 1000000" | bc -l } encode_segments() { cd "$SEGMENT_DIR" || exit mkdir ./encoded || exit echo > "$GRAIN_LOG" for VIDEO in $(ls segment*.mkv) do echo "$VIDEO" >> "$GRAIN_LOG" for GRAIN in $(seq $LOW_GRAIN $STEP_GRAIN $HIGH_GRAIN) do OUTPUT_VIDEO="encoded/encoded_$VIDEO" encode -i "$VIDEO" -g $GRAIN "$OUTPUT_VIDEO" BITRATE="$(get_output_bitrate "$OUTPUT_VIDEO")" echo -e "\tgrain: $GRAIN, bitrate: $BITRATE" >> "$GRAIN_LOG" done echo >> "$GRAIN_LOG" done test -n "$OUTPUT_FILE" && cp "$GRAIN_LOG" "$CALLING_DIR/$OUTPUT_FILE" less "$GRAIN_LOG" } get_avg_bitrate "$INPUT" segment_video encode_segments