summaryrefslogtreecommitdiff
path: root/bin/stream
blob: 9df3b567f65430f117956c06dd754d7070a90993 (plain)
  1. #!/bin/sh
  2. #
  3. # TODO: Set predictive GOP size for WebM (when tested that it works)
  4. set -eu
  5. # TODO: Externalize to site-specific configfile
  6. [ $# -gt 0 ] || set -- morla 5002 -- dvcam ../../content/icon_small.png
  7. exit1() {
  8. echo >&2 "ERROR: $1"
  9. exit 1
  10. }
  11. while [ $# -gt 0 ]; do
  12. case $1 in
  13. --)
  14. shift; break;;
  15. *)
  16. if [ -z "${HOST:-}" ]; then
  17. HOST=$1
  18. elif [ -z "${FIRSTPORT:-}" ]; then
  19. FIRSTPORT=$1
  20. else
  21. exit1 "Too many arguments: Max. 2 about target"
  22. fi
  23. ;;
  24. esac
  25. shift
  26. done
  27. # TODO: Externalize to site-specific configfile
  28. [ $# -gt 0 ] || set -- dvcam ../../content/icon_small.png
  29. AINPUT=
  30. VINPUT=
  31. WINPUT=
  32. XINPUT=
  33. while [ $# -gt 0 ]; do
  34. case $1 in
  35. alsa=*) ALSA=${1#*=}; AINPUT=$((AINPUT+1));;
  36. alsa) ALSA=default; AINPUT=$((AINPUT+1));;
  37. dvcam=*) DVCAM=${1#*=}; DEINT=${DEINT:-yadif}; XINPUT=$((XINPUT+1));;
  38. dvcam) DVCAM=auto; XINPUT=$((XINPUT+1));;
  39. dc=*) IIDC=${1#*=}; VINPUT=$((VINPUT+1));;
  40. dc) IIDC=/dev/fw1; VINPUT=$((VINPUT+1));;
  41. videofile=*) VFILE=${1#*=}; VINPUT=$((VINPUT+1));;
  42. *.ffv1|*.yuv|*.vp8|*.vp9) VFILE=$1; VINPUT=$((VINPUT+1));;
  43. container=*) XFILE=${1#*=}; XINPUT=$((XINPUT+1));;
  44. *.avi|*.dv|*.mkv|*.mov|*.mp4|*.nut|*.ogg|*.ogv|*.webm) XFILE=$1; XINPUT=$((XINPUT+1));;
  45. *.png) LOGO=$1; WINPUT=$((WINPUT+1));;
  46. --) shift; break;;
  47. *) exit1 "Unsupported input: $1";;
  48. esac
  49. shift
  50. done
  51. HOST=${HOST:-127.0.0.1}
  52. if [ "$HOST" = "$(hostname --short)" ]; then
  53. IP=127.0.0.1
  54. else
  55. IP=$(host "$HOST" | grep -Po 'address \K\S+')
  56. fi
  57. [ -n "$AINPUT$VINPUT$XINPUT" ] || exit1 "Too few arguments: Min. 1 A/V source"
  58. [ -z "$AINPUT" ] || [ -z "$VINPUT" ] || [ -z "$XINPUT" ] || exit1 "Too many arguments: Max. 2 A/V sources"
  59. [ -z "$AINPUT" ] || [ $AINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 audio source"
  60. [ -z "$VINPUT" ] || [ $VINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 video source"
  61. [ -z "$WINPUT" ] || [ $WINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 watermark source"
  62. [ -z "$XINPUT" ] || [ $XINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 multimedia source"
  63. [ -n "${NOAUDIO:-}" ] || [ -z "$AINPUT$XINPUT" ] || HASAUDIO=1
  64. [ -n "${NOVIDEO:-}" ] || [ -z "$VINPUT$XINPUT" ] || HASVIDEO=1
  65. [ "$AINPUT$VINPUT$XINPUT" = "1" ] || TWOSOURCES=1
  66. VSTREAMINDEX=1
  67. [ -n "$AINPUT" ] || VSTREAMINDEX=0
  68. FIRSTPORT=${FIRSTPORT:-5002} # even number - next 7 ports used too
  69. ACHANNELS=1
  70. AFRAMERATE_SRC=48000
  71. AFRAMERATE_OPUS=48000
  72. AFRAMERATE_AAC=44100
  73. ABITRATE_OPUS=48000
  74. ABITRATE_AAC=64000
  75. # FIXME: support multiple heights
  76. HEIGHT=270
  77. HEIGHTS_WEBM="$HEIGHT"
  78. HEIGHTS_MPEG="$HEIGHT"
  79. # * height steps and bitrates based on https://developer.apple.com/library/content/documentation/General/Reference/HLSAuthoringSpec/Requirements.html
  80. # * speeds tuned to just below 100% cpu usage for each combination on a multi-core computer
  81. # TODO: adjust height steps or bitrates for 4:3 aspect ratio
  82. # TODO: resolve steps from source height using http://aarmstrong.org/tutorials/aspect-ratios-and-h264
  83. # TODO: Externalize speeds to site-specific configfile
  84. if [ $HEIGHT -le 234 ]; then
  85. VBITRATE=145000
  86. SPEED_X264=fast; SPEED_X264_ALONE=fast
  87. SPEED_VP8=3; SPEED_VP8_ALONE=2
  88. elif [ $HEIGHT -le 270 ]; then
  89. VBITRATE=365000
  90. SPEED_X264=faster; SPEED_X264_ALONE=fast
  91. SPEED_VP8=4; SPEED_VP8_ALONE=2
  92. elif [ $HEIGHT -le 360 ]; then
  93. VBITRATE=730000
  94. SPEED_X264=veryfast; SPEED_X264_ALONE=fast
  95. SPEED_VP8=5; SPEED_VP8_ALONE=3
  96. elif [ $HEIGHT -le 432 ]; then
  97. VBITRATE=1100000
  98. SPEED_X264=ultrafast; SPEED_X264_ALONE=fast
  99. SPEED_VP8=8; SPEED_VP8_ALONE=4
  100. elif [ $HEIGHT -le 540 ]; then
  101. VBITRATE=2000000
  102. SPEED_X264=toofast; SPEED_X264_ALONE=veryfast
  103. SPEED_VP8_ALONE=5
  104. elif [ $HEIGHT -le 720 ]; then
  105. VBITRATE=3000000
  106. SPEED_X264=toofast; SPEED_X264_ALONE=ultrafast
  107. SPEED_VP8_ALONE=15
  108. fi
  109. [ toofast != "$SPEED_X264" ] || HEIGHTS_MPEG=
  110. [ -n "$HEIGHTS_MPEG" ] || SPEED_VP8="$SPEED_VP8_ALONE"
  111. [ -n "$HEIGHTS_WEBM" ] || SPEED_X264="$SPEED_X264_ALONE"
  112. # shellcheck disable=SC2048,SC2059
  113. echo_n() {
  114. printf "$*"
  115. }
  116. # shellcheck disable=SC2048,SC2059
  117. printf_each() {
  118. skel=$1; shift
  119. for string in $*; do
  120. printf "$skel" "$string"
  121. done
  122. }
  123. uniqwords() {
  124. echo "$@" | tr ' ' '\012' | sort -u
  125. }
  126. valuedargcount() {
  127. nonempty=
  128. while [ $# -gt 0 ]; do
  129. [ -z "$1" ] || nonempty=$((nonempty+1))
  130. shift
  131. done
  132. echo_n "$nonempty"
  133. }
  134. HEIGHTS=$(uniqwords "$HEIGHTS_WEBM $HEIGHTS_MPEG")
  135. [ -z "${SAVEDIR:-}" ] || SAVESTEM="${SAVEDIR:-}/$(date +%Y%m%d-%H%M%S)"
  136. [ -z "${SAVEDIR:-}" ] || export FFREPORT=file="$SAVESTEM.log"
  137. filter_split_height() {
  138. heightcount=$(echo "$HEIGHTS" | wc --words)
  139. echo_n "[$VSTREAMINDEX:v]${DEINT:+$DEINT,}split=$heightcount"
  140. printf_each '[s%s]' "$HEIGHTS"
  141. }
  142. filter_scale() { outstem=${1:-v};
  143. for height in $HEIGHTS; do
  144. echo_n "[s$height]scale=iw*sar*$height/ih:$height,setsar=1[$outstem$height]"
  145. done
  146. }
  147. filter_watermark() {
  148. for height in $HEIGHTS; do
  149. echo_n "[bg$height][$((VSTREAMINDEX+1)):v]overlay=main_w-overlay_w-20:main_h-overlay_h-20[v$height]"
  150. done
  151. }
  152. filter_split_codec() { heights_webm=$1; heights_mpeg=$2;
  153. codeccount=$(valuedargcount "$@")
  154. printf_each "[v%s]split=$codeccount$(printf_each '[v%swebm]' "$heights_webm")$(printf_each '[v%smpeg]' "$heights_mpeg")" "$HEIGHTS"
  155. }
  156. tee_rtp() { stream=$1; pt=$2; port=$3;
  157. echo_n "[select=\'$stream\':f=rtp:payload_type=$pt]rtp://$IP:$port?pkt_size=1200"
  158. }
  159. # * scale+watermark trick based on http://stackoverflow.com/a/10937357
  160. # + scale to square pixels as needed
  161. # * routing based on http://trac.ffmpeg.org/wiki/Creating%20multiple%20outputs#Teepseudo-muxer
  162. # * VP8 encoding based on http://www.webmproject.org/docs/encoder-parameters/#real-time-cbr-encoding-and-streaming
  163. # + Add 1s latency (deadline)
  164. # * Predictive GOP size preserving scene-change GOP based on https://superuser.com/a/1098329
  165. # * Use same RTP payload types as GStreamer
  166. ffmpeg -hide_banner -threads auto \
  167. ${ALSA:+-f alsa -sample_rate "$AFRAMERATE_SRC" -channels "$ACHANNELS" -thread_queue_size 2048 -i "$ALSA"} \
  168. ${DVCAM:+-f iec61883 -thread_queue_size 64 -i $DVCAM} \
  169. ${XFILE:+-re${SEEK:+ -ss $SEEK} -i "$XFILE"} \
  170. ${IIDC:+-f libdc1394 -video_size 640x480 -framerate 15 -thread_queue_size 256 -i "$IIDC"} \
  171. ${VFILE:+-re${SEEK:+ -ss $SEEK} -i "$VFILE"} \
  172. ${LOGO:+-i "$LOGO"} \
  173. ${HASVIDEO:+-filter_complex \
  174. "$(filter_split_height);
  175. $(filter_scale "${WINPUT:+bg}")${WINPUT:+${LOGO:+;
  176. $(filter_watermark)}};
  177. $(filter_split_codec "$HEIGHTS_WEBM" "$HEIGHTS_MPEG")" } \
  178. ${SAVEDIR:+\
  179. ${DVCAM:+-map $((0${ALSA:++1})) \
  180. -codec copy -f dv "$SAVESTEM.dv" }} \
  181. ${HEIGHTS_WEBM:+\
  182. ${HASAUDIO:+-map '0:a' }${HASVIDEO:+$(printf_each ' -map [v%swebm]' "$HEIGHTS_WEBM") } -shortest \
  183. ${HASAUDIO:+\
  184. -codec:a libopus -ac "$ACHANNELS" -ar "$AFRAMERATE_OPUS" -b:a "$ABITRATE_OPUS" } \
  185. ${HASVIDEO:+\
  186. -pix_fmt yuv420p \
  187. -codec:v vp8 -quality realtime -deadline 1000000 -cpu-used "$SPEED_VP8" \
  188. -b:v "$VBITRATE" -minrate "$VBITRATE" -maxrate "$VBITRATE" \
  189. -undershoot-pct 95 -bufsize $((6000*VBITRATE/1000)) -rc_init_occupancy $((4000*VBITRATE/1000)) \
  190. -max-intra-rate 0 \
  191. -qmin 4 -qmax 56 } \
  192. -f tee \
  193. "${HASAUDIO:+\
  194. $(tee_rtp a 111 "$FIRSTPORT")|\
  195. }${HASVIDEO:+\
  196. $(tee_rtp v 100 $((FIRSTPORT+2)))}" } \
  197. ${HEIGHTS_MPEG:+\
  198. ${HASAUDIO:+-map '0:a' }${HASVIDEO:+$(printf_each ' -map [v%smpeg]' "$HEIGHTS_MPEG") } -shortest \
  199. ${HASAUDIO:+\
  200. -codec:a aac -strict experimental -ac "$ACHANNELS" -ar "$AFRAMERATE_AAC" -b:a $((ACHANNELS*ABITRATE_AAC)) } \
  201. ${HASVIDEO:+\
  202. -pix_fmt yuv420p \
  203. -codec:v libx264 -preset "$SPEED_X264" -tune zerolatency \
  204. -maxrate "$VBITRATE" -bufsize "$((VBITRATE*2))" -crf 23 } \
  205. -force_key_frames 'expr:gte(t,n_forced*2)' \
  206. -f rtp_mpegts rtp://$IP:10000 }