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