Frame Rate Hitch

I am experiencing what appears to be a periodic (~15 second interval) frame rate hitch in my app when running on a 2nd generation ipod touch. When it occurs the frame rate momentarily drops from 30fps to about 27.5. As the game I am working on relies on smooth controls, this hitch is rather disturbing.

I have tried disabling pretty much everything to isloate the cause, down to the point where I have nothing in motion and am only rendering a single piece of static geometry.

I also notice the same periodic hitch in the front-end screen which only has 2 guiText objects and a simple script to detect input.

Does anyone have any suggestions for what might be causing this, or how to track down the problem?

Why are your controls framerate dependent? Visually, why would anyone even notice a drop of that small an increment, for a short time; are you VBL syncing?

Um, can you even sync with VBL? I thought all of the Player Quality Settings were irrelevant when building for iPhone.

I often see a drop like that, though it is BIGGER… 30 fps, down to 11 fps… then right back up again. Have not worked out what it is yet…

I would assume it would drop to 20, instead of 27.5, based on this, but the official info is scarce:

30 to 20 is better than 30 to 11, though. (which is neither 12 nor 10… :? )

Most likely he’s using the frame rate tracker that’s posted on the wiki (or something similar) that polls the framerate over a short time, which gives him an average for that time period. He’s probably actually dropping to 15 fps during that time.

I’ve seen this quite a bit in my own game. Things are generally smooth, but every now and then some unexplained hitches occur. Garbage collection and heap allocations do a lot to make this occur, but I feel that there are some other, unexplained issues occurring as well.

Yes, I can confirm that I am using the frame rate tracker from the wiki which reports an average.

The controls are not frame-rate dependent, however the camera (which may be tracking the player for instance), will have a much larger displacement when this hiccup occurs, which is visually disturbing for the player.

As I am also seeing this hitch in the front-end with very little going on, (2xguiTextObjects and a simple script to get input and load the next scene), I am wondering if it is either:
a) something unusual in my project settings?
or
b) something systemic to unity iphone

I am really hoping it is a! :slight_smile:

Could you enable #define ENABLE_INTERNAL_PROFILER 1 in AppController.mm, try to “catch” moment when framerate drops and post results from XCode console.

From the top of my head I would suspect GarbageCollector. But it can be anything else :slight_smile: First of all iPhone OS is not really a realtime OS, plus there are number of processes running in the background which can affect performance of your game. So drops from 30fps to occasional 20fps could be considered a norm.

In general your rendering code should be running frametime independently. You should multiply your camera movements by the Time.deltaTime.

NSTimers (like the timer that triggers rendering) are just events in the runloop, like touch events, and anything that can delay the runloop can delay the repaint.

ReJ: have you thought about using a more reliable timer, like maybe a CoreAudio timer? Not sure if that’s even possible, just a thought.

Hi ReJ,

Thanks for looking into this. I setup a very simple test scene containing a camera and a textured plane. The problem still persists on a roughly 15 second interval.

The profiler output was:

iPhone Unity internal profiler stats:
cpu-player> min: 5.4 max: 12.4 avg: 6.9
cpu-ogles-drv> min: 0.5 max: 1.6 avg: 0.6
cpu-present> min: 0.6 max: 1.7 avg: 0.7
frametime> min: 15.1 max: 18.1 avg: 16.7
draw-call #> min: 4 max: 4 avg: 4
tris #> min: 42 max: 42 avg: 42
verts #> min: 64 max: 64 avg: 64
player-detail> physx: 1.1 animation: 0.0 skinning: 0.0 render: 5.3 fixed-update-count: 2 … 2
mono-scripts> update: 0.2 fixedUpdate: 0.0 coroutines: 0.0
mono-memory> used heap: 274432 allocated heap: 3657728

iPhone Unity internal profiler stats:
cpu-player> min: 5.0 max: 91.2 avg: 9.5
cpu-ogles-drv> min: 0.5 max: 1.4 avg: 0.6
cpu-present> min: 0.6 max: 2.0 avg: 0.7
frametime> min: 15.3 max: 101.1 avg: 19.4
draw-call #> min: 4 max: 4 avg: 4
tris #> min: 42 max: 42 avg: 42
verts #> min: 64 max: 64 avg: 64
player-detail> physx: 1.1 animation: 0.0 skinning: 0.0 render: 7.9 fixed-update-count: 2 … 12
mono-scripts> update: 0.2 fixedUpdate: 0.0 coroutines: 0.0
mono-memory> used heap: 278528 allocated heap: 3657728

iPhone Unity internal profiler stats:
cpu-player> min: 5.0 max: 9.8 avg: 6.6
cpu-ogles-drv> min: 0.5 max: 1.5 avg: 0.6
cpu-present> min: 0.6 max: 1.9 avg: 0.7
frametime> min: 15.3 max: 17.5 avg: 16.7
draw-call #> min: 4 max: 4 avg: 4
tris #> min: 42 max: 42 avg: 42
verts #> min: 64 max: 64 avg: 64
player-detail> physx: 1.0 animation: 0.0 skinning: 0.0 render: 5.1 fixed-update-count: 2 … 2
mono-scripts> update: 0.2 fixedUpdate: 0.0 coroutines: 0.0
mono-memory> used heap: 319488 allocated heap: 3657728

As you can see, the frametime spikes quite badly.

ReJ, Is there any solution for this in the works?

I had these same “spikes” in my project, and it was related to having multiple levels in my game. For instance, if I loaded a ‘menu’ level first, and then from there loaded my ‘game’ level, the game level suffered from dramatic framerate spikes every 10-15 seconds or so, regardless of what was going on on screen. However if I just ran the game level first, everything was fine.

I eventually solved this by consolidating my menu and game into a single level, which was a major hassle, but it worked.

I’m seeing something similar to this and it’s driving me crazy. The frame rate hiccups from 30 to 15 and either some of the geometry or the sprites being managed by the SpriteManger drop out for a frame. It’s very disturbing and I wonder how to fix it.

PirateNinjaAlliance - right now I don’t have a GUI front end, I just load into the level I’m working on.

I’ve also seen these spikes and at the moment put this down to allocations, eg instantiating object. Boggybear in your profile the memory useGe is going up from 27 to 31 when the spike in main CPU occurs. Is it possible that a large alloc is occuring?

No, nothing on the face of it. The scene literally just had a camera and a plane dropped into it. Nothing dynamic at all.

Curious. Your profile is very similar to what I have seen: an increase in heap allocation, a big peak in CPU and frametime, and a peak in the number of fixed updates. I put this down to various script behaviour in my game but as you only had a camera + plane, it makes it hard to attribute something to the performance hit.

I’m also not seeing the ~15 second interrupt.

I was going to suggest iPhone 3g protocol interruption but you’re also running on iPod touch which, I assume, doesn’t have the 3g protocol stack. So the only other thing I can think of is the garbage collector, which really should not be running during game play. For example, at the moment, with the profiler running and the game idling, I’m seeing a continuous increase in heap usage. I can only assume one of my scripts is doing that, otherwise there’s a deeper problem.

Edit: re: increasing heap usage, i.e. it’s leaking memory

there is no leaked memory.

Used heap is the amount of memory from its allocated part that the managed memory currently is using. The GC will not start to flush until that gets critically commonly.

Leaks would not appear in that statistic, they can only be found through Instruments which measures the real ram handling

But the behavior is still odd.

Maybe “leak” is not the correct word in this sense, however, used heap memory is increasing by either 4096 or 8192 bytes every frame (where “frame” is the frequency of the output from the internal profiler). Therefore, something is allocating from managed memory. Over time, the GC will run and a big CPU hit will occur, e.g.


iPhone Unity internal profiler stats:
cpu-player> min: 15.5 max: 22.2 avg: 18.8
cpu-ogles-drv> min: 2.6 max: 5.8 avg: 3.1
cpu-present> min: 0.9 max: 2.4 avg: 1.1
frametime> min: 31.5 max: 34.7 avg: 33.3
draw-call #> min: 15 max: 15 avg: 15
tris #> min: 1188 max: 1188 avg: 1188
verts #> min: 880 max: 880 avg: 880
player-detail> physx: 1.1 animation: 0.0 skinning: 0.0 render: 8.4 fixed-update-count: 0 … 2
mono-scripts> update: 7.6 fixedUpdate: 0.4 coroutines: 0.4
mono-memory> used heap: 3653632 allocated heap: 3657728

iPhone Unity internal profiler stats:
cpu-player> min: 15.5 max: 118.9 avg: 22.2
cpu-ogles-drv> min: 2.6 max: 5.1 avg: 2.9
cpu-present> min: 0.9 max: 2.5 avg: 1.1
frametime> min: 31.3 max: 134.1 avg: 36.6
draw-call #> min: 15 max: 15 avg: 15
tris #> min: 1188 max: 1188 avg: 1188
verts #> min: 880 max: 880 avg: 880
player-detail> physx: 1.5 animation: 0.0 skinning: 0.0 render: 7.9 fixed-update-count: 1 … 4
mono-scripts> update: 11.2 fixedUpdate: 0.3 coroutines: 0.5
mono-memory> used heap: 245760 allocated heap: 3657728

Here it shows that the heap has reached its maximum allocation size and the GC has been run to clear out the unused memory blocks, i.e. the “leaks”.

IMHO, the GC should not run during game play, it is not predictable and causes a large hit on frame rate. The GC should be performed under programmer control at timely points in the game, e.g. between level loads, with something like this in C#:

System.GC.Collect();
System.GC.WaitForPendingFinalizers();

To do this however requires that allocations between GC runs do not exceed the maximum heap size, which in this case, because the engine is leaking, can’t happen.

I solved my issue and it was a problem with using the smooth follow camera script in a 2d game. I’m not entirely sure what was happening but while using instruments I noticed that it happened more often and I finally saw the camera “twist” in one of the hitches so I took off the script and the hitches disappeared. I honestly had no idea that’s what was going on and I’m glad I found at least the problem in my game but it took me disassembling the whole thing to find it. I was convinced it was just the way I was handling data and the garbage collector was the cause but I was wrong. I stopped using the smooth follow and just wrote my own. I hope this can help at least one person.

Hmm, interesting. Same hitches here too - and I use the Smooth Camera myself too. Weird.

You guys mean the SmoothFollow2 camera from the wiki?

In what way did you change the smooth follow camera code? The wiki version is only using 5 functions. My hunch, is that one of those functions either has a bug or is doing a pointless allocation which overtime is causing the GC to run :? If that is the case, that is both interesting to know and also frustrating.