Skip to content

Conversation

@Mairramer
Copy link
Contributor

@Mairramer Mairramer commented Oct 3, 2025

This change replaces the previous planesToNV21() implementation with a direct and stride-aware conversion.

The prior version assumed that the U and V planes followed the same layout used by Camera2, which is not always true for CameraX. This assumption could cause color distortion or corrupted frames because CameraX applies internal transformations (rotation, cropping, or proxy wrapping) that modify rowStride, pixelStride, and overall buffer layout.

After investigating inconsistencies between the original areUVPlanesNV21()-based logic and the actual ImageProxy buffers returned by CameraX, this version explicitly reconstructs NV21 by reading each plane row by row while respecting their individual strides, rather than simply validating whether the format already matches NV21.

Fixes flutter/flutter#176410

Pre-Review Checklist

If you need help, consider asking for advice on the #hackers-new channel on Discord.

Note: The Flutter team is currently trialing the use of Gemini Code Assist for GitHub. Comments from the gemini-code-assist bot should not be taken as authoritative feedback from the Flutter team. If you find its comments useful you can update your code accordingly, but if you are unsure or disagree with the feedback, please feel free to wait for a Flutter team member's review for guidance on which automated comments should be addressed.

Footnotes

  1. Regular contributors who have demonstrated familiarity with the repository guidelines only need to comment if the PR is not auto-exempted by repo tooling. 2 3

@gemini-code-assist
Copy link

Using Gemini Code Assist

The full guide for Gemini Code Assist can be found on our documentation page, here are some quick tips.

Invoking Gemini

You can request assistance from Gemini at any point by creating a comment using either /gemini <command> or @gemini-code-assist <command>. Below is a summary of the supported commands on the current page.

Feature Command Description

Customization

To customize Gemini Code Assist for GitHub experience, repository maintainers can create a configuration file and/or provide a custom code review style guide (such as PEP-8 for Python) by creating and adding files to a .gemini/ folder in the base of the repository. Detailed instructions can be found here.

Limitations & Feedback

Gemini Code Assist may make mistakes. Please leave feedback on any instances where its feedback is incorrect or counter productive. You can react with 👍 and 👎 on @gemini-code-assist comments. If you're interested in giving your feedback about your experience with Gemini Code Assist for Github and other Google products, sign up here.

@Mairramer Mairramer marked this pull request as ready for review October 3, 2025 18:24
@Mairramer Mairramer requested a review from camsim99 as a code owner October 3, 2025 18:24
@gemini-code-assist
Copy link

Using Gemini Code Assist
The full guide for Gemini Code Assist can be found on our [documentation page](https://developers.google.com/gemini-code-assist/docs/review-github-code), here are some quick tips.Invoking GeminiYou can request assistance from Gemini at any point by creating a comment using either `/gemini ` or `@gemini-code-assist `. Below is a summary of the supported commands on the current page.Feature | Command | Description--- | --- | ---CustomizationTo customize Gemini Code Assist for GitHub experience, repository maintainers can create a configuration file and/or provide a custom code review style guide (such as PEP-8 for Python) by creating and adding files to a `.gemini/` folder in the base of the repository. Detailed instructions can be found [here](https://developers.google.com/gemini-code-assist/docs/customize-gemini-behavior-github).Limitations & FeedbackGemini Code Assist may make mistakes. Please leave feedback on any instances where its feedback is incorrect or counter productive. You can react with 👍 and 👎 on @gemini-code-assist comments. If you're interested in giving your feedback about your experience with Gemini Code Assist for Github and other Google products, sign up [here](https://google.qualtrics.com/jfe/form/SV_2cyuGuTWsEw84yG).


List<PlaneProxy> planes = Arrays.asList(yPlane, uPlane, vPlane);

ByteBuffer nv21Buffer = ImageProxyUtils.planesToNV21(planes, width, height);
Copy link
Contributor Author

Choose a reason for hiding this comment

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

1280x720 → ~4.6 ms per frame. Further evaluation is needed to assess the impact on the processing pipeline.

Copy link
Contributor

Choose a reason for hiding this comment

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

I assume not but are there any optimizations you know of that can be made?

I'll think on it more (because frankly I'm not sure how big an impact this would cause to google-mlkit users versus say using their own NV21 conversion code in Dart, for example) but right now I definitely lean on the side of landing this to correct the implementation. Then, we can investigate how to make this faster.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I’m considering using one or two ThreadLocal instances here. This might reduce memory allocation and garbage collection (GC) pressure. I still need to validate it later.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I ran some tests on my S23 and LG K12+, both in debug mode. The results are as follows:

Device Frames Recorded First Frame (ms) Subsequent Frame Time (ms) Average (excluding first frame)
S23 740 21.9 ~1.0 to ~2.0 ~1.35
LG K12+ 540 172.42 6.83 to 16.19 ~10

I considered using ThreadLocal, but it didn’t work very well. I made some adjustments in the iteration and used System.arraycopy. This reduced the processing time by 50% on the LG K12+. On the S23 in profile mode, it dropped below 1 ms.

@stuartmorgan-g stuartmorgan-g added the triage-android Should be looked at in Android triage label Oct 7, 2025
@camsim99
Copy link
Contributor

camsim99 commented Oct 7, 2025

@Mairramer two things -- First, can you possibly update to the PR description to describe how you came to these changes? I'm not an expert in the conversion so it would help me review this PR.

Second, I assumed the fix would be using this logic I left out initially:
https://github.com/googlesamples/mlkit/blob/da17257a78b9beedb57b7a9795b911296ae970a0/android/automl/app/src/main/java/com/google/mlkit/vision/automl/demo/BitmapUtils.java#L322-L330

Thoughts on that versus your approach?

@Mairramer
Copy link
Contributor Author

Mairramer commented Oct 7, 2025

Second, I assumed the fix would be using this logic I left out initially: https://github.com/googlesamples/mlkit/blob/da17257a78b9beedb57b7a9795b911296ae970a0/android/automl/app/src/main/java/com/google/mlkit/vision/automl/demo/BitmapUtils.java#L322-L330

Thoughts on that versus your approach?

While this approach is valid, it still encounters the same issue when checking areUVPlanesNV21().

My implementation essentially replicates part of the unpackPlane() logic, avoiding assumptions about the U/V buffer layout and ensuring a deterministic NV21 output. Additionally, it leverages CameraX’s internal ImageUtil
for a more robust solution.

Both approaches perform a complete manual reconstruction of the image, guaranteeing correctness regardless of how the U/V planes are arranged.

I am still investigating why position == limit occurs in some CameraX buffers. Understanding this behavior could help refine the reconstruction logic.

@Mairramer
Copy link
Contributor Author

My only concern is the per-frame processing time.

@camsim99
Copy link
Contributor

camsim99 commented Oct 9, 2025

Thanks for the clarification! This fix makes sense at a high level to me. I'll start reviewing this!

@mboetger
Copy link
Contributor

@camsim99 - Can you take another look?

Copy link
Contributor

@camsim99 camsim99 left a comment

Choose a reason for hiding this comment

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

This LGTM thanks! Just left some nits + a question both for clarity.

System.arraycopy(row, 0, nv21Bytes, position, width);
position += width;
if (rowIndex < height - 1) {
yBuffer.position(yBuffer.position() - width + yRowStride);
Copy link
Contributor

Choose a reason for hiding this comment

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

I assume this line is what makes the case with padding different from the case without passing. Can you explain why? Maybe leave a comment too?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yes, exactly. I'll add a brief comment to the code.

Copy link
Contributor

Choose a reason for hiding this comment

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

Gotcha thanks!

// Restore buffers to their initial state.
vBuffer.position(vBufferPosition);
uBuffer.limit(uBufferLimit);
for (int col = 0; col < width / 2; col++) {
Copy link
Contributor

Choose a reason for hiding this comment

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

Nit: can you add a comment to signal (if I'm understanding correctly) that this is adding the overlapping V and U buffers to the NV21 buffer?

Copy link
Contributor

@camsim99 camsim99 left a comment

Choose a reason for hiding this comment

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

Thanks for the quick fixes and apologies for the slow review on my part! @bparrishMines would you mind giving this a second review?

@reidbaker
Copy link
Contributor

Approved and I resolved the incoming conflicts. Full transparency I also added to the changelog that the error would happen when using NV21 format.

@camsim99 this feels like a patch version change since no outward facing apis were modified and this literally fixes a bug. That said I left the version along since you had already approved it.

@camsim99
Copy link
Contributor

Ahh that's a mistake on my part; a patch version makes sense. Just went ahead and modified it so we can get this landed!

@camsim99 camsim99 added the autosubmit Merge PR when tree becomes green via auto submit App label Oct 23, 2025
@auto-submit auto-submit bot merged commit 5502041 into flutter:main Oct 23, 2025
80 checks passed
engine-flutter-autoroll added a commit to engine-flutter-autoroll/flutter that referenced this pull request Oct 24, 2025
github-merge-queue bot pushed a commit to flutter/flutter that referenced this pull request Oct 24, 2025
flutter/packages@9ec29b6...53d6138

2025-10-23 49699333+dependabot[bot]@users.noreply.github.com
[dependabot]: Bump com.google.ads.interactivemedia.v3:interactivemedia
from 3.37.0 to 3.38.0 in /packages/interactive_media_ads/android
(flutter/packages#10262)
2025-10-23 [email protected]
[camera_android_camerax] Modify plugin to assume mp4 format for captured
videos (flutter/packages#10273)
2025-10-23 [email protected]
[camera_android_x] Refactor ImageProxyUtils.planesToNV21 to prevent
buffer issues (flutter/packages#10163)

If this roll has caused a breakage, revert this CL and stop the roller
using the controls here:
https://autoroll.skia.org/r/flutter-packages-flutter-autoroll
Please CC [email protected] on the revert to ensure that a
human
is aware of the problem.

To file a bug in Flutter:
https://github.com/flutter/flutter/issues/new/choose

To report a problem with the AutoRoller itself, please file a bug:
https://issues.skia.org/issues/new?component=1389291&template=1850622

Documentation for the AutoRoller is here:
https://skia.googlesource.com/buildbot/+doc/main/autoroll/README.md
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

autosubmit Merge PR when tree becomes green via auto submit App p: camera platform-android triage-android Should be looked at in Android triage

Projects

None yet

Development

Successfully merging this pull request may close these issues.

[camera_android_camerax] Setting NV21 format for image streaming sometimes throws java.lang.IllegalArgumentException: newPosition > limit

5 participants