mirror of
https://github.com/nitraiolo/CfgUSBLoader.git
synced 2025-01-24 08:51:13 +01:00
2002 lines
46 KiB
C
2002 lines
46 KiB
C
|
|
||
|
// by oggzee & usptactical
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <ogcsys.h>
|
||
|
#include <malloc.h>
|
||
|
#include <string.h>
|
||
|
#include <unistd.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <fcntl.h>
|
||
|
|
||
|
#include "cache.h"
|
||
|
#include "mem.h"
|
||
|
#include "cfg.h"
|
||
|
#include "gui.h"
|
||
|
#include "coverflow.h"
|
||
|
//#include "disc.h"
|
||
|
//#include "gettext.h"
|
||
|
|
||
|
//#define cc_dbg dbg_printf
|
||
|
#define cc_dbg(...)
|
||
|
|
||
|
//0x2000000 = 32 MB
|
||
|
//0x1700000 = 23 MB
|
||
|
#define COVER_CACHE_DATA_SIZE 0x1900000 // 25 MB
|
||
|
|
||
|
extern struct discHdr *all_gameList;
|
||
|
extern struct discHdr *gameList;
|
||
|
extern s32 gameCnt, all_gameCnt;
|
||
|
|
||
|
#if 0
|
||
|
|
||
|
struct Cover_State
|
||
|
{
|
||
|
char used;
|
||
|
char id[8];
|
||
|
int gid; // ccache.game[gid]
|
||
|
GRRLIB_texImg tx;
|
||
|
//int lru;
|
||
|
};
|
||
|
|
||
|
struct Game_State
|
||
|
{
|
||
|
int request;
|
||
|
int state; // idle, loading, present, missing
|
||
|
int cid; // cache index
|
||
|
int agi; // actual all_gameList[agi] (mapping gameList -> all_gameList)
|
||
|
char id[8];
|
||
|
};
|
||
|
|
||
|
struct Cover_Cache
|
||
|
{
|
||
|
void *data;
|
||
|
int width4, height4;
|
||
|
int csize;
|
||
|
int num, cover_alloc, lru;
|
||
|
struct Cover_State *cover;
|
||
|
int num_game, game_alloc;
|
||
|
struct Game_State *game;
|
||
|
volatile int run;
|
||
|
volatile int idle;
|
||
|
volatile int restart;
|
||
|
volatile int quit;
|
||
|
lwp_t lwp;
|
||
|
lwpq_t queue;
|
||
|
mutex_t mutex;
|
||
|
};
|
||
|
|
||
|
struct Cover_Cache ccache;
|
||
|
int ccache_init = 0;
|
||
|
int ccache_inv = 0;
|
||
|
|
||
|
//****************************************************************************
|
||
|
//************************** START CACHE CODE ******************************
|
||
|
//****************************************************************************
|
||
|
|
||
|
void game_populate_ids()
|
||
|
{
|
||
|
int i;
|
||
|
char *id;
|
||
|
|
||
|
for (i=0; i<all_gameCnt; i++) {
|
||
|
memcheck_ptr(all_gameList, &all_gameList[i]);
|
||
|
id = (char*)all_gameList[i].id;
|
||
|
memcheck_ptr(ccache.game,&ccache.game[i]);
|
||
|
STRCOPY(ccache.game[i].id, id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int cache_game_find_id(int gi)
|
||
|
{
|
||
|
int i;
|
||
|
bool getNewIds = true;
|
||
|
char *id = (char*)gameList[gi].id;
|
||
|
|
||
|
retry:;
|
||
|
for (i=0; i<all_gameCnt; i++) {
|
||
|
memcheck_ptr(ccache.game,&ccache.game[i]);
|
||
|
if (strcmp(ccache.game[i].id, id) == 0) {
|
||
|
ccache.game[gi].agi = i;
|
||
|
return i;
|
||
|
}
|
||
|
}
|
||
|
//didn't find the game so lets load up the
|
||
|
//ids in case they're not loaded yet
|
||
|
if (getNewIds) {
|
||
|
game_populate_ids();
|
||
|
getNewIds = false;
|
||
|
goto retry;
|
||
|
}
|
||
|
//should never get here...
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
// find gameList[gi] in all_gameList[]
|
||
|
int cache_game_find(int gi)
|
||
|
{
|
||
|
// try if direct mapping exists
|
||
|
int agi = ccache.game[gi].agi;
|
||
|
if ( (strcmp(ccache.game[agi].id, (char*)gameList[gi].id) == 0) &&
|
||
|
(strcmp(ccache.game[agi].id, (char*)all_gameList[agi].id) == 0) )
|
||
|
{
|
||
|
return agi;
|
||
|
}
|
||
|
// direct mapping not found, search for id
|
||
|
return cache_game_find_id(gi);
|
||
|
}
|
||
|
|
||
|
int cache_find(char *id)
|
||
|
{
|
||
|
int i;
|
||
|
for (i=0; i<ccache.num; i++) {
|
||
|
if (strcmp(ccache.cover[i].id, id) == 0) {
|
||
|
return i;
|
||
|
}
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
int cache_find_free()
|
||
|
{
|
||
|
int i;
|
||
|
for (i=0; i<ccache.num; i++) {
|
||
|
if (*ccache.cover[i].id == 0) {
|
||
|
return i;
|
||
|
}
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
int cache_find_lru()
|
||
|
{
|
||
|
int i, idx;
|
||
|
//int lru = -1;
|
||
|
//int found = -1;
|
||
|
// not a true lru, but good enough
|
||
|
for (i=0; i<ccache.num; i++) {
|
||
|
idx = (ccache.lru + i) % ccache.num;
|
||
|
if (ccache.cover[idx].used == 0) {
|
||
|
ccache.lru = (idx + 1) % ccache.num;
|
||
|
return idx;
|
||
|
}
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
void cache_free(int i)
|
||
|
{
|
||
|
memcheck_ptr(ccache.cover, &ccache.cover[i]);
|
||
|
int gid = ccache.cover[i].gid;
|
||
|
if (*ccache.cover[i].id && gid >= 0) {
|
||
|
memcheck_ptr(ccache.game, &ccache.game[gid]);
|
||
|
ccache.game[gid].state = CS_IDLE;
|
||
|
}
|
||
|
*ccache.cover[i].id = 0;
|
||
|
ccache.cover[i].used = 0;
|
||
|
//ccache.cover[i].lru = 0;
|
||
|
ccache.cover[i].gid = -1;
|
||
|
}
|
||
|
|
||
|
int cache_alloc(char *id)
|
||
|
{
|
||
|
int i;
|
||
|
i = cache_find(id);
|
||
|
if (i >= 0) goto found;
|
||
|
i = cache_find_free();
|
||
|
if (i >= 0) goto found;
|
||
|
i = cache_find_lru();
|
||
|
if (i >= 0) goto found;
|
||
|
return -1;
|
||
|
found:
|
||
|
cache_free(i);
|
||
|
return i;
|
||
|
}
|
||
|
|
||
|
#ifdef DEBUG_CACHE
|
||
|
|
||
|
struct Cache_Debug
|
||
|
{
|
||
|
int cc_rq_prio;
|
||
|
int cc_i;
|
||
|
int cc_line;
|
||
|
int cc_line_lock;
|
||
|
int cc_line_unlock;
|
||
|
int cc_locked;
|
||
|
int cc_cid;
|
||
|
void *cc_img;
|
||
|
void *cc_buf;
|
||
|
};
|
||
|
|
||
|
struct Cache_Debug ccdbg;
|
||
|
|
||
|
#define ___ ccdbg.cc_line = __LINE__
|
||
|
#define CACHE_LOCK() do{ \
|
||
|
ccdbg.cc_line_lock = __LINE__; \
|
||
|
LWP_MutexLock(ccache.mutex); \
|
||
|
ccdbg.cc_locked = 1; }while(0)
|
||
|
|
||
|
#define CACHE_UNLOCK() do{ \
|
||
|
ccdbg.cc_line_unlock = __LINE__; \
|
||
|
LWP_MutexUnlock(ccache.mutex); \
|
||
|
ccdbg.cc_locked = 0; }while(0)
|
||
|
|
||
|
#define CCDBG(X) X
|
||
|
|
||
|
#else
|
||
|
|
||
|
#define ___
|
||
|
#define CACHE_LOCK() LWP_MutexLock(ccache.mutex)
|
||
|
#define CACHE_UNLOCK() LWP_MutexUnlock(ccache.mutex)
|
||
|
#define CCDBG(X)
|
||
|
|
||
|
#endif
|
||
|
|
||
|
|
||
|
/**
|
||
|
* This is the main cache thread. This method loops through all the game indexes
|
||
|
* (0 through ccache.num_game) and loads each requested cover. The covers are loaded
|
||
|
* based on the priority level (rq_prio) -> 1 is highest priority, 3 is lowest.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
void* cache_thread(void *arg)
|
||
|
{
|
||
|
int i, ret, actual_i;
|
||
|
char *id;
|
||
|
void *img;
|
||
|
void *buf;
|
||
|
int cid;
|
||
|
int rq_prio;
|
||
|
bool resizeToFullCover = false;
|
||
|
int cover_height, cover_width;
|
||
|
char path[200];
|
||
|
//cc_dbg("thread started\n");
|
||
|
|
||
|
___;
|
||
|
while (!ccache.quit) {
|
||
|
___;
|
||
|
CACHE_LOCK();
|
||
|
ccache.idle = 0;
|
||
|
//cc_dbg("thread running\n");
|
||
|
|
||
|
restart:
|
||
|
|
||
|
ccache.restart = 0;
|
||
|
___;
|
||
|
//if (0) // disable
|
||
|
for (rq_prio=1; rq_prio<=3; rq_prio++) {
|
||
|
___;
|
||
|
for (i=0; i<ccache.num_game; i++) {
|
||
|
___;
|
||
|
CCDBG(ccdbg.cc_rq_prio = rq_prio);
|
||
|
CCDBG(ccdbg.cc_i = i);
|
||
|
if (ccache.restart) goto restart;
|
||
|
//get the actual game index, in case we're using favorites
|
||
|
actual_i = cache_game_find(i);
|
||
|
memcheck_ptr(ccache.game, &ccache.game[actual_i]);
|
||
|
if (ccache.game[actual_i].request != rq_prio) continue; //was "<"
|
||
|
// load
|
||
|
ccache.game[actual_i].state = CS_LOADING;
|
||
|
memcheck_ptr(gameList, &gameList[i]);
|
||
|
id = (char*)gameList[i].id;
|
||
|
|
||
|
//cc_dbg("thread processing %d %s\n", i, id);
|
||
|
img = NULL;
|
||
|
___;
|
||
|
CACHE_UNLOCK();
|
||
|
___;
|
||
|
|
||
|
//capture the current cover width and height
|
||
|
cover_height = COVER_HEIGHT;
|
||
|
cover_width = COVER_WIDTH;
|
||
|
|
||
|
//load the cover image
|
||
|
if (CFG.cover_style == CFG_COVER_STYLE_FULL) {
|
||
|
//try to load the full cover
|
||
|
ret = Gui_LoadCover_style((u8*)id, &img, false, CFG_COVER_STYLE_FULL, path);
|
||
|
if (ret < 0) {
|
||
|
//try to load the 2D cover
|
||
|
ret = Gui_LoadCover_style((u8*)id, &img, true, CFG_COVER_STYLE_2D, path);
|
||
|
if (ret && img) resizeToFullCover = true;
|
||
|
}
|
||
|
} else {
|
||
|
ret = Gui_LoadCover_style((u8*)id, &img, false, CFG.cover_style, path);
|
||
|
}
|
||
|
//sleep(1);//dbg
|
||
|
___;
|
||
|
if (ccache.quit) goto quit;
|
||
|
___;
|
||
|
CACHE_LOCK();
|
||
|
___;
|
||
|
if (ret > 0 && img) {
|
||
|
___;
|
||
|
cid = cache_alloc(id);
|
||
|
if (cid < 0) {
|
||
|
// should not happen
|
||
|
free(img);
|
||
|
ccache.game[actual_i].state = CS_IDLE;
|
||
|
goto end_request;
|
||
|
}
|
||
|
buf = ccache.data + ccache.csize * cid;
|
||
|
___;
|
||
|
CACHE_UNLOCK();
|
||
|
___;
|
||
|
memcheck_ptr(ccache.cover, &ccache.cover[cid]);
|
||
|
CCDBG(ccdbg.cc_img = img);
|
||
|
CCDBG(ccdbg.cc_buf = buf);
|
||
|
CCDBG(ccdbg.cc_cid = cid);
|
||
|
//sleep(3);//dbg
|
||
|
|
||
|
LWP_MutexUnlock(ccache.mutex);
|
||
|
|
||
|
//make sure the cover height and width didn't change on us before we resize it
|
||
|
if ((cover_height == COVER_HEIGHT) &&
|
||
|
(cover_width == COVER_WIDTH)) {
|
||
|
|
||
|
if (resizeToFullCover) {
|
||
|
ccache.cover[cid].tx = Gui_LoadTexture_fullcover(img, COVER_WIDTH, COVER_HEIGHT, COVER_WIDTH_FRONT, buf, path);
|
||
|
resizeToFullCover = false;
|
||
|
} else {
|
||
|
if (CFG.cover_style == CFG_COVER_STYLE_FULL && CFG.gui_compress_covers) {
|
||
|
ccache.cover[cid].tx = Gui_LoadTexture_CMPR(img, COVER_WIDTH, COVER_HEIGHT, buf, path);
|
||
|
} else {
|
||
|
ccache.cover[cid].tx = Gui_LoadTexture_RGBA8(img, COVER_WIDTH, COVER_HEIGHT, buf, path);
|
||
|
}
|
||
|
if (!ccache.cover[cid].tx.data && CFG.cover_style == CFG_COVER_STYLE_FULL) {
|
||
|
//corrupted image - try to load the 2D cover
|
||
|
ret = Gui_LoadCover_style((u8*)id, &img, true, CFG_COVER_STYLE_2D, path);
|
||
|
if (ret && img)
|
||
|
ccache.cover[cid].tx = Gui_LoadTexture_fullcover(img, COVER_WIDTH, COVER_HEIGHT, COVER_WIDTH_FRONT, buf, path);
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
CCDBG(ccdbg.cc_img = NULL);
|
||
|
CCDBG(ccdbg.cc_buf = NULL);
|
||
|
free(img);
|
||
|
if (ccache.quit) goto quit;
|
||
|
CACHE_LOCK();
|
||
|
goto end_request;
|
||
|
}
|
||
|
|
||
|
CCDBG(ccdbg.cc_img = NULL);
|
||
|
CCDBG(ccdbg.cc_buf = NULL);
|
||
|
free(img);
|
||
|
___;
|
||
|
if (ccache.quit) goto quit;
|
||
|
___;
|
||
|
CACHE_LOCK();
|
||
|
___;
|
||
|
if (ccache.cover[cid].tx.data == NULL) {
|
||
|
cache_free(cid);
|
||
|
goto noimg;
|
||
|
}
|
||
|
// mark
|
||
|
STRCOPY(ccache.cover[cid].id, id);
|
||
|
// check if req change
|
||
|
if (ccache.game[actual_i].request) {
|
||
|
ccache.cover[cid].used = 1;
|
||
|
ccache.cover[cid].gid = actual_i;
|
||
|
ccache.game[actual_i].cid = cid;
|
||
|
ccache.game[actual_i].state = CS_PRESENT;
|
||
|
//cc_dbg("Load OK %d %d %s\n", i, cid, id);
|
||
|
|
||
|
} else {
|
||
|
ccache.game[actual_i].state = CS_IDLE;
|
||
|
}
|
||
|
} else {
|
||
|
noimg:
|
||
|
ccache.game[actual_i].state = CS_MISSING;
|
||
|
//cc_dbg("Load FAIL %d %s\n", i, id);
|
||
|
}
|
||
|
end_request:
|
||
|
// request processed.
|
||
|
ccache.game[actual_i].request = 0;
|
||
|
___;
|
||
|
}
|
||
|
}
|
||
|
___;
|
||
|
CCDBG(ccdbg.cc_rq_prio = 0);
|
||
|
CCDBG(ccdbg.cc_i = -1);
|
||
|
ccache.idle = 1;
|
||
|
//cc_dbg("thread idle\n");
|
||
|
// all processed. wait.
|
||
|
while (!ccache.restart && !ccache.quit) {
|
||
|
___;
|
||
|
CACHE_UNLOCK();
|
||
|
//cc_dbg("thread sleep\n");
|
||
|
___;
|
||
|
LWP_ThreadSleep(ccache.queue);
|
||
|
//cc_dbg("thread wakeup\n");
|
||
|
___;
|
||
|
CACHE_LOCK();
|
||
|
___;
|
||
|
}
|
||
|
___;
|
||
|
ccache.restart = 0;
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
|
||
|
quit:
|
||
|
___;
|
||
|
ccache.quit = 0;
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
#undef ___
|
||
|
|
||
|
void cache_wait_idle()
|
||
|
{
|
||
|
int i;
|
||
|
if (!ccache_init) return;
|
||
|
// wait till idle
|
||
|
i = 1000; // 1 second
|
||
|
while (!ccache.idle && i) {
|
||
|
usleep(1000);
|
||
|
i--;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This method is used to set the caching priority of the passed in game index.
|
||
|
* If the image is already cached or missing then the CoverCache is updated for
|
||
|
* the cover.
|
||
|
*
|
||
|
* @param game_i the game index
|
||
|
* @param rq_prio the cache priority: 1 = high priority
|
||
|
*/
|
||
|
void cache_request_1(int game_i, int rq_prio)
|
||
|
{
|
||
|
int cid, actual_i;
|
||
|
if (game_i < 0 || game_i >= ccache.num_game) return;
|
||
|
|
||
|
actual_i = cache_game_find(game_i);
|
||
|
if (ccache.game[actual_i].request>0) return;
|
||
|
// check if already present
|
||
|
if (ccache.game[actual_i].state == CS_PRESENT) {
|
||
|
cid = ccache.game[actual_i].cid;
|
||
|
ccache.cover[cid].used = 1;
|
||
|
return;
|
||
|
}
|
||
|
// check if already missing
|
||
|
if (ccache.game[actual_i].state == CS_MISSING) {
|
||
|
return;
|
||
|
}
|
||
|
// check if already cached
|
||
|
cid = cache_find((char*)gameList[game_i].id);
|
||
|
if (cid >= 0) {
|
||
|
ccache.cover[cid].used = 1;
|
||
|
ccache.cover[cid].gid = actual_i;
|
||
|
ccache.game[actual_i].cid = cid;
|
||
|
ccache.game[actual_i].state = CS_PRESENT;
|
||
|
return;
|
||
|
}
|
||
|
// add request
|
||
|
ccache.game[actual_i].request = rq_prio;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* This method is used to set the caching priority of the passed in game index
|
||
|
* as well as the next x number of games.
|
||
|
*
|
||
|
* @param game_i the starting game index
|
||
|
* @param num the number of images to set the priority on
|
||
|
* @param rq_prio the cache priority: 1 = high priority
|
||
|
*/
|
||
|
void cache_request(int game_i, int num, int rq_prio)
|
||
|
{
|
||
|
int i, idle;
|
||
|
// setup requests
|
||
|
CACHE_LOCK();
|
||
|
for (i=0; i<num; i++) {
|
||
|
cache_request_1(game_i + i, rq_prio);
|
||
|
}
|
||
|
// restart thread
|
||
|
ccache.restart = 1;
|
||
|
idle = ccache.idle;
|
||
|
CACHE_UNLOCK();
|
||
|
//cc_dbg("cache restart\n");
|
||
|
LWP_ThreadSignal(ccache.queue);
|
||
|
if (idle) {
|
||
|
//cc_dbg("thread idle wait restart\n");
|
||
|
i = 10;
|
||
|
while (ccache.restart && i) {
|
||
|
usleep(10);
|
||
|
LWP_ThreadSignal(ccache.queue);
|
||
|
i--;
|
||
|
}
|
||
|
if (ccache.restart) {
|
||
|
//cc_dbg("thread fail restart\n");
|
||
|
}
|
||
|
}
|
||
|
//cc_dbg("cache restart done\n");
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* This method is used to set the caching priority of the passed in game index
|
||
|
* as well as the next x number of games before and after the index. All the
|
||
|
* rest of the covers are given a lower priority.
|
||
|
*
|
||
|
* @param game_i the starting game index
|
||
|
* @param num the number of images before and after the game_i to set the priority on
|
||
|
* @param rq_prio the cache priority: 1 = high priority
|
||
|
*/
|
||
|
void cache_request_before_and_after(int game_i, int num, int rq_prio)
|
||
|
{
|
||
|
int i, idle;
|
||
|
int nextRight, nextLeft;
|
||
|
|
||
|
// setup requests
|
||
|
CACHE_LOCK();
|
||
|
|
||
|
// set the first one high
|
||
|
cache_request_1(game_i, rq_prio);
|
||
|
|
||
|
// alternate the rest
|
||
|
nextRight = game_i + 1;
|
||
|
nextLeft = game_i - 1;
|
||
|
for (i=0; i<num; i++) {
|
||
|
if (nextRight >= ccache.num_game)
|
||
|
nextRight = 0;
|
||
|
cache_request_1(nextRight, 2);
|
||
|
|
||
|
if (nextLeft < 0)
|
||
|
nextLeft = ccache.num_game - 1;
|
||
|
cache_request_1(nextLeft, 2);
|
||
|
|
||
|
nextRight++;
|
||
|
nextLeft--;
|
||
|
}
|
||
|
|
||
|
// now set the priority to low for +-10 covers
|
||
|
for (i=0; i<10; i++) {
|
||
|
if (nextRight >= ccache.num_game)
|
||
|
nextRight = 0;
|
||
|
cache_request_1(nextRight, 3);
|
||
|
|
||
|
if (nextLeft < 0)
|
||
|
nextLeft = ccache.num_game - 1;
|
||
|
cache_request_1(nextLeft, 3);
|
||
|
|
||
|
nextRight++;
|
||
|
nextLeft--;
|
||
|
}
|
||
|
|
||
|
// restart thread
|
||
|
ccache.restart = 1;
|
||
|
idle = ccache.idle;
|
||
|
CACHE_UNLOCK();
|
||
|
LWP_ThreadSignal(ccache.queue);
|
||
|
if (idle) {
|
||
|
//cc_dbg("thread idle wait restart\n");
|
||
|
i = 10;
|
||
|
while (ccache.restart && i) {
|
||
|
usleep(10);
|
||
|
LWP_ThreadSignal(ccache.queue);
|
||
|
i--;
|
||
|
}
|
||
|
if (ccache.restart) {
|
||
|
//cc_dbg("thread fail restart\n");
|
||
|
}
|
||
|
}
|
||
|
//cc_dbg("cache restart done\n");
|
||
|
}
|
||
|
|
||
|
|
||
|
void cache_release_1(int game_i)
|
||
|
{
|
||
|
int cid;
|
||
|
if (game_i < 0 || game_i >= ccache.num_game) return;
|
||
|
memcheck_ptr(ccache.game, &ccache.game[game_i]);
|
||
|
ccache.game[game_i].request = 0;
|
||
|
if (ccache.game[game_i].state == CS_PRESENT) {
|
||
|
// release if already cached
|
||
|
cid = ccache.game[game_i].cid;
|
||
|
memcheck_ptr(ccache.cover, &ccache.cover[cid]);
|
||
|
ccache.cover[cid].used = 0; //--
|
||
|
ccache.game[game_i].state = CS_IDLE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void cache_release(int game_i, int num)
|
||
|
{
|
||
|
int i;
|
||
|
// cancel requests
|
||
|
CACHE_LOCK();
|
||
|
for (i=0; i<num; i++) {
|
||
|
cache_release_1(game_i + i);
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
|
||
|
void cache_release_all()
|
||
|
{
|
||
|
int i;
|
||
|
// cancel all requests
|
||
|
CACHE_LOCK();
|
||
|
for (i=0; i<ccache.num_game; i++) {
|
||
|
cache_release_1(i);
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
|
||
|
void cache_num_game()
|
||
|
{
|
||
|
ccache.num_game = gameCnt;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Reallocates cache tables, if gamelist resized
|
||
|
*/
|
||
|
int cache_resize_tables()
|
||
|
{
|
||
|
bool resized = false;
|
||
|
int size, new_num;
|
||
|
|
||
|
//cc_dbg("\ncache_resize %p %d %d %p %d %d\n",
|
||
|
// ccache.game, ccache.game_alloc, ccache.num_game,
|
||
|
// ccache.cover, ccache.cover_alloc, ccache.num);
|
||
|
|
||
|
|
||
|
memcheck();
|
||
|
// game table
|
||
|
// resize if games added
|
||
|
if (all_gameCnt > ccache.game_alloc || ccache.game == NULL)
|
||
|
{
|
||
|
ccache.game_alloc = all_gameCnt;
|
||
|
size = sizeof(struct Game_State) * ccache.game_alloc;
|
||
|
ccache.game = realloc(ccache.game, size);
|
||
|
if (ccache.game == NULL) goto err;
|
||
|
resized = true;
|
||
|
}
|
||
|
cache_num_game();
|
||
|
|
||
|
memcheck();
|
||
|
// cover table
|
||
|
// check cover size
|
||
|
|
||
|
if (CFG.cover_style == CFG_COVER_STYLE_FULL && CFG.gui_compress_covers) {
|
||
|
ccache.width4 = COVER_WIDTH;
|
||
|
ccache.height4 = COVER_HEIGHT;
|
||
|
ccache.csize = (ccache.width4 * ccache.height4)/2;
|
||
|
} else {
|
||
|
ccache.width4 = (COVER_WIDTH + 3) >> 2 << 2;
|
||
|
ccache.height4 = (COVER_HEIGHT + 3) >> 2 << 2;
|
||
|
ccache.csize = ccache.width4 * ccache.height4 * 4;
|
||
|
}
|
||
|
|
||
|
new_num = COVER_CACHE_DATA_SIZE / ccache.csize;
|
||
|
if (new_num > ccache.cover_alloc || ccache.cover == NULL) {
|
||
|
ccache.cover_alloc = new_num;
|
||
|
size = sizeof(struct Cover_State) * ccache.cover_alloc;
|
||
|
ccache.cover = realloc(ccache.cover, size);
|
||
|
if (ccache.cover == NULL) goto err;
|
||
|
resized = true;
|
||
|
}
|
||
|
ccache.num = new_num;
|
||
|
|
||
|
memcheck();
|
||
|
if (resized) {
|
||
|
// clear both tables
|
||
|
size = sizeof(struct Game_State) * ccache.game_alloc;
|
||
|
memset(ccache.game, 0, size);
|
||
|
size = sizeof(struct Cover_State) * ccache.cover_alloc;
|
||
|
memset(ccache.cover, 0, size);
|
||
|
ccache.lru = 0;
|
||
|
}
|
||
|
memcheck();
|
||
|
|
||
|
//cc_dbg("cache_resize %p %d %d %p %d %d\n",
|
||
|
// ccache.game, ccache.game_alloc, ccache.num_game,
|
||
|
// ccache.cover, ccache.cover_alloc, ccache.num);
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
err:
|
||
|
// fatal
|
||
|
printf(gt("ERROR: cache: out of memory"));
|
||
|
printf("\n");
|
||
|
sleep(1);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Initializes the cover cache
|
||
|
*/
|
||
|
int Cache_Init()
|
||
|
{
|
||
|
int ret;
|
||
|
|
||
|
//update the cache tables size
|
||
|
if (ccache_inv || !ccache_init) {
|
||
|
// call invalidate again, for safety
|
||
|
// this will cancel all requests and wait till idle
|
||
|
Cache_Invalidate();
|
||
|
ret = cache_resize_tables();
|
||
|
if (ret) return -1;
|
||
|
ccache_inv = 0;
|
||
|
}
|
||
|
ccache.num_game = gameCnt;
|
||
|
|
||
|
if (ccache_init) return 0;
|
||
|
ccache_init = 1;
|
||
|
|
||
|
ccache.data = LARGE_memalign(32, COVER_CACHE_DATA_SIZE);
|
||
|
|
||
|
ccache.lwp = LWP_THREAD_NULL;
|
||
|
ccache.queue = LWP_TQUEUE_NULL;
|
||
|
ccache.mutex = LWP_MUTEX_NULL;
|
||
|
|
||
|
// start thread
|
||
|
LWP_InitQueue(&ccache.queue);
|
||
|
LWP_MutexInit(&ccache.mutex, FALSE);
|
||
|
ccache.run = 1;
|
||
|
//cc_dbg("running thread\n");
|
||
|
ret = LWP_CreateThread(&ccache.lwp, cache_thread, NULL, NULL, 32*1024, 40);
|
||
|
if (ret < 0) {
|
||
|
//cache_thread_stop();
|
||
|
return -1;
|
||
|
}
|
||
|
//cc_dbg("lwp ret: %d id: %x\n", ret, ccache.lwp);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void Cache_Invalidate()
|
||
|
{
|
||
|
if (!ccache_init) return;
|
||
|
CACHE_LOCK();
|
||
|
// clear requests
|
||
|
memset(ccache.game, 0, sizeof(struct Game_State) * ccache.game_alloc);
|
||
|
CACHE_UNLOCK();
|
||
|
// wait till idle
|
||
|
cache_wait_idle();
|
||
|
if (!ccache.idle) {
|
||
|
//printf("\nERROR: cache not idle\n"); sleep(3);
|
||
|
}
|
||
|
// clear all
|
||
|
memset(ccache.game, 0, sizeof(struct Game_State) * ccache.game_alloc);
|
||
|
memset(ccache.cover, 0, sizeof(struct Cover_State) * ccache.cover_alloc);
|
||
|
|
||
|
ccache_inv = 1;
|
||
|
}
|
||
|
|
||
|
void Cache_Close()
|
||
|
{
|
||
|
int i;
|
||
|
if (!ccache_init) return;
|
||
|
ccache_init = 0;
|
||
|
CACHE_LOCK();
|
||
|
ccache.quit = 1;
|
||
|
CACHE_UNLOCK();
|
||
|
LWP_ThreadSignal(ccache.queue);
|
||
|
i = 1000; // 1 second
|
||
|
while (ccache.quit && i) {
|
||
|
usleep(1000);
|
||
|
LWP_ThreadSignal(ccache.queue);
|
||
|
i--;
|
||
|
}
|
||
|
if (ccache.quit) {
|
||
|
printf("\n");
|
||
|
printf(gt("ERROR: Cache Close"));
|
||
|
printf("\n");
|
||
|
sleep(5);
|
||
|
} else {
|
||
|
LWP_JoinThread(ccache.lwp, NULL);
|
||
|
}
|
||
|
LWP_CloseQueue(ccache.queue);
|
||
|
LWP_MutexDestroy(ccache.mutex);
|
||
|
ccache.lwp = LWP_THREAD_NULL;
|
||
|
ccache.queue = LWP_TQUEUE_NULL;
|
||
|
ccache.mutex = LWP_MUTEX_NULL;
|
||
|
SAFE_FREE(ccache.cover);
|
||
|
SAFE_FREE(ccache.game);
|
||
|
}
|
||
|
|
||
|
void draw_Cache()
|
||
|
{
|
||
|
#ifdef DEBUG_CACHE
|
||
|
unsigned int i, x, y, c, cid;
|
||
|
|
||
|
// thread state
|
||
|
x = 15;
|
||
|
y = 30;
|
||
|
c = ccache.idle ? 0x00FF00FF : 0xFF0000FF;
|
||
|
GRRLIB_Rectangle(x, y, 8, 8, c, 1);
|
||
|
x += 10;
|
||
|
c = !ccache.restart ? 0x00FF00FF : 0xFF0000FF;
|
||
|
GRRLIB_Rectangle(x, y, 8, 8, c, 1);
|
||
|
x += 10;
|
||
|
c = !ccache.quit ? 0x00FF00FF : 0xFF0000FF;
|
||
|
GRRLIB_Rectangle(x, y, 8, 8, c, 1);
|
||
|
x += 20;
|
||
|
GRRLIB_Printf(x, y, tx_cfont, 0xFFFFFFFF, 1, "%d %d %d",
|
||
|
ccdbg.cc_rq_prio, ccdbg.cc_i, ccdbg.cc_line);
|
||
|
// lock state
|
||
|
x = 15;
|
||
|
y = 40;
|
||
|
c = !ccdbg.cc_locked ? 0x00FF00FF : 0xFF0000FF;
|
||
|
GRRLIB_Rectangle(x, y, 8, 8, c, 1);
|
||
|
x += 40;
|
||
|
GRRLIB_Printf(x, y, tx_cfont, 0xFFFFFFFF, 1, "%d %d %p %p %d",
|
||
|
ccdbg.cc_line_lock, ccdbg.cc_line_unlock,
|
||
|
ccdbg.cc_img, ccdbg.cc_buf, ccdbg.cc_cid);
|
||
|
|
||
|
// cover state
|
||
|
for (i=0; i<ccache.num; i++) {
|
||
|
x = 15 + (i % 10) * 2;
|
||
|
y = 50 + (i / 10) * 2;
|
||
|
// free: black
|
||
|
// used: blue
|
||
|
// present: green
|
||
|
// add lru to red?
|
||
|
if (ccache.cover[i].id[0] == 0) {
|
||
|
c = 0x000000FF;
|
||
|
} else if (ccache.cover[i].used) {
|
||
|
c = 0x8080FFFF;
|
||
|
} else {
|
||
|
c = 0x80FF80FF;
|
||
|
}
|
||
|
GRRLIB_Rectangle(x, y, 2, 2, c, 1);
|
||
|
}
|
||
|
|
||
|
// game state
|
||
|
for (i=0; i<ccache.num_game; i++) {
|
||
|
x = 15 + (i % 10)*2;
|
||
|
y = 110 + (i / 10)*2;
|
||
|
// idle: black
|
||
|
// loading: yellow
|
||
|
// used: blue
|
||
|
// present: green
|
||
|
// missing: red
|
||
|
// request: violet
|
||
|
if (ccache.game[i].request) {
|
||
|
c = 0xFF00FFFF;
|
||
|
} else
|
||
|
switch (ccache.game[i].state) {
|
||
|
case CS_LOADING: c = 0xFFFF80FF; break;
|
||
|
case CS_PRESENT:
|
||
|
cid = ccache.game[i].cid;
|
||
|
if (ccache.cover[cid].used) {
|
||
|
c = 0x8080FFFF;
|
||
|
} else {
|
||
|
c = 0x80FF80FF;
|
||
|
}
|
||
|
break;
|
||
|
case CS_MISSING: c = 0xFF8080FF; break;
|
||
|
default:
|
||
|
case CS_IDLE: c = 0x000000FF; break;
|
||
|
}
|
||
|
//GRRLIB_Plot(x, y, c);
|
||
|
GRRLIB_Rectangle(x, y, 2, 2, c, 1);
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
GRRLIB_texImg* cache_request_cover(int game_index, int style, int format, int priority, int *state)
|
||
|
{
|
||
|
GRRLIB_texImg *tx = NULL;
|
||
|
int actual_i = cache_game_find(game_index);
|
||
|
int cstate = ccache.game[actual_i].state;
|
||
|
if (cstate == CS_PRESENT) {
|
||
|
tx = &ccache.cover[ccache.game[actual_i].cid].tx;
|
||
|
// mark used
|
||
|
}
|
||
|
if (state) *state = cstate;
|
||
|
return tx;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
#else // newcache
|
||
|
|
||
|
|
||
|
|
||
|
/*
|
||
|
sizes
|
||
|
2d: 64 kb 160x224 = 140 kb rgba
|
||
|
3d: 67 kb 176x248 = 170 kb rgba
|
||
|
full: 315 kb 512x340 = 680 kb rgba cmpr: 512x336/2 = 84 kb
|
||
|
hq: 1081 kb 1024x680 = 2720 kb rgba cmpr: 340 kb
|
||
|
mipmap full : 512x512 + LOD0 = 128 kb cmpr
|
||
|
mipmap full : 512x512 + LOD4 = 170 kb cmpr
|
||
|
mipmap hq : 1024x1024 + LOD5 = 682 kb cmpr
|
||
|
|
||
|
mipmap LOD 4 : 512, 256, 128, 64, 32
|
||
|
|
||
|
num of covers that fit in 25MB:
|
||
|
25MB / 140 = 128 2d rgba
|
||
|
25MB / 170 = 150 3d rgba
|
||
|
25MB / 84 = 304 full cmpr
|
||
|
25MB / 128 = 200 full mipmap lod0
|
||
|
25MB / 170 = 150 full mipmap lod4
|
||
|
25MB / 340 = 75 hq cmpr
|
||
|
25MB / 682 = 37 hq mipmap lod5
|
||
|
|
||
|
times: (seconds) (from SD)
|
||
|
load,decode,sum,covers, ms/c
|
||
|
3d: 1.114 + 1.228 = 2.342 / 71 = 32.9
|
||
|
full: 4.344 + 7.064 = 11.408 / 68 = 167.7
|
||
|
.ccc: 2.363 + 0 = 2.363 / 68 = 34.7
|
||
|
*/
|
||
|
|
||
|
// sizeof(GRRLIB_texImg) = 60
|
||
|
|
||
|
|
||
|
#define CACHE_STATE_NUM 500
|
||
|
#define LRU_MAX 0xFF
|
||
|
|
||
|
typedef struct Cover_Key
|
||
|
{
|
||
|
u8 id[7];
|
||
|
char style; // 2d, 3d, disc, full, hq
|
||
|
char format; // RGBA, C4x4, CMPR, MIPMAP
|
||
|
} Cover_Key;
|
||
|
|
||
|
typedef struct Cover_State
|
||
|
{
|
||
|
Cover_Key key;
|
||
|
char used; // 0=unused, 1=used
|
||
|
char request; // request priority: 3=hi 2=med 1=lo
|
||
|
char state; // free, loading, present, missing
|
||
|
char hq_available;
|
||
|
u8 lru; // least recently used
|
||
|
int hnext; // hash next
|
||
|
GRRLIB_texImg tx;
|
||
|
} Cover_State;
|
||
|
|
||
|
struct Cover_Cache
|
||
|
{
|
||
|
void *data;
|
||
|
heap heap;
|
||
|
Cover_State cstate[CACHE_STATE_NUM];
|
||
|
HashTable hash;
|
||
|
lwp_t lwp;
|
||
|
mutex_t mutex;
|
||
|
cond_t cond;
|
||
|
bool stop;
|
||
|
bool waiting;
|
||
|
int new_request; // highest priority of a new request
|
||
|
long long stat_load;
|
||
|
long long stat_decode;
|
||
|
int stat_n2d;
|
||
|
};
|
||
|
|
||
|
struct Cover_Cache ccache;
|
||
|
int ccache_init = 0;
|
||
|
int ccache_inv = 0;
|
||
|
|
||
|
|
||
|
#define CACHE_LOCK() LWP_MutexLock(ccache.mutex)
|
||
|
#define CACHE_UNLOCK() LWP_MutexUnlock(ccache.mutex)
|
||
|
|
||
|
u32 cover_key_hash(void *cb, void *key)
|
||
|
{
|
||
|
Cover_Key *ck = key;
|
||
|
return hash_string((char*)ck->id);
|
||
|
}
|
||
|
|
||
|
bool cover_key_cmp(void *cb, void *key, int handle)
|
||
|
{
|
||
|
Cover_Key *k1 = key;
|
||
|
Cover_Key *k2 = &ccache.cstate[handle].key;
|
||
|
return memcmp(k1, k2, sizeof(*k1)) == 0;
|
||
|
}
|
||
|
|
||
|
int* cover_hnext(void *cb, int handle)
|
||
|
{
|
||
|
return &ccache.cstate[handle].hnext;
|
||
|
}
|
||
|
|
||
|
void make_cover_key(Cover_Key *ck, u8 *id, int style, int format)
|
||
|
{
|
||
|
memset(ck, 0, sizeof(*ck));
|
||
|
strcopy((char*)ck->id, (char*)id, sizeof(ck->id));
|
||
|
ck->style = style;
|
||
|
ck->format = format;
|
||
|
}
|
||
|
|
||
|
Cover_State* cache_find_cover(u8 *id, int style, int format)
|
||
|
{
|
||
|
Cover_Key ck;
|
||
|
int i;
|
||
|
make_cover_key(&ck, id, style, format);
|
||
|
CACHE_LOCK();
|
||
|
i = hash_get(&ccache.hash, &ck);
|
||
|
CACHE_UNLOCK();
|
||
|
if (i < 0) return NULL;
|
||
|
return &ccache.cstate[i];
|
||
|
}
|
||
|
|
||
|
Cover_State* cache_find_free()
|
||
|
{
|
||
|
int i;
|
||
|
Cover_State *cs = NULL;
|
||
|
for (i=0; i<CACHE_STATE_NUM; i++) {
|
||
|
cs = &ccache.cstate[i];
|
||
|
if (cs->used) continue;
|
||
|
if (cs->request) continue;
|
||
|
if (cs->state != CS_IDLE) continue;
|
||
|
if (cs->key.id[0]) continue;
|
||
|
return cs;
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
Cover_State* cache_find_lru(bool present)
|
||
|
{
|
||
|
Cover_State *cs = NULL;
|
||
|
Cover_State *lru_cs = NULL;
|
||
|
Cover_State *hq_cs = NULL;
|
||
|
int lru = -1;
|
||
|
int hq_lru = -1;
|
||
|
int hq_cnt = 0;
|
||
|
int rgb_cnt = 0;
|
||
|
int i;
|
||
|
CACHE_LOCK();
|
||
|
for (i=0; i<CACHE_STATE_NUM; i++) {
|
||
|
cs = &ccache.cstate[i];
|
||
|
// don't return a cover that is:
|
||
|
// used or requested or being loaded
|
||
|
// or the present flag doesn't match the present state
|
||
|
if (cs->used) continue;
|
||
|
if (cs->request) continue;
|
||
|
if (cs->state == CS_LOADING) continue;
|
||
|
if (present) {
|
||
|
if (cs->state != CS_PRESENT) continue;
|
||
|
} else {
|
||
|
if (cs->state == CS_PRESENT) continue;
|
||
|
}
|
||
|
if (cs->lru > lru) {
|
||
|
lru = cs->lru;
|
||
|
lru_cs = cs;
|
||
|
//if (lru >= LRU_MAX / 2) break;
|
||
|
}
|
||
|
if (present) {
|
||
|
// limit FULL/HQ RGBA to 3
|
||
|
if ( (cs->key.style == CFG_COVER_STYLE_FULL
|
||
|
|| cs->key.style == CFG_COVER_STYLE_HQ)
|
||
|
&& cs->key.format == CC_FMT_RGBA)
|
||
|
{
|
||
|
rgb_cnt++;
|
||
|
if (rgb_cnt > 3) {
|
||
|
lru_cs = cs;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
// limit HQ to 10
|
||
|
if (cs->key.style == CFG_COVER_STYLE_HQ) {
|
||
|
hq_cnt++;
|
||
|
if (cs->lru > hq_lru) {
|
||
|
hq_lru = cs->lru;
|
||
|
hq_cs = cs;
|
||
|
}
|
||
|
}
|
||
|
if (hq_cnt > 10) {
|
||
|
lru_cs = hq_cs;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
return lru_cs;
|
||
|
}
|
||
|
|
||
|
void cache_free_buf(void *ptr)
|
||
|
{
|
||
|
CACHE_LOCK();
|
||
|
if (ptr) heap_free(&ccache.heap, ptr);
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
|
||
|
#define CACHE_SAFE_FREE(ptr) if(ptr){cache_free_buf(ptr);ptr=NULL;}
|
||
|
|
||
|
void* cache_alloc_buf(int size)
|
||
|
{
|
||
|
CACHE_LOCK();
|
||
|
void *ptr = heap_alloc(&ccache.heap, size);
|
||
|
CACHE_UNLOCK();
|
||
|
return ptr;
|
||
|
}
|
||
|
|
||
|
void cache_free_cover(Cover_State *cs)
|
||
|
{
|
||
|
int i = cs - ccache.cstate;
|
||
|
CACHE_LOCK();
|
||
|
hash_remove(&ccache.hash, &cs->key, i);
|
||
|
CACHE_SAFE_FREE(cs->tx.data);
|
||
|
memset(cs, 0, sizeof(*cs));
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
|
||
|
Cover_State* cache_free_lru(bool present)
|
||
|
{
|
||
|
Cover_State *cs;
|
||
|
cs = cache_find_lru(present);
|
||
|
if (cs) {
|
||
|
cache_free_cover(cs);
|
||
|
}
|
||
|
return cs;
|
||
|
}
|
||
|
|
||
|
void* cache_alloc_data(int size)
|
||
|
{
|
||
|
void *ptr = NULL;
|
||
|
CACHE_LOCK();
|
||
|
do {
|
||
|
ptr = cache_alloc_buf(size);
|
||
|
if (!ptr) {
|
||
|
// try to free
|
||
|
// find lru with data
|
||
|
Cover_State *cs = cache_free_lru(true);
|
||
|
if (!cs) break;
|
||
|
}
|
||
|
} while (!ptr);
|
||
|
CACHE_UNLOCK();
|
||
|
if (!ptr) dbg_printf("cc: alloc err %d\n", size);
|
||
|
return ptr;
|
||
|
}
|
||
|
|
||
|
Cover_State* cache_get_free_cover()
|
||
|
{
|
||
|
Cover_State *cs = NULL;
|
||
|
cs = cache_find_free();
|
||
|
if (!cs) {
|
||
|
cs = cache_free_lru(false);
|
||
|
}
|
||
|
if (!cs) {
|
||
|
cs = cache_free_lru(true);
|
||
|
}
|
||
|
return cs;
|
||
|
}
|
||
|
|
||
|
Cover_State* cache_alloc_cover(u8 *id, int style, int format)
|
||
|
{
|
||
|
Cover_State *cs = NULL;
|
||
|
CACHE_LOCK();
|
||
|
cs = cache_get_free_cover();
|
||
|
if (cs) {
|
||
|
make_cover_key(&cs->key, id, style, format);
|
||
|
int i = cs - ccache.cstate;
|
||
|
hash_add(&ccache.hash, &cs->key, i);
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
return cs;
|
||
|
}
|
||
|
|
||
|
// find existing mark as used and return
|
||
|
// or create new, set key, mark as used and return
|
||
|
// return NULL on error
|
||
|
Cover_State* cache_get_cover(u8 *id, int style, int format)
|
||
|
{
|
||
|
Cover_State *cs = NULL;
|
||
|
CACHE_LOCK();
|
||
|
cs = cache_find_cover(id, style, format);
|
||
|
if (!cs) {
|
||
|
cs = cache_alloc_cover(id, style, format);
|
||
|
}
|
||
|
if (cs) {
|
||
|
cs->used = 1;
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
return cs;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
|
||
|
struct CIMG_HDR
|
||
|
{
|
||
|
char tag[4];
|
||
|
char id[8];
|
||
|
short version;
|
||
|
short gxformat;
|
||
|
short width;
|
||
|
short height;
|
||
|
short lod;
|
||
|
short hq;
|
||
|
short pad[4];
|
||
|
};
|
||
|
|
||
|
char* cache_image_path(u8 *id, char *path, int size)
|
||
|
{
|
||
|
char *coverpath = cfg_get_covers_path(CFG_COVER_STYLE_CACHE);
|
||
|
snprintf(path, size, "%s/%.6s.ccc", coverpath, (char*)id);
|
||
|
return coverpath;
|
||
|
}
|
||
|
|
||
|
// the normal quality 512x512 mipmap is saved first followed by HQ
|
||
|
// so that it loads faster since HQ is used less frequently
|
||
|
int cache_save_image(u8 *id, GRRLIB_texImg *tx)
|
||
|
{
|
||
|
char filepath[200];
|
||
|
char *dir;
|
||
|
struct CIMG_HDR hdr;
|
||
|
int fd, ret;
|
||
|
int size, hq_size = 0;
|
||
|
u8 *data;
|
||
|
|
||
|
if (!id || !*id || !tx || !tx->data) return -1;
|
||
|
dir = cache_image_path(id, filepath, sizeof(filepath));
|
||
|
dbg_printf("cc save: %s\n", filepath);
|
||
|
mkdir(dir, 0777);
|
||
|
fd = open(filepath, O_CREAT | O_WRONLY, 0333);
|
||
|
if (fd < 0) return -1;
|
||
|
|
||
|
memset(&hdr, 0, sizeof(hdr));
|
||
|
memcpy(hdr.tag, "CIMG", 4);
|
||
|
memcpy(hdr.id, id, 6);
|
||
|
hdr.version = 1;
|
||
|
hdr.gxformat = tx->tex_format;
|
||
|
hdr.width = tx->w;
|
||
|
hdr.height = tx->h;
|
||
|
hdr.lod = tx->tex_lod;
|
||
|
hdr.hq = 0;
|
||
|
data = tx->data;
|
||
|
if (tx->w > 512 && tx->tex_lod) {
|
||
|
// hq
|
||
|
hq_size = fixGX_GetTexBufferSize(hdr.width, hdr.height,
|
||
|
hdr.gxformat, GX_FALSE, 1);
|
||
|
data += hq_size;
|
||
|
hdr.width /= 2;
|
||
|
hdr.height /= 2;
|
||
|
hdr.lod--;
|
||
|
hdr.hq = 1;
|
||
|
}
|
||
|
|
||
|
ret = write(fd, &hdr, sizeof(hdr));
|
||
|
if (ret != sizeof(hdr)) goto error;
|
||
|
|
||
|
size = fixGX_GetTexBufferSize(hdr.width, hdr.height,
|
||
|
hdr.gxformat, hdr.lod ? GX_TRUE : GX_FALSE, hdr.lod);
|
||
|
ret = write(fd, data, size);
|
||
|
if (ret != size) goto error;
|
||
|
|
||
|
if (hdr.hq) {
|
||
|
ret = write(fd, tx->data, hq_size);
|
||
|
if (ret != hq_size) goto error;
|
||
|
}
|
||
|
close(fd);
|
||
|
return 0;
|
||
|
|
||
|
error:
|
||
|
if (fd >= 0) close(fd);
|
||
|
remove(filepath);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
// style: FULL OR HQ
|
||
|
// FULL is always loaded, HQ only if requested and available in file
|
||
|
// hqavail is set to what is available in the file
|
||
|
int cache_load_image(u8 *id, GRRLIB_texImg *tx, int style, bool *hqavail)
|
||
|
{
|
||
|
char filepath[200];
|
||
|
struct CIMG_HDR hdr;
|
||
|
int fd, ret;
|
||
|
int size, hq_size = 0, data_size;
|
||
|
u8 *data = NULL;
|
||
|
|
||
|
if (!id || !*id || !tx || tx->data) return -1;
|
||
|
memset(tx, 0, sizeof(*tx));
|
||
|
memset(&hdr, 0, sizeof(hdr));
|
||
|
*hqavail = false;
|
||
|
cache_image_path(id, filepath, sizeof(filepath));
|
||
|
fd = open(filepath, O_RDONLY);
|
||
|
if (fd < 0) return -1;
|
||
|
dbg_printf("cc load: %s\n", filepath);
|
||
|
ret = read(fd, &hdr, sizeof(hdr));
|
||
|
if (ret != sizeof(hdr)) goto error;
|
||
|
if (memcmp(hdr.tag, "CIMG", 4)) goto error;
|
||
|
if (memcmp(hdr.id, id, 6)) goto error;
|
||
|
if (hdr.version != 1) goto error;
|
||
|
if (hdr.gxformat != GX_TF_CMPR) goto error;
|
||
|
if (hdr.width > 512) goto error;
|
||
|
if (hdr.height > 512) goto error;
|
||
|
|
||
|
size = fixGX_GetTexBufferSize(hdr.width, hdr.height,
|
||
|
hdr.gxformat, hdr.lod ? GX_TRUE : GX_FALSE, hdr.lod);
|
||
|
|
||
|
if (style == CFG_COVER_STYLE_HQ && hdr.hq) {
|
||
|
// load HQ too
|
||
|
hdr.width *= 2;
|
||
|
hdr.height *= 2;
|
||
|
hdr.lod++;
|
||
|
hq_size = fixGX_GetTexBufferSize(hdr.width, hdr.height,
|
||
|
hdr.gxformat, GX_FALSE, 1);
|
||
|
}
|
||
|
|
||
|
data_size = size + hq_size;
|
||
|
data = cache_alloc_data(data_size);
|
||
|
if (!data) {
|
||
|
close(fd);
|
||
|
return -2; // alloc error
|
||
|
}
|
||
|
|
||
|
ret = read(fd, data + hq_size, size);
|
||
|
if (ret != size) goto error;
|
||
|
|
||
|
if (hq_size) {
|
||
|
ret = read(fd, data, hq_size);
|
||
|
if (ret != hq_size) goto error;
|
||
|
//memset(data, 0xAA, 16000); // dbg red stripe
|
||
|
}
|
||
|
|
||
|
tx->data = data;
|
||
|
tx->w = hdr.width;
|
||
|
tx->h = hdr.height;
|
||
|
tx->tex_lod = hdr.lod;
|
||
|
tx->tex_format = hdr.gxformat;
|
||
|
GRRLIB_FlushTex(tx);
|
||
|
*hqavail = hdr.hq ? true : false;
|
||
|
close(fd);
|
||
|
return 0;
|
||
|
|
||
|
error:
|
||
|
CACHE_SAFE_FREE(data);
|
||
|
memset(tx, 0, sizeof(*tx));
|
||
|
close(fd);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
|
||
|
int cache_convert_image(u8 *id)
|
||
|
{
|
||
|
char path[200];
|
||
|
void *img = NULL;
|
||
|
GRRLIB_texImg tx;
|
||
|
u32 width=0, height=0;
|
||
|
int gx_lod = CC_MIPMAP_LOD;
|
||
|
int ret;
|
||
|
memset(&tx, 0, sizeof(tx));
|
||
|
ret = Gui_LoadCover_style(id, &img, false, CFG_COVER_STYLE_HQ, path);
|
||
|
if (ret <= 0) return -1;
|
||
|
ret = __Gui_GetPngDimensions(img, &width, &height);
|
||
|
if (ret < 0) goto out;
|
||
|
width = upperPower(width);
|
||
|
height = upperPower(height);
|
||
|
if (width > 512) gx_lod++; // +1 if HQ
|
||
|
tx = Gui_LoadTexture_MIPMAP(img, width, height, gx_lod, NULL, path);
|
||
|
if (!tx.data) { ret = -1; goto out; }
|
||
|
ret = cache_save_image(id, &tx);
|
||
|
out:
|
||
|
SAFE_FREE(img);
|
||
|
SAFE_FREE(tx.data);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void cache_check_convert_image(u8 *id)
|
||
|
{
|
||
|
struct stat st1, st2;
|
||
|
char path1[200], path2[200];
|
||
|
int ret1, ret2;
|
||
|
ret1 = find_cover_path(id, CFG_COVER_STYLE_FULL, path1, sizeof(path1), &st1);
|
||
|
cache_image_path(id, path2, sizeof(path2));
|
||
|
ret1 = stat(path1, &st1);
|
||
|
if (ret1 == 0) {
|
||
|
ret2 = stat(path2, &st2);
|
||
|
if (ret2 == 0 && st1.st_mtime <= st2.st_mtime) return;
|
||
|
// ignore if modify time in future
|
||
|
if (ret2 == 0 && st1.st_mtime > time(NULL)) return;
|
||
|
cache_convert_image(id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void cache_cover_load(Cover_State *cs)
|
||
|
{
|
||
|
char path[200];
|
||
|
void *img = NULL;
|
||
|
void *buf = NULL;
|
||
|
int size;
|
||
|
bool resizeToFullCover = false;
|
||
|
GRRLIB_texImg tx;
|
||
|
int ret;
|
||
|
int state = CS_IDLE;
|
||
|
long long t1, tload = 0, tdecode = 0;
|
||
|
bool hqavail = false;
|
||
|
Cover_State *csfull = NULL;
|
||
|
memset(&tx, 0, sizeof(tx));
|
||
|
|
||
|
if (cs->key.style == CFG_COVER_STYLE_HQ) {
|
||
|
//sleep(2);//dbg
|
||
|
// if FULL is present it will have the info if HQ is available
|
||
|
CACHE_LOCK();
|
||
|
csfull = cache_find_cover(cs->key.id, CFG_COVER_STYLE_FULL, cs->key.format);
|
||
|
if (csfull) hqavail = csfull->hq_available;
|
||
|
CACHE_UNLOCK();
|
||
|
if (csfull && !hqavail) {
|
||
|
state = CS_MISSING;
|
||
|
goto out;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// use cached converted image
|
||
|
if (cs->key.style == CFG_COVER_STYLE_FULL || cs->key.style == CFG_COVER_STYLE_HQ) {
|
||
|
if (cs->key.format == CC_FMT_MIPMAP) {
|
||
|
t1 = gettime();
|
||
|
cache_check_convert_image(cs->key.id);
|
||
|
tdecode = diff_usec(t1, gettime());
|
||
|
t1 = gettime();
|
||
|
ret = cache_load_image(cs->key.id, &tx, cs->key.style, &hqavail);
|
||
|
tload = diff_usec(t1, gettime());
|
||
|
if (ret == 0 && tx.data) goto out;
|
||
|
if (ret == -2) {
|
||
|
// alloc error
|
||
|
state = CS_IDLE;
|
||
|
goto out;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
t1 = gettime();
|
||
|
//load the cover image
|
||
|
ret = Gui_LoadCover_style((u8*)cs->key.id, &img, false, cs->key.style, path);
|
||
|
tload = diff_usec(t1, gettime());
|
||
|
//cc_dbg("cc ld: %.6s %d %d = %d %p\n",
|
||
|
// cs->key.id, cs->key.style, cs->key.format, ret, img);
|
||
|
if (ret <= 0 && cs->key.style == CFG_COVER_STYLE_FULL) {
|
||
|
//try to load the 2D cover
|
||
|
;overlay_2d:;
|
||
|
tload = 0;
|
||
|
SAFE_FREE(img);
|
||
|
ret = Gui_LoadCover_style((u8*)cs->key.id, &img, true, CFG_COVER_STYLE_2D, path);
|
||
|
//cc_dbg("cc ld 2d: %.6s %d %d = %d %p\n",
|
||
|
// cs->key.id, cs->key.style, cs->key.format, ret, img);
|
||
|
resizeToFullCover = true;
|
||
|
}
|
||
|
if (ret <= 0 || !img) {
|
||
|
state = CS_MISSING;
|
||
|
goto out;
|
||
|
}
|
||
|
if (resizeToFullCover) {
|
||
|
int cover_width_front = (int)(tx_nocover_full.h / 1.4) >> 2 << 2;
|
||
|
size = tx_nocover_full.w * tx_nocover_full.h * 4;
|
||
|
if (CFG.gui_compress_covers) {
|
||
|
size /= 8;
|
||
|
}
|
||
|
buf = cache_alloc_data(size);
|
||
|
if (!buf) {
|
||
|
state = CS_IDLE;
|
||
|
goto out;
|
||
|
}
|
||
|
// XXX format depends on CFG.gui_compress_covers
|
||
|
// needs to be an arg.
|
||
|
tx = Gui_LoadTexture_fullcover(img,
|
||
|
tx_nocover_full.w, tx_nocover_full.h,
|
||
|
cover_width_front, buf, path);
|
||
|
if (!tx.data) {
|
||
|
CACHE_SAFE_FREE(buf);
|
||
|
} else {
|
||
|
ccache.stat_n2d++;
|
||
|
}
|
||
|
} else {
|
||
|
u32 width=0, height=0;
|
||
|
// Get image dimensions
|
||
|
ret = __Gui_GetPngDimensions(img, &width, &height);
|
||
|
if (ret < 0) goto invalid_img;
|
||
|
//cc_dbg("cc sz %d x %d / %d x %d\n", width, height, COVER_WIDTH, COVER_HEIGHT);
|
||
|
if (width > 512) hqavail = true;
|
||
|
if (cs->key.style == CFG_COVER_STYLE_FULL) {
|
||
|
// scale down HQ if FULL requested
|
||
|
if (width > 512) {
|
||
|
width = 512;
|
||
|
height = 340;
|
||
|
//if (cs->key.format == CC_FMT_CMPR) height = 336;
|
||
|
}
|
||
|
//if (width > 64) { width = 64; height = 64; } // dbg
|
||
|
}
|
||
|
int gx_fmt = GX_TF_RGBA8;
|
||
|
int gx_mipmap = GX_FALSE;
|
||
|
int gx_lod = 0;
|
||
|
switch (cs->key.format) {
|
||
|
case CC_FMT_CMPR:
|
||
|
gx_fmt = GX_TF_CMPR;
|
||
|
// 512x512 CMPR looks better than 512x340
|
||
|
// but is slower - old AA with 512x512 is too slow
|
||
|
//height = width;
|
||
|
break;
|
||
|
case CC_FMT_MIPMAP:
|
||
|
gx_fmt = GX_TF_CMPR;
|
||
|
gx_mipmap = GX_TRUE;
|
||
|
gx_lod = CC_MIPMAP_LOD;
|
||
|
width = upperPower(width);
|
||
|
height = upperPower(height);
|
||
|
if (width > 512) gx_lod++; // +1 if HQ
|
||
|
break;
|
||
|
}
|
||
|
size = fixGX_GetTexBufferSize(width, height, gx_fmt, gx_mipmap, gx_lod);
|
||
|
buf = cache_alloc_data(size);
|
||
|
if (!buf) {
|
||
|
state = CS_IDLE;
|
||
|
goto out;
|
||
|
}
|
||
|
t1 = gettime();
|
||
|
switch (cs->key.format) {
|
||
|
default:
|
||
|
case CC_FMT_C4x4:
|
||
|
tx = Gui_LoadTexture_RGBA8(img, width, height, buf, path);
|
||
|
break;
|
||
|
case CC_FMT_CMPR:
|
||
|
tx = Gui_LoadTexture_CMPR(img, width, height, buf, path);
|
||
|
break;
|
||
|
case CC_FMT_MIPMAP:
|
||
|
tx = Gui_LoadTexture_MIPMAP(img, width, height, gx_lod, buf, path);
|
||
|
break;
|
||
|
}
|
||
|
tdecode = diff_usec(t1, gettime());
|
||
|
//cc_dbg("cc tx %d (%p %d %d %p) = %p\n", cs->key.format,
|
||
|
// img, COVER_WIDTH, COVER_HEIGHT, buf, tx.data);
|
||
|
if (!tx.data) {
|
||
|
;invalid_img:;
|
||
|
tdecode = 0;
|
||
|
state = CS_MISSING;
|
||
|
CACHE_SAFE_FREE(buf);
|
||
|
// corrupted image or out of mem?
|
||
|
dbg_printf("cc error %.6s\n", cs->key.id);
|
||
|
if (cs->key.style == CFG_COVER_STYLE_FULL) {
|
||
|
// try to load the 2D cover
|
||
|
goto overlay_2d;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
out:
|
||
|
SAFE_FREE(img);
|
||
|
CACHE_LOCK();
|
||
|
if (tx.data) {
|
||
|
state = CS_PRESENT;
|
||
|
if (cs->key.style == CFG_COVER_STYLE_HQ && !hqavail) {
|
||
|
// HQ requested but only full found
|
||
|
// if full exists then discard currently loaded
|
||
|
// else store as full and mark HQ as not available
|
||
|
csfull = cache_find_cover(cs->key.id, CFG_COVER_STYLE_FULL, cs->key.format);
|
||
|
if (!csfull) {
|
||
|
csfull = cache_get_cover(cs->key.id, CFG_COVER_STYLE_FULL, cs->key.format);
|
||
|
csfull->used = 0;
|
||
|
}
|
||
|
if (csfull) {
|
||
|
if (csfull->state != CS_PRESENT) {
|
||
|
csfull->state = CS_PRESENT;
|
||
|
csfull->tx = tx;
|
||
|
csfull->request = 0;
|
||
|
buf = NULL;
|
||
|
}
|
||
|
csfull->hq_available = false;
|
||
|
}
|
||
|
CACHE_SAFE_FREE(buf);
|
||
|
memset(&tx, 0, sizeof(tx));
|
||
|
state = CS_MISSING;
|
||
|
}
|
||
|
}
|
||
|
cs->state = state;
|
||
|
cs->request = 0;
|
||
|
cs->tx = tx;
|
||
|
cs->hq_available = hqavail;
|
||
|
if (tload && tx.data) {
|
||
|
ccache.stat_load += tload;
|
||
|
ccache.stat_decode += tdecode;
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
if (tx.data) cc_dbg("cc load: %.6s %d %d = %p %d\n",
|
||
|
cs->key.id, cs->key.style, cs->key.format, tx.data, state);
|
||
|
}
|
||
|
|
||
|
void cache_process_requests()
|
||
|
{
|
||
|
Cover_State *cs;
|
||
|
int prio;
|
||
|
int i;
|
||
|
do {
|
||
|
ccache.new_request = 0;
|
||
|
for (prio = 3; prio > 0; prio--) {
|
||
|
;restart:;
|
||
|
for (i=0; i<CACHE_STATE_NUM; i++) {
|
||
|
if (ccache.stop) break;
|
||
|
cs = &ccache.cstate[i];
|
||
|
CACHE_LOCK();
|
||
|
if (cs->request == prio) {
|
||
|
cs->state = CS_LOADING;
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
if (cs->state == CS_LOADING) {
|
||
|
// load
|
||
|
//sleep(1);//dbg
|
||
|
cache_cover_load(cs);
|
||
|
// need to restart loop at higher priority?
|
||
|
if (ccache.new_request > prio) {
|
||
|
//cc_dbg("new rq %d %d\n", ccache.new_request, prio);
|
||
|
prio = ccache.new_request;
|
||
|
ccache.new_request = 0;
|
||
|
goto restart;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
//if (ccache.new_request) cc_dbg("new rq %d\n", ccache.new_request);
|
||
|
} while (ccache.new_request);
|
||
|
}
|
||
|
|
||
|
void* cache_thread(void *arg)
|
||
|
{
|
||
|
cc_dbg("cc started\n");
|
||
|
while (1) {
|
||
|
// lock
|
||
|
CACHE_LOCK();
|
||
|
if (!ccache.stop && ccache.new_request == 0) {
|
||
|
cc_dbg("cc waiting\n");
|
||
|
ccache.waiting = true;
|
||
|
// LWP_CondWait unlocks mutex on enter
|
||
|
LWP_CondWait(ccache.cond, ccache.mutex);
|
||
|
// LWP_CondWait locks mutex on exit
|
||
|
ccache.waiting = false;
|
||
|
cc_dbg("cc -> running\n");
|
||
|
}
|
||
|
// unlock
|
||
|
CACHE_UNLOCK();
|
||
|
if (ccache.stop) {
|
||
|
break;
|
||
|
}
|
||
|
cache_process_requests();
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
int Cache_Init()
|
||
|
{
|
||
|
if (ccache_init) return 0;
|
||
|
int ret;
|
||
|
ccache.data = mem2_alloc(COVER_CACHE_DATA_SIZE);
|
||
|
if (!ccache.data) return -1;
|
||
|
heap_init(&ccache.heap, ccache.data, COVER_CACHE_DATA_SIZE);
|
||
|
hash_init(&ccache.hash, 0, NULL, cover_key_hash, cover_key_cmp, cover_hnext);
|
||
|
ccache.lwp = LWP_THREAD_NULL;
|
||
|
ccache.mutex = LWP_MUTEX_NULL;
|
||
|
ccache.cond = LWP_COND_NULL;
|
||
|
LWP_MutexInit(&ccache.mutex, true);
|
||
|
LWP_CondInit(&ccache.cond);
|
||
|
// start thread
|
||
|
dbg_printf("cc. start\n");
|
||
|
ret = LWP_CreateThread(&ccache.lwp, cache_thread, NULL, NULL, 32*1024, 40);
|
||
|
if (ret < 0) {
|
||
|
return -1;
|
||
|
}
|
||
|
//cc_dbg("lwp ret: %d id: %x\n", ret, ccache.lwp);
|
||
|
ccache_init = 1;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void Cache_Wakeup()
|
||
|
{
|
||
|
//cc_dbg("cc wakeup\n");
|
||
|
CACHE_LOCK();
|
||
|
bool waiting = ccache.waiting;
|
||
|
CACHE_UNLOCK();
|
||
|
// wake
|
||
|
if (waiting) {
|
||
|
cc_dbg("cc signal\n");
|
||
|
LWP_CondSignal(ccache.cond);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Cache_Close()
|
||
|
{
|
||
|
dbg_printf("cc close\n");
|
||
|
if (!ccache_init) return;
|
||
|
if (ccache.lwp == LWP_THREAD_NULL) return;
|
||
|
CACHE_LOCK();
|
||
|
ccache.stop = true;
|
||
|
CACHE_UNLOCK();
|
||
|
Cache_Wakeup();
|
||
|
// wait for exit and cleanup
|
||
|
LWP_JoinThread(ccache.lwp, NULL);
|
||
|
LWP_MutexDestroy(ccache.mutex);
|
||
|
LWP_CondDestroy(ccache.cond);
|
||
|
hash_close(&ccache.hash);
|
||
|
//heap_close
|
||
|
SAFE_FREE(ccache.data);
|
||
|
memset(&ccache, 0, sizeof(ccache));
|
||
|
ccache.lwp = LWP_THREAD_NULL;
|
||
|
ccache_init = 0;
|
||
|
dbg_printf("cc stop\n");
|
||
|
}
|
||
|
|
||
|
GRRLIB_texImg* cache_request_cover1(int game_index, int style, int format, int priority, int *state)
|
||
|
{
|
||
|
Cover_State *cs = NULL;
|
||
|
GRRLIB_texImg *tx = NULL;
|
||
|
bool need_wake = false;
|
||
|
u8 *id = NULL;
|
||
|
if (game_index < 0 || game_index >= gameCnt) goto error;
|
||
|
if (style > CFG_COVER_STYLE_HQ) goto error;
|
||
|
id = gameList[game_index].id;
|
||
|
//cc_dbg("cc req: %d %.6s %d %d %d\n", game_index, id, style, format, priority);
|
||
|
if (priority) {
|
||
|
cs = cache_get_cover(id, style, format);
|
||
|
if (!cs) goto error;
|
||
|
// cs will already be marked as used by cache_get_cover
|
||
|
} else {
|
||
|
// if priority == CC_PRIO_NONE then
|
||
|
// return the cover only if already loaded
|
||
|
// but don't updte LRU
|
||
|
CACHE_LOCK();
|
||
|
cs = cache_find_cover(id, style, format);
|
||
|
if (cs) cs->used = 1;
|
||
|
CACHE_UNLOCK();
|
||
|
if (!cs) {
|
||
|
if (state) *state = CS_IDLE;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
if (priority > 0) {
|
||
|
CACHE_LOCK();
|
||
|
if (cs->state == CS_IDLE) {
|
||
|
if (cs->request == 0) {
|
||
|
need_wake = true;
|
||
|
//cc_dbg("cc rq: %d %.6s %d %d %d\n", game_index, id, style, format, priority);
|
||
|
}
|
||
|
if (priority > cs->request) {
|
||
|
cs->request = priority;
|
||
|
// mark new request
|
||
|
if (priority > ccache.new_request) {
|
||
|
ccache.new_request = priority;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
cs->lru = 0;
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
if (state) *state = cs->state;
|
||
|
tx = &cs->tx;
|
||
|
if (need_wake) Cache_Wakeup();
|
||
|
//cc_dbg("cc req = %d %p\n", cs->state, tx);
|
||
|
return tx;
|
||
|
error:
|
||
|
// out of cover slots or invalid index
|
||
|
if (state) *state = CS_ERROR;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
void cache_remap_style_fmt(int *cstyle, int *fmt)
|
||
|
{
|
||
|
switch (*cstyle) {
|
||
|
case CFG_COVER_STYLE_HQ_OR_FULL_RGB:
|
||
|
*cstyle = CFG_COVER_STYLE_HQ_OR_FULL;
|
||
|
*fmt = CC_FMT_C4x4;
|
||
|
break;
|
||
|
case CFG_COVER_STYLE_FULL_RGB:
|
||
|
*cstyle = CFG_COVER_STYLE_FULL;
|
||
|
*fmt = CC_FMT_C4x4;
|
||
|
break;
|
||
|
case CFG_COVER_STYLE_HQ_RGB:
|
||
|
*cstyle = CFG_COVER_STYLE_HQ;
|
||
|
*fmt = CC_FMT_C4x4;
|
||
|
break;
|
||
|
case CFG_COVER_STYLE_FULL_CMPR:
|
||
|
*cstyle = CFG_COVER_STYLE_FULL;
|
||
|
*fmt = CC_FMT_CMPR;
|
||
|
break;
|
||
|
case CFG_COVER_STYLE_HQ_CMPR:
|
||
|
*cstyle = CFG_COVER_STYLE_HQ;
|
||
|
*fmt = CC_FMT_CMPR;
|
||
|
break;
|
||
|
case CFG_COVER_STYLE_FULL_MIPMAP:
|
||
|
*cstyle = CFG_COVER_STYLE_FULL;
|
||
|
*fmt = CC_FMT_MIPMAP;
|
||
|
break;
|
||
|
case CFG_COVER_STYLE_HQ_MIPMAP:
|
||
|
*cstyle = CFG_COVER_STYLE_HQ;
|
||
|
*fmt = CC_FMT_MIPMAP;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
GRRLIB_texImg* cache_request_cover(int game_index, int style, int format, int priority, int *state)
|
||
|
{
|
||
|
GRRLIB_texImg *tx;
|
||
|
int my_state;
|
||
|
cache_remap_style_fmt(&style, &format);
|
||
|
if (style == CFG_COVER_STYLE_HQ_OR_FULL) {
|
||
|
// HQ or FULL requested, try HQ first:
|
||
|
tx = cache_request_cover1(game_index, CFG_COVER_STYLE_HQ,
|
||
|
format, priority, &my_state);
|
||
|
if (my_state == CS_PRESENT) {
|
||
|
// HQ present: return it
|
||
|
if (state) *state = my_state;
|
||
|
return tx;
|
||
|
}
|
||
|
if ((my_state == CS_IDLE && priority > 0) || my_state == CS_LOADING) {
|
||
|
// HQ requested or loading: return FULL if present
|
||
|
// but don't add a request for FULL (CC_PRIO_NONE)
|
||
|
return cache_request_cover1(game_index, CFG_COVER_STYLE_FULL,
|
||
|
format, CC_PRIO_NONE, state);
|
||
|
}
|
||
|
// HQ not found: request FULL
|
||
|
style = CFG_COVER_STYLE_FULL;
|
||
|
}
|
||
|
return cache_request_cover1(game_index, style, format, priority, state);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This method is used to set the caching priority of the passed in game index
|
||
|
* as well as the next x number of games.
|
||
|
*
|
||
|
* @param game_i the starting game index
|
||
|
* @param num the number of images to set the priority on
|
||
|
* @param rq_prio the cache priority: 1 = high priority
|
||
|
*/
|
||
|
void cache_request(int game_i, int num, int rq_prio)
|
||
|
{
|
||
|
int i;
|
||
|
//cc_dbg("cc_req(%d %d %d)\n", game_i, num, rq_prio);
|
||
|
for (i = game_i; i < game_i + num; i++) {
|
||
|
cache_request_cover(i, CFG.cover_style, CC_FMT_C4x4, CC_PRIO_HIGH, NULL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* This method is used to set the caching priority of the passed in game index
|
||
|
* as well as the next x number of games before and after the index. All the
|
||
|
* rest of the covers are given a lower priority.
|
||
|
*
|
||
|
* @param game_i the starting game index
|
||
|
* @param num the number of images before and after the game_i to set the priority on
|
||
|
* @param rq_prio the cache priority: 1 = high priority
|
||
|
*/
|
||
|
void cache_request_before_and_after(int game_i, int num, int rq_prio)
|
||
|
{
|
||
|
int i;
|
||
|
int format = CC_FMT_C4x4;
|
||
|
if (CFG.gui_compress_covers) format = CC_COVERFLOW_FMT;
|
||
|
//cc_dbg("cc_rqba(%d %d %d)\n", game_i, num, format);
|
||
|
if (showingFrontCover) {
|
||
|
cache_request_cover(game_i, CFG_COVER_STYLE_FULL, format, CC_PRIO_HIGH, NULL);
|
||
|
} else {
|
||
|
cache_request_cover(game_i, CFG_COVER_STYLE_HQ_OR_FULL, format, CC_PRIO_HIGH, NULL);
|
||
|
}
|
||
|
for (i = game_i - num; i < game_i + num; i++) {
|
||
|
if (i == game_i) continue;
|
||
|
cache_request_cover(i, CFG_COVER_STYLE_FULL, format, CC_PRIO_MED, NULL);
|
||
|
}
|
||
|
num += 5;
|
||
|
for (i = game_i - num; i < game_i + num; i++) {
|
||
|
if (i == game_i) continue;
|
||
|
cache_request_cover(i, CFG_COVER_STYLE_FULL, format, CC_PRIO_LOW, NULL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void cache_release_cover(int game_index, int style, int format)
|
||
|
{
|
||
|
Cover_State *cs = NULL;
|
||
|
u8 *id = gameList[game_index].id;
|
||
|
cs = cache_get_cover(id, style, format);
|
||
|
if (cs) {
|
||
|
CACHE_LOCK();
|
||
|
cs->used = 0;
|
||
|
if (cs->request > 0) cs->request--;
|
||
|
if (cs->lru < LRU_MAX) cs->lru++;
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void cache_release_full(bool full)
|
||
|
{
|
||
|
Cover_State *cs;
|
||
|
int i;
|
||
|
CACHE_LOCK();
|
||
|
for (i=0; i<CACHE_STATE_NUM; i++) {
|
||
|
cs = &ccache.cstate[i];
|
||
|
cs->used = 0;
|
||
|
if (full) {
|
||
|
// completely release
|
||
|
cs->request = 0;
|
||
|
} else {
|
||
|
// decrease priority
|
||
|
if (cs->request > 0) cs->request--;
|
||
|
}
|
||
|
if (cs->lru < LRU_MAX) cs->lru++;
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
|
||
|
void cache_release_all()
|
||
|
{
|
||
|
cache_release_full(false);
|
||
|
}
|
||
|
|
||
|
void Cache_Invalidate()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void cache_wait_idle()
|
||
|
{
|
||
|
cache_release_full(true);
|
||
|
while (!ccache.waiting) {
|
||
|
LWP_YieldThread();
|
||
|
usleep(1000);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Cache_Invalidate_Cover(u8 *id, int style)
|
||
|
{
|
||
|
int i;
|
||
|
bool do_free;
|
||
|
if (!ccache_init) return;
|
||
|
Cover_State *cs;
|
||
|
cache_wait_idle();
|
||
|
CACHE_LOCK();
|
||
|
for (i = 0; i < CACHE_STATE_NUM; i++) {
|
||
|
cs = &ccache.cstate[i];
|
||
|
if (strcmp((char*)cs->key.id, (char*)id) == 0) {
|
||
|
do_free = false;
|
||
|
if (cs->key.style == style) do_free = true;
|
||
|
if (cs->key.style == CFG_COVER_STYLE_FULL
|
||
|
&& (style == CFG_COVER_STYLE_2D || style == CFG_COVER_STYLE_HQ))
|
||
|
{
|
||
|
do_free = true;
|
||
|
}
|
||
|
if (do_free) cache_free_cover(cs);
|
||
|
}
|
||
|
}
|
||
|
if (style == CFG_COVER_STYLE_FULL || style == CFG_COVER_STYLE_HQ) {
|
||
|
cache_check_convert_image(id);
|
||
|
}
|
||
|
CACHE_UNLOCK();
|
||
|
}
|
||
|
|
||
|
void cache_num_game()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void cache_stats(char *str, int size)
|
||
|
{
|
||
|
int i;
|
||
|
int present = 0;
|
||
|
int missing = 0;
|
||
|
int nfree = 0;
|
||
|
int nstyle[5];
|
||
|
int style;
|
||
|
memset(nstyle, 0, sizeof(nstyle));
|
||
|
Cover_State *cs;
|
||
|
for (i = 0; i < CACHE_STATE_NUM; i++) {
|
||
|
cs = &ccache.cstate[i];
|
||
|
if (cs->state == CS_MISSING) missing++;
|
||
|
if (cs->state == CS_PRESENT) {
|
||
|
present++;
|
||
|
style = cs->key.style;
|
||
|
if (style >= 0 && style < 5) {
|
||
|
nstyle[style]++;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
heap_stats hs;
|
||
|
heap_stat(&ccache.heap, &hs);
|
||
|
#define MBf (1024.0 * 1024.0)
|
||
|
nfree = CACHE_STATE_NUM - (present + missing);
|
||
|
snprintf(str, size,
|
||
|
"ccache p: %d m: %d f: %d / %d\n"
|
||
|
"cc mem: s:%.1f u:%.1f f:%.1f [%d,%d]\n"
|
||
|
"cc tm: load: %.3f decode: %.3f\n"
|
||
|
"cc 2d:%d 3d:%d d:%d f:%d hq:%d f2d:%d\n",
|
||
|
present, missing, nfree, CACHE_STATE_NUM,
|
||
|
hs.size / MBf, hs.used / MBf, hs.free / MBf,
|
||
|
ccache.heap.used_list.num,
|
||
|
ccache.heap.free_list.num,
|
||
|
(float)ccache.stat_load / 1000000.0,
|
||
|
(float)ccache.stat_decode / 1000000.0,
|
||
|
nstyle[0], nstyle[1], nstyle[2], nstyle[3], nstyle[4],
|
||
|
ccache.stat_n2d
|
||
|
);
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
//****************************************************************************
|
||
|
//**************************** END CACHE CODE ******************************
|
||
|
//****************************************************************************
|