X-Git-Url: https://git.lizzy.rs/?a=blobdiff_plain;f=src%2Fsettings.h;h=80d41fd790f18b5c66a387a577c5ea0e2702f04f;hb=7dcc115b02af411844152ae49c8579d1bd146108;hp=53e8d8ef5c85d43b30fa7c6889bb5b539824f5e2;hpb=3570f3e396acad4a6b5381d06c2dae5cf4e95fbd;p=dragonfireclient.git diff --git a/src/settings.h b/src/settings.h index 53e8d8ef5..80d41fd79 100644 --- a/src/settings.h +++ b/src/settings.h @@ -21,837 +21,213 @@ with this program; if not, write to the Free Software Foundation, Inc., #define SETTINGS_HEADER #include "irrlichttypes_bloated.h" -#include "exceptions.h" -#include -#include "jthread/jmutex.h" -#include "jthread/jmutexautolock.h" -#include "strfnd.h" -#include -#include -#include -#include "debug.h" -#include "log.h" #include "util/string.h" -#include "util/serialize.h" -#include +#include "threading/mutex.h" +#include #include +#include #include -#include "filesys.h" -enum ValueType -{ +class Settings; +struct NoiseParams; + +// Global objects +extern Settings *g_settings; +extern std::string g_settings_path; + +/** function type to register a changed callback */ +typedef void (*setting_changed_callback)(const std::string &name, void *data); + +enum ValueType { VALUETYPE_STRING, VALUETYPE_FLAG // Doesn't take any arguments }; -struct ValueSpec -{ +enum SettingsParseEvent { + SPE_NONE, + SPE_INVALID, + SPE_COMMENT, + SPE_KVPAIR, + SPE_END, + SPE_GROUP, + SPE_MULTILINE, +}; + +struct ValueSpec { ValueSpec(ValueType a_type, const char *a_help=NULL) { type = a_type; help = a_help; } + ValueType type; const char *help; }; -class Settings -{ -public: - Settings() +struct SettingsEntry { + SettingsEntry() { + group = NULL; + is_group = false; } - void writeLines(std::ostream &os) + SettingsEntry(const std::string &value_) { - JMutexAutoLock lock(m_mutex); - - for(std::map::iterator - i = m_settings.begin(); - i != m_settings.end(); ++i) - { - std::string name = i->first; - std::string value = i->second; - os< getNames(){ - std::vector names; - for(std::map::iterator - i = m_settings.begin(); - i != m_settings.end(); ++i) - { - names.push_back(i->first); - } - return names; + value = value_; + group = NULL; + is_group = false; } - // remove a setting - bool remove(const std::string& name) + SettingsEntry(Settings *group_) { - return m_settings.erase(name); + group = group_; + is_group = true; } + std::string value; + Settings *group; + bool is_group; +}; - bool parseConfigLine(const std::string &line) - { - JMutexAutoLock lock(m_mutex); - - std::string trimmedline = trim(line); - - // Ignore empty lines and comments - if(trimmedline.size() == 0 || trimmedline[0] == '#') - return true; - - //infostream<<"trimmedline=\""< &dst, - std::set &updated, - bool &value_changed) - { - JMutexAutoLock lock(m_mutex); - - if(is.eof()) - return false; - - // NOTE: This function will be expanded to allow multi-line settings - std::string line; - std::getline(is, line); - - std::string trimmedline = trim(line); - - std::string line_end = ""; - if(is.eof() == false) - line_end = "\n"; - - // Ignore empty lines and comments - if(trimmedline.size() == 0 || trimmedline[0] == '#') - { - dst.push_back(line+line_end); - return true; - } - - Strfnd sf(trim(line)); - - std::string name = sf.next("="); - name = trim(name); - - if(name == "") - { - dst.push_back(line+line_end); - return true; - } - - std::string value = sf.next("\n"); - value = trim(value); - - if(m_settings.find(name) != m_settings.end()) - { - std::string newvalue = m_settings[name]; - - if(newvalue != value) - { - infostream<<"Changing value of \""< \""< objects; - std::set updated; - bool something_actually_changed = false; - - // Read and modify stuff - { - std::ifstream is(filename); - if(is.good() == false) - { - infostream<<"updateConfigFile():" - " Error opening configuration file" - " for reading: \"" - <::iterator - i = m_settings.begin(); - i != m_settings.end(); ++i) - { - if(updated.find(i->first) != updated.end()) - continue; - something_actually_changed = true; - break; - } - } - - // If nothing was actually changed, skip writing the file - if(!something_actually_changed){ - infostream<<"Skipping writing of "<::iterator - i = objects.begin(); - i != objects.end(); ++i) - { - ss<<(*i); - } - - /* - Write stuff that was not already in the file - */ - for(std::map::iterator - i = m_settings.begin(); - i != m_settings.end(); ++i) - { - if(updated.find(i->first) != updated.end()) - continue; - std::string name = i->first; - std::string value = i->second; - infostream<<"Adding \""< &allowed_options) - { - int nonopt_index = 0; - int i=1; - for(;;) - { - if(i >= argc) - break; - std::string argname = argv[i]; - if(argname.substr(0, 2) != "--") - { - // If option doesn't start with -, read it in as nonoptX - if(argname[0] != '-'){ - std::string name = "nonopt"; - name += itos(nonopt_index); - set(name, argname); - nonopt_index++; - i++; - continue; - } - errorstream<<"Invalid command-line parameter \"" - < expected."<::iterator n; - n = allowed_options.find(name); - if(n == allowed_options.end()) - { - errorstream<<"Unknown command-line parameter \"" - <second.type; - - std::string value = ""; - - if(type == VALUETYPE_FLAG) - { - value = "true"; - } - else - { - if(i >= argc) - { - errorstream<<"Invalid command-line parameter \"" - <::iterator n; - n = m_settings.find(name); - if(n == m_settings.end()) - { - n = m_defaults.find(name); - if(n == m_defaults.end()) - { - throw SettingNotFoundException(("Setting [" + name + "] not found ").c_str()); - } - } - - return n->second; - } - - //////////// Get setting - bool getBool(std::string name) - { - return is_yes(get(name)); - } - - bool getFlag(std::string name) - { - try - { - return getBool(name); - } - catch(SettingNotFoundException &e) - { - return false; - } - } - - // Asks if empty - bool getBoolAsk(std::string name, std::string question, bool def) - { - // If it is in settings - if(exists(name)) - return getBool(name); - - std::string s; - char templine[10]; - std::cout<>value; - return value; - } - - u32 getFlagStr(std::string name, FlagDesc *flagdesc, u32 *flagmask) - { - std::string val = get(name); - return (isdigit(val[0])) ? stoi(val) : - readFlagString(val, flagdesc, flagmask); - } - + std::map &allowed_options); + bool parseConfigLines(std::istream &is, const std::string &end = ""); + void writeLines(std::ostream &os, u32 tab_depth=0) const; + + SettingsParseEvent parseConfigObject(const std::string &line, + const std::string &end, std::string &name, std::string &value); + bool updateConfigObject(std::istream &is, std::ostream &os, + const std::string &end, u32 tab_depth=0); + + static bool checkNameValid(const std::string &name); + static bool checkValueValid(const std::string &value); + static std::string sanitizeName(const std::string &name); + static std::string sanitizeValue(const std::string &value); + static std::string getMultiline(std::istream &is, size_t *num_lines=NULL); + static void printEntry(std::ostream &os, const std::string &name, + const SettingsEntry &entry, u32 tab_depth=0); + + /*********** + * Getters * + ***********/ + + const SettingsEntry &getEntry(const std::string &name) const; + Settings *getGroup(const std::string &name) const; + std::string get(const std::string &name) const; + bool getBool(const std::string &name) const; + u16 getU16(const std::string &name) const; + s16 getS16(const std::string &name) const; + s32 getS32(const std::string &name) const; + u64 getU64(const std::string &name) const; + float getFloat(const std::string &name) const; + v2f getV2F(const std::string &name) const; + v3f getV3F(const std::string &name) const; + u32 getFlagStr(const std::string &name, const FlagDesc *flagdesc, + u32 *flagmask) const; // N.B. if getStruct() is used to read a non-POD aggregate type, // the behavior is undefined. - bool getStruct(std::string name, std::string format, void *out, size_t olen) - { - std::string valstr; - - try { - valstr = get(name); - } catch (SettingNotFoundException &e) { - return false; - } - - if (!deSerializeStringToStruct(valstr, format, out, olen)) - return false; - - return true; - } - - //////////// Try to get value, no exception thrown - bool getNoEx(std::string name, std::string &val) - { - try { - val = get(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } + bool getStruct(const std::string &name, const std::string &format, + void *out, size_t olen) const; + bool getNoiseParams(const std::string &name, NoiseParams &np) const; + bool getNoiseParamsFromValue(const std::string &name, NoiseParams &np) const; + bool getNoiseParamsFromGroup(const std::string &name, NoiseParams &np) const; + // return all keys used + std::vector getNames() const; + bool exists(const std::string &name) const; + + + /*************************************** + * Getters that don't throw exceptions * + ***************************************/ + + bool getEntryNoEx(const std::string &name, SettingsEntry &val) const; + bool getGroupNoEx(const std::string &name, Settings *&val) const; + bool getNoEx(const std::string &name, std::string &val) const; + bool getFlag(const std::string &name) const; + bool getU16NoEx(const std::string &name, u16 &val) const; + bool getS16NoEx(const std::string &name, s16 &val) const; + bool getS32NoEx(const std::string &name, s32 &val) const; + bool getU64NoEx(const std::string &name, u64 &val) const; + bool getFloatNoEx(const std::string &name, float &val) const; + bool getV2FNoEx(const std::string &name, v2f &val) const; + bool getV3FNoEx(const std::string &name, v3f &val) const; // N.B. getFlagStrNoEx() does not set val, but merely modifies it. Thus, // val must be initialized before using getFlagStrNoEx(). The intention of // this is to simplify modifying a flags field from a default value. - bool getFlagStrNoEx(std::string name, u32 &val, FlagDesc *flagdesc) - { - try { - u32 flags, flagmask; - - flags = getFlagStr(name, flagdesc, &flagmask); - - val &= ~flagmask; - val |= flags; - - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getFloatNoEx(std::string name, float &val) - { - try { - val = getFloat(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getU16NoEx(std::string name, int &val) - { - try { - val = getU16(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getU16NoEx(std::string name, u16 &val) - { - try { - val = getU16(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getS16NoEx(std::string name, int &val) - { - try { - val = getU16(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getS16NoEx(std::string name, s16 &val) - { - try { - val = getS16(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getS32NoEx(std::string name, s32 &val) - { - try { - val = getS32(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getV3FNoEx(std::string name, v3f &val) - { - try { - val = getV3F(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getV2FNoEx(std::string name, v2f &val) - { - try { - val = getV2F(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - bool getU64NoEx(std::string name, u64 &val) - { - try { - val = getU64(name); - return true; - } catch (SettingNotFoundException &e) { - return false; - } - } - - //////////// Set setting - + bool getFlagStrNoEx(const std::string &name, u32 &val, FlagDesc *flagdesc) const; + + + /*********** + * Setters * + ***********/ + + // N.B. Groups not allocated with new must be set to NULL in the settings + // tree before object destruction. + bool setEntry(const std::string &name, const void *entry, + bool set_group, bool set_default); + bool set(const std::string &name, const std::string &value); + bool setDefault(const std::string &name, const std::string &value); + bool setGroup(const std::string &name, Settings *group); + bool setGroupDefault(const std::string &name, Settings *group); + bool setBool(const std::string &name, bool value); + bool setS16(const std::string &name, s16 value); + bool setU16(const std::string &name, u16 value); + bool setS32(const std::string &name, s32 value); + bool setU64(const std::string &name, u64 value); + bool setFloat(const std::string &name, float value); + bool setV2F(const std::string &name, v2f value); + bool setV3F(const std::string &name, v3f value); + bool setFlagStr(const std::string &name, u32 flags, + const FlagDesc *flagdesc, u32 flagmask); + bool setNoiseParams(const std::string &name, const NoiseParams &np, + bool set_default=false); // N.B. if setStruct() is used to write a non-POD aggregate type, // the behavior is undefined. - bool setStruct(std::string name, std::string format, void *value) - { - std::string structstr; - if (!serializeStructToString(&structstr, format, value)) - return false; - - set(name, structstr); - return true; - } - - void setFlagStr(std::string name, u32 flags, - FlagDesc *flagdesc, u32 flagmask) - { - set(name, writeFlagString(flags, flagdesc, flagmask)); - } - - void setBool(std::string name, bool value) - { - if(value) - set(name, "true"); - else - set(name, "false"); - } - - void setFloat(std::string name, float value) - { - set(name, ftos(value)); - } - - void setV3F(std::string name, v3f value) - { - std::ostringstream os; - os<<"("< m_settings; + std::map m_defaults; - clear(); - (*this) += other; + std::map > > m_callbacks; - return *this; - } + mutable Mutex m_callbackMutex; + mutable Mutex m_mutex; // All methods that access m_settings/m_defaults directly should lock this. -private: - std::map m_settings; - std::map m_defaults; - // All methods that access m_settings/m_defaults directly should lock this. - JMutex m_mutex; }; #endif