Skip to content

Commit c4a9665

Browse files
committed
FreeJ2ME + Anbu + Libretro: Implement a better frame limit logic
Previously, frame limiting in FreeJ2ME relied on a fixed interval for Thread.sleep(), which is already a bit innacurate by itself, but with a fixed sleep interval meant that the time the game logic took to render also influenced the reslulting framerate. For example: If a game took 3ms to render and the fps limit was set to 60 fps, the resulting framerate would be around 50 to 51fps, as the limit would actually be ~19,666ms (3ms + 16,666ms fixed). This new logic works by calculating the time it took to render the last frame and subtracting it from the time we should wait for the next one, which will then be our Thread.sleep() interval. While it still relies on Thread.sleep() and its inaccuracies, this approach is miles better than what was already in place, case in point, jbenchmark 2.0 now locks around 59.5 to 60.3 fps in all tests on my machine, whereas it ran at about 48.6fps on the first test, 55.3 on the second, 56.4 on the third and 52.5 on the fourth.
1 parent 572a6c5 commit c4a9665

File tree

3 files changed

+55
-13
lines changed

3 files changed

+55
-13
lines changed

src/org/recompile/freej2me/Anbu.java

+18-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,13 @@ public static void main(String[] args)
4444
private boolean useSiemensControls = false;
4545
private boolean useMotorolaControls = false;
4646
private boolean rotateDisplay = false;
47+
48+
// Frame Limit Variables
4749
private int limitFPS = 0;
50+
private long lastRenderTime = 0;
51+
private long requiredFrametime = 0;
52+
private long elapsedTime = 0;
53+
private long sleepTime = 0;
4854

4955
private boolean[] pressedKeys = new boolean[128];
5056

@@ -75,6 +81,16 @@ public void run()
7581
try
7682
{
7783
int[] data;
84+
85+
if(limitFPS>0)
86+
{
87+
requiredFrametime = 1000 / limitFPS;
88+
elapsedTime = System.currentTimeMillis() - lastRenderTime;
89+
sleepTime = requiredFrametime - elapsedTime;
90+
91+
if (sleepTime > 0) { Thread.sleep(sleepTime); }
92+
}
93+
7894
// Send Frame to SDL interface
7995
if(!config.isRunning) { data = Mobile.getPlatform().getLCD().getRGB(0, 0, lcdWidth, lcdHeight, null, 0, lcdWidth); }
8096
else { data = config.getLCD().getRGB(0, 0, lcdWidth, lcdHeight, null, 0, lcdWidth);}
@@ -88,6 +104,8 @@ public void run()
88104
cb += 3;
89105
}
90106
sdl.frame.write(frame);
107+
108+
lastRenderTime = System.currentTimeMillis();
91109
}
92110
catch (Exception e) { }
93111
}
@@ -365,7 +383,6 @@ else if(useMotorolaControls)
365383
void settingsChanged()
366384
{
367385
limitFPS = Integer.parseInt(config.settings.get("fps"));
368-
if(limitFPS>0) { limitFPS = 1000 / limitFPS; }
369386

370387
String sound = config.settings.get("sound");
371388
Mobile.sound = false;

src/org/recompile/freej2me/FreeJ2ME.java

+18-7
Original file line numberDiff line numberDiff line change
@@ -53,9 +53,15 @@ public class FreeJ2ME
5353
private boolean useSiemensControls = false;
5454
private boolean useMotorolaControls = false;
5555
private boolean rotateDisplay = false;
56+
57+
// Frame Limit Variables
5658
private int limitFPS = 0;
57-
private int renderHint = 0;
59+
private long lastRenderTime = 0;
60+
private long requiredFrametime = 0;
61+
private long elapsedTime = 0;
62+
private long sleepTime = 0;
5863

64+
// AWT GUI
5965
private AWTGUI awtGUI;
6066

6167
private boolean[] pressedKeys = new boolean[128];
@@ -374,7 +380,6 @@ private void settingsChanged()
374380
int h = Integer.parseInt(config.settings.get("height"));
375381

376382
limitFPS = Integer.parseInt(config.settings.get("fps"));
377-
if(limitFPS>0) { limitFPS = 1000 / limitFPS; }
378383

379384
String sound = config.settings.get("sound");
380385
Mobile.sound = false;
@@ -531,6 +536,16 @@ public void paint(Graphics g)
531536
try
532537
{
533538
Graphics2D cgc = (Graphics2D)this.getGraphics();
539+
540+
if(limitFPS>0)
541+
{
542+
requiredFrametime = 1000 / limitFPS;
543+
elapsedTime = System.currentTimeMillis() - lastRenderTime;
544+
sleepTime = requiredFrametime - elapsedTime;
545+
546+
if (sleepTime > 0) { Thread.sleep(sleepTime); }
547+
}
548+
534549
if (config.isRunning)
535550
{
536551
if(!rotateDisplay)
@@ -556,12 +571,8 @@ public void paint(Graphics g)
556571
// Draw the rotated FB with adjusted cy and cx values
557572
cgc.drawImage(Mobile.getPlatform().getLCD(), 0, cx, ch, cw, null);
558573
}
559-
560-
if(limitFPS>0)
561-
{
562-
Thread.sleep(limitFPS);
563-
}
564574
}
575+
lastRenderTime = System.currentTimeMillis();
565576
}
566577
catch (Exception e)
567578
{

src/org/recompile/freej2me/Libretro.java

+19-5
Original file line numberDiff line numberDiff line change
@@ -51,7 +51,14 @@ public class Libretro
5151
private boolean useMotorolaControls = false;
5252
private boolean rotateDisplay = false;
5353
private boolean soundEnabled = true;
54+
55+
// Frame Limit Variables
5456
private int limitFPS = 0;
57+
private long lastRenderTime = 0;
58+
private long requiredFrametime = 0;
59+
private long elapsedTime = 0;
60+
private long sleepTime = 0;
61+
5562
private int maxmidistreams = 32;
5663

5764
private boolean[] pressedKeys = new boolean[128];
@@ -378,17 +385,23 @@ public void run()
378385
try
379386
{
380387
int[] data;
388+
389+
if(limitFPS>0)
390+
{
391+
requiredFrametime = 1000 / limitFPS;
392+
elapsedTime = System.currentTimeMillis() - lastRenderTime;
393+
sleepTime = requiredFrametime - elapsedTime;
394+
395+
if (sleepTime > 0) { Thread.sleep(sleepTime); }
396+
}
397+
381398
if(config.isRunning)
382399
{
383400
data = config.getLCD().getRGB(0, 0, lcdWidth, lcdHeight, null, 0, lcdWidth);
384401
}
385402
else
386403
{
387404
data = surface.getRGB(0, 0, lcdWidth, lcdHeight, null, 0, lcdWidth);
388-
if(limitFPS>0)
389-
{
390-
Thread.sleep(limitFPS);
391-
}
392405
}
393406
int bufferLength = data.length*3;
394407
int cb = 0;
@@ -408,6 +421,8 @@ public void run()
408421
System.out.write(frameHeader, 0, 6);
409422
System.out.write(frameBuffer, 0, bufferLength);
410423
System.out.flush();
424+
425+
lastRenderTime = System.currentTimeMillis();
411426
}
412427
catch (Exception e)
413428
{
@@ -432,7 +447,6 @@ private void settingsChanged()
432447
int h = Integer.parseInt(config.settings.get("height"));
433448

434449
limitFPS = Integer.parseInt(config.settings.get("fps"));
435-
if(limitFPS>0) { limitFPS = 1000 / limitFPS; }
436450

437451
String sound = config.settings.get("sound");
438452
Mobile.sound = false;

0 commit comments

Comments
 (0)