summaryrefslogtreecommitdiff
path: root/bin/stream
blob: 452f47ed25164b1191b9bca492b93d11490fb50d (plain)
  1. #!/bin/sh
  2. set -e
  3. # TODO: Externalize to site-specific configfile
  4. [ $# -gt 0 ] || set -- morla 5002 -- dvcam ../../content/icon_small.png
  5. exit1() {
  6. echo >&2 "ERROR: $1"
  7. exit 1
  8. }
  9. while [ $# -gt 0 ]; do
  10. case $1 in
  11. --)
  12. shift; break;;
  13. *)
  14. if [ -z "$HOST" ]; then
  15. HOST=$1
  16. elif [ -z "$FIRSTPORT" ]; then
  17. FIRSTPORT=$1
  18. else
  19. exit1 "Too many arguments: Max. 2 about target"
  20. fi
  21. ;;
  22. esac
  23. shift
  24. done
  25. # TODO: Externalize to site-specific configfile
  26. [ $# -gt 0 ] || set -- dvcam ../../content/icon_small.png
  27. while [ $# -gt 0 ]; do
  28. case $1 in
  29. alsa=*) ALSA=${1#*=}; AINPUT=$((AINPUT+1));;
  30. alsa) ALSA=default; AINPUT=$((AINPUT+1));;
  31. dvcam=*) DVCAM=${1#*=}; XINPUT=$((XINPUT+1));;
  32. dvcam) DVCAM=auto; XINPUT=$((XINPUT+1));;
  33. dc=*) IIDC=${1#*=}; VINPUT=$((VINPUT+1));;
  34. dc) IIDC=/dev/fw1; VINPUT=$((VINPUT+1));;
  35. videofile=*) VFILE=${1#*=}; VINPUT=$((VINPUT+1));;
  36. *.ffv1|*.yuv|*.vp8|*.vp9) VFILE=$1; VINPUT=$((VINPUT+1));;
  37. container=*) XFILE=${1#*=}; XINPUT=$((XINPUT+1));;
  38. *.avi|*.mkv|*.mov|*.mp4|*.ogg|*.ogv|*.webm) XFILE=$1; XINPUT=$((XINPUT+1));;
  39. *.png) LOGO=$1; WINPUT=$((WINPUT+1));;
  40. --) shift; break;;
  41. *) exit1 "Unsupported input: $1";;
  42. esac
  43. shift
  44. done
  45. HOST=${HOST:-127.0.0.1}
  46. if [ "$HOST" = "$(hostname --short)" ]; then
  47. IP=127.0.0.1
  48. else
  49. IP=$(host "$HOST" | grep -Po 'address \K\S+')
  50. fi
  51. [ -n "$AINPUT$VINPUT$XINPUT" ] || exit1 "Too few arguments: Min. 1 A/V source"
  52. [ -z "$AINPUT" ] || [ -z "$VINPUT" ] || [ -z "$XINPUT" ] || exit1 "Too many arguments: Max. 2 A/V sources"
  53. [ -z "$AINPUT" ] || [ $AINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 audio source"
  54. [ -z "$VINPUT" ] || [ $VINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 video source"
  55. [ -z "$WINPUT" ] || [ $WINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 watermark source"
  56. [ -z "$XINPUT" ] || [ $XINPUT -eq 1 ] || exit1 "Too many arguments: Max. 1 multimedia source"
  57. [ -n "$NOAUDIO" ] || [ -z "$AINPUT$XINPUT" ] || HASAUDIO=1
  58. [ -n "$NOVIDEO" ] || [ -z "$VINPUT$XINPUT" ] || HASVIDEO=1
  59. [ "$AINPUT$VINPUT$XINPUT" = "1" ] || TWOSOURCES=1
  60. VSTREAMINDEX=1
  61. [ -n "$AINPUT" ] || VSTREAMINDEX=0
  62. FIRSTPORT=${FIRSTPORT:-5002} # even number - next 7 ports used too
  63. ACHANNELS=2
  64. AFRAMERATE=48000
  65. ABITRATE=32000
  66. HEIGHT=360
  67. VBITRATE=256000
  68. # FIXME: support multiple heights
  69. HEIGHTCOUNT=1
  70. height=$HEIGHT
  71. MKWEBM="[v${height}webm]"
  72. MKMPEG="[v${height}mpeg]"
  73. [ -z "$MKWEBM" ] || VCODECCOUNT=$((VCODECCOUNT+1))
  74. [ -z "$MKMPEG" ] || VCODECCOUNT=$((VCODECCOUNT+1))
  75. SAVESTEM="$SAVEDIR/$(date +%Y%m%d-%H%M%S)"
  76. ${SAVEDIR:+export FFREPORT=file="$SAVESTEM.log"}
  77. # * scale+watermark trick based on http://stackoverflow.com/a/10937357
  78. # * routing based on http://trac.ffmpeg.org/wiki/Creating%20multiple%20outputs#Teepseudo-muxer
  79. # * VP8 encoding based on http://www.webmproject.org/docs/encoder-parameters/#real-time-cbr-encoding-and-streaming
  80. # + Add 1s latency (deadline)
  81. # * Use same RTP payload types as GStreamer
  82. ffmpeg -hide_banner -threads auto -re \
  83. ${ALSA:+-f alsa -sample_rate "$AFRAMERATE" -channels "$ACHANNELS" -thread_queue_size 1024 -i "$ALSA"} \
  84. ${DVCAM:+-f iec61883 -thread_queue_size 64 -i $DVCAM} \
  85. ${XFILE:+-i "$XFILE"} \
  86. ${IIDC:+-f libdc1394 -video_size 640x480 -framerate 15 -thread_queue_size 256 -i "$IIDC"} \
  87. ${VFILE:+-i "$VFILE"} \
  88. ${LOGO:+-i "$LOGO"} \
  89. ${HASVIDEO:+-filter_complex \
  90. "[$VSTREAMINDEX:v]split=$HEIGHTCOUNT[s$height];
  91. [s$height]scale=-2:$height${WATERMARK:-[v$height]}${WATERMARK:+[bg$height]${LOGO:+;
  92. [bg$height][$((VSTREAMINDEX+1)):v]overlay=main_w-overlay_w-20:main_h-overlay_h-20[v$height]}};
  93. [v$height]split=$VCODECCOUNT$MKWEBM$MKMPEG" } \
  94. ${SAVEDIR:+-map '0:' ${TWOSOURCES:+-map '1:'} \
  95. -codec copy \
  96. -f segment -segment_format matroska -segment_format_options live=1:reserve_index_space=512kB \
  97. -segment_list "$SAVESTEM.ffconcat" -segment_list_flags live \
  98. -segment_time 600 -segment_atclocktime 1 -strftime 1 "$SAVESTEM-%H%M.mkv"} \
  99. ${MKWEBM:+\
  100. ${HASAUDIO:+-map '0:a' } ${HASVIDEO:+-map "[v${height}webm]" } \
  101. ${HASAUDIO:+\
  102. -codec:a libopus -ac "$ACHANNELS" -ar "$AFRAMERATE" -b:a "$ABITRATE" } \
  103. ${HASVIDEO:+\
  104. -pix_fmt yuv420p \
  105. -codec:v vp8 -quality realtime -deadline 1000000 -cpu-used 15 \
  106. -b:v "$VBITRATE" -minrate "$VBITRATE" -maxrate "$VBITRATE" \
  107. -undershoot-pct 95 -bufsize $((6000*VBITRATE/1000)) -rc_init_occupancy $((4000*VBITRATE/1000)) \
  108. -max-intra-rate 0 \
  109. -qmin 4 -qmax 56 } \
  110. -f tee \
  111. "${HASAUDIO:+[select=\'a\':f=rtp:payload_type=111]rtp://$IP:$FIRSTPORT?pkt_size=1200| \
  112. }${HASVIDEO:+[select=\'v\':f=rtp:payload_type=100]rtp://$IP:$((FIRSTPORT+2))?pkt_size=1200}" } \
  113. ${MKMPEG:+\
  114. ${HASAUDIO:+-map '0:a' } ${HASVIDEO:+-map "[v${height}mpeg]" } \
  115. ${HASAUDIO:+\
  116. -codec:a aac -strict experimental -ac "$ACHANNELS" -ar 44100 -b:a $((ACHANNELS*64))k } \
  117. ${HASVIDEO:+\
  118. -pix_fmt yuv420p \
  119. -codec:v libx264 -tune zerolatency -preset ultrafast \
  120. -maxrate "$VBITRATE" -bufsize "$((VBITRATE*2))" -crf 23 } \
  121. -f tee \
  122. "${HASAUDIO:+[select=\'a\':f=rtp:payload_type=96]rtp://$IP:$((FIRSTPORT+4))?pkt_size=1200| \
  123. }${HASVIDEO:+[select=\'v\':f=rtp:payload_type=97]rtp://$IP:$((FIRSTPORT+6))?pkt_size=1200}" }