You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

263 lines
6.9 KiB
Go

3 years ago
package icsmediaconv
import (
"fmt"
"sync"
"gitlab.com/ics_cinnamon/joy4/av"
"gitlab.com/ics_cinnamon/joy4/cgo/ffmpeg"
"gitlab.com/ics_cinnamon/joy4/codec"
"gitlab.com/ics_cinnamon/voicegateway/icserror"
"gitlab.com/ics_cinnamon/voicegateway/icslog"
"gitlab.com/ics_cinnamon/voicegateway/icspacketparser"
)
const (
ICS_PT_MULAW = 0
ICS_PT_ALAW = 8
ICS_PT_G729 = 18
ICS_PT_END = ICS_PT_G729
)
const PCM_8K_16BIT_10MS_SIZE = 160
type Converter struct {
payloadtype icspacketparser.PayloadType
codec av.AudioCodecData
name string
decoder *ffmpeg.AudioDecoder
encoder *ffmpeg.AudioEncoder
samplingRate int
onePacketSize int
//frame *av.AudioFrame
isStart bool
m *sync.Mutex
ID int
}
func NewConverter(id int, pt icspacketparser.PayloadType, dec bool) (*Converter, *icserror.IcsError) {
conv := &Converter{payloadtype: pt}
conv.ID = id
conv.isStart = false
conv.m = &sync.Mutex{}
if dec { //decoding
switch pt {
case ICS_PT_MULAW:
conv.codec = codec.NewPCMMulawCodecData()
conv.samplingRate = 8000
conv.onePacketSize = 80
//conv.onePacketSize = 160
case ICS_PT_ALAW:
conv.codec = codec.NewPCMAlawCodecData()
conv.samplingRate = 8000
conv.onePacketSize = 80
//conv.onePacketSize = 160
case ICS_PT_G729:
conv.codec = codec.NewG729CodecData()
conv.samplingRate = 8000
conv.onePacketSize = 10
default:
return nil, icserror.ICSERRCONVNotSupportedCodec
}
var err error
conv.decoder, err = ffmpeg.NewAudioDecoder(conv.codec)
if err != nil {
icserror.ICSERRCONVNotSupportedCodec.SetError(err)
return nil, icserror.ICSERRCONVNotSupportedCodec
}
} else { //encoding
switch pt {
case ICS_PT_MULAW:
conv.codec = codec.NewPCMMulawCodecData()
conv.name = "mulaw"
conv.samplingRate = 8000
conv.onePacketSize = 160
case ICS_PT_ALAW:
conv.codec = codec.NewPCMAlawCodecData()
conv.name = "alaw"
conv.samplingRate = 8000
conv.onePacketSize = 160
case ICS_PT_G729:
conv.codec = codec.NewG729CodecData()
conv.name = "g729"
conv.samplingRate = 8000
conv.onePacketSize = 10
default:
return nil, icserror.ICSERRCONVNotSupportedCodec
}
var err error
conv.encoder, err = ffmpeg.NewAudioEncoderByCodecType(conv.codec.Type())
if err != nil {
fmt.Println("########111", err)
icserror.ICSERRCONVNotSupportedCodec.SetError(err)
return nil, icserror.ICSERRCONVNotSupportedCodec
}
conv.encoder.SetChannelLayout(av.CH_MONO)
conv.encoder.SetSampleRate(conv.samplingRate)
conv.encoder.SetSampleFormat(av.S16)
// fmt.Println("########222", conv.name)
}
//ffmpeg.SetLogLevel(ffmpeg.QUIET)
conv.Start()
return conv, nil
}
func (c *Converter) Start() {
c.m.Lock()
c.isStart = true
c.m.Unlock()
}
func (c *Converter) Stop() {
c.m.Lock()
c.isStart = false
c.m.Unlock()
}
func (c *Converter) IsStart() bool {
return c.isStart
}
func (c *Converter) Close() {
c.Stop()
if c.decoder != nil {
c.decoder.Close()
}
if c.encoder != nil {
c.encoder.Close()
}
l := icslog.GetIcsLog()
l.Print(icslog.LOG_LEVEL_INFO, c.ID, "Closed Converter")
}
func (c *Converter) Decode(packet []byte) ([]byte, *icserror.IcsError) {
if c == nil {
return nil, icserror.ICSERRInvalidParam
}
//l := icslog.GetIcsLog()
//l.Printf(icslog.LOG_LEVEL_DEBUG2, -1, "converter### Decode() packet length: %d", len(packet))
retBuf := make([]byte, PCM_8K_16BIT_10MS_SIZE*2)
packetLen := len(packet)
iter := 0
for packetLen >= c.onePacketSize {
packetLen -= c.onePacketSize
//fmt.Printf("### Decode() iter(%d) packetlen(%d)\n", iter, packetLen)
buf := packet[c.onePacketSize*iter : c.onePacketSize*(iter+1)]
//fmt.Printf("### Decode() iter(%d), buf length %d %v\n", iter, len(buf), buf)
//l.Printf(icslog.LOG_LEVEL_DEBUG2, c.ID, "### Decode() iter(%d), buf length %d %v", iter, len(buf), buf)
c.m.Lock()
if c.IsStart() {
ok, frame, errDec := c.decoder.Decode(buf)
if !ok {
icserror.ICSERRCONVDecodeFail.SetError(errDec)
//icserror.ICSERRCONVDecodeFail.Print()
c.m.Unlock()
return nil, icserror.ICSERRCONVDecodeFail
}
//fmt.Println("###frame len", iter, PCM_8K_16BIT_10MS_SIZE*iter, PCM_8K_16BIT_10MS_SIZE*(iter+1))
//fmt.Println("###frame len", len(frame.Data[0]), len(frame.Data), frame)
copy(retBuf[PCM_8K_16BIT_10MS_SIZE*iter:PCM_8K_16BIT_10MS_SIZE*(iter+1)], frame.Data[0][:PCM_8K_16BIT_10MS_SIZE])
}
c.m.Unlock()
/*
f1, err := os.OpenFile("./tx.voice.raw", os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
if err != nil {
log.Fatal(err)
}
f1.Write(frame.Data[0][:PCM_8K_16BIT_10MS_SIZE])
f1.Sync()
f1.Close()
*/
iter++
}
//fmt.Println("###retBuf len", len(retBuf), retBuf)
return retBuf, nil
}
func (c *Converter) Encode(packet []byte) ([]byte, *icserror.IcsError) {
if c == nil {
return nil, icserror.ICSERRInvalidParam
}
//l := icslog.GetIcsLog()
//l.Printf(icslog.LOG_LEVEL_DEBUG2, -1, "converter### Decode() packet length: %d", len(packet))
retBuf := make([]byte, PCM_8K_16BIT_10MS_SIZE)
//retBuf := make([]byte, PCM_8K_16BIT_10MS_SIZE*2)
packetLen := len(packet)
iter := 0
var onePacketSize int = c.onePacketSize
if c.codec.Type() == av.PCM_ALAW || c.codec.Type() == av.PCM_MULAW {
onePacketSize = onePacketSize
//onePacketSize = onePacketSize / 2
}
for packetLen >= onePacketSize {
packetLen -= onePacketSize
//for packetLen >= c.onePacketSize {
//packetLen -= c.onePacketSize
//fmt.Printf("### Decode() iter(%d) packetlen(%d)\n", iter, packetLen)
buf := packet[onePacketSize*iter : onePacketSize*(iter+1)]
//buf := packet[c.onePacketSize*iter : c.onePacketSize*(iter+1)]
//fmt.Printf("### Decode() iter(%d), buf length %d %v\n", iter, len(buf), buf)
//l.Printf(icslog.LOG_LEVEL_DEBUG2, c.ID, "### Decode() iter(%d), buf length %d %v", iter, len(buf), buf)
tbuf := make([][]byte, 1)
tbuf[0] = buf
frame := av.AudioFrame{
SampleFormat: av.S16,
ChannelLayout: av.CH_MONO,
SampleCount: c.onePacketSize / 2,
SampleRate: c.samplingRate,
Data: tbuf,
}
//fmt.Printf(">>>%+v\n", frame)
c.m.Lock()
if c.IsStart() {
pcmFrame, errEnc := c.encoder.Encode(frame)
//ok, frame, errDec := c.decoder.Decode(buf)
if errEnc != nil {
icserror.ICSERRCONVEncodeFail.SetError(errEnc)
//icserror.ICSERRCONVDecodeFail.Print()
c.m.Unlock()
return nil, icserror.ICSERRCONVEncodeFail
}
//fmt.Println("###frame len", iter, PCM_8K_16BIT_10MS_SIZE*iter, PCM_8K_16BIT_10MS_SIZE*(iter+1))
//fmt.Println("###frame len", len(frame.Data[0]), len(frame.Data), frame)
copy(retBuf[(PCM_8K_16BIT_10MS_SIZE/2)*iter:(PCM_8K_16BIT_10MS_SIZE/2)*(iter+1)], pcmFrame[0][:(PCM_8K_16BIT_10MS_SIZE/2)])
//copy(retBuf[PCM_8K_16BIT_10MS_SIZE*iter:PCM_8K_16BIT_10MS_SIZE*(iter+1)], pcmFrame[0][:PCM_8K_16BIT_10MS_SIZE])
//fmt.Printf("%d$$$%v", len(pcmFrame[0]), pcmFrame[0])
}
c.m.Unlock()
iter++
}
//fmt.Println("###retBuf len", len(retBuf), retBuf)
return retBuf, nil
}