I encountered the following 2 (huge!) memory leaks in Chrome:
- When editing the 'src' of an existing image, with new bytes
- When using clone() to clone an image
Note that in Internet Explorer there is NO memory leak what so ever!
Some background: I'm working on a project where an external camera provides a live feed of images (let's say 100 frames per second).
The main 3 functionalities of the project are:
- play live feed
- record live feed
- show recorded feed
You are welcome to download the following standalone code (simply save it as "leak.html" and execute it), and see for yourself:
<!DOCTYPE html>
<html>
<body>
<canvas id="meCanvas" width="526" height="395"></canvas>
<script src="http://code.jquery.com/jquery-2.0.3.min.js" type="text/javascript"> </script>
<script>
var meContext = document.getElementById("meCanvas").getContext("2d");
// Bytes array representing a chair image
var chairImgSrc = "";
var image = new Image();
image.onload = drawNewImage;
var RECORD_LEN = 20;
var recordedImages = new Array(RECORD_LEN);
var count = 0;
function drawNewImage() {
meContext.clearRect(0, 0, meContext.canvas.width, meContext.canvas.height);
meContext.drawImage(image, 0, 0, meContext.canvas.width, meContext.canvas.height);
setTimeout(nextImage, 10); // Simulates 100 frames per second
}
function drawOldImage() {
var curImage = count % RECORD_LEN; // Cyclic loop over the array
meContext.clearRect(0, 0, meContext.canvas.width, meContext.canvas.height);
meContext.drawImage(recordedImages[curImage], 0, 0, meContext.canvas.width, meContext.canvas.height);
setTimeout(nextImage, 10); // Simulates 100 frames per second
}
function nextImage() {
count++;
if (count <= 1000) // Phase I (during first 10 seconds): use live camera feed
{
// Generating a random image src (Just for this example!!, instead of using the real camera feed)
var newImgSrc = chairImgSrc.slice(0, -11) + ("00000" + count).slice(-6) + "/2Q==";
// (CHROME MEMORY LEAK #1: editing the 'src' of an existing image with new bytes creates a new memory that never gets released)
image.src = newImgSrc;
// Cloning the image, to keep a recorded array of the last N frames
var $tmpImage = $(image);
// (CHROME MEMORY LEAK #2: clone() creates a new memory that never gets released
var clonedImage = $tmpImage.clone()[0];
recordedImages[count % RECORD_LEN] = clonedImage;
}
else // Phase II: use recorded feed
{
drawOldImage();
}
}
window.onload = nextImage;
</script>
</body>
</html>
This example code takes a static image (of a chair) and modifies it randomly every frame (just to simulate my real camera feed).
For the first 1000 frames it shows the image, and stores the last 10 frames in a cyclic array, and from then on it just shows the 10 frames last recorded (in a loop).
(Obviously my real project is much more complicated, I just simplified it to illustrates the problem).
The question is - please suggest an alternative way (preferably - based on the provided source code) to perform the exact same functionality, without causing a memory leak in Chrome.
PS 1:
In chromium I found the following 2 related bugs, which were NOT really fixed (evidence - my code still leaks...):
- "manipulating img src via javascript will generate a massive memory leak" - https://code.google.com/p/chromium/issues/detail?id=36142
- "Memory usage grows infinitely when changing img.src" - https://code.google.com/p/chromium/issues/detail?id=114570
PS 2:
I'm fully aware of existing, similar questions in stackoverflow, and I made a lot of attempts, but none of them helped me solve my problem:
- Rapidly updating image with Data URI causes caching, memory leak
- Canvas even Img eating RAM and CPU
- Refresh image with a new one at the same url
- Setting img.src to dataUrl Leaks Memory
- Memory leak when loading images with javascript's settimeout
Some attempts I made, for example:
- To make sure that the cache is not the cause, I work in Incognito mode of chrome, so cache is not relevant here.
- Instead of setting bytes array as the src, I tried using blob URLs (but a similar leak still occurs):
- img.src = window.URL.createObjectURL(new Blob([bytes.buffer], {type : "image/jpeg"}));
- Tried putting the image in an iframe, and reloading it every X frames: this partially helps, but it's practically impossible for me to use this 'workaround'.
* UPDATE 29/Jan *
I replaced the following lines:
var $tmpImage = $(image);
var clonedImage = $tmpImage.clone()[0];
With:
var clonedImage = new Image();
clonedImage.src = newImgSrc;
and the leak is the same.
=> So I am down to 'only' 1 bug that requires a workaround (in 2 places): leak when editing an image's src.