mirror of https://bitbucket.org/ausocean/av.git
Fixed errors and bugs and it seems like flv packetization is working
This commit is contained in:
parent
ec796bd9ae
commit
c46a8d8f08
65
flv/FLV.go
65
flv/FLV.go
|
@ -2,6 +2,7 @@ package flv
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"../tools"
|
"../tools"
|
||||||
|
"fmt"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -12,7 +13,11 @@ const (
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
videoTagType = 9
|
VideoTagType = 9
|
||||||
|
KeyFrameType = 1
|
||||||
|
H264 = 7
|
||||||
|
AVCNALU = 1
|
||||||
|
DataHeaderLength = 5
|
||||||
)
|
)
|
||||||
|
|
||||||
type Header struct {
|
type Header struct {
|
||||||
|
@ -20,13 +25,14 @@ type Header struct {
|
||||||
VideoFlag bool
|
VideoFlag bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *Header) toByteSlice() []byte {
|
func (h *Header) ToByteSlice() (output []byte) {
|
||||||
output = make([]byte, 0, headerLength)
|
output = make([]byte, 0, headerLength)
|
||||||
output = append(output, []byte{0x46, 0x4C, 0x56,
|
output = append(output, []byte{0x46, 0x4C, 0x56,
|
||||||
version,
|
version,
|
||||||
0x00 | tools.boolToByte(h.audioFlag)<<3 | tools.boolToByte(h.videoFlag),
|
0x00 | tools.BoolToByte(h.AudioFlag)<<2 | tools.BoolToByte(h.VideoFlag),
|
||||||
0x00, 0x00, 0x00, byte(72),
|
0x00, 0x00, 0x00, byte(9),
|
||||||
}...)
|
}...)
|
||||||
|
fmt.Println(output)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,41 +42,48 @@ type VideoTag struct {
|
||||||
DataSize uint32
|
DataSize uint32
|
||||||
Timestamp uint32
|
Timestamp uint32
|
||||||
TimestampExtended uint32
|
TimestampExtended uint32
|
||||||
|
FrameType byte
|
||||||
|
Codec byte
|
||||||
|
PacketType byte
|
||||||
|
CompositionTime uint32
|
||||||
Data []byte
|
Data []byte
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *VideoTag) toByteSlice() (output []byte) {
|
func (t *VideoTag) ToByteSlice() (output []byte) {
|
||||||
output = make([]byte, 0, maxVideoTagSize)
|
output = make([]byte, 0, maxVideoTagSize)
|
||||||
output = append(output, []byte{
|
output = append(output, []byte{
|
||||||
byte(t.prevTagSize >> 24),
|
byte(t.PrevTagSize >> 24),
|
||||||
byte(t.prevTagSize >> 16),
|
byte(t.PrevTagSize >> 16),
|
||||||
byte(t.prevTagSize >> 8),
|
byte(t.PrevTagSize >> 8),
|
||||||
byte(t.prevTagSize),
|
byte(t.PrevTagSize),
|
||||||
byte(t.tageType),
|
byte(t.TagType),
|
||||||
byte(t.dataSize >> 16),
|
byte(t.DataSize >> 16),
|
||||||
byte(t.dataSize >> 8),
|
byte(t.DataSize >> 8),
|
||||||
byte(t.dataSize),
|
byte(t.DataSize),
|
||||||
byte(t.timeStamp >> 16),
|
byte(t.Timestamp >> 16),
|
||||||
byte(t.timeStamp >> 8),
|
byte(t.Timestamp >> 8),
|
||||||
byte(t.timeStamp),
|
byte(t.Timestamp),
|
||||||
byte(t.timestampExtended),
|
byte(t.TimestampExtended),
|
||||||
0x00, 0x00, 0x00,
|
0x00, 0x00, 0x00,
|
||||||
|
byte(t.FrameType << 4) | byte(t.Codec),
|
||||||
|
t.PacketType,
|
||||||
|
byte(t.CompositionTime >> 16),byte(t.CompositionTime >> 8),byte(t.CompositionTime),
|
||||||
}...)
|
}...)
|
||||||
output = append(output, data...)
|
output = append(output, t.Data...)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
type AudioTag struct {
|
type AudioTag struct {
|
||||||
soundFormat uint8
|
SoundFormat uint8
|
||||||
soundRate uint8
|
SoundRate uint8
|
||||||
soundSize uint8
|
SoundSize uint8
|
||||||
soundType uint8
|
SoundType uint8
|
||||||
data []byte
|
Data []byte
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *AudioTage) toByteSlice() (output []byte) {
|
func (t *AudioTag) ToByteSlice() (output []byte) {
|
||||||
output = make([]byte, 0, maxAudioTagSize)
|
output = make([]byte, 0, maxAudioTagSize)
|
||||||
output = append(output, byte(soundFormat<<4)|byte(soundRate<<2)|byte(soundSize<<1)|byte(soundType))
|
output = append(output, byte(t.SoundFormat<<4)|byte(t.SoundRate<<2)|byte(t.SoundSize<<1)|byte(t.SoundType))
|
||||||
output = append(output, data...)
|
output = append(output, t.Data...)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,23 @@
|
||||||
package generator
|
package generator
|
||||||
|
|
||||||
|
import (
|
||||||
|
"../flv"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
inputChanLength = 1000
|
||||||
|
outputChanLength = 1000
|
||||||
|
)
|
||||||
|
|
||||||
type flvGenerator struct {
|
type flvGenerator struct {
|
||||||
fps uint
|
fps uint
|
||||||
inputChan chan []byte
|
inputChan chan []byte
|
||||||
outputChan chan []byte
|
outputChan chan []byte
|
||||||
header Header
|
audioFlag bool
|
||||||
|
videoFlag bool
|
||||||
|
lastTagSize int
|
||||||
|
currentTimestamp uint32
|
||||||
|
header flv.Header
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *flvGenerator)GetInputChan() chan []byte {
|
func (g *flvGenerator)GetInputChan() chan []byte {
|
||||||
|
@ -15,10 +28,15 @@ func (g *flvGenerator)GetOutputChan() chan []byte {
|
||||||
return g.outputChan
|
return g.outputChan
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewFlvGenerator() (g *flvGenerator) {
|
func NewFlvGenerator(audio bool, video bool, fps uint) (g *flvGenerator) {
|
||||||
g = new(flvGenerator)
|
g = new(flvGenerator)
|
||||||
g.timestamp = 0
|
g.fps = fps
|
||||||
|
g.audioFlag = audio
|
||||||
|
g.videoFlag = video
|
||||||
|
g.currentTimestamp = 0
|
||||||
g.lastTagSize = 0
|
g.lastTagSize = 0
|
||||||
|
g.inputChan = make(chan []byte, inputChanLength)
|
||||||
|
g.outputChan = make(chan []byte, outputChanLength)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -28,36 +46,42 @@ func (g *flvGenerator) Start(){
|
||||||
|
|
||||||
func (g *flvGenerator) GenHeader(){
|
func (g *flvGenerator) GenHeader(){
|
||||||
header := flv.Header{
|
header := flv.Header{
|
||||||
AudioFlag: true,
|
AudioFlag: g.audioFlag,
|
||||||
VideoFlag: true,
|
VideoFlag: g.videoFlag,
|
||||||
}
|
}
|
||||||
g.outputChan <- header.toByteSlice()
|
g.outputChan <- header.ToByteSlice()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *flvGenerator) getNextTimestamp() (timestamp uint32){
|
func (g *flvGenerator) getNextTimestamp() (timestamp uint32){
|
||||||
timestamp = g.currentTimestamp
|
timestamp = g.currentTimestamp
|
||||||
g.currentTimeStamp += 100*time.Millisecond() / g.fps
|
g.currentTimestamp += uint32(1000) / uint32(g.fps)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *flvGenerator) ResetTimestamp() {
|
func (g *flvGenerator) ResetTimestamp() {
|
||||||
g.timestamp = 0
|
g.currentTimestamp = 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *tsGenerator) generate() {
|
func (g *flvGenerator) generate() {
|
||||||
g.GenHeader()
|
g.GenHeader()
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case videoFrame := <-g.inputChan
|
case videoFrame := <-g.inputChan:
|
||||||
tag := VideoTage{
|
tag := flv.VideoTag{
|
||||||
PrevTagSize: g.lastTagSize,
|
PrevTagSize: uint32(g.lastTagSize),
|
||||||
TagType: flv.videoTagType,
|
TagType: uint8(flv.VideoTagType),
|
||||||
DataSize: len(videoFrame),
|
DataSize: uint32(len(videoFrame)) + flv.DataHeaderLength,
|
||||||
Timestamp: g.getNextTimestamp(),
|
Timestamp: g.getNextTimestamp(),
|
||||||
TimestampExtended: 0,
|
TimestampExtended: 0,
|
||||||
Data: videoFrame
|
FrameType: flv.KeyFrameType,
|
||||||
|
Codec: flv.H264,
|
||||||
|
PacketType: flv.AVCNALU,
|
||||||
|
CompositionTime: 0,
|
||||||
|
Data: videoFrame,
|
||||||
}
|
}
|
||||||
g.outputChan<-tag.toByteSlice()
|
tagAsByteSlice := tag.ToByteSlice()
|
||||||
|
g.lastTagSize = len(tagAsByteSlice)
|
||||||
|
g.outputChan<-tagAsByteSlice
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,22 +28,8 @@ LICENSE
|
||||||
|
|
||||||
package generator
|
package generator
|
||||||
|
|
||||||
import (
|
|
||||||
_"fmt"
|
|
||||||
_"os"
|
|
||||||
//"bitbucket.org/ausocean/av/mpegts"
|
|
||||||
//"bitbucket.org/ausocean/av/pes"
|
|
||||||
//"bitbucket.org/ausocean/av/tools"
|
|
||||||
//"bitbucket.org/ausocean/av/rtp"
|
|
||||||
"../mpegts"
|
|
||||||
"../pes"
|
|
||||||
"../tools"
|
|
||||||
"../rtp"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Generator interface {
|
type Generator interface {
|
||||||
GetInputChan() chan []byte
|
GetInputChan() chan []byte
|
||||||
GetOutputChan() <-chan *mpegts.MpegTsPacket
|
GetOutputChan() chan []byte
|
||||||
Start()
|
Start()
|
||||||
Stop()
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,18 @@
|
||||||
package generator
|
package generator
|
||||||
|
|
||||||
|
import (
|
||||||
|
_"fmt"
|
||||||
|
_"os"
|
||||||
|
//"bitbucket.org/ausocean/av/mpegts"
|
||||||
|
//"bitbucket.org/ausocean/av/pes"
|
||||||
|
//"bitbucket.org/ausocean/av/tools"
|
||||||
|
//"bitbucket.org/ausocean/av/rtp"
|
||||||
|
"../mpegts"
|
||||||
|
"../pes"
|
||||||
|
"../tools"
|
||||||
|
"../rtp"
|
||||||
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
PatTable = []byte{0, 0, 176, 13, 0, 1, 193, 0, 0, 0, 1, 240, 0, 42, 177, 4, 178, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,}
|
PatTable = []byte{0, 0, 176, 13, 0, 1, 193, 0, 0, 0, 1, 240, 0, 42, 177, 4, 178, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,}
|
||||||
|
|
||||||
|
@ -14,10 +27,8 @@ const (
|
||||||
)
|
)
|
||||||
|
|
||||||
type tsGenerator struct {
|
type tsGenerator struct {
|
||||||
TsChan <-chan *mpegts.MpegTsPacket
|
rtpInputChan chan rtp.RtpPacket
|
||||||
tsChan chan<- *mpegts.MpegTsPacket
|
outputChan chan []byte
|
||||||
InputChan chan<- rtp.RtpPacket
|
|
||||||
inputChan <-chan rtp.RtpPacket
|
|
||||||
nalInputChan chan []byte
|
nalInputChan chan []byte
|
||||||
currentTsPacket *mpegts.MpegTsPacket
|
currentTsPacket *mpegts.MpegTsPacket
|
||||||
payloadByteChan chan byte
|
payloadByteChan chan byte
|
||||||
|
@ -33,18 +44,14 @@ func (g *tsGenerator)GetInputChan() chan []byte {
|
||||||
return g.nalInputChan
|
return g.nalInputChan
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *tsGenerator)GetOutputChan() <-chan *mpegts.MpegTsPacket {
|
func (g *tsGenerator)GetOutputChan() chan []byte {
|
||||||
return g.TsChan
|
return g.outputChan
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTsGenerator(fps uint) (g *tsGenerator) {
|
func NewTsGenerator(fps uint) (g *tsGenerator) {
|
||||||
g = new(tsGenerator)
|
g = new(tsGenerator)
|
||||||
tsChan := make(chan *mpegts.MpegTsPacket, 100)
|
g.outputChan = make(chan []byte, 100)
|
||||||
g.TsChan = tsChan
|
g.rtpInputChan = make(chan rtp.RtpPacket, 100)
|
||||||
g.tsChan = tsChan
|
|
||||||
inputChan := make(chan rtp.RtpPacket, 100)
|
|
||||||
g.InputChan = inputChan
|
|
||||||
g.inputChan = inputChan
|
|
||||||
g.nalInputChan = make(chan []byte, 10000)
|
g.nalInputChan = make(chan []byte, 10000)
|
||||||
g.currentCC = 0
|
g.currentCC = 0
|
||||||
g.fps = fps
|
g.fps = fps
|
||||||
|
@ -80,7 +87,7 @@ func (g *tsGenerator) generate() {
|
||||||
var rtpBuffer [](*rtp.RtpPacket)
|
var rtpBuffer [](*rtp.RtpPacket)
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case rtpPacket := <-g.inputChan:
|
case rtpPacket := <-g.rtpInputChan:
|
||||||
rtpBuffer = append(rtpBuffer, &rtpPacket)
|
rtpBuffer = append(rtpBuffer, &rtpPacket)
|
||||||
if len(rtpBuffer) > 2 {
|
if len(rtpBuffer) > 2 {
|
||||||
// if there's something weird going on with sequence numbers then
|
// if there's something weird going on with sequence numbers then
|
||||||
|
@ -189,7 +196,9 @@ func (g *tsGenerator) generate() {
|
||||||
AFC: 1,
|
AFC: 1,
|
||||||
Payload: PatTable,
|
Payload: PatTable,
|
||||||
}
|
}
|
||||||
g.tsChan <- &patPkt
|
|
||||||
|
patPktAsByteSlice, _ := patPkt.ToByteSlice()
|
||||||
|
g.outputChan <- patPktAsByteSlice
|
||||||
|
|
||||||
// Create pmt table and send off
|
// Create pmt table and send off
|
||||||
pmtPkt := mpegts.MpegTsPacket{
|
pmtPkt := mpegts.MpegTsPacket{
|
||||||
|
@ -199,12 +208,14 @@ func (g *tsGenerator) generate() {
|
||||||
AFC: 1,
|
AFC: 1,
|
||||||
Payload: PmtTable,
|
Payload: PmtTable,
|
||||||
}
|
}
|
||||||
g.tsChan <- &pmtPkt
|
pmtPktAsByteSlice, _ := pmtPkt.ToByteSlice()
|
||||||
|
g.outputChan <- pmtPktAsByteSlice
|
||||||
|
|
||||||
pkt.PCR = g.genPcr()
|
pkt.PCR = g.genPcr()
|
||||||
pusi = false
|
pusi = false
|
||||||
}
|
}
|
||||||
|
pktAsBytelice, _ := pkt.ToByteSlice()
|
||||||
g.tsChan <- &pkt
|
g.outputChan<-pktAsBytelice
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,4 +26,4 @@ LICENSE
|
||||||
along with revid in gpl.txt. If not, see [GNU licenses](http://www.gnu.org/licenses).
|
along with revid in gpl.txt. If not, see [GNU licenses](http://www.gnu.org/licenses).
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package tsgenerator
|
package generator
|
||||||
|
|
|
@ -1,5 +1,11 @@
|
||||||
package parser
|
package parser
|
||||||
|
|
||||||
|
import (
|
||||||
|
//"bitbucket.org/ausocean/av/itut"
|
||||||
|
"../itut"
|
||||||
|
_"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
type h264Parser struct {
|
type h264Parser struct {
|
||||||
inputBuffer []byte
|
inputBuffer []byte
|
||||||
isParsing bool
|
isParsing bool
|
||||||
|
|
|
@ -1,5 +1,10 @@
|
||||||
package parser
|
package parser
|
||||||
|
|
||||||
|
import (
|
||||||
|
//"bitbucket.org/ausocean/av/itut"
|
||||||
|
_"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
type mjpegParser struct {
|
type mjpegParser struct {
|
||||||
inputBuffer []byte
|
inputBuffer []byte
|
||||||
isParsing bool
|
isParsing bool
|
||||||
|
|
|
@ -29,7 +29,6 @@ package parser
|
||||||
|
|
||||||
import (
|
import (
|
||||||
//"bitbucket.org/ausocean/av/itut"
|
//"bitbucket.org/ausocean/av/itut"
|
||||||
"../itut"
|
|
||||||
"log"
|
"log"
|
||||||
"sync"
|
"sync"
|
||||||
_"fmt"
|
_"fmt"
|
||||||
|
|
|
@ -1,5 +1,17 @@
|
||||||
package revid
|
package revid
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
//"bitbucket.org/ausocean/av/parser"
|
||||||
|
//"bitbucket.org/ausocean/av/tsgenerator"
|
||||||
|
|
||||||
|
//"bitbucket.org/ausocean/av/ringbuffer"
|
||||||
|
//"bitbucket.org/ausocean/utils/smartLogger"
|
||||||
|
"../../utils/smartLogger"
|
||||||
|
)
|
||||||
|
|
||||||
// Config provides parameters relevant to a revid instance. A new config must
|
// Config provides parameters relevant to a revid instance. A new config must
|
||||||
// be passed to the constructor.
|
// be passed to the constructor.
|
||||||
type Config struct {
|
type Config struct {
|
||||||
|
@ -7,7 +19,7 @@ type Config struct {
|
||||||
InputCodec uint8
|
InputCodec uint8
|
||||||
Output uint8
|
Output uint8
|
||||||
RtmpEncodingMethod uint8
|
RtmpEncodingMethod uint8
|
||||||
FramesPerClip uint
|
FramesPerClip int
|
||||||
RtmpUrl string
|
RtmpUrl string
|
||||||
Bitrate string
|
Bitrate string
|
||||||
OutputFileName string
|
OutputFileName string
|
||||||
|
@ -30,7 +42,7 @@ const (
|
||||||
Rtp = 2
|
Rtp = 2
|
||||||
H264Codec = 3
|
H264Codec = 3
|
||||||
File = 4
|
File = 4
|
||||||
HttpOut = 5
|
Http = 5
|
||||||
H264 = 6
|
H264 = 6
|
||||||
Mjpeg = 7
|
Mjpeg = 7
|
||||||
None = 8
|
None = 8
|
||||||
|
@ -38,7 +50,7 @@ const (
|
||||||
Rtmp = 10
|
Rtmp = 10
|
||||||
Ffmpeg = 11
|
Ffmpeg = 11
|
||||||
Revid = 12
|
Revid = 12
|
||||||
Flv = 13
|
Flv = 13
|
||||||
)
|
)
|
||||||
|
|
||||||
// Default config settings
|
// Default config settings
|
||||||
|
@ -93,7 +105,7 @@ func (config *Config) Validate(r *revidInst) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
switch config.Output {
|
switch config.Output {
|
||||||
case HttpOut:
|
case Http:
|
||||||
case File:
|
case File:
|
||||||
case Rtmp:
|
case Rtmp:
|
||||||
switch config.RtmpEncodingMethod {
|
switch config.RtmpEncodingMethod {
|
||||||
|
@ -108,7 +120,7 @@ func (config *Config) Validate(r *revidInst) error {
|
||||||
}
|
}
|
||||||
case NothingDefined:
|
case NothingDefined:
|
||||||
r.Log(Warning, "No output defined, defaulting to httpOut!")
|
r.Log(Warning, "No output defined, defaulting to httpOut!")
|
||||||
config.Output = HttpOut
|
config.Output = Http
|
||||||
default:
|
default:
|
||||||
return errors.New("Bad output type defined in config!")
|
return errors.New("Bad output type defined in config!")
|
||||||
}
|
}
|
||||||
|
@ -116,6 +128,7 @@ func (config *Config) Validate(r *revidInst) error {
|
||||||
switch config.Packetization {
|
switch config.Packetization {
|
||||||
case None:
|
case None:
|
||||||
case Mpegts:
|
case Mpegts:
|
||||||
|
case Flv:
|
||||||
case NothingDefined:
|
case NothingDefined:
|
||||||
r.Log(Warning, "No packetization option defined, defaulting to none!")
|
r.Log(Warning, "No packetization option defined, defaulting to none!")
|
||||||
config.Packetization = None
|
config.Packetization = None
|
||||||
|
@ -185,4 +198,5 @@ func (config *Config) Validate(r *revidInst) error {
|
||||||
return errors.New("Bad quantization defined in config!")
|
return errors.New("Bad quantization defined in config!")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,12 +44,11 @@ import (
|
||||||
|
|
||||||
//"bitbucket.org/ausocean/av/parser"
|
//"bitbucket.org/ausocean/av/parser"
|
||||||
//"bitbucket.org/ausocean/av/tsgenerator"
|
//"bitbucket.org/ausocean/av/tsgenerator"
|
||||||
|
"../generator"
|
||||||
"../parser"
|
"../parser"
|
||||||
"../tsgenerator"
|
|
||||||
|
|
||||||
//"bitbucket.org/ausocean/av/ringbuffer"
|
//"bitbucket.org/ausocean/av/ringbuffer"
|
||||||
//"bitbucket.org/ausocean/utils/smartLogger"
|
//"bitbucket.org/ausocean/utils/smartLogger"
|
||||||
"../../utils/smartLogger"
|
|
||||||
"../ringbuffer"
|
"../ringbuffer"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -58,9 +57,9 @@ const (
|
||||||
clipDuration = 1 // s
|
clipDuration = 1 // s
|
||||||
mp2tPacketSize = 188 // MPEG-TS packet size
|
mp2tPacketSize = 188 // MPEG-TS packet size
|
||||||
mp2tMaxPackets = 2016 * clipDuration // # first multiple of 7 and 8 greater than 2000
|
mp2tMaxPackets = 2016 * clipDuration // # first multiple of 7 and 8 greater than 2000
|
||||||
ringBufferSize = 100 / clipDuration
|
ringBufferSize = 1000 / clipDuration
|
||||||
ringBufferElementSize = 10000000
|
ringBufferElementSize = 10000000
|
||||||
maxClipSize = 100000
|
maxClipSize = 100000
|
||||||
httpTimeOut = 5 // s
|
httpTimeOut = 5 // s
|
||||||
packetsPerFrame = 7
|
packetsPerFrame = 7
|
||||||
h264BufferSize = 1000000
|
h264BufferSize = 1000000
|
||||||
|
@ -90,23 +89,25 @@ type RevidInst interface {
|
||||||
|
|
||||||
// The revidInst struct provides fields to describe the state of a RevidInst.
|
// The revidInst struct provides fields to describe the state of a RevidInst.
|
||||||
type revidInst struct {
|
type revidInst struct {
|
||||||
ffmpegPath string
|
ffmpegPath string
|
||||||
tempDir string
|
tempDir string
|
||||||
ringBuffer ringbuffer.RingBuffer
|
ringBuffer ringbuffer.RingBuffer
|
||||||
config Config
|
config Config
|
||||||
isRunning bool
|
isRunning bool
|
||||||
outputFile *os.File
|
outputFile *os.File
|
||||||
inputFile *os.File
|
inputFile *os.File
|
||||||
generator generator.TsGenerator
|
generator generator.Generator
|
||||||
parser parser.Parser
|
parser parser.Parser
|
||||||
cmd *exec.Cmd
|
cmd *exec.Cmd
|
||||||
ffmpegCmd *exec.Cmd
|
ffmpegCmd *exec.Cmd
|
||||||
inputReader *bufio.Reader
|
inputReader *bufio.Reader
|
||||||
ffmpegStdin io.WriteCloser
|
ffmpegStdin io.WriteCloser
|
||||||
outputChan chan []byte
|
outputChan chan []byte
|
||||||
configureOutput func()
|
setupInput func() error
|
||||||
getFrame func()[]byte
|
setupOutput func() error
|
||||||
flushData func()
|
getFrame func() []byte
|
||||||
|
flushData func()
|
||||||
|
sendClip func(clip []byte) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewRevidInstance returns a pointer to a new revidInst with the desired
|
// NewRevidInstance returns a pointer to a new revidInst with the desired
|
||||||
|
@ -140,47 +141,48 @@ func (r *revidInst) ChangeState(config Config) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.New("Config struct is bad!: " + err.Error())
|
return errors.New("Config struct is bad!: " + err.Error())
|
||||||
}
|
}
|
||||||
|
r.config = config
|
||||||
switch r.config.Output {
|
switch r.config.Output {
|
||||||
case File:
|
case File:
|
||||||
r.outputFile, err = os.Create(r.config.OutputFileName)
|
r.sendClip = r.sendClipToFile
|
||||||
if err != nil {
|
r.setupOutput = r.setupOutputForFile
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
configureOutput = configureOutputForFile
|
|
||||||
case Rtmp:
|
case Rtmp:
|
||||||
configureOutput = configureOutputForRtmp
|
r.setupOutput = r.setupOutputForRtmp
|
||||||
|
r.sendClip = r.sendClipToRtmp
|
||||||
|
case Http:
|
||||||
|
r.sendClip = r.sendClipToHTTP
|
||||||
}
|
}
|
||||||
switch r.config.Input {
|
switch r.config.Input {
|
||||||
case Raspivid:
|
case Raspivid:
|
||||||
configureInput = configureInputForRaspivid
|
r.setupInput = r.setupInputForRaspivid
|
||||||
case File:
|
case File:
|
||||||
configureInput = configureInputForFile
|
r.setupInput = r.setupInputForFile
|
||||||
}
|
}
|
||||||
switch r.config.InputCodec {
|
switch r.config.InputCodec {
|
||||||
case H264:
|
case H264:
|
||||||
r.Log(Info, "Using H264 parser!")
|
r.Log(Info, "Using H264 parser!")
|
||||||
r.parser = parser.NewH264Parser()
|
r.parser = parser.NewH264Parser()
|
||||||
|
fmt.Println("here")
|
||||||
case Mjpeg:
|
case Mjpeg:
|
||||||
r.Log(Info, "Using MJPEG parser!")
|
r.Log(Info, "Using MJPEG parser!")
|
||||||
r.parser = parser.NewMJPEGParser(mjpegParserInChanLen)
|
r.parser = parser.NewMJPEGParser(mjpegParserInChanLen)
|
||||||
}
|
}
|
||||||
if r.config.Packetization == None {
|
if r.config.Packetization == None {
|
||||||
r.parser.SetOutputChan(r.outputChan)
|
r.parser.SetOutputChan(r.outputChan)
|
||||||
getFrame = getFrameForNoPacketization
|
r.getFrame = r.getFrameNoPacketization
|
||||||
} else {
|
} else {
|
||||||
switch r.config.Packetization {
|
switch r.config.Packetization {
|
||||||
case Mpegts:
|
case Mpegts:
|
||||||
frameRateAsInt, _ := strconv.Atoi(r.config.FrameRate)
|
frameRateAsInt, _ := strconv.Atoi(r.config.FrameRate)
|
||||||
r.generator = tsgenerator.NewTsGenerator(uint(frameRateAsInt))
|
r.generator = generator.NewTsGenerator(uint(frameRateAsInt))
|
||||||
case Flv:
|
case Flv:
|
||||||
frameRateAsInt, _ := strconv.Atoi(r.config.FrameRate)
|
frameRateAsInt, _ := strconv.Atoi(r.config.FrameRate)
|
||||||
r.generator = flvGenerator.NewFlvGenerator(uint(frameRateAsInt))
|
r.generator = generator.NewFlvGenerator(false, true, uint(frameRateAsInt))
|
||||||
}
|
}
|
||||||
getFrame = getFrameForPacketization
|
r.getFrame = r.getFramePacketization
|
||||||
r.parser.SetOutputChan(r.generator.GetInputChan())
|
r.parser.SetOutputChan(r.generator.GetInputChan())
|
||||||
r.generator.Start()
|
r.generator.Start()
|
||||||
}
|
}
|
||||||
r.config = config
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -230,27 +232,27 @@ func (r *revidInst) Stop() {
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
// and packetising to a defined output.
|
// and packetising to a defined output.
|
||||||
func (r *revidInst) getFrameNoPacketization() []byte {
|
func (r *revidInst) getFrameNoPacketization() []byte {
|
||||||
return <-r.outputChan
|
return <-r.outputChan
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
// and packetising to a defined output.
|
// and packetising to a defined output.
|
||||||
func (r *revidInst) getFrameWithPacketization() []byte {
|
func (r *revidInst) getFramePacketization() []byte {
|
||||||
return <-(r.generator.GetOutputChan())
|
return <-(r.generator.GetOutputChan())
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
// and packetising to a defined output.
|
// and packetising to a defined output.
|
||||||
func (r *revidInst) flushDataNoPacketisation(){
|
func (r *revidInst) flushDataNoPacketisation() {
|
||||||
for len(r.outputChan) > 0 {
|
for len(r.outputChan) > 0 {
|
||||||
<-(r.outputChan)
|
<-(r.outputChan)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
func (r *revidInst) flushDataMpegtsPacketisation(){
|
func (r *revidInst) flushDataMpegtsPacketisation() {
|
||||||
for len(r.generator.GetOutputChan()) > 0 {
|
for len(r.generator.GetOutputChan()) > 0 {
|
||||||
<-(r.generator.GetTsOutputChan())
|
<-(r.generator.GetOutputChan())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -259,13 +261,11 @@ func (r *revidInst) flushDataMpegtsPacketisation(){
|
||||||
func (r *revidInst) packClips() {
|
func (r *revidInst) packClips() {
|
||||||
clipSize := 0
|
clipSize := 0
|
||||||
packetCount := 0
|
packetCount := 0
|
||||||
now := time.Now()
|
|
||||||
prevTime := now
|
|
||||||
for {
|
for {
|
||||||
if clip, err := r.ringBuffer.Get(); err != nil {
|
if clip, err := r.ringBuffer.Get(); err != nil {
|
||||||
r.Log(Error, err.Error())
|
r.Log(Error, err.Error())
|
||||||
r.Log(Warning, "Clearing output chan!")
|
r.Log(Warning, "Clearing output chan!")
|
||||||
r.flushOutputData()
|
r.flushData()
|
||||||
} else {
|
} else {
|
||||||
for {
|
for {
|
||||||
frame := r.getFrame()
|
frame := r.getFrame()
|
||||||
|
@ -274,7 +274,6 @@ func (r *revidInst) packClips() {
|
||||||
copy(clip[clipSize:upperBound], frame)
|
copy(clip[clipSize:upperBound], frame)
|
||||||
packetCount++
|
packetCount++
|
||||||
clipSize += lenOfFrame
|
clipSize += lenOfFrame
|
||||||
now = time.Now()
|
|
||||||
if packetCount >= r.config.FramesPerClip {
|
if packetCount >= r.config.FramesPerClip {
|
||||||
if err := r.ringBuffer.DoneWriting(clipSize); err != nil {
|
if err := r.ringBuffer.DoneWriting(clipSize); err != nil {
|
||||||
r.Log(Error, err.Error())
|
r.Log(Error, err.Error())
|
||||||
|
@ -282,7 +281,6 @@ func (r *revidInst) packClips() {
|
||||||
}
|
}
|
||||||
clipSize = 0
|
clipSize = 0
|
||||||
packetCount = 0
|
packetCount = 0
|
||||||
prevTime = now
|
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -329,6 +327,9 @@ func (r *revidInst) outputClips() {
|
||||||
prevTime = now
|
prevTime = now
|
||||||
bytes = 0
|
bytes = 0
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
r.Log(Debug, err.Error())
|
||||||
|
time.Sleep(1*time.Second)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -336,7 +337,7 @@ func (r *revidInst) outputClips() {
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
// and packetising to a defined output.
|
// and packetising to a defined output.
|
||||||
func (r *revidInst) sendClipToFile(clip []byte) error {
|
func (r *revidInst) sendClipToFile(clip []byte) error {
|
||||||
err := r.outputFile.Write(clip)
|
_,err := r.outputFile.Write(clip)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -349,10 +350,11 @@ func (r *revidInst) sendClipToHTTP(clip []byte) error {
|
||||||
client := http.Client{
|
client := http.Client{
|
||||||
Timeout: timeout,
|
Timeout: timeout,
|
||||||
}
|
}
|
||||||
|
url := r.config.HttpAddress+strconv.Itoa(len(clip))
|
||||||
r.Log(Debug, fmt.Sprintf("Posting %s (%d bytes)\n", url, len(clip)))
|
r.Log(Debug, fmt.Sprintf("Posting %s (%d bytes)\n", url, len(clip)))
|
||||||
resp, err := client.Post(r.config.HttpAddress + strconv.Itoa(len(clip)), "video/mp2t", bytes.NewReader(clip)) // lighter than NewBuffer
|
resp, err := client.Post(url, "video/mp2t", bytes.NewReader(clip)) // lighter than NewBuffer
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Error posting to %s: %s", output, err)
|
return fmt.Errorf("Error posting to %s: %s", url, err)
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
body, err := ioutil.ReadAll(resp.Body)
|
body, err := ioutil.ReadAll(resp.Body)
|
||||||
|
@ -377,7 +379,7 @@ func (r *revidInst) sendClipToRtmp(clip []byte) error {
|
||||||
|
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
// and packetising to a defined output.
|
// and packetising to a defined output.
|
||||||
func (r *revidInst) setupOutputForRtmp(){
|
func (r *revidInst) setupOutputForRtmp() error {
|
||||||
r.ffmpegCmd = exec.Command(ffmpegPath,
|
r.ffmpegCmd = exec.Command(ffmpegPath,
|
||||||
"-f", "h264",
|
"-f", "h264",
|
||||||
"-r", r.config.FrameRate,
|
"-r", r.config.FrameRate,
|
||||||
|
@ -398,19 +400,25 @@ func (r *revidInst) setupOutputForRtmp(){
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.Log(Error, err.Error())
|
r.Log(Error, err.Error())
|
||||||
r.Stop()
|
r.Stop()
|
||||||
return
|
return err
|
||||||
}
|
}
|
||||||
err = r.ffmpegCmd.Start()
|
err = r.ffmpegCmd.Start()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.Log(Error, err.Error())
|
r.Log(Error, err.Error())
|
||||||
r.Stop()
|
r.Stop()
|
||||||
return
|
return err
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *revidInst) setupOutputForFile() (err error) {
|
||||||
|
r.outputFile, err = os.Create(r.config.OutputFileName)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
// and packetising to a defined output.
|
// and packetising to a defined output.
|
||||||
func (r *revidInst) setupInputForRaspivid(){
|
func (r *revidInst) setupInputForRaspivid() error {
|
||||||
r.Log(Info, "Starting raspivid!")
|
r.Log(Info, "Starting raspivid!")
|
||||||
switch r.config.InputCodec {
|
switch r.config.InputCodec {
|
||||||
case H264:
|
case H264:
|
||||||
|
@ -441,14 +449,16 @@ func (r *revidInst) setupInputForRaspivid(){
|
||||||
r.inputReader = bufio.NewReader(stdout)
|
r.inputReader = bufio.NewReader(stdout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.Log(Error, err.Error())
|
r.Log(Error, err.Error())
|
||||||
return
|
return err
|
||||||
}
|
}
|
||||||
go r.readCamera()
|
go r.readCamera()
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start invokes a revidInst to start processing video from a defined input
|
// Start invokes a revidInst to start processing video from a defined input
|
||||||
func (r *revidInst) setupInputForFile(){
|
func (r *revidInst) setupInputForFile() error {
|
||||||
go r.readFile()
|
go r.readFile()
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// readCamera reads data from the defined camera while the revidInst is running.
|
// readCamera reads data from the defined camera while the revidInst is running.
|
||||||
|
@ -471,33 +481,32 @@ func (r *revidInst) readCamera() {
|
||||||
}
|
}
|
||||||
|
|
||||||
// readFile reads data from the defined file while the revidInst is running.
|
// readFile reads data from the defined file while the revidInst is running.
|
||||||
func (r *revidInst) readFile() {
|
func (r *revidInst) readFile() error {
|
||||||
for {
|
if len(r.parser.GetInputChan()) == 0 {
|
||||||
if len(r.parser.GetInputChan()) == 0 {
|
var err error
|
||||||
var err error
|
r.inputFile, err = os.Open(r.config.InputFileName)
|
||||||
r.inputFile, err = os.Open(r.config.InputFileName)
|
if err != nil {
|
||||||
if err != nil {
|
r.Log(Error, err.Error())
|
||||||
r.Log(Error, err.Error())
|
r.Stop()
|
||||||
r.Stop()
|
return err
|
||||||
return
|
|
||||||
}
|
|
||||||
stats, err := r.inputFile.Stat()
|
|
||||||
if err != nil {
|
|
||||||
r.Log(Error, "Could not get input file stats!")
|
|
||||||
r.Stop()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
data := make([]byte, stats.Size())
|
|
||||||
_, err = r.inputFile.Read(data)
|
|
||||||
if err != nil {
|
|
||||||
r.Log(Error, err.Error())
|
|
||||||
r.Stop()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
for i := range data {
|
|
||||||
r.parser.GetInputChan() <- data[i]
|
|
||||||
}
|
|
||||||
r.inputFile.Close()
|
|
||||||
}
|
}
|
||||||
|
stats, err := r.inputFile.Stat()
|
||||||
|
if err != nil {
|
||||||
|
r.Log(Error, "Could not get input file stats!")
|
||||||
|
r.Stop()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
data := make([]byte, stats.Size())
|
||||||
|
_, err = r.inputFile.Read(data)
|
||||||
|
if err != nil {
|
||||||
|
r.Log(Error, err.Error())
|
||||||
|
r.Stop()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for i := range data {
|
||||||
|
r.parser.GetInputChan() <- data[i]
|
||||||
|
}
|
||||||
|
r.inputFile.Close()
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,8 +29,8 @@ LICENSE
|
||||||
package revid
|
package revid
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -123,22 +123,22 @@ func TestRtmpOutput(t *testing.T){
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// Test h264 inputfile to flv output files
|
// Test h264 inputfile to flv output files
|
||||||
func TestFlvOutputFile(t *testing.T){
|
func TestFlvOutputFile(t *testing.T) {
|
||||||
config := Config{
|
config := Config{
|
||||||
Input: File,
|
Input: File,
|
||||||
InputFileName: "testInput.h264",
|
InputFileName: "testInput.h264",
|
||||||
InputCodec: H264,
|
InputCodec: H264,
|
||||||
Output: File,
|
Output: File,
|
||||||
OutputFileName: "testOutput.flv",
|
OutputFileName: "testOutput.flv",
|
||||||
Packetization: Flv,
|
Packetization: Flv,
|
||||||
FrameRate: "25",
|
FrameRate: "25",
|
||||||
}
|
}
|
||||||
revidInst, err := NewRevidInstance(config)
|
revidInst, err := NewRevidInstance(config)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("Should not of have got an error!: %v\n", err.Error())
|
t.Errorf("Should not of have got an error!: %v\n", err.Error())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
revidInst.Start()
|
revidInst.Start()
|
||||||
time.Sleep(5*time.Second)
|
time.Sleep(5 * time.Second)
|
||||||
revidInst.Stop()
|
revidInst.Stop()
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,7 +45,7 @@ type RingBuffer interface {
|
||||||
DoneReading() error
|
DoneReading() error
|
||||||
IsReadable() bool
|
IsReadable() bool
|
||||||
IsWritable() bool
|
IsWritable() bool
|
||||||
GetNoOfElements() int
|
GetNoOfElements() int
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rb *ringBuffer)GetNoOfElements() int {
|
func (rb *ringBuffer)GetNoOfElements() int {
|
||||||
|
@ -100,7 +100,7 @@ func (rb *ringBuffer) Get() ([]byte, error) {
|
||||||
if !rb.IsWritable() {
|
if !rb.IsWritable() {
|
||||||
return nil, errors.New("Buffer full!")
|
return nil, errors.New("Buffer full!")
|
||||||
}
|
}
|
||||||
var nextlast int
|
var nextlast int
|
||||||
if !rb.currentlyWriting {
|
if !rb.currentlyWriting {
|
||||||
rb.currentlyWriting = true
|
rb.currentlyWriting = true
|
||||||
nextlast = rb.last + 1
|
nextlast = rb.last + 1
|
||||||
|
|
Loading…
Reference in New Issue