Merge pull request #11103 from JosJuice/android-gamefilecache-not-null

Android: Allocate GameFileCache on GUI thread
This commit is contained in:
Mai 2022-09-29 09:02:47 -04:00 committed by GitHub
commit 865348cfb1
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 78 additions and 78 deletions

View File

@ -74,7 +74,7 @@ public class AppLinkActivity extends FragmentActivity
});
DirectoryInitialization.start(this);
GameFileCacheManager.startLoad(this);
GameFileCacheManager.startLoad();
}
/**

View File

@ -233,7 +233,7 @@ public class Settings implements Closeable
if (mLoadedRecursiveIsoPathsValue != BooleanSetting.MAIN_RECURSIVE_ISO_PATHS.getBoolean(this))
{
// Refresh game library
GameFileCacheManager.startRescan(context);
GameFileCacheManager.startRescan();
}
}
else

View File

@ -109,11 +109,11 @@ public class GameFileCache
public static native String[] getAllGamePaths(String[] folderPaths, boolean recursiveScan);
public native int getSize();
public synchronized native int getSize();
public native GameFile[] getAllGames();
public synchronized native GameFile[] getAllGames();
public native GameFile addOrGet(String gamePath);
public synchronized native GameFile addOrGet(String gamePath);
/**
* Sets the list of games to cache.
@ -123,7 +123,7 @@ public class GameFileCache
*
* @return true if the cache was modified
*/
public native boolean update(String[] gamePaths);
public synchronized native boolean update(String[] gamePaths);
/**
* For each game that already is in the cache, scans the folder that contains the game
@ -131,9 +131,9 @@ public class GameFileCache
*
* @return true if the cache was modified
*/
public native boolean updateAdditionalMetadata();
public synchronized native boolean updateAdditionalMetadata();
public native boolean load();
public synchronized native boolean load();
public native boolean save();
public synchronized native boolean save();
}

View File

@ -2,8 +2,6 @@
package org.dolphinemu.dolphinemu.services;
import android.content.Context;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.MutableLiveData;
@ -23,14 +21,15 @@ import java.util.concurrent.Executors;
*/
public final class GameFileCacheManager
{
private static GameFileCache gameFileCache = null;
private static final MutableLiveData<GameFile[]> gameFiles =
private static GameFileCache sGameFileCache = null;
private static final MutableLiveData<GameFile[]> sGameFiles =
new MutableLiveData<>(new GameFile[]{});
private static boolean runRescanAfterLoad = false;
private static boolean sFirstLoadDone = false;
private static boolean sRunRescanAfterLoad = false;
private static final ExecutorService executor = Executors.newFixedThreadPool(1);
private static final MutableLiveData<Boolean> loadInProgress = new MutableLiveData<>(false);
private static final MutableLiveData<Boolean> rescanInProgress = new MutableLiveData<>(false);
private static final ExecutorService sExecutor = Executors.newFixedThreadPool(1);
private static final MutableLiveData<Boolean> sLoadInProgress = new MutableLiveData<>(false);
private static final MutableLiveData<Boolean> sRescanInProgress = new MutableLiveData<>(false);
private GameFileCacheManager()
{
@ -38,12 +37,12 @@ public final class GameFileCacheManager
public static LiveData<GameFile[]> getGameFiles()
{
return gameFiles;
return sGameFiles;
}
public static List<GameFile> getGameFilesForPlatform(Platform platform)
{
GameFile[] allGames = gameFiles.getValue();
GameFile[] allGames = sGameFiles.getValue();
ArrayList<GameFile> platformGames = new ArrayList<>();
for (GameFile game : allGames)
{
@ -57,7 +56,7 @@ public final class GameFileCacheManager
public static GameFile getGameFileByGameId(String gameId)
{
GameFile[] allGames = gameFiles.getValue();
GameFile[] allGames = sGameFiles.getValue();
for (GameFile game : allGames)
{
if (game.getGameId().equals(gameId))
@ -72,7 +71,7 @@ public final class GameFileCacheManager
{
GameFile matchWithoutRevision = null;
GameFile[] allGames = gameFiles.getValue();
GameFile[] allGames = sGameFiles.getValue();
for (GameFile otherGame : allGames)
{
if (game.getGameId().equals(otherGame.getGameId()) &&
@ -102,7 +101,7 @@ public final class GameFileCacheManager
*/
public static LiveData<Boolean> isLoading()
{
return loadInProgress;
return sLoadInProgress;
}
/**
@ -110,12 +109,12 @@ public final class GameFileCacheManager
*/
public static LiveData<Boolean> isRescanning()
{
return rescanInProgress;
return sRescanInProgress;
}
public static boolean isLoadingOrRescanning()
{
return loadInProgress.getValue() || rescanInProgress.getValue();
return sLoadInProgress.getValue() || sRescanInProgress.getValue();
}
/**
@ -123,13 +122,15 @@ public final class GameFileCacheManager
* if the games are still present in the user's configured folders.
* If this has already been called, calling it again has no effect.
*/
public static void startLoad(Context context)
public static void startLoad()
{
if (!loadInProgress.getValue())
createGameFileCacheIfNeeded();
if (!sLoadInProgress.getValue())
{
loadInProgress.setValue(true);
sLoadInProgress.setValue(true);
new AfterDirectoryInitializationRunner().runWithoutLifecycle(
() -> executor.execute(GameFileCacheManager::load));
() -> sExecutor.execute(GameFileCacheManager::load));
}
}
@ -139,13 +140,15 @@ public final class GameFileCacheManager
* If loading the game file cache hasn't started or hasn't finished,
* the execution of this will be postponed until it finishes.
*/
public static void startRescan(Context context)
public static void startRescan()
{
if (!rescanInProgress.getValue())
createGameFileCacheIfNeeded();
if (!sRescanInProgress.getValue())
{
rescanInProgress.setValue(true);
sRescanInProgress.setValue(true);
new AfterDirectoryInitializationRunner().runWithoutLifecycle(
() -> executor.execute(GameFileCacheManager::rescan));
() -> sExecutor.execute(GameFileCacheManager::rescan));
}
}
@ -153,8 +156,8 @@ public final class GameFileCacheManager
{
// Common case: The game is in the cache, so just grab it from there.
// (Actually, addOrGet already checks for this case, but we want to avoid calling it if possible
// because onHandleIntent may hold a lock on gameFileCache for extended periods of time.)
GameFile[] allGames = gameFiles.getValue();
// because the executor thread may hold a lock on sGameFileCache for extended periods of time.)
GameFile[] allGames = sGameFiles.getValue();
for (GameFile game : allGames)
{
if (game.getPath().equals(gamePath))
@ -165,10 +168,8 @@ public final class GameFileCacheManager
// Unusual case: The game wasn't found in the cache.
// Scan the game and add it to the cache so that we can return it.
synchronized (gameFileCache)
{
return gameFileCache.addOrGet(gamePath);
}
createGameFileCacheIfNeeded();
return sGameFileCache.addOrGet(gamePath);
}
/**
@ -178,30 +179,26 @@ public final class GameFileCacheManager
*/
private static void load()
{
if (gameFileCache == null)
if (!sFirstLoadDone)
{
GameFileCache temp = new GameFileCache();
synchronized (temp)
{
gameFileCache = temp;
gameFileCache.load();
if (gameFileCache.getSize() != 0)
sFirstLoadDone = true;
sGameFileCache.load();
if (sGameFileCache.getSize() != 0)
{
updateGameFileArray();
}
}
if (sRunRescanAfterLoad)
{
sRescanInProgress.postValue(true);
}
if (runRescanAfterLoad)
{
rescanInProgress.postValue(true);
}
sLoadInProgress.postValue(false);
loadInProgress.postValue(false);
if (runRescanAfterLoad)
if (sRunRescanAfterLoad)
{
runRescanAfterLoad = false;
sRunRescanAfterLoad = false;
rescan();
}
}
@ -214,25 +211,21 @@ public final class GameFileCacheManager
*/
private static void rescan()
{
if (gameFileCache == null)
if (!sFirstLoadDone)
{
runRescanAfterLoad = true;
sRunRescanAfterLoad = true;
}
else
{
String[] gamePaths = GameFileCache.getAllGamePaths();
boolean changed;
synchronized (gameFileCache)
{
changed = gameFileCache.update(gamePaths);
}
boolean changed = sGameFileCache.update(gamePaths);
if (changed)
{
updateGameFileArray();
}
boolean additionalMetadataChanged = gameFileCache.updateAdditionalMetadata();
boolean additionalMetadataChanged = sGameFileCache.updateAdditionalMetadata();
if (additionalMetadataChanged)
{
updateGameFileArray();
@ -240,17 +233,29 @@ public final class GameFileCacheManager
if (changed || additionalMetadataChanged)
{
gameFileCache.save();
sGameFileCache.save();
}
}
rescanInProgress.postValue(false);
sRescanInProgress.postValue(false);
}
private static void updateGameFileArray()
{
GameFile[] gameFilesTemp = gameFileCache.getAllGames();
GameFile[] gameFilesTemp = sGameFileCache.getAllGames();
Arrays.sort(gameFilesTemp, (lhs, rhs) -> lhs.getTitle().compareToIgnoreCase(rhs.getTitle()));
gameFiles.postValue(gameFilesTemp);
sGameFiles.postValue(gameFilesTemp);
}
private static void createGameFileCacheIfNeeded()
{
// Creating the GameFileCache in the static initializer may be unsafe, because GameFileCache
// relies on native code, and the native library isn't loaded right when the app starts.
// We create it here instead.
if (sGameFileCache == null)
{
sGameFileCache = new GameFileCache();
}
}
}

View File

@ -301,7 +301,7 @@ public final class MainActivity extends AppCompatActivity
public void onRefresh()
{
setRefreshing(true);
GameFileCacheManager.startRescan(this);
GameFileCacheManager.startRescan();
}
/**
@ -368,7 +368,7 @@ public final class MainActivity extends AppCompatActivity
mViewPager.setCurrentItem(IntSetting.MAIN_LAST_PLATFORM_TAB.getIntGlobal());
showGames();
GameFileCacheManager.startLoad(this);
GameFileCacheManager.startLoad();
}
@Override

View File

@ -96,7 +96,7 @@ public final class MainPresenter
case R.id.menu_refresh:
mView.setRefreshing(true);
GameFileCacheManager.startRescan(activity);
GameFileCacheManager.startRescan();
return true;
case R.id.button_add_directory:
@ -146,7 +146,7 @@ public final class MainPresenter
if (sShouldRescanLibrary)
{
GameFileCacheManager.startRescan(mActivity);
GameFileCacheManager.startRescan();
}
sShouldRescanLibrary = true;

View File

@ -79,7 +79,7 @@ public final class TvMainActivity extends FragmentActivity
if (DirectoryInitialization.shouldStart(this))
{
DirectoryInitialization.start(this);
GameFileCacheManager.startLoad(this);
GameFileCacheManager.startLoad();
}
mPresenter.onResume();
@ -292,7 +292,7 @@ public final class TvMainActivity extends FragmentActivity
}
DirectoryInitialization.start(this);
GameFileCacheManager.startLoad(this);
GameFileCacheManager.startLoad();
}
}
@ -303,7 +303,7 @@ public final class TvMainActivity extends FragmentActivity
public void onRefresh()
{
setRefreshing(true);
GameFileCacheManager.startRescan(this);
GameFileCacheManager.startRescan();
}
private void buildRowsAdapter()
@ -313,7 +313,7 @@ public final class TvMainActivity extends FragmentActivity
if (!DirectoryInitialization.isWaitingForWriteAccess(this))
{
GameFileCacheManager.startLoad(this);
GameFileCacheManager.startLoad();
}
for (Platform platform : Platform.values())

View File

@ -13,8 +13,6 @@ if(CMAKE_SYSTEM_NAME MATCHES "Windows")
add_definitions(-D_CRT_SECURE_NO_DEPRECATE)
add_definitions(-D_CRT_NONSTDC_NO_WARNINGS)
add_definitions(-D_SILENCE_CXX17_CODECVT_HEADER_DEPRECATION_WARNING)
# The replacement for the old atomic shared_ptr functions was added in C++20, so we can't use it yet
add_definitions(-D_SILENCE_CXX20_OLD_SHARED_PTR_ATOMIC_SUPPORT_DEPRECATION_WARNING)
endif()
if (NOT MSVC)

View File

@ -4,7 +4,6 @@
#include "UICommon/GameFileCache.h"
#include <algorithm>
#include <atomic>
#include <cstddef>
#include <functional>
#include <list>
@ -204,7 +203,7 @@ bool GameFileCache::UpdateAdditionalMetadata(std::shared_ptr<GameFile>* game_fil
if (custom_cover_changed)
copy->CustomCoverCommit();
std::atomic_store(game_file, std::move(copy));
*game_file = std::move(copy);
return true;
}

View File

@ -29,8 +29,6 @@
<PreprocessorDefinitions>_WINSOCK_DEPRECATED_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<!--Currently needed for some code in StringUtil used only on Android-->
<PreprocessorDefinitions>_SILENCE_CXX17_CODECVT_HEADER_DEPRECATION_WARNING;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<!--The replacement for the old atomic shared_ptr functions was added in C++20, so we can't use it yet-->
<PreprocessorDefinitions>_SILENCE_CXX20_OLD_SHARED_PTR_ATOMIC_SUPPORT_DEPRECATION_WARNING;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<!--Dolphin-specific definitions-->
<PreprocessorDefinitions Condition="'$(Platform)'=='x64'">_ARCH_64=1;_M_X86=1;_M_X86_64=1;%(PreprocessorDefinitions)</PreprocessorDefinitions>