У меня получилось искаженное изображение при попытке конвертировать YUV420p в RGB24 с использованием sws_scale.ffmpeg sws_scale получил искаженное изображение от YUV420P до RGB24
Код:
ret = avcodec_decode_video2(video_dec_ctx, frame, got_frame, &pkt);
if (ret < 0) {
fprintf(stderr, "Error decoding video frame\n");
return ret;
}
if (*got_frame)
{
printf("video_frame%s n:%d coded_n:%d pts:%s\n",
cached ? "(cached)" : "",
video_frame_count++, frame->coded_picture_number,
"#"/*av_ts2timestr(frame->pts, &video_dec_ctx->time_base)*/);
/* copy decoded frame to destination buffer:
* this is required since rawvideo expects non aligned data */
av_image_copy(video_dst_data, video_dst_linesize,
(const uint8_t **)(frame->data), frame->linesize,
video_dec_ctx->pix_fmt, video_dec_ctx->width, video_dec_ctx->height);
/* write to rawvideo file */
fwrite(video_dst_data[0], 1, video_dst_bufsize, video_dst_file);
AVPicture pic;
avpicture_alloc(&pic, AV_PIX_FMT_RGB24, frame->width, frame->height);
SwsContext *ctxt = sws_getContext(frame->width, frame->height, static_cast<AVPixelFormat>(frame->format),
frame->width, frame->height, AV_PIX_FMT_RGB24, SWS_BILINEAR, NULL, NULL, NULL);
if (NULL == ctxt)
{
//Log("failed to get sws context");
}
if (0 < sws_scale(ctxt, frame->data, frame->linesize, 0, frame->height, pic.data, pic.linesize))
{
char szPic[256] = { 0 };
sprintf(szPic, "decoded/%d.bmp", video_frame_count);
FILE *pf = fopen(szPic,"w");
if (NULL != pf)
{
BITMAPFILEHEADER bmpFileHeader = {0};
bmpFileHeader.bfReserved1 = 0;
bmpFileHeader.bfReserved2 = 0;
bmpFileHeader.bfType = 0x4D42;
bmpFileHeader.bfSize = sizeof(bmpFileHeader) + sizeof(BITMAPINFOHEADER) + pic.linesize[0] * frame->height;
bmpFileHeader.bfOffBits = sizeof(bmpFileHeader) + sizeof(BITMAPINFOHEADER);
BITMAPINFOHEADER bmiHeader = { 0 };
bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
bmiHeader.biWidth = frame->width;
bmiHeader.biHeight = 0 - frame->height;
bmiHeader.biPlanes = 1;
bmiHeader.biBitCount = 24;
bmiHeader.biCompression = BI_RGB;
bmiHeader.biSizeImage = pic.linesize[0] * frame->height;
bmiHeader.biXPelsPerMeter = 0;
bmiHeader.biYPelsPerMeter = 0;
bmiHeader.biClrUsed = 0;
bmiHeader.biClrImportant = 0;
fwrite(&bmpFileHeader, 1, sizeof(bmpFileHeader), pf);
fwrite(&bmiHeader, 1, sizeof(bmiHeader), pf);
fwrite(pic.data[0], 1, pic.linesize[0] * frame->height, pf);
fclose(pf);
}
}
// pic.data[0] now contains the image data in RGB format (3 bytes)
// and pic.linesize[0] is the pitch of the data (ie. size of a row in memory, which can be larger than width*sizeof(pixel))
avpicture_free(&pic);
sws_freeContext(ctxt);
}
выше декодировать только кадр затем преобразовать это от RGB24, затем записать растровое изображение. оригинальная видеокадра, подобная этому, но преобразованное изображение, есть ли какой-либо код или какой-либо код является неправильным?
благодарит заранее.
Несколько вещей здесь: '1' неправильный порядок компонентов RGB vs BGR,' 2' высота в файле должна быть положительной, размер строки '3' в файле должен иметь минимальное отступление до 32-битной границы, хотя SWS-контекст может быть больше. Все вместе вы получите правильный BMP-файл. –
thks, Roman R. Но высота может быть отрицательной, если здесь я даю ей положительное значение, растровое изображение будет обратным изображением; BTW, размер изображения составляет 1280 x 760, поэтому он 32-битный; Я также попробовал BGR24, тот же результат. –
Код примерно вправо, поэтому во время выполнения есть что-то не так. Например, SWS-контекст может иметь ширину 1288, а не 1280 и т. Д. Это нормально для обработки, но не для файла. Возможно, вы можете разместить ссылку на файл, который вы получаете. –