《Ndk中使用Mediacode解码》
《android mediacodec 编码demo(java)》
《NDK中使用mediacodec编码h264》
《Android native 层使用opengl渲染YUV420p和NV12》
《android 使用NativeWindow渲染RGB视频》
《opengl 叠加显示文字》
《android studio 编译freeType》
《最原始的yuv图像叠加文字的实现--手动操作像素》
Ndk native c使用MediaCodec编码:(在ndk直接编译成可执行文件,push到设备上可运行)
ndkmediacode 是android在framwork层对native 层mediacodec 的一层封装,源码在 framework/av/media/ndk/NdkMediacodec.cpp
对应的库 :/system/lib64/libmediandk.so
//canok 20210316
//NdkMediacodec.cpp
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<pthread.h>
#include <unistd.h>
#include "media/NdkMediaCodec.h"
#define LOGD printf
bool bRun = true;
AMediaCodec* pMediaCodec;
AMediaFormat *format ;
FILE *fp_in = NULL;
FILE *fp_out = NULL;
int mW = 1920;
int mH=1080;
int64_t getNowUs(){
timeval tv;
gettimeofday(&tv, 0);
return (int64_t)tv.tv_sec * 1000000 + (int64_t)tv.tv_usec;
}
void *run(void*pram){
fp_in = fopen("/storage/emulated/0/canok/1080p60.yuv","r");
if(NULL == fp_in){
LOGD("[%s][%d]fopen erro, no inputfile!\n",__FUNCTION__ ,__LINE__);
exit(-1);
}
fp_out = fopen("/storage/emulated/0/canok/out_1080p60.h264","w+");
if(NULL == fp_out){
LOGD("[%s][%d]fopen erro\n",__FUNCTION__ ,__LINE__);
exit(-1);
}
//https://github.com/android/ndk-samples/blob/main/native-codec/app/src/main/cpp/native-codec-jni.cpp
//decode
//这里设定名称
pMediaCodec = AMediaCodec_createEncoderByType("video/avc");//h264 // 创建 codec 编码器
if(pMediaCodec == NULL){
LOGD("createEncoder erro[%s%d]\n",__FUNCTION__ ,__LINE__);
}
format = AMediaFormat_new();
AMediaFormat_setString(format, "mime", "video/avc");
AMediaFormat_setInt32(format,AMEDIAFORMAT_KEY_WIDTH,mW);
AMediaFormat_setInt32(format,AMEDIAFORMAT_KEY_HEIGHT,mH);
//在OMX_IVCommon.h https://www.androidos.net.cn/android/9.0.0_r8/xref/frameworks/native/headers/media_plugin/media/openmax/OMX_IVCommon.h
// AMediaFormat_setInt32(format,AMEDIAFORMAT_KEY_COLOR_FORMAT,OMX_COLOR_FormatYUV420Planar);
AMediaFormat_setInt32(format,AMEDIAFORMAT_KEY_COLOR_FORMAT,19);
AMediaFormat_setInt32(format,AMEDIAFORMAT_KEY_FRAME_RATE,25);
AMediaFormat_setInt32(format,AMEDIAFORMAT_KEY_BIT_RATE,4000*1000);
AMediaFormat_setInt32(format,AMEDIAFORMAT_KEY_I_FRAME_INTERVAL,5);
LOGD("[%s][%s][%d]fromat:%s\n",__FUNCTION__ ,__DATE__,__LINE__,AMediaFormat_toString(format));
//这里配置 format
media_status_t status = AMediaCodec_configure(pMediaCodec,format,NULL,NULL,AMEDIACODEC_CONFIGURE_FLAG_ENCODE);//解码,flags 给0,编码给AMEDIACODEC_CONFIGURE_FLAG_ENCODE
if(status!=0){
LOGD("erro config %d\n",status);
return NULL;
}
AMediaCodec_start(pMediaCodec);
if(status!=0){
LOGD("start erro %d\n",status);
return NULL;
}
int in_frame=0;
int out_frame=0;
int64_t lastOuttime=0;
while(bRun){
//请求buffer
ssize_t bufidx = AMediaCodec_dequeueInputBuffer(pMediaCodec,0);
//LOGD("input buffer %zd\n",bufidx);
if(bufidx>=0) {
size_t bufsize;
int64_t pts = getNowUs();
uint8_t *buf = AMediaCodec_getInputBuffer(pMediaCodec, bufidx, &bufsize);
//填充yuv数据
int frameLenYuv = mW * mH * 3 / 2;
if (fread(buf, 1, frameLenYuv, fp_in) < frameLenYuv) {
fseek(fp_in, 0, SEEK_SET);
fread(buf, 1, frameLenYuv, fp_in);
}
//入队列
LOGD("in[%d] pts:%lld\n", in_frame++, pts);
AMediaCodec_queueInputBuffer(pMediaCodec, bufidx, 0, frameLenYuv, pts, 0);
}
AMediaCodecBufferInfo info;
//取输出buffer
auto outindex = AMediaCodec_dequeueOutputBuffer(pMediaCodec, &info, 0);
if (outindex >= 0) {
//在这里取走编码后的数据
//释放buffer给编码器
size_t outsize;
uint8_t *buf = AMediaCodec_getOutputBuffer(pMediaCodec,outindex,&outsize);
fwrite(buf,1,info.size,fp_out);
if(1) {
AMediaFormat *format2 = AMediaCodec_getOutputFormat(pMediaCodec);
int32_t frameRate,w,h,color;
AMediaFormat_getInt32(format2,AMEDIAFORMAT_KEY_FRAME_RATE,&frameRate);
AMediaFormat_getInt32(format2,AMEDIAFORMAT_KEY_WIDTH,&w);
AMediaFormat_getInt32(format2,AMEDIAFORMAT_KEY_HEIGHT,&h);
LOGD("out[%d] pts %lld %dX%[email protected]%d %d \n",out_frame++,info.presentationTimeUs,w,h,frameRate,info.size);
int64_t nowtime = getNowUs();
LOGD("frame stay times:%lld, out_gap:%lld\n",nowtime-info.presentationTimeUs,nowtime-lastOuttime);
lastOuttime = nowtime;
}
AMediaCodec_releaseOutputBuffer(pMediaCodec, outindex, false);
}
}
}
int main(int argc, const char*argv[]){
int ret =0;
pthread_t pid;
if((ret=pthread_create(&pid,NULL,run,NULL)) !=0 ){
LOGD("thread_create err\n");
return -1;
}
while(1){
usleep(1000*1000);
}
}
#Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE:= codec_demo
LOCAL_SRC_FILES := NdkMediacodec.cpp
#LOCAL_SHARED_LIBRARIES := libandroid libmediandk liblog
LOCAL_LDLIBS := -lmediandk
LOCAL_LDFLAGS += -pie -fPIE
include $(BUILD_EXECUTABLE)
android studio ndk编译出来的文件在 /app/build/intermediates/ndkBuild/debug/obj/local/arm64-v8a/codec_demo
