Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Support for YUV420 and YVU variants. #276

Open
wants to merge 3 commits into
base: master
Choose a base branch
from

Conversation

samlii
Copy link

@samlii samlii commented Jun 7, 2024

This adds encoder support for the following planar formats from v4l.

  • YUV420 (YU12)
  • YVU420 (YV12)

Hopefully its pretty straight forward.

I needed this because scrcpy v4l2 sink (through v4l2loopback) uses the YU12 format on my android phones.

I did notice that I once and a while get -- ERROR [47432.502 ] -- CAP: V4L2 error: grabbed HW buffer=0 is already used. I'm not sure why these happen, however I do think these should probably be Warnings and just continue the loop instead of stopping execution to rebuild the capture. Has this grabbed problem been seen where it didn't resolve itself?

@samlii samlii marked this pull request as ready for review June 7, 2024 17:43
@samlii samlii changed the title Add Support fo YUV420 and 410 and YVU variants. Add Support for YUV420 and 410 and YVU variants. Jun 7, 2024
@mdevaev
Copy link
Member

mdevaev commented Jun 7, 2024

I'm not sure why these happen, however I do think these should probably be Warnings and just continue the loop instead of stopping execution to rebuild the capture.

This is a bug in v4l2loopback. It re-gives a buffer that has already been taken earlier, but has not yet been released. This should never happen, otherwise multithreaded frame processing will be impossible.

@mdevaev
Copy link
Member

mdevaev commented Jun 7, 2024

Btw what about HW and M2M encoders? H.264?

@samlii
Copy link
Author

samlii commented Jun 7, 2024

This is a bug in v4l2loopback. It re-gives a buffer that has already been taken earlier, but has not yet been released. This should never happen, otherwise multithreaded frame processing will be impossible.

Interesting I see the threads now. I wonder if there would be a workaround if we knew it was a loopback (or at least handle it differently for a loopback device.)

Btw what about HW and M2M encoders? H.264?

As I read it hw doesn't make sense since that is only useful for M/JPEG input streams which this is not. The other 2 I didn't understand how they were working.
I don't see anything being handled specially for the packed YUV formats, so I figured it wasn't something I needed to worry about. Where in the M2M stuff is the conversion actually happening? Why it would be different than YUYV?

@mdevaev
Copy link
Member

mdevaev commented Jun 8, 2024

Interesting I see the threads now. I wonder if there would be a workaround if we knew it was a loopback (or at least handle it differently for a loopback device.)

We don't need workaround here, it should be fixed in v4l2loopback. Don't get me wrong, I don't want to disable protection and limit the number of threads simply because someone else's software is not working properly. Bugs should be fixed, not ignored.

As I read it hw doesn't make sense since that is only useful for M/JPEG input streams which this is not.

This is not quite true. M2M works for RAW formats, this is their main purpose. I use this mainly with UYVY. My main goal is that any format should be supported by any encoder.

You can grep format here: https://github.com/pikvm/ustreamer/blob/master/src/ustreamer/m2m.c

@samlii
Copy link
Author

samlii commented Jun 8, 2024

We don't need workaround here, it should be fixed in v4l2loopback. Don't get me wrong, I don't want to disable protection and limit the number of threads simply because someone else's software is not working properly. Bugs should be fixed, not ignored.

I agree it should be fixed upstream, was just postulating whether we could make our experience better until then. It looks like there isn't a good workaround (eventually it still segfaults no matter what I do), so this is a non-issue. I'll see if there is any movement on the v4l2loopback side and try to understand the issue better.

As I read it hw doesn't make sense since that is only useful for M/JPEG input streams which this is not.

This is not quite true. M2M works for RAW formats, this is their main purpose. I use this mainly with UYVY. My main goal is that any format should be supported by any encoder.

You can grep format here: https://github.com/pikvm/ustreamer/blob/master/src/ustreamer/m2m.c

That line was in reference to the encoders/hw/encoder.c code, which explicitly says if it's not jpeg don't use it. As for the M2M interface, I still don't see any checks or things to do for raw formats. It appears the code simply sets the input format to whatever the frame is and hopes that the encoder hardware doesn't fail. I don't have a Pi handy so I can't test the M2M interface as is used in the code, however I have seen references to ffmpeg using the hardware encoder with YUV420 (YU12) based input formats successfully, I don't think this should be any different. Is there a location where I can see what input formats the hardware encoder of the Pi supports?

Sorry not trying to be obtuse here, just trying to understand the M2M stuff and how it is used. My use case is not on a Pi, and I realize that isn't the typical case here, so I need a little guidance when it comes to the Pi. Do you know if there is a decent way to test the M2M stuff if I am not on a Pi?

@mdevaev
Copy link
Member

mdevaev commented Jun 8, 2024

Yes, you are right about hw. I didn't get enough sleep. I don't know how to test M2M without Pi, we probably need something like Nvidia or ATI with the same M2M interface, I'm sure the driver is exists.

Here are all formats:

https://github.com/raspberrypi/linux/blob/rpi-6.6.y/drivers/staging/vc04_services/bcm2835-codec/bcm2835-v4l2-codec.c

@samlii
Copy link
Author

samlii commented Jun 9, 2024

Yes, you are right about hw. I didn't get enough sleep. I don't know how to test M2M without Pi, we probably need something like Nvidia or ATI with the same M2M interface, I'm sure the driver is exists.

I think the problem here is that the M2M available source formats are dependent on the ones that are available to the Pi itself.

https://github.com/raspberrypi/linux/blob/rpi-6.6.y/drivers/staging/vc04_services/bcm2835-codec/bcm2835-v4l2-codec.c

Thanks for this, so according to this source the Pi will support the 420 formats, but not the 410 formats for M2M conversion. I'm okay with removing the 410 formats if you want to try to keep the list of formats to only what can be done in M2M. Just let me know.

I am trying to figure out why the CI build keeps failing. It seems to fail on linting of things that weren't touched in my code. Am I missing something?

@mdevaev
Copy link
Member

mdevaev commented Jun 11, 2024

I'll take a look later, sorry, very busy now.

@samlii samlii changed the title Add Support for YUV420 and 410 and YVU variants. Add Support for YUV420 and YVU variants. Jun 13, 2024
@samlii
Copy link
Author

samlii commented Sep 19, 2024

Hey @mdevaev, just wondering if you had time and if you saw anything else that needed to change?

@mdevaev
Copy link
Member

mdevaev commented Sep 20, 2024

I totally forgot about it ._. Hopefully by the end of this month.

Comment on lines +421 to +424
_LOG_DEBUG("Grabbed HW buffer=%u: bytesused=%u, length=%u, grab_ts=%.3Lf, latency=%.3Lf, skipped=%u, "
"width=%u, height=%u, stride=%u",
buf.index, buf.bytesused, buf.length, (*hw)->raw.grab_ts, us_get_now_monotonic() - (*hw)->raw.grab_ts, skipped,
(*hw)->raw.width, (*hw)->raw.height, (*hw)->raw.stride);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please revert this, there is no point to log resolution because it was reported earlier on configuration stage and didn't changing while working.

Comment on lines +42 to +43
#define US_FORMATS_STR "YUYV, YVYU, UYVY, YU12, YV12,"
#define US_FORMATS2_STR "RGB565, RGB24, BGR24, MJPEG, JPEG"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do it in one line, as it was before. I'll think about how to form this list beautifully automatically, but then.

@@ -85,7 +88,9 @@ void us_cpu_encoder_compress(const us_frame_s *src, us_frame_s *dest, unsigned q
// https://www.fourcc.org/yuv.php
case V4L2_PIX_FMT_YUYV:
case V4L2_PIX_FMT_YVYU:
case V4L2_PIX_FMT_UYVY: _jpeg_write_scanlines_yuv(&jpeg, src); break;
case V4L2_PIX_FMT_UYVY: _jpeg_write_scanlines_yuv(&jpeg, src); break;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't see why this line changed?..

@@ -167,6 +172,62 @@ static void _jpeg_write_scanlines_yuv(struct jpeg_compress_struct *jpeg, const u
free(line_buf);
}

static void _jpeg_write_scanlines_yuv_planar(struct jpeg_compress_struct *jpeg, const us_frame_s *frame) {
uint8_t *line_buf;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Use u8 type.

uint8_t *line_buf;
US_CALLOC(line_buf, frame->width * 3);

US_LOG_DEBUG("Using Planar Encoder");
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you really need logging here?

@@ -377,7 +377,7 @@ int options_parse(us_options_s *options, us_capture_s *cap, us_encoder_s *enc, u
case _O_RESOLUTION: OPT_RESOLUTION("--resolution", cap->width, cap->height, true);
# pragma GCC diagnostic ignored "-Wsign-compare"
# pragma GCC diagnostic push
case _O_FORMAT: OPT_PARSE_ENUM("pixel format", cap->format, us_capture_parse_format, US_FORMATS_STR);
case _O_FORMAT: OPT_PARSE_ENUM("pixel format", cap->format, us_capture_parse_format, US_FORMATS_STR " " US_FORMATS2_STR);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Single macro, as mentioned.

@@ -621,7 +621,9 @@ static void _help(FILE *fp, const us_capture_s *cap, const us_encoder_s *enc, co
SAY(" -i|--input <N> ────────────────────── Input channel. Default: %u.\n", cap->input);
SAY(" -r|--resolution <WxH> ─────────────── Initial image resolution. Default: %ux%u.\n", cap->width, cap->height);
SAY(" -m|--format <fmt> ─────────────────── Image format.");
SAY(" Available: %s; default: YUYV.\n", US_FORMATS_STR);
SAY(" Available: %s", US_FORMATS_STR);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Single macro, as mentioned.

@mdevaev
Copy link
Member

mdevaev commented Dec 15, 2024

Sorry for the long-long delay. Unfortunately, only I can do some things with PiKVM and sometimes priorities are terrifying. Please check my comments on the patch.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

2 participants