forked from mirrors/gecko-dev
		
	Backed out changeset fccadc284fd9 (bug 1753275) for causing perma mochitest failures @ dom/canvas/test/test_capture_throttled.html CLOSED TREE
This commit is contained in:
		
							parent
							
								
									45ead03a6d
								
							
						
					
					
						commit
						51479c81df
					
				
					 3 changed files with 65 additions and 37 deletions
				
			
		|  | @ -1,35 +1,45 @@ | ||||||
| <!DOCTYPE html> | <!DOCTYPE HTML> | ||||||
| <meta charset="utf-8" /> | <meta http-equiv="content-type" content="text/html; charset=utf-8" /> | ||||||
| 
 | 
 | ||||||
| <title>Canvas2D test: CaptureStream() with throttled rAF</title> | <title>Canvas2D test: CaptureStream() with throttled rAF</title> | ||||||
| 
 | 
 | ||||||
| <script src="/tests/SimpleTest/SimpleTest.js"></script> | <script src="/tests/SimpleTest/SimpleTest.js"></script> | ||||||
| <script src="captureStream_common.js"></script> | <script src="captureStream_common.js"></script> | ||||||
| <link rel="stylesheet" href="/tests/SimpleTest/test.css" /> | <link rel="stylesheet" href="/tests/SimpleTest/test.css"> | ||||||
|  | <body> | ||||||
| <script> | <script> | ||||||
| SimpleTest.waitForExplicitFinish(); | SimpleTest.waitForExplicitFinish(); | ||||||
| SimpleTest.requestFlakyTimeout( | SimpleTest.requestFlakyTimeout("Ensuring nothing happens until timing out with good margin"); | ||||||
|   "Ensuring nothing happens until timing out with good margin" |  | ||||||
| ); |  | ||||||
| 
 | 
 | ||||||
| // CaptureStreamTestHelper holding utility test functions. | // CaptureStreamTestHelper holding utility test functions. | ||||||
| const h = new CaptureStreamTestHelper2D(); | const h = new CaptureStreamTestHelper2D(); | ||||||
| 
 | 
 | ||||||
| async function measureRequestAnimationFrameRate() { | async function measureRequestAnimationFrameRate() { | ||||||
|   const start = await new Promise(r => requestAnimationFrame(r)); |   const frameRate = await new Promise(resolve => { | ||||||
|   for (let count = 1; ; count++) { |     let start; | ||||||
|     const time = await new Promise(r => requestAnimationFrame(r)); |     let count; | ||||||
|  |     const tick = time => { | ||||||
|  |       if (!start) { | ||||||
|  |         start = time; | ||||||
|  |         count = 0; | ||||||
|  |       } else { | ||||||
|  |         count += 1; | ||||||
|  |       } | ||||||
|       if (time - start > 1000) { |       if (time - start > 1000) { | ||||||
|         // One second has passed, break. |         // One second has passed, break. | ||||||
|       return count / ((time - start) / 1000); |         resolve(count / ((time - start) / 1000)); | ||||||
|     } |         return; | ||||||
|       } |       } | ||||||
|  |       window.requestAnimationFrame(tick); | ||||||
|  |     }; | ||||||
|  |     window.requestAnimationFrame(tick); | ||||||
|  |   }); | ||||||
|  |   return frameRate; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async function measureSetTimeoutRate() { | async function measureSetTimeoutRate() { | ||||||
|   // The average isn't accurate at low counts. |  | ||||||
|   const COUNT = 25; |  | ||||||
|   const start = performance.now(); |   const start = performance.now(); | ||||||
|  |   const COUNT = 5; | ||||||
|   for(let i = 0; i < COUNT; ++i) { |   for(let i = 0; i < COUNT; ++i) { | ||||||
|     await new Promise(resolve => setTimeout(resolve, 0)); |     await new Promise(resolve => setTimeout(resolve, 0)); | ||||||
|   } |   } | ||||||
|  | @ -38,7 +48,7 @@ async function measureSetTimeoutRate() { | ||||||
| 
 | 
 | ||||||
| async function measureCanvasCaptureFrameRate(captureRate) { | async function measureCanvasCaptureFrameRate(captureRate) { | ||||||
|   // Canvas element captured by streams. |   // Canvas element captured by streams. | ||||||
|   const c = h.createAndAppendElement("canvas", "c"); |   const c = h.createAndAppendElement('canvas', 'c'); | ||||||
| 
 | 
 | ||||||
|   // Since we are in a background tab, the video element won't get composited, |   // Since we are in a background tab, the video element won't get composited, | ||||||
|   // so we cannot look for a frame count there. Instead we use RTCPeerConnection |   // so we cannot look for a frame count there. Instead we use RTCPeerConnection | ||||||
|  | @ -47,42 +57,59 @@ async function measureCanvasCaptureFrameRate(captureRate) { | ||||||
|   const pc2 = new RTCPeerConnection(); |   const pc2 = new RTCPeerConnection(); | ||||||
| 
 | 
 | ||||||
|   // Add the canvas.captureStream track. |   // Add the canvas.captureStream track. | ||||||
|   const ctx = c.getContext("2d"); |   const ctx = c.getContext('2d'); | ||||||
|   const [track] = c.captureStream(captureRate).getTracks(); |   const [track] = c.captureStream(captureRate).getTracks(); | ||||||
|   const sender = pc1.addTrack(track); |   const sender = pc1.addTrack(track); | ||||||
| 
 | 
 | ||||||
|   // Ice candidates signaling |   // Ice candidates signaling | ||||||
|   pc1.onicecandidate = e => pc2.addIceCandidate(e.candidate); |   for (const [local, remote] of [[pc1, pc2], [pc2, pc1]]) { | ||||||
|   pc2.onicecandidate = e => pc1.addIceCandidate(e.candidate); |     local.addEventListener("icecandidate", ({candidate}) => { | ||||||
|  |       if (!candidate || remote.signalingState == "closed") { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       remote.addIceCandidate(candidate); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|   // Offer/Answer exchange |   // Offer/Answer exchange | ||||||
|   await pc1.setLocalDescription(); |   await pc1.setLocalDescription(await pc1.createOffer()); | ||||||
|   await pc2.setRemoteDescription(pc1.localDescription); |   await pc2.setRemoteDescription(pc1.localDescription); | ||||||
|   await pc2.setLocalDescription(); |   await pc2.setLocalDescription(await pc2.createAnswer()); | ||||||
|   await pc1.setRemoteDescription(pc2.localDescription); |   await pc1.setRemoteDescription(pc2.localDescription); | ||||||
| 
 | 
 | ||||||
|   // Wait for RTP packets to arrive. |   // Wait for connection | ||||||
|   const event = await new Promise(r => pc2.ontrack = r); |   while ([pc1, pc2].some(pc => pc.iceConnectionState == "new" || | ||||||
|   await new Promise(r => event.track.onunmute = r); |                                pc.iceConnectionState == "checking")) { | ||||||
|  |     await Promise.any( | ||||||
|  |       [pc1, pc2].map(p => new Promise(r => p.oniceconnectionstatechange = r))); | ||||||
|  |   } | ||||||
|  |   for (const [pc, name] of [[pc1, "pc1"], [pc2, "pc2"]]) { | ||||||
|  |     ok(["connected", "completed"].includes(pc.iceConnectionState), | ||||||
|  |       `${name} connection established (${pc.iceConnectionState})`); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|   // Draw to the canvas |   // Draw to the canvas | ||||||
|   const intervalMillis = 1000 / 60; |   const intervalMillis = 1000 / 60; | ||||||
|   const getFrameCount = async () => { |   const getFrameCount = async () => { | ||||||
|     const stats = await sender.getStats(); |     const stats = await sender.getStats(); | ||||||
|     const outbound = [...stats.values()].find( |     const outbound = | ||||||
|       ({ type }) => type == "outbound-rtp" |       [...stats.values()].find(({type}) => type == "outbound-rtp"); | ||||||
|     ); |     return outbound?.framesEncoded ?? 0; | ||||||
|     return outbound?.framesEncoded ?? 0; // See bug 1789768. |  | ||||||
|   }; |   }; | ||||||
|   is(await getFrameCount(), 0, "frame count starts at 0"); |   // Wait for frame count change to ensure sender is working. | ||||||
|  |   while (await getFrameCount() == 0) { | ||||||
|  |     h.drawColor(c, h.green); | ||||||
|  |     await new Promise(resolve => setTimeout(resolve, intervalMillis)); | ||||||
|  |   } | ||||||
|  |   const startFrameCount = await getFrameCount(); | ||||||
|   const start = performance.now(); |   const start = performance.now(); | ||||||
|   let end; |   let end = start; | ||||||
|   do { |   while(end - start <= 1000) { | ||||||
|     h.drawColor(c, h.green); |     h.drawColor(c, h.green); | ||||||
|     await new Promise(resolve => setTimeout(resolve, intervalMillis)); |     await new Promise(resolve => setTimeout(resolve, intervalMillis)); | ||||||
|     end = performance.now(); |     end = performance.now(); | ||||||
|   } while (end - start <= 1000); |   } | ||||||
|   const framerate = (await getFrameCount()) / ((end - start) / 1000); |   const framerate = (await getFrameCount() - startFrameCount) / ((end - start) / 1000); | ||||||
|   pc1.close(); |   pc1.close(); | ||||||
|   pc2.close(); |   pc2.close(); | ||||||
|   return framerate; |   return framerate; | ||||||
|  | @ -121,3 +148,4 @@ async function measureCanvasCaptureFrameRate(captureRate) { | ||||||
|   SimpleTest.finish(); |   SimpleTest.finish(); | ||||||
| })(); | })(); | ||||||
| </script> | </script> | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | @ -163,7 +163,7 @@ class RequestedFrameRefreshObserver : public nsARefreshObserver { | ||||||
|             ? now |             ? now | ||||||
|             : mLastCaptureTime + TimeDuration::FromMilliseconds( |             : mLastCaptureTime + TimeDuration::FromMilliseconds( | ||||||
|                                      nsRefreshDriver::DefaultInterval()); |                                      nsRefreshDriver::DefaultInterval()); | ||||||
|     if (next <= now) { |     if (mLastCaptureTime.IsNull() || next <= now) { | ||||||
|       AUTO_PROFILER_MARKER_TEXT("Canvas CaptureStream", MEDIA_RT, {}, |       AUTO_PROFILER_MARKER_TEXT("Canvas CaptureStream", MEDIA_RT, {}, | ||||||
|                                 "CaptureFrame direct while throttled"_ns); |                                 "CaptureFrame direct while throttled"_ns); | ||||||
|       CaptureFrame(now); |       CaptureFrame(now); | ||||||
|  |  | ||||||
|  | @ -88,7 +88,7 @@ class TimerDriver : public OutputStreamDriver { | ||||||
|       str.AppendPrintf( |       str.AppendPrintf( | ||||||
|           "TimerDriver %staking frame (%sexplicitly requested; after %.2fms; " |           "TimerDriver %staking frame (%sexplicitly requested; after %.2fms; " | ||||||
|           "interval cap %.2fms)", |           "interval cap %.2fms)", | ||||||
|           FrameCaptureRequested(aTime) ? "" : "NOT ", |           sinceLast >= mFrameInterval ? "" : "NOT ", | ||||||
|           mExplicitCaptureRequested ? "" : "NOT ", sinceLast.ToMilliseconds(), |           mExplicitCaptureRequested ? "" : "NOT ", sinceLast.ToMilliseconds(), | ||||||
|           mFrameInterval.ToMilliseconds()); |           mFrameInterval.ToMilliseconds()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Sandor Molnar
						Sandor Molnar