加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
Jessibuca.cpp 10.60 KB
一键复制 编辑 原始数据 按行查看 历史
langhuihui 提交于 2021-07-07 22:40 . 播放pcma成功
#ifdef EDITTIME
#undef __cplusplus
#define __cplusplus 201703L
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string>
// #include <functional>
// #include <map>
// #include <queue>
#include <emscripten.h>
#include <emscripten/bind.h>
#include <emscripten/val.h>
// #include <time.h>
// #include <regex>
using namespace std;
using namespace emscripten;
// #include "slice.h"
typedef unsigned char u8;
// typedef signed char i8;
// typedef unsigned short u16;
// typedef signed short i16;
typedef unsigned int u32;
// typedef signed int i32;
#define PROP(name, type) \
type name; \
val get##name() const \
{ \
return val(name); \
} \
void set##name(val value) \
{ \
name = value.as<type>(); \
emscripten_log(0, #name " = %d", name); \
}
extern "C"
{
#include <libavcodec/avcodec.h>
#include <libswresample/swresample.h>
}
//const int SamplingFrequencies[] = {96000, 88200, 64000, 48000, 44100, 32000, 24000, 22050, 16000, 12000, 11025, 8000, 7350, 0, 0, 0};
//const int AudioObjectTypes[] = {};
class FFmpeg
{
public:
AVCodec *codec = nullptr;
AVCodecParserContext *parser = nullptr;
AVCodecContext *dec_ctx = nullptr;
AVFrame *frame;
AVPacket *pkt;
val jsObject;
bool initialized = false;
FFmpeg(val &&v) : jsObject(forward<val>(v))
{
}
void initCodec(enum AVCodecID id)
{
if (dec_ctx != nullptr)
{
clear();
}
pkt = av_packet_alloc();
codec = avcodec_find_decoder(id);
parser = av_parser_init(codec->id);
dec_ctx = avcodec_alloc_context3(codec);
frame = av_frame_alloc();
}
void initCodec(enum AVCodecID id, string input)
{
initCodec(id);
dec_ctx->extradata_size = input.length();
dec_ctx->extradata = (u8 *)input.data();
// // dec_ctx->extradata = (u8 *)malloc(dec_ctx->extradata_size);
// // memcpy(dec_ctx->extradata, input.c_str(), dec_ctx->extradata_size);
avcodec_open2(dec_ctx, codec, NULL);
initialized = true;
}
virtual ~FFmpeg()
{
clear();
}
virtual int decode(string input)
{
int ret = 0;
pkt->data = (u8 *)(input.data());
pkt->size = input.length();
ret = avcodec_send_packet(dec_ctx, pkt);
while (ret >= 0)
{
ret = avcodec_receive_frame(dec_ctx, frame);
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
return 0;
_decode();
}
return 0;
}
virtual void _decode(){};
virtual void clear()
{
if (parser)
{
av_parser_close(parser);
parser = nullptr;
}
if (dec_ctx)
{
avcodec_free_context(&dec_ctx);
}
if (frame)
{
av_frame_free(&frame);
}
if (pkt)
{
av_packet_free(&pkt);
}
codec = nullptr;
initialized = false;
}
};
class FFmpegAudioDecoder : public FFmpeg
{
SwrContext *au_convert_ctx = nullptr;
u8 *out_buffer[2];
int output_nb_samples;
int n_channel;
public:
PROP(sample_rate, int)
// struct SwrContext *au_convert_ctx = nullptr;
FFmpegAudioDecoder(val &&v) : FFmpeg(move(v))
{
// emscripten_log(0, "FFMpegAudioDecoder init");
}
~FFmpegAudioDecoder()
{
if (au_convert_ctx)
swr_free(&au_convert_ctx);
if (out_buffer[0])
free(out_buffer[0]);
// emscripten_log(0, "FFMpegAudioDecoder destory");
}
void clear() override
{
FFmpeg::clear();
}
int decode(string input) override
{
u8 flag = (u8)input[0];
u8 audioType = flag >> 4;
if (initialized)
{
return FFmpeg::decode(input.substr(audioType == 10 ? 2 : 1));
}
else
{
switch (audioType)
{
case 10:
if (!input[1])
{
initCodec(AV_CODEC_ID_AAC, input.substr(2));
n_channel = 2;
}
break;
case 7:
initCodec(AV_CODEC_ID_PCM_ALAW);
dec_ctx->channel_layout = AV_CH_LAYOUT_MONO;
dec_ctx->sample_rate = 8000;
dec_ctx->channels = 1;
avcodec_open2(dec_ctx, codec, NULL);
n_channel = 1;
initialized = true;
break;
case 8:
initCodec(AV_CODEC_ID_PCM_MULAW);
dec_ctx->channel_layout = AV_CH_LAYOUT_MONO;
dec_ctx->sample_rate = 8000;
dec_ctx->channels = 1;
avcodec_open2(dec_ctx, codec, NULL);
n_channel = 1;
initialized = true;
break;
default:
emscripten_log(0, "audio type not support:%d", audioType);
break;
}
if (initialized)
{
jsObject.call<void>("initAudioPlanar", n_channel, sample_rate);
}
}
return 0;
}
void _decode() override
{
auto nb_samples = frame->nb_samples;
auto bytes_per_sample = av_get_bytes_per_sample(AV_SAMPLE_FMT_FLTP);
if (dec_ctx->sample_fmt == AV_SAMPLE_FMT_FLTP && sample_rate == dec_ctx->sample_rate && dec_ctx->channel_layout == n_channel)
{
jsObject.call<void>("playAudioPlanar", int(frame->data), nb_samples *bytes_per_sample *n_channel);
return;
}
if (!au_convert_ctx)
{
// out_buffer = (uint8_t *)av_malloc(av_get_bytes_per_sample(dec_ctx->sample_fmt)*dec_ctx->channels*dec_ctx->frame_size);
emscripten_log(0, "au_convert_ctx channel_layout:%d,sample_fmt:%d", dec_ctx->channel_layout, dec_ctx->sample_fmt);
au_convert_ctx = swr_alloc_set_opts(NULL, n_channel == 2 ? AV_CH_LAYOUT_STEREO : AV_CH_LAYOUT_MONO, AV_SAMPLE_FMT_FLTP, sample_rate,
dec_ctx->channel_layout, dec_ctx->sample_fmt, dec_ctx->sample_rate,
0, NULL);
auto ret = swr_init(au_convert_ctx);
auto out_buffer_size = av_samples_get_buffer_size(NULL, n_channel, nb_samples, AV_SAMPLE_FMT_FLTP, 0);
auto buffer = (uint8_t *)av_malloc(out_buffer_size);
out_buffer[0] = buffer;
out_buffer[1] = buffer + (out_buffer_size / 2);
emscripten_log(0, "au_convert_ctx init sample_rate:%d->%d,ret:%d", dec_ctx->sample_rate, sample_rate, ret);
}
// // 转换
auto ret = swr_convert(au_convert_ctx, out_buffer, nb_samples, (const uint8_t **)frame->data, nb_samples);
while (ret > 0)
{
jsObject.call<void>("playAudioPlanar", int(&out_buffer), ret);
ret = swr_convert(au_convert_ctx, out_buffer, nb_samples, (const uint8_t **)frame->data, 0);
}
}
};
class FFmpegVideoDecoder : public FFmpeg
{
public:
u32 videoWidth = 0;
u32 videoHeight = 0;
u32 y = 0;
u32 u = 0;
u32 v = 0;
int NAL_unit_length = 0;
u32 compositionTime = 0;
FFmpegVideoDecoder(val &&v) : FFmpeg(move(v))
{
// emscripten_log(0, "FFMpegVideoDecoder init");
}
~FFmpegVideoDecoder()
{
// emscripten_log(0, "FFMpegVideoDecoder destory");
}
void clear() override
{
videoWidth = 0;
videoHeight = 0;
FFmpeg::clear();
if (y)
{
free((void *)y);
y = 0;
}
}
int decode(string data) override
{
if (!initialized)
{
int codec_id = ((int)data[0]) & 0x0F;
if (((int)(data[0]) >> 4) == 1 && data[1] == 0)
{
// emscripten_log(0, "codec = %d", codec_id);
switch (codec_id)
{
case 7:
initCodec(AV_CODEC_ID_H264, data.substr(5));
break;
case 12:
initCodec(AV_CODEC_ID_H265, data.substr(5));
break;
default:
emscripten_log(0, "codec not support: %d", codec_id);
return -1;
}
}
}
else
{
compositionTime = (((u32)data[2]) << 16) + (((u32)data[3]) << 8) + (u32)data[4];
return FFmpeg::decode(data.substr(5));
}
return 0;
}
void _decode() override
{
if (videoWidth != frame->width || videoHeight != frame->height)
{
videoWidth = frame->width;
videoHeight = frame->height;
jsObject.call<void>("setVideoSize", videoWidth, videoHeight);
int size = videoWidth * videoHeight;
if (y)
free((void *)y);
y = (u32)malloc(size * 3 >> 1);
u = y + size;
v = u + (size >> 2);
}
u32 dst = y;
for (int i = 0; i < videoHeight; i++)
{
memcpy((u8 *)dst, (const u8 *)(frame->data[0] + i * frame->linesize[0]), videoWidth);
dst += videoWidth;
}
dst = u;
int halfh = videoHeight >> 1;
int halfw = videoWidth >> 1;
for (int i = 0; i < halfh; i++)
{
memcpy((u8 *)dst, (const u8 *)(frame->data[1] + i * frame->linesize[1]), halfw);
dst += halfw;
}
for (int i = 0; i < halfh; i++)
{
memcpy((u8 *)dst, (const u8 *)(frame->data[2] + i * frame->linesize[2]), halfw);
dst += halfw;
}
jsObject.call<void>("draw", compositionTime, y, u, v);
}
};
#define FUNC(name) function(#name, &FFmpegAudioDecoder::name)
#undef PROP
#define PROP(name) property(#name, &FFmpegAudioDecoder::get##name, &FFmpegAudioDecoder::set##name)
EMSCRIPTEN_BINDINGS(FFmpegAudioDecoder)
{
class_<FFmpegAudioDecoder>("AudioDecoder")
.constructor<val>()
.PROP(sample_rate)
.FUNC(clear)
.FUNC(decode);
}
#undef FUNC
#define FUNC(name) function(#name, &FFmpegVideoDecoder::name)
#undef PROP
#define PROP(name) property(#name, &FFmpegVideoDecoder::get##name, &FFmpegVideoDecoder::set##name)
EMSCRIPTEN_BINDINGS(FFmpegVideoDecoder)
{
class_<FFmpegVideoDecoder>("VideoDecoder")
.constructor<val>()
.FUNC(clear)
.FUNC(decode);
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化