Switch to a LRU cache for beatmap background images.
Created a separate BeatmapImageCache class to handle cache operations. The cache now uses File objects as keys, rather than Beatmap objects (which was buggy). Also renamed "OsuHitObjectResult" helper class to "HitObjectResult". Signed-off-by: Jeffrey Han <itdelatrisu@gmail.com>
This commit is contained in:
parent
74f7ac18b9
commit
92f4a5176d
|
@ -124,7 +124,7 @@ public class Container extends AppGameContainer {
|
||||||
// reset image references
|
// reset image references
|
||||||
GameImage.clearReferences();
|
GameImage.clearReferences();
|
||||||
GameData.Grade.clearReferences();
|
GameData.Grade.clearReferences();
|
||||||
Beatmap.resetImageCache();
|
Beatmap.getBackgroundImageCache().clear();
|
||||||
|
|
||||||
// prevent loading tracks from re-initializing OpenAL
|
// prevent loading tracks from re-initializing OpenAL
|
||||||
MusicController.reset();
|
MusicController.reset();
|
||||||
|
|
|
@ -191,7 +191,7 @@ public class GameData {
|
||||||
private int[] hitResultOffset;
|
private int[] hitResultOffset;
|
||||||
|
|
||||||
/** List of hit result objects associated with hit objects. */
|
/** List of hit result objects associated with hit objects. */
|
||||||
private LinkedBlockingDeque<OsuHitObjectResult> hitResultList;
|
private LinkedBlockingDeque<HitObjectResult> hitResultList;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class to store hit error information.
|
* Class to store hit error information.
|
||||||
|
@ -230,7 +230,7 @@ public class GameData {
|
||||||
public enum HitObjectType { CIRCLE, SLIDERTICK, SLIDER_FIRST, SLIDER_LAST, SPINNER }
|
public enum HitObjectType { CIRCLE, SLIDERTICK, SLIDER_FIRST, SLIDER_LAST, SPINNER }
|
||||||
|
|
||||||
/** Hit result helper class. */
|
/** Hit result helper class. */
|
||||||
private class OsuHitObjectResult {
|
private class HitObjectResult {
|
||||||
/** Object start time. */
|
/** Object start time. */
|
||||||
public int time;
|
public int time;
|
||||||
|
|
||||||
|
@ -265,7 +265,7 @@ public class GameData {
|
||||||
* @param curve the slider curve (or null if not applicable)
|
* @param curve the slider curve (or null if not applicable)
|
||||||
* @param expand whether or not the hit result animation should expand (if applicable)
|
* @param expand whether or not the hit result animation should expand (if applicable)
|
||||||
*/
|
*/
|
||||||
public OsuHitObjectResult(int time, int result, float x, float y, Color color,
|
public HitObjectResult(int time, int result, float x, float y, Color color,
|
||||||
HitObjectType hitResultType, Curve curve, boolean expand) {
|
HitObjectType hitResultType, Curve curve, boolean expand) {
|
||||||
this.time = time;
|
this.time = time;
|
||||||
this.result = result;
|
this.result = result;
|
||||||
|
@ -375,12 +375,12 @@ public class GameData {
|
||||||
healthDisplay = 100f;
|
healthDisplay = 100f;
|
||||||
hitResultCount = new int[HIT_MAX];
|
hitResultCount = new int[HIT_MAX];
|
||||||
if (hitResultList != null) {
|
if (hitResultList != null) {
|
||||||
for (OsuHitObjectResult hitResult : hitResultList) {
|
for (HitObjectResult hitResult : hitResultList) {
|
||||||
if (hitResult.curve != null)
|
if (hitResult.curve != null)
|
||||||
hitResult.curve.discardCache();
|
hitResult.curve.discardCache();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hitResultList = new LinkedBlockingDeque<OsuHitObjectResult>();
|
hitResultList = new LinkedBlockingDeque<HitObjectResult>();
|
||||||
hitErrorList = new LinkedBlockingDeque<HitErrorInfo>();
|
hitErrorList = new LinkedBlockingDeque<HitErrorInfo>();
|
||||||
fullObjectCount = 0;
|
fullObjectCount = 0;
|
||||||
combo = 0;
|
combo = 0;
|
||||||
|
@ -872,9 +872,9 @@ public class GameData {
|
||||||
* @param trackPosition the current track position (in ms)
|
* @param trackPosition the current track position (in ms)
|
||||||
*/
|
*/
|
||||||
public void drawHitResults(int trackPosition) {
|
public void drawHitResults(int trackPosition) {
|
||||||
Iterator<OsuHitObjectResult> iter = hitResultList.iterator();
|
Iterator<HitObjectResult> iter = hitResultList.iterator();
|
||||||
while (iter.hasNext()) {
|
while (iter.hasNext()) {
|
||||||
OsuHitObjectResult hitResult = iter.next();
|
HitObjectResult hitResult = iter.next();
|
||||||
if (hitResult.time + HITRESULT_TIME > trackPosition) {
|
if (hitResult.time + HITRESULT_TIME > trackPosition) {
|
||||||
// spinner
|
// spinner
|
||||||
if (hitResult.hitResultType == HitObjectType.SPINNER && hitResult.result != HIT_MISS) {
|
if (hitResult.hitResultType == HitObjectType.SPINNER && hitResult.result != HIT_MISS) {
|
||||||
|
@ -1217,7 +1217,7 @@ public class GameData {
|
||||||
if (!Options.isPerfectHitBurstEnabled())
|
if (!Options.isPerfectHitBurstEnabled())
|
||||||
; // hide perfect hit results
|
; // hide perfect hit results
|
||||||
else
|
else
|
||||||
hitResultList.add(new OsuHitObjectResult(time, result, x, y, null, HitObjectType.SLIDERTICK, null, false));
|
hitResultList.add(new HitObjectResult(time, result, x, y, null, HitObjectType.SLIDERTICK, null, false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1338,14 +1338,14 @@ public class GameData {
|
||||||
else if (result == HIT_MISS && (GameMod.RELAX.isActive() || GameMod.AUTOPILOT.isActive()))
|
else if (result == HIT_MISS && (GameMod.RELAX.isActive() || GameMod.AUTOPILOT.isActive()))
|
||||||
; // "relax" and "autopilot" mods: hide misses
|
; // "relax" and "autopilot" mods: hide misses
|
||||||
else {
|
else {
|
||||||
hitResultList.add(new OsuHitObjectResult(time, result, x, y, color, hitResultType, curve, expand));
|
hitResultList.add(new HitObjectResult(time, result, x, y, color, hitResultType, curve, expand));
|
||||||
|
|
||||||
// sliders: add the other curve endpoint for the hit animation
|
// sliders: add the other curve endpoint for the hit animation
|
||||||
if (curve != null) {
|
if (curve != null) {
|
||||||
boolean isFirst = (hitResultType == HitObjectType.SLIDER_FIRST);
|
boolean isFirst = (hitResultType == HitObjectType.SLIDER_FIRST);
|
||||||
float[] p = curve.pointAt((isFirst) ? 1f : 0f);
|
float[] p = curve.pointAt((isFirst) ? 1f : 0f);
|
||||||
HitObjectType type = (isFirst) ? HitObjectType.SLIDER_LAST : HitObjectType.SLIDER_FIRST;
|
HitObjectType type = (isFirst) ? HitObjectType.SLIDER_LAST : HitObjectType.SLIDER_FIRST;
|
||||||
hitResultList.add(new OsuHitObjectResult(time, result, p[0], p[1], color, type, null, expand));
|
hitResultList.add(new HitObjectResult(time, result, p[0], p[1], color, type, null, expand));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,12 +22,10 @@ import itdelatrisu.opsu.Options;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.LinkedList;
|
import java.util.LinkedList;
|
||||||
|
|
||||||
import org.newdawn.slick.Color;
|
import org.newdawn.slick.Color;
|
||||||
import org.newdawn.slick.Image;
|
import org.newdawn.slick.Image;
|
||||||
import org.newdawn.slick.SlickException;
|
|
||||||
import org.newdawn.slick.util.Log;
|
import org.newdawn.slick.util.Log;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -37,11 +35,13 @@ public class Beatmap implements Comparable<Beatmap> {
|
||||||
/** Game modes. */
|
/** Game modes. */
|
||||||
public static final byte MODE_OSU = 0, MODE_TAIKO = 1, MODE_CTB = 2, MODE_MANIA = 3;
|
public static final byte MODE_OSU = 0, MODE_TAIKO = 1, MODE_CTB = 2, MODE_MANIA = 3;
|
||||||
|
|
||||||
/** Map of all loaded background images. */
|
/** Background image cache. */
|
||||||
private static HashMap<Beatmap, Image> bgImageMap = new HashMap<Beatmap, Image>();
|
private static final BeatmapImageCache bgImageCache = new BeatmapImageCache();
|
||||||
|
|
||||||
/** Maximum number of cached images before all get erased. */
|
/**
|
||||||
private static final int MAX_CACHE_SIZE = 10;
|
* Returns the background image cache.
|
||||||
|
*/
|
||||||
|
public static BeatmapImageCache getBackgroundImageCache() { return bgImageCache; }
|
||||||
|
|
||||||
/** The OSU File object associated with this beatmap. */
|
/** The OSU File object associated with this beatmap. */
|
||||||
private File file;
|
private File file;
|
||||||
|
@ -156,11 +156,11 @@ public class Beatmap implements Comparable<Beatmap> {
|
||||||
* [Events]
|
* [Events]
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** Background image file name. */
|
/** Background image file. */
|
||||||
public String bg;
|
public File bg;
|
||||||
|
|
||||||
/** Background video file name. */
|
/** Background video file. */
|
||||||
// public String video;
|
// public File video;
|
||||||
|
|
||||||
/** All break periods (start time, end time, ...). */
|
/** All break periods (start time, end time, ...). */
|
||||||
public ArrayList<Integer> breaks;
|
public ArrayList<Integer> breaks;
|
||||||
|
@ -201,30 +201,6 @@ public class Beatmap implements Comparable<Beatmap> {
|
||||||
/** Last object end time (in ms). */
|
/** Last object end time (in ms). */
|
||||||
public int endTime = -1;
|
public int endTime = -1;
|
||||||
|
|
||||||
/**
|
|
||||||
* Destroys all cached background images and resets the cache.
|
|
||||||
*/
|
|
||||||
public static void clearImageCache() {
|
|
||||||
for (Image img : bgImageMap.values()) {
|
|
||||||
if (img != null && !img.isDestroyed()) {
|
|
||||||
try {
|
|
||||||
img.destroy();
|
|
||||||
} catch (SlickException e) {
|
|
||||||
Log.warn(String.format("Failed to destroy image '%s'.", img.getResourceReference()), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
resetImageCache();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Resets the image cache.
|
|
||||||
* This does NOT destroy images, so be careful of memory leaks!
|
|
||||||
*/
|
|
||||||
public static void resetImageCache() {
|
|
||||||
bgImageMap = new HashMap<Beatmap, Image>();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor.
|
* Constructor.
|
||||||
* @param file the file associated with this beatmap
|
* @param file the file associated with this beatmap
|
||||||
|
@ -287,12 +263,10 @@ public class Beatmap implements Comparable<Beatmap> {
|
||||||
if (bg == null)
|
if (bg == null)
|
||||||
return false;
|
return false;
|
||||||
try {
|
try {
|
||||||
Image bgImage = bgImageMap.get(this);
|
Image bgImage = bgImageCache.get(this);
|
||||||
if (bgImage == null) {
|
if (bgImage == null) {
|
||||||
if (bgImageMap.size() > MAX_CACHE_SIZE)
|
bgImage = new Image(bg.getAbsolutePath());
|
||||||
clearImageCache();
|
bgImageCache.put(this, bgImage);
|
||||||
bgImage = new Image(new File(file.getParentFile(), bg).getAbsolutePath());
|
|
||||||
bgImageMap.put(this, bgImage);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int swidth = width;
|
int swidth = width;
|
||||||
|
|
86
src/itdelatrisu/opsu/beatmap/BeatmapImageCache.java
Normal file
86
src/itdelatrisu/opsu/beatmap/BeatmapImageCache.java
Normal file
|
@ -0,0 +1,86 @@
|
||||||
|
/*
|
||||||
|
* opsu! - an open-source osu! client
|
||||||
|
* Copyright (C) 2014, 2015 Jeffrey Han
|
||||||
|
*
|
||||||
|
* opsu! is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* opsu! is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with opsu!. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package itdelatrisu.opsu.beatmap;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.newdawn.slick.Image;
|
||||||
|
import org.newdawn.slick.SlickException;
|
||||||
|
import org.newdawn.slick.util.Log;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* LRU cache for beatmap background images.
|
||||||
|
*/
|
||||||
|
public class BeatmapImageCache {
|
||||||
|
/** Maximum number of cached images. */
|
||||||
|
private static final int MAX_CACHE_SIZE = 10;
|
||||||
|
|
||||||
|
/** Map of all loaded background images. */
|
||||||
|
private LinkedHashMap<File, Image> cache;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("serial")
|
||||||
|
public BeatmapImageCache() {
|
||||||
|
this.cache = new LinkedHashMap<File, Image>(MAX_CACHE_SIZE + 1, 1.1f, true) {
|
||||||
|
@Override
|
||||||
|
protected boolean removeEldestEntry(Map.Entry<File, Image> eldest) {
|
||||||
|
if (size() > MAX_CACHE_SIZE) {
|
||||||
|
// destroy the eldest image
|
||||||
|
Image img = eldest.getValue();
|
||||||
|
if (img != null && !img.isDestroyed()) {
|
||||||
|
try {
|
||||||
|
img.destroy();
|
||||||
|
} catch (SlickException e) {
|
||||||
|
Log.warn(String.format("Failed to destroy image '%s'.", img.getResourceReference()), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the image mapped to the specified beatmap.
|
||||||
|
* @param beatmap the Beatmap
|
||||||
|
* @return the Image, or {@code null} if no such mapping exists
|
||||||
|
*/
|
||||||
|
public Image get(Beatmap beatmap) { return cache.get(beatmap.bg); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a mapping from the specified beatmap to the given image.
|
||||||
|
* @param beatmap the Beatmap
|
||||||
|
* @param image the Image
|
||||||
|
* @return the previously mapped Image, or {@code null} if no such mapping existed
|
||||||
|
*/
|
||||||
|
public Image put(Beatmap beatmap, Image image) { return cache.put(beatmap.bg, image); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes all entries from the cache.
|
||||||
|
* <p>
|
||||||
|
* NOTE: This does NOT destroy the images in the cache, and will cause
|
||||||
|
* memory leaks if all images have not been destroyed.
|
||||||
|
*/
|
||||||
|
public void clear() { cache.clear(); }
|
||||||
|
}
|
|
@ -440,7 +440,7 @@ public class BeatmapParser {
|
||||||
tokens[2] = tokens[2].replaceAll("^\"|\"$", "");
|
tokens[2] = tokens[2].replaceAll("^\"|\"$", "");
|
||||||
String ext = BeatmapParser.getExtension(tokens[2]);
|
String ext = BeatmapParser.getExtension(tokens[2]);
|
||||||
if (ext.equals("jpg") || ext.equals("png"))
|
if (ext.equals("jpg") || ext.equals("png"))
|
||||||
beatmap.bg = getDBString(tokens[2]);
|
beatmap.bg = new File(dir, getDBString(tokens[2]));
|
||||||
break;
|
break;
|
||||||
case "2": // break periods
|
case "2": // break periods
|
||||||
try {
|
try {
|
||||||
|
|
|
@ -335,7 +335,7 @@ public class BeatmapDB {
|
||||||
stmt.setBoolean(33, beatmap.letterboxInBreaks);
|
stmt.setBoolean(33, beatmap.letterboxInBreaks);
|
||||||
stmt.setBoolean(34, beatmap.widescreenStoryboard);
|
stmt.setBoolean(34, beatmap.widescreenStoryboard);
|
||||||
stmt.setBoolean(35, beatmap.epilepsyWarning);
|
stmt.setBoolean(35, beatmap.epilepsyWarning);
|
||||||
stmt.setString(36, beatmap.bg);
|
stmt.setString(36, beatmap.bg.getName());
|
||||||
stmt.setString(37, beatmap.sliderBorderToString());
|
stmt.setString(37, beatmap.sliderBorderToString());
|
||||||
stmt.setString(38, beatmap.timingPointsToString());
|
stmt.setString(38, beatmap.timingPointsToString());
|
||||||
stmt.setString(39, beatmap.breaksToString());
|
stmt.setString(39, beatmap.breaksToString());
|
||||||
|
@ -476,7 +476,7 @@ public class BeatmapDB {
|
||||||
beatmap.letterboxInBreaks = rs.getBoolean(33);
|
beatmap.letterboxInBreaks = rs.getBoolean(33);
|
||||||
beatmap.widescreenStoryboard = rs.getBoolean(34);
|
beatmap.widescreenStoryboard = rs.getBoolean(34);
|
||||||
beatmap.epilepsyWarning = rs.getBoolean(35);
|
beatmap.epilepsyWarning = rs.getBoolean(35);
|
||||||
beatmap.bg = BeatmapParser.getDBString(rs.getString(36));
|
beatmap.bg = new File(beatmap.getFile().getParentFile(), BeatmapParser.getDBString(rs.getString(36)));
|
||||||
beatmap.sliderBorderFromString(rs.getString(37));
|
beatmap.sliderBorderFromString(rs.getString(37));
|
||||||
} catch (SQLException e) {
|
} catch (SQLException e) {
|
||||||
throw e;
|
throw e;
|
||||||
|
|
Loading…
Reference in New Issue
Block a user