I have a raw yuv420p file at 1860x1920, fill it to cv::Mat, then use libyuv::I420Scale
shrunk to a quarter of its original size,but get wired outout, for example there are some green stripes like this:
Use ffplay -f rawvideo -i yuv_image.yuv -video_size 1860x1920
, the origin image here:
The scaled image is not normal, it looks like the yuv data is not aligned
What am I doing wrong, code below:
#define ALIGN32(n) (((n) >> 5) << 5)
// yuv420_image is yuv420 cv::Mat
int src_width = yuv420_image.cols, src_height = yuv420_image.rows / 3 * 2;
const uint8_t* src = yuv420_image.data;
const uint8_t* src_y = src;
const uint8_t* src_u = src_y + src_width * src_width;
const uint8_t* src_v = src_u + src_width * src_width / 4;
int dst_width = ALIGN32(src_width / kScaleRatio), dst_height = ALIGN32(src_height / kScaleRatio);
cv::Mat yuv420_scaled(dst_height * 3 / 2, dst_width,
CV_8UC1);
uint8_t* dst = yuv420_scaled.data;
uint8_t* dst_y = dst;
uint8_t* dst_u = dst_y + dst_width * dst_height;
uint8_t* dst_v = dst_u + dst_width * dst_height / 4;
int result = libyuv::I420Scale(src_y, src_width,
src_u, src_width / 2,
src_v, src_width / 2,
src_width, src_width,
dst_y, dst_width,
dst_u, dst_width / 2,
dst_v, dst_width / 2,
dst_width, dst_height,
libyuv::FilterModeEnum::kFilterNone);
The issue is mixing
width
andheight
in 3 places.Issues:
src_width
andsrc_height
:const uint8_t* src_u = src_y + src_width * src_width;
const uint8_t* src_v = src_u + src_width * src_width / 4;
Supposed to be:
src_width
andsrc_height
:int result = libyuv::I420Scale(src_y, src_width,
src_u, src_width / 2,
src_v, src_width / 2,
src_width, src_width,
Supposed to be:
For testing I created a synthetic video frame in YUV420p (I420) pixel format using FFmpeg:
ffmpeg -f lavfi -i testsrc=1860x1920:rate=1:duration=1 -vf scale=out_color_matrix=bt709:out_range=full -pix_fmt yuv420p yuv_image.yuv
The following code sample downscales the image to 448x480.
The code uses OpenCV for converting the result to BGR (for testing).
Output (after converting to BGR):
