У меня есть AV_PIX_FMT_YUV420P AVFrame, и я хочу записать его как файл PNG или JPEG, неважно, какой формат внутри (BGR, BGRA, RGB или RGBA). Я попытался сохранить его как AV_PIX_FMT_YUV420P, но у меня было 3 изображения в одном, поэтому мне нужно сначала преобразовать его в RGB24. Я делаю это через:
int destW = decCtx->width;
int destH = decCtx->height;
AVFrame * resFrame = av_frame_alloc();
resFrame->linesize[0] = destW * 3;
resFrame->data[0] = (uint8_t*)malloc(resFrame->linesize[0] * destH);
SwsContext * ctx = sws_getContext(decCtx->width,
decCtx->height,
AV_PIX_FMT_YUV420P,
decCtx->width,
decCtx->height,
AV_PIX_FMT_RGB24,
SWS_FAST_BILINEAR, 0, 0, 0);
sws_scale(ctx, frame->data, frame->linesize, 0,
decCtx->height, resFrame->data, resFrame->linesize);
sws_freeContext(ctx);
int64_t pts = frame->pts;
resFrame->format = AV_PIX_FMT_RGB24;
resFrame->pts = pts;
Похоже, у меня есть правильный преобразованный кадр, но теперь кодировщик возвращает мне пакет с размером 86, так что мое изображение фактически пустое или близко к нему. Я кодирую через:
encCodec = avcodec_find_encoder(AV_CODEC_ID_PNG);
if (!encCodec) {
return false;
}
encCtx = avcodec_alloc_context3(encCodec);
if (!encCtx) {
return false;
}
encCtx->bit_rate = decCtx->bit_rate;
encCtx->codec_type = AVMEDIA_TYPE_VIDEO;
encCtx->thread_count = 1;
encCtx->width = decCtx->width;
encCtx->height = decCtx->height;
int fps = fmtc->streams[iVideoStream]->r_frame_rate.num / fmtc->streams[iVideoStream]->r_frame_rate.den;
encCtx->time_base = (AVRational){1, fps};
encCtx->framerate = (AVRational){fps, 1};
encCtx->gop_size = decCtx->gop_size;
encCtx->max_b_frames = decCtx->max_b_frames;
encCtx->pix_fmt = AV_PIX_FMT_RGB24;
int ret = avcodec_parameters_from_context(fmtc->streams[iVideoStream]->codecpar, encCtx);
if (ret < 0) {
return false;
}
ret = avcodec_open2(encCtx, encCodec, nullptr);
if (ret < 0) {
return false;
}
ret = avcodec_send_frame(encCtx, source);
if (ret < 0) {
return false;
}
av_packet_unref(pkt);
av_init_packet(pkt);
ret = avcodec_receive_packet(encCtx, pkt);
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
return false;
else if (ret < 0) {
return false;
}
if (ret >= 0)
{
FILE * outPng = fopen("./sample.png", "wb");
fwrite(pkt->data, pkt->size, 1, outPng);
fclose(outPng);
}
Что я делаю не так с преобразованием изображения в RGB24 или с его кодированием?