cloud-game/pkg/encoder/vpx-encoder/encoder.go
giongto35 0b9d628ec2
Refactor: Fix channel init order (#141)
* Fix channel order

* Remove Done in encoder

* Update h264 encoder done code

* remove Unnecessary IsRunning
2019-11-27 02:38:13 +08:00

183 lines
4.4 KiB
Go

package vpxencoder
import (
"fmt"
"image"
"log"
"unsafe"
"github.com/giongto35/cloud-game/pkg/encoder"
"github.com/giongto35/cloud-game/pkg/util"
)
// https://chromium.googlesource.com/webm/libvpx/+/master/examples/simple_encoder.c
/*
#cgo pkg-config: vpx
#include <stdlib.h>
#include "vpx/vpx_encoder.h"
#include "tools_common.h"
typedef struct GoBytes {
void *bs;
int size;
} GoBytesType;
vpx_codec_err_t call_vpx_codec_enc_config_default(const VpxInterface *encoder, vpx_codec_enc_cfg_t *cfg) {
return vpx_codec_enc_config_default(encoder->codec_interface(), cfg, 0);
}
vpx_codec_err_t call_vpx_codec_enc_init(vpx_codec_ctx_t *codec, const VpxInterface *encoder, vpx_codec_enc_cfg_t *cfg) {
return vpx_codec_enc_init(codec, encoder->codec_interface(), cfg, 0);
}
GoBytesType get_frame_buffer(vpx_codec_ctx_t *codec, vpx_codec_iter_t *iter) {
// iter has set to NULL when after add new image
GoBytesType bytes = {NULL, 0};
const vpx_codec_cx_pkt_t *pkt = vpx_codec_get_cx_data(codec, iter);
if (pkt != NULL && pkt->kind == VPX_CODEC_CX_FRAME_PKT) {
bytes.bs = pkt->data.frame.buf;
bytes.size = pkt->data.frame.sz;
}
return bytes;
}
*/
import "C"
const chanSize = 2
// VpxEncoder yuvI420 image to vp8 video
type VpxEncoder struct {
Output chan []byte // frame
Input chan *image.RGBA // yuvI420
width int
height int
// C
fps C.int
bitrate C.uint
keyFrameInterval C.int
frameCount C.int
vpxCodexCtx C.vpx_codec_ctx_t
vpxImage C.vpx_image_t
vpxCodexIter C.vpx_codec_iter_t
}
// NewVpxEncoder create vp8 encoder
func NewVpxEncoder(w, h, fps, bitrate, keyframe int) (encoder.Encoder, error) {
v := &VpxEncoder{
Output: make(chan []byte, 5*chanSize),
Input: make(chan *image.RGBA, chanSize),
// C
width: w,
height: h,
fps: C.int(fps),
bitrate: C.uint(bitrate),
keyFrameInterval: C.int(keyframe),
frameCount: C.int(0),
}
if err := v.init(); err != nil {
return nil, err
}
return v, nil
}
func (v *VpxEncoder) init() error {
v.frameCount = 0
codecName := C.CString("vp8")
encoder := C.get_vpx_encoder_by_name(codecName)
C.free(unsafe.Pointer(codecName))
if encoder == nil {
return fmt.Errorf("get_vpx_encoder_by_name failed")
}
if C.vpx_img_alloc(&v.vpxImage, C.VPX_IMG_FMT_I420, C.uint(v.width), C.uint(v.height), 0) == nil {
return fmt.Errorf("vpx_img_alloc failed")
}
var cfg C.vpx_codec_enc_cfg_t
if C.call_vpx_codec_enc_config_default(encoder, &cfg) != 0 {
return fmt.Errorf("Failed to get default codec config")
}
cfg.g_w = C.uint(v.width)
cfg.g_h = C.uint(v.height)
cfg.g_timebase.num = 1
cfg.g_timebase.den = v.fps
cfg.rc_target_bitrate = v.bitrate
cfg.g_error_resilient = 1
if C.call_vpx_codec_enc_init(&v.vpxCodexCtx, encoder, &cfg) != 0 {
return fmt.Errorf("Failed to initialize encoder")
}
go v.startLooping()
return nil
}
func (v *VpxEncoder) startLooping() {
defer func() {
if r := recover(); r != nil {
log.Println("Warn: Recovered panic in encoding ", r)
}
}()
size := int(float32(v.width*v.height) * 1.5)
yuv := make([]byte, size, size)
for img := range v.Input {
util.RgbaToYuvInplace(img, yuv, v.width, v.height)
// Add Image
v.vpxCodexIter = nil
C.vpx_img_read(&v.vpxImage, unsafe.Pointer(&yuv[0]))
var flags C.int
if v.keyFrameInterval > 0 && v.frameCount%v.keyFrameInterval == 0 {
flags |= C.VPX_EFLAG_FORCE_KF
}
if C.vpx_codec_encode(&v.vpxCodexCtx, &v.vpxImage, C.vpx_codec_pts_t(v.frameCount), 1, C.vpx_enc_frame_flags_t(flags), C.VPX_DL_REALTIME) != 0 {
fmt.Println("Failed to encode frame")
}
v.frameCount++
// Get Frame
goBytes := C.get_frame_buffer(&v.vpxCodexCtx, &v.vpxCodexIter)
if goBytes.bs == nil {
continue
}
bs := C.GoBytes(goBytes.bs, goBytes.size)
// if buffer is full skip frame
if len(v.Output) >= cap(v.Output) {
continue
}
v.Output <- bs
}
}
// Release release memory and stop loop
func (v *VpxEncoder) release() {
log.Println("Releasing encoder")
C.vpx_img_free(&v.vpxImage)
C.vpx_codec_destroy(&v.vpxCodexCtx)
// TODO: Bug here, after close it will signal
close(v.Output)
if v.Input != nil {
close(v.Input)
}
}
// GetInputChan returns input channel
func (v *VpxEncoder) GetInputChan() chan *image.RGBA {
return v.Input
}
// GetInputChan returns output channel
func (v *VpxEncoder) GetOutputChan() chan []byte {
return v.Output
}
// GetDoneChan returns done channel
func (v *VpxEncoder) Stop() {
v.release()
}