From b1126ce19c58fc4403283ba8193bf7073b1037be Mon Sep 17 00:00:00 2001 From: Corentin Wallez Date: Wed, 20 Nov 2024 18:34:42 +0100 Subject: [PATCH] wip: textureLoad + aspect for importExternalTexture --- src/resources/README.md | 43 +++--- .../four-colors-h264-bt601-scaled-10x1.mp4 | Bin 0 -> 3192 bytes .../four-colors-vp9-bt601-scaled-10x1.mp4 | Bin 0 -> 2072 bytes .../external_texture/video.spec.ts | 146 ++++++++++++++++++ src/webgpu/web_platform/util.ts | 32 ++++ 5 files changed, 202 insertions(+), 19 deletions(-) create mode 100644 src/resources/four-colors-h264-bt601-scaled-10x1.mp4 create mode 100644 src/resources/four-colors-vp9-bt601-scaled-10x1.mp4 diff --git a/src/resources/README.md b/src/resources/README.md index a1ed0604170c..1db940714778 100644 --- a/src/resources/README.md +++ b/src/resources/README.md @@ -1,93 +1,98 @@ Always use `getResourcePath()` to get the appropriate path to these resources depending on the context (WPT, standalone, worker, etc.) - The test video files were generated with by ffmpeg cmds below: ``` // Generate four-colors-vp8-bt601.webm, mimeType: 'video/webm; codecs=vp8' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-vp8-bt601.webm +ffmpeg -loop 1 -i four-colors.png -c:v libvpx -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-vp8-bt601.webm // Generate four-colors-h264-bt601.mp4, mimeType: 'video/mp4; codecs=avc1.4d400c' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-h264-bt601.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-h264-bt601.mp4 // Generate four-colors-vp9-bt601.webm, mimeType: 'video/webm; codecs=vp9' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-vp9-bt601.webm +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-vp9-bt601.webm // Generate four-colors-vp9-bt709.webm, mimeType: 'video/webm; codecs=vp9' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace bt709 -color_primaries bt709 -color_trc bt709 -color_range tv -vf scale=out_color_matrix=bt709:out_range=tv four-colors-vp9-bt709.webm +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace bt709 -color_primaries bt709 -color_trc bt709 -color_range tv -vf scale=out_color_matrix=bt709:out_range=tv four-colors-vp9-bt709.webm // Generate four-colors-vp9-bt601.mp4, mimeType: 'video/mp4; codecs=vp9' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-vp9-bt601.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv four-colors-vp9-bt601.mp4 ``` Generate video files to test rotation behaviour. Use ffmepg to rotate video content x degrees in cw direction (by using `transpose`) and update transform matrix in metadata through `display_rotation` to x degrees to apply ccw direction rotation. H264 rotated video files are generated by ffmpeg cmds below: + ``` // Generate four-colors-h264-bt601-rotate-90.mp4, mimeType: 'video/mp4; codecs=avc1.4d400c' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2 temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2 temp.mp4 ffmpeg -display_rotation 270 -i temp.mp4 -c copy four-colors-h264-bt601-rotate-90.mp4 rm temp.mp4 // Generate four-colors-h264-bt601-rotate-180.mp4, mimeType: 'video/mp4; codecs=avc1.4d400c' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2,transpose=2 temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2,transpose=2 temp.mp4 ffmpeg -display_rotation 180 -i temp.mp4 -c copy four-colors-h264-bt601-rotate-180.mp4 rm temp.mp4 // Generate four-colors-h264-bt601-rotate-270.mp4, mimeType: 'video/mp4; codecs=avc1.4d400c' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=1 temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=1 temp.mp4 ffmpeg -display_rotation 90 -i temp.mp4 -c copy four-colors-h264-bt601-rotate-270.mp4 rm temp.mp4 - ``` Vp9 rotated video files are generated by ffmpeg cmds below: + ``` // Generate four-colors-vp9-bt601-rotate-90.mp4, mimeType: 'video/mp4; codecs=vp9' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2 temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2 temp.mp4 ffmpeg -display_rotation 270 -i temp.mp4 -c copy four-colors-vp9-bt601-rotate-90.mp4 rm temp.mp4 // Generate four-colors-vp9-bt601-rotate-180.mp4, mimeType: 'video/mp4; codecs=vp9' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2,transpose=2 temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2,transpose=2 temp.mp4 ffmpeg -display_rotation 180 -i temp.mp4 -c copy four-colors-vp9-bt601-rotate-180.mp4 rm temp.mp4 // Generate four-colors-vp9-bt601-rotate-270.mp4, mimeType: 'video/mp4; codecs=vp9' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=1 temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=1 temp.mp4 ffmpeg -display_rotation 90 -i temp.mp4 -c copy four-colors-vp9-bt601-rotate-270.mp4 rm temp.mp4 - ``` Generate video files to test flip behaviour. Use ffmpeg to flip video content. Using `display_hflip` to do horizontal flip and `display_vflip` to do vertical flip. H264 flip video files are generated by ffmpeg cmds below: + ``` // Generate four-colors-h264-bt601-hflip.mp4, mimeType: 'video/mp4; codecs=avc1.4d400c' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 ffmpeg -display_hflip -i temp.mp4 -c copy four-colors-h264-bt601-hflip.mp4 rm temp.mp4 // Generate four-colors-h264-bt601-vflip.mp4, mimeType: 'video/mp4; codecs=avc1.4d400c' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 ffmpeg -display_vflip -i temp.mp4 -c copy four-colors-h264-bt601-vflip.mp4 rm temp.mp4 - ``` Vp9 flip video files are generated by ffmpeg cmds below: + ``` // Generate four-colors-vp9-bt601-hflip.mp4, mimeType: 'video/mp4; codecs=vp09.00.10.08' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 ffmpeg -display_hflip -i temp.mp4 -c copy four-colors-vp9-bt601-hflip.mp4 rm temp.mp4 // Generate four-colors-vp9-bt601-vflip.mp4, mimeType: 'video/mp4; codecs=vp09.00.10.08' -ffmpeg.exe -loop 1 -i .\four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 +ffmpeg -loop 1 -i four-colors.png -c:v libvpx-vp9 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv temp.mp4 ffmpeg -display_vflip -i temp.mp4 -c copy four-colors-vp9-bt601-vflip.mp4 rm temp.mp4 +``` +Scaled 10x1 video files are generated by ffmpeg cmds below: ``` +// Generate four-colors-h264-bt601-scaled-10x1.mp4, mimeType: 'video/mp4; codecs=avc1.4d400c' +ffmpeg -loop 1 -i four-colors.png -c:v libx264 -pix_fmt yuv420p -frames 50 -colorspace smpte170m -color_primaries smpte170m -color_trc smpte170m -color_range tv -vf transpose=2 temp.mp4 +``` \ No newline at end of file diff --git a/src/resources/four-colors-h264-bt601-scaled-10x1.mp4 b/src/resources/four-colors-h264-bt601-scaled-10x1.mp4 new file mode 100644 index 0000000000000000000000000000000000000000..f78540e33898b1c506c34b6233b35747441bcf7e GIT binary patch literal 3192 zcmeHKeQXp(6rbz0r6H66Z4t521sj90z3$$f{n8CRnucnsMNtgVxX$g)^_K45Zg;1> zJBUfKl@Me6!;q?NawiF-eCmR;w1YoUzz_)M#fe5XURxot=z@hP(HJz&) zf?Q}_2Y4z(21AEoCN&bFoS)|d4SwDq0A-X?v#qf)9*_G5RfQNDj`<9$w~_7Qi&9+! zAH$@o5wqJk8Hcef3YB8g_DwO=q9Eh%a9{#VKoL2;Ammy z6S1&PY{*tvGLyiois%O(E7)Ym2uooV+iVlnBF@3$MD2~zFxU)}#H3!s1gonwVQC*p z0=DRXoX@h7uEs#*Y?;J}JV3=D&%(4Y%c(`|C@i;R>GpA%-I7Jwig0=u63dPSg6I$U zc&?uvM&x}>z?ux$p*SHnwSvN?#1w;^YC@HuBcOj-)xGka;z z&FXKbwx5pgc&>TsSGnhO^?TYci;lN@%60}P-=)6Up8nMNh8wSY2by0!b9loJx_8Bm z6Fan2e9Pfyua}%mtSxGWa zgoZaW$&_;&{(c7q|EHw`;fQKLAdv}SQHu;vZ;6}GuzURD!ULWo6rqo5Rf!|FscB} z8{W$Z&G`m-5c|6`3ItFW$?bPM@5YrNRC)%kHWvLWw9nboo9CECgMEuf(LQ(Ecis?$ z)9x&+BMP;Ft0kHZA7WdrM#$5WeuBaF+oURm~-4AL7y?DY|*AP`S%i-X`KCwBT z(RJ*T+l6C_MwpNHH@I^nz*GD2P+zi!@S%d2EVvtqKAfLJO~ zIa4h#v26lf0h3j<8a_?g%r6~5h)uM_rcPCWbq)b(hK2xNI4r9c8|-i}Js-+GD6o0!$nq^ilvBU-wEpRD!r7f!9d^bUX2!v~7)G*E zPn81YvK&xH>m6@iB3~^Xnu6vQq$Ov$wYn``IUB z-}lbeXDHA$JlcQkO(OmL+n()4i ze(Mf(qqN)q*XV~~njSt2y5aRJSPP_LhN$u_1&Dvk zvf=@7(Q|w`ay2W6oMG3N({P@5(;{a@L13N@d5fhe&uV!$^kUPT*ceq6*MSvLb;lKm zs5H+jxPmtgMU&x^45hgD7QhOfnG!o2`xV{JbC9Pg_sWvu56Rje>+SK-~meYIuF;_LB8 d8noO6k3MisCXC>3ok^IKa0ybl1PAyI{04Uy$Atg@ literal 0 HcmV?d00001 diff --git a/src/webgpu/web_platform/external_texture/video.spec.ts b/src/webgpu/web_platform/external_texture/video.spec.ts index b5796f0a1800..e6da48166f05 100644 --- a/src/webgpu/web_platform/external_texture/video.spec.ts +++ b/src/webgpu/web_platform/external_texture/video.spec.ts @@ -252,6 +252,152 @@ for several combinations of video format, video color spaces and dst color space }); }); +g.test('importExternalTexture,loadAndDimensions') + .desc( + ` +Tests that we can import an HTMLVideoElement/VideoFrame into a GPUExternalTexture, textureLoad from it +for several combinations of video format, video color spaces and dst color spaces, and get the correct textureDimensions() +in the shader. +` + ) + .params(u => + u // + .combine('videoName', kVideoNames) + .combine('sourceType', ['VideoElement', 'VideoFrame'] as const) + ) + .fn(async t => { + const { videoName, sourceType } = t.params; + const kDstColorSpace = 'srgb'; + + if (sourceType === 'VideoFrame' && typeof VideoFrame === 'undefined') { + t.skip('WebCodec is not supported'); + } + + const videoElement = getVideoElement(t, videoName); + + await startPlayingAndWaitForVideo(videoElement, async () => { + // Get the video source and its size + let source: HTMLVideoElement | VideoFrame = videoElement; + let expectedWidth = videoElement.videoWidth; + let expectedHeight = videoElement.videoHeight; + if (sourceType === 'VideoFrame') { + const frame = await getVideoFrameFromVideoElement(t, source); + source = frame; + expectedWidth = frame.displayWidth; + expectedHeight = frame.displayHeight; + } + + // Create the pipeline + const module = t.device.createShaderModule({ + code: ` + @vertex fn vs(@builtin(vertex_index) i : u32) -> @builtin(position) vec4f { + const pos = array( + vec2(2, 0), + vec2(-1, 3), + vec2(-1, -3), + ); + return vec4f(vec2f(pos[i]), 0, 1); + } + + @group(0) @binding(0) var dimension : vec2u; + @group(0) @binding(1) var t : texture_external; + @fragment fn fs(@builtin(position) pos : vec4f) -> @location(0) vec4f { + dimension = textureDimensions(t); + let normCoord = pos.xy / vec2f(${kWidth}, ${kHeight}); + let loadCoord = normCoord * vec2f(dimension - vec2u(1, 1)); + return textureLoad(t, vec2u(loadCoord)); + } + `, + }); + const pipeline = t.device.createRenderPipeline({ + layout: 'auto', + vertex: { module }, + fragment: { module, targets: [{ format: kFormat }] }, + }); + + // Create the test resources + const dimensionsBuffer = t.device.createBufferTracked({ + size: 8, + usage: GPUBufferUsage.STORAGE | GPUBufferUsage.COPY_SRC, + }); + + const bindGroup = t.device.createBindGroup({ + layout: pipeline.getBindGroupLayout(0), + entries: [ + { + binding: 0, + resource: { buffer: dimensionsBuffer }, + }, + { + binding: 1, + resource: t.device.importExternalTexture({ source, colorSpace: kDstColorSpace }), + }, + ], + }); + + // Run the test + const colorAttachment = t.createTextureTracked({ + format: kFormat, + size: { width: kWidth, height: kHeight, depthOrArrayLayers: 1 }, + usage: GPUTextureUsage.COPY_SRC | GPUTextureUsage.RENDER_ATTACHMENT, + }); + + const commandEncoder = t.device.createCommandEncoder(); + const passEncoder = commandEncoder.beginRenderPass({ + colorAttachments: [ + { + view: colorAttachment.createView(), + clearValue: { r: 0.0, g: 0.0, b: 0.0, a: 1.0 }, + loadOp: 'clear', + storeOp: 'store', + }, + ], + }); + passEncoder.setPipeline(pipeline); + passEncoder.setBindGroup(0, bindGroup); + passEncoder.draw(6); + passEncoder.end(); + t.device.queue.submit([commandEncoder.finish()]); + + const srcColorSpace = kVideoInfo[videoName].colorSpace; + const presentColors = kVideoExpectedColors[srcColorSpace][kDstColorSpace]; + + // visible rect is whole frame, no clipping. + const expect = kVideoInfo[videoName].display; + + // For validation, we sample a few pixels away from the edges to avoid compression + // artifacts. + t.expectSinglePixelComparisonsAreOkInTexture({ texture: colorAttachment }, [ + // Top-left. + { + coord: { x: kWidth * 0.25, y: kHeight * 0.25 }, + exp: convertToUnorm8(presentColors[expect.topLeftColor]), + }, + // Top-right. + { + coord: { x: kWidth * 0.75, y: kHeight * 0.25 }, + exp: convertToUnorm8(presentColors[expect.topRightColor]), + }, + // Bottom-left. + { + coord: { x: kWidth * 0.25, y: kHeight * 0.75 }, + exp: convertToUnorm8(presentColors[expect.bottomLeftColor]), + }, + // Bottom-right. + { + coord: { x: kWidth * 0.75, y: kHeight * 0.75 }, + exp: convertToUnorm8(presentColors[expect.bottomRightColor]), + }, + ]); + + // Check that we got the correct values reflected in textureDimensions + t.expectGPUBufferValuesEqual( + dimensionsBuffer, + new Uint32Array([expectedWidth, expectedHeight]) + ); + }); + }); + g.test('importExternalTexture,sample_non_YUV_video_frame') .desc( ` diff --git a/src/webgpu/web_platform/util.ts b/src/webgpu/web_platform/util.ts index f000d80e56b9..0eb0a3cc7eb1 100644 --- a/src/webgpu/web_platform/util.ts +++ b/src/webgpu/web_platform/util.ts @@ -343,6 +343,38 @@ export const kVideoInfo = makeTable({ bottomRightColor: 'red', }, }, + 'four-colors-h264-bt601-scaled-10x1.mp4': { + mimeType: 'video/mp4; codecs=avc1.4d400c', + colorSpace: 'bt601', + coded: { + topLeftColor: 'yellow', + topRightColor: 'red', + bottomLeftColor: 'blue', + bottomRightColor: 'green', + }, + display: { + topLeftColor: 'yellow', + topRightColor: 'red', + bottomLeftColor: 'blue', + bottomRightColor: 'green', + }, + }, + 'four-colors-vp9-bt601-scaled-10x1.mp4': { + mimeType: 'video/webm; codecs=vp9', + colorSpace: 'bt601', + coded: { + topLeftColor: 'yellow', + topRightColor: 'red', + bottomLeftColor: 'blue', + bottomRightColor: 'green', + }, + display: { + topLeftColor: 'yellow', + topRightColor: 'red', + bottomLeftColor: 'blue', + bottomRightColor: 'green', + }, + }, }, } as const);