flycast/core/cfg/cfg.cpp

204 lines
4.9 KiB
C++
Raw Normal View History

2013-12-19 17:10:14 +00:00
/*
Config file crap
Supports various things, as virtual config entries and such crap
Works surprisingly well considering how old it is ...
2013-12-19 17:10:14 +00:00
*/
#define _CRT_SECURE_NO_DEPRECATE (1)
#include <errno.h>
2013-12-19 17:10:14 +00:00
#include "cfg.h"
#include "ini.h"
2013-12-19 17:10:14 +00:00
static string cfgPath;
static bool save_config = true;
2013-12-19 17:10:14 +00:00
static ConfigFile cfgdb;
static string game_id;
static bool has_game_specific_config = false;
2013-12-19 17:10:14 +00:00
void savecfgf()
{
FILE* cfgfile = fopen(cfgPath.c_str(),"wt");
if (!cfgfile)
2015-08-24 12:46:45 +00:00
{
2013-12-19 17:10:14 +00:00
printf("Error : Unable to open file for saving \n");
2015-08-24 12:46:45 +00:00
}
2013-12-19 17:10:14 +00:00
else
{
cfgdb.save(cfgfile);
2013-12-19 17:10:14 +00:00
fclose(cfgfile);
}
}
void cfgSaveStr(const wchar * Section, const wchar * Key, const wchar * String)
{
const std::string section(Section);
const std::string key(Key);
const std::string value(String);
if (cfgHasGameSpecificConfig())
{
if (cfgdb.get(section, key, "") == value)
// Same value as main config: delete entry
cfgdb.delete_entry(game_id, key);
else
cfgdb.set(game_id, key, value);
}
else
cfgdb.set(section, key, value);
if(save_config)
{
savecfgf();
}
2013-12-19 17:10:14 +00:00
}
//New config code
/*
I want config to be really flexible .. so , here is the new implementation :
2013-12-19 17:10:14 +00:00
Functions :
cfgLoadInt : Load an int , if it does not exist save the default value to it and return it
cfgSaveInt : Save an int
cfgLoadStr : Load a str , if it does not exist save the default value to it and return it
cfgSaveStr : Save a str
cfgExists : Returns true if the Section:Key exists. If Key is null , it retuns true if Section exists
2013-12-19 17:10:14 +00:00
Config parameters can be read from the config file , and can be given at the command line
2013-12-19 17:10:14 +00:00
-cfg section:key=value -> defines a value at command line
If a cfgSave* is made on a value defined by command line , then the command line value is replaced by it
cfg values set by command line are not written to the cfg file , unless a cfgSave* is used
2013-12-19 17:10:14 +00:00
*/
///////////////////////////////
/*
** This will verify there is a working file @ ./szIniFn
** - if not present, it will write defaults
*/
bool cfgOpen()
{
const char* filename = "/emu.cfg";
string config_path_read = get_readonly_config_path(filename);
cfgPath = get_writable_config_path(filename);
FILE* cfgfile = fopen(config_path_read.c_str(),"r");
if(cfgfile != NULL) {
cfgdb.parse(cfgfile);
fclose(cfgfile);
2015-08-24 12:05:51 +00:00
}
else
{
// Config file can't be opened
int error_code = errno;
printf("Warning: Unable to open the config file '%s' for reading (%s)\n", config_path_read.c_str(), strerror(error_code));
if (error_code == ENOENT || cfgPath != config_path_read)
{
// Config file didn't exist
printf("Creating new empty config file at '%s'\n", cfgPath.c_str());
savecfgf();
}
else
{
// There was some other error (may be a permissions problem or something like that)
save_config = false;
}
2013-12-19 17:10:14 +00:00
}
return true;
}
//Implementations of the interface :)
//Section must be set
//If key is 0 , it looks for the section
//0 : not found
//1 : found section , key was 0
//2 : found section & key
s32 cfgExists(const wchar * Section, const wchar * Key)
{
if(cfgdb.has_entry(string(Section), string(Key)))
{
return 2;
}
else
{
return (cfgdb.has_section(string(Section)) ? 1 : 0);
}
2013-12-19 17:10:14 +00:00
}
void cfgLoadStr(const wchar * Section, const wchar * Key, wchar * Return,const wchar* Default)
2013-12-19 17:10:14 +00:00
{
string value = cfgdb.get(Section, Key, Default);
// FIXME: Buffer overflow possible
strcpy(Return, value.c_str());
2013-12-19 17:10:14 +00:00
}
string cfgLoadStr(const wchar * Section, const wchar * Key, const wchar* Default)
2015-03-22 00:16:28 +00:00
{
std::string v = cfgdb.get(string(Section), string(Key), string(Default));
if (cfgHasGameSpecificConfig())
v = cfgdb.get(game_id, string(Key), v);
return v;
2015-03-22 00:16:28 +00:00
}
2013-12-19 17:10:14 +00:00
//These are helpers , mainly :)
void cfgSaveInt(const wchar * Section, const wchar * Key, s32 Int)
2013-12-19 17:10:14 +00:00
{
char str[32];
sprintf(str, "%d", Int);
cfgSaveStr(Section, Key, str);
2013-12-19 17:10:14 +00:00
}
s32 cfgLoadInt(const wchar * Section, const wchar * Key,s32 Default)
2013-12-19 17:10:14 +00:00
{
s32 v = cfgdb.get_int(string(Section), string(Key), Default);
if (cfgHasGameSpecificConfig())
v = cfgdb.get_int(game_id, string(Key), v);
return v;
}
2018-04-11 19:37:31 +00:00
void cfgSaveBool(const wchar * Section, const wchar * Key, bool BoolValue)
2016-03-02 05:48:34 +00:00
{
cfgSaveStr(Section, Key, BoolValue ? "yes" : "no");
2016-03-02 05:48:34 +00:00
}
bool cfgLoadBool(const wchar * Section, const wchar * Key,bool Default)
{
bool v = cfgdb.get_bool(string(Section), string(Key), Default);
if (cfgHasGameSpecificConfig())
v = cfgdb.get_bool(game_id, string(Key), v);
return v;
2016-03-02 05:48:34 +00:00
}
2015-07-17 21:56:51 +00:00
void cfgSetVirtual(const wchar * Section, const wchar * Key, const wchar * String)
2013-12-19 17:10:14 +00:00
{
2015-08-19 02:43:54 +00:00
cfgdb.set(string(Section), string(Key), string(String), true);
}
void cfgSetGameId(const char *id)
{
game_id = id;
}
const char *cfgGetGameId()
{
return game_id.c_str();
}
bool cfgHasGameSpecificConfig()
{
return has_game_specific_config || cfgdb.has_section(game_id);
}
void cfgMakeGameSpecificConfig()
{
has_game_specific_config = true;
}
void cfgDeleteGameSpecificConfig()
{
has_game_specific_config = false;
cfgdb.delete_section(game_id);
}