summaryrefslogtreecommitdiff
path: root/bin/decode+edit+loop+encode+stream
blob: 3ebb7a160d1451dcbccd4bd2f14e37157804845e (plain)
  1. #!/bin/sh
  2. # FIXME: Handle audio track in main input
  3. # TODO: Handle audio as secondary input
  4. set -e
  5. ##
  6. ## defaults
  7. ##
  8. INPUT=../src/omni/show0/earth.mp4
  9. LOGO=../../content/icon_small.png
  10. RATIO=4:3
  11. ##
  12. ## generic functions
  13. ##
  14. # shellcheck disable=SC2048,SC2059
  15. echo_n() {
  16. printf -- "$*"
  17. }
  18. HOST=${1:-morla}
  19. if [ "$HOST" = "$(hostname --short)" ]; then
  20. IP=127.0.0.1
  21. else
  22. IP=$(host "$HOST" | grep -Po 'address \K\S+')
  23. fi
  24. FIRSTPORT=${2:-5002} # even number - next 3 ports used too
  25. ITERATIONS=${3-0} # endless by default
  26. HEIGHT=384
  27. videosize() { height=$1; width_or_ratio=$2;
  28. case $width_or_ratio in
  29. *:*) num=${width_or_ratio%:*}; den=${width_or_ratio#*:}; width=$((height*num/den));;
  30. *) width=$width_or_ratio;;
  31. esac
  32. echo_n $((height*width))
  33. }
  34. BITS=$(videosize "$HEIGHT" "$RATIO")
  35. # * bitrates and bits in parens based on https://developer.apple.com/library/content/documentation/General/Reference/HLSAuthoringSpec/Requirements.html
  36. # + bits rounded up to include nearby modulo 16 formats
  37. # + best (i.e. high-modulo) 16:9 heights: 216 288 360 432 576 720
  38. # + best (i.e. high-modulo) 4:3 heights: 288 312 384 480 624 816
  39. # * speeds tuned to just below 100% cpu usage for each combination on a multi-core computer
  40. # TODO: Externalize speeds to site-specific configfile
  41. if [ $BITS -le 110592 ]; then # 234p → 97344
  42. VBITRATE=145000
  43. SPEED_VP8=2
  44. elif [ $BITS -le 150528 ]; then # 270p → 129600
  45. VBITRATE=365000
  46. SPEED_VP8=2
  47. elif [ $BITS -le 196608 ]; then # 360p → 172800
  48. VBITRATE=730000
  49. SPEED_VP8=3
  50. elif [ $BITS -le 331776 ]; then # 432p → 331776
  51. VBITRATE=1100000
  52. SPEED_VP8=4
  53. elif [ $BITS -le 589824 ]; then # 540p → 518400
  54. VBITRATE=2000000
  55. SPEED_VP8=5
  56. elif [ $BITS -le 921600 ]; then # 720p → 921600
  57. VBITRATE=3000000
  58. SPEED_VP8=15
  59. fi
  60. encode_vp8() { bitrate=$1; speed=$2;
  61. echo_n "-codec:v vp8 -quality realtime -deadline 1000000 -cpu-used $speed \
  62. -b:v $bitrate -minrate $bitrate -maxrate $bitrate \
  63. -undershoot-pct 95 -bufsize $((6000*bitrate/1000)) -rc_init_occupancy $((4000*bitrate/1000)) \
  64. -max-intra-rate 0 \
  65. -qmin 4 -qmax 56"
  66. }
  67. # based on http://www.webmproject.org/docs/encoder-parameters/#real-time-cbr-encoding-and-streaming
  68. # + loop-in-filter trick based on http://video.stackexchange.com/a/16933
  69. # + scale+watermark trick based on http://stackoverflow.com/a/10937357
  70. # + scale to square pixels as needed
  71. # + Add 1s latency (deadline)
  72. # + Use same RTP payload types as GStreamer
  73. ffmpeg -hide_banner -threads auto -re \
  74. -f lavfi -i "movie=filename=$INPUT:loop=$ITERATIONS, setpts=N/(FRAME_RATE*TB)" \
  75. -i "$LOGO" \
  76. -filter_complex \
  77. "[0:v]format=pix_fmts=yuv420p,scale=iw*sar*$HEIGHT/ih:$HEIGHT,setsar=1[bg];
  78. [bg][1:v]overlay=main_w-overlay_w-20:main_h-overlay_h-20[v]" \
  79. -map '[v]' \
  80. $(encode_vp8 "$VBITRATE" "$SPEED_VP8") \
  81. -f rtp -payload_type 100 "rtp://$IP:$((FIRSTPORT+2))?pkt_size=1200"