While profiling my Google Cardboard application, I found out a very large memory leak (15Mb!) each time I left the activity with the 3D graphics.
After a long and grievous investigation, I found out that the source of the problem was a Context leak that happened each time I closed my CardboardActivity
subclass.
The solution can be found in the accepted answer*
*
wow... this is awkward... Note for any kind (and experienced) reviewer: I am writing a question to whom I know the answer already: am I supposed to do something for style, like add some fake suspense ("will our heroes prevail?! Find out in the accepted answer!"), like in a old Batman TV series or something?
After dicing and slicing my
CardboardActivity
subclass, until nothing else but the base class remained, I had to conclude that the base class itself was leaking the context.I searched the web and found this post explaining how the activity in question leaked the context by failing to un-register a listener with a private instance of a class.
Upon trying to invoke said method manually (using reflection), I found out that in the current version of the Cardboard SDK (0.5.4 at the time of writing), the field is not present anymore.
Long story short: all sensors are now handled by an undocumented (yet public) SensorConnection class instantiated in
CardboardActivity
as asensorConnection
field, which is still plagued by the bug detailed in my first link.This led me to this solution:
sensorConnection
field in theCardboardActivity
by reflectionmagneticSensor
field, again by reflectionsetOnCardboardTheaterListener
withnull
argument, to clear the binding holding the reference to theContext
in theActivity
onDestroy
method.this boils down to the following code:
which solved the problem entirely.
A word to the wise: since this solution relies on reflection, it might break (without consequences other than doing nothing, likely) as soon as Google will update the SDK (possibly fixing the issue in a clean way).
Hope this helps someone