From 5159241db723c91a05dadb2827c2e63f7b124d56 Mon Sep 17 00:00:00 2001 From: Steve Reinhardt Date: Tue, 21 Oct 2003 21:24:34 -0700 Subject: New ini-file feature: += appends RHS to LHS. Added doxygen comments to inifile.hh. Updated initest. Some other minor cleanup. base/inifile.cc: Add support for '+=' append operation. Factor common code from IniFile::load() and IniFile::add() into new Section::add(). Rename ConfigTable to SectionTable (more descriptive). Fix bug in Section::dump(). base/inifile.hh: Add doxygen comments. Add support for '+=' append operation. Factor common code from IniFile::load() and IniFile::add() into new Section::add(). Rename ConfigTable to SectionTable (more descriptive). test/Makefile: initest needs cprintf.o now. test/foo.ini: Add test of '+=' operator. test/initest.cc: Bring this up-to-date. Steal main loop from main.cc so we can test multiple .ini files and command-line assignments too. --HG-- extra : convert_revision : 982521677fbf464e93aa93798ff7d9611826f17c --- base/inifile.cc | 91 +++++++++++++++++++++++++++------------------- base/inifile.hh | 110 ++++++++++++++++++++++++++++++++++++++++++++++++++++---- 2 files changed, 158 insertions(+), 43 deletions(-) (limited to 'base') diff --git a/base/inifile.cc b/base/inifile.cc index d5436fba8..7fd2f5568 100644 --- a/base/inifile.cc +++ b/base/inifile.cc @@ -59,8 +59,8 @@ IniFile::IniFile() IniFile::~IniFile() { - ConfigTable::iterator i = table.begin(); - ConfigTable::iterator end = table.end(); + SectionTable::iterator i = table.begin(); + SectionTable::iterator end = table.end(); while (i != end) { delete (*i).second; @@ -75,6 +75,16 @@ IniFile::loadCPP(const string &file, vector &cppArgs) { int fd[2]; + // Open the file just to verify that we can. Otherwise if the + // file doesn't exist or has bad permissions the user will get + // confusing errors from cpp/g++. + ifstream tmpf(file.c_str()); + + if (!tmpf.is_open()) + return false; + + tmpf.close(); + #ifdef CPP_PIPE if (pipe(fd) == -1) return false; @@ -183,7 +193,8 @@ IniFile::Entry::getValue() const void IniFile::Section::addEntry(const std::string &entryName, - const std::string &value) + const std::string &value, + bool append) { EntryTable::iterator ei = table.find(entryName); @@ -191,6 +202,10 @@ IniFile::Section::addEntry(const std::string &entryName, // new entry table[entryName] = new Entry(value); } + else if (append) { + // append new reult to old entry + ei->second->appendValue(value); + } else { // override old entry ei->second->setValue(value); @@ -198,6 +213,27 @@ IniFile::Section::addEntry(const std::string &entryName, } +bool +IniFile::Section::add(const std::string &assignment) +{ + string::size_type offset = assignment.find('='); + if (offset == string::npos) // no '=' found + return false; + + // if "+=" rather than just "=" then append value + bool append = (assignment[offset-1] == '+'); + + string entryName = assignment.substr(0, append ? offset-1 : offset); + string value = assignment.substr(offset + 1); + + eat_white(entryName); + eat_white(value); + + addEntry(entryName, value, append); + return true; +} + + IniFile::Entry * IniFile::Section::findEntry(const std::string &entryName) const { @@ -212,10 +248,10 @@ IniFile::Section::findEntry(const std::string &entryName) const IniFile::Section * IniFile::addSection(const string §ionName) { - ConfigTable::iterator ci = table.find(sectionName); + SectionTable::iterator i = table.find(sectionName); - if (ci != table.end()) { - return ci->second; + if (i != table.end()) { + return i->second; } else { // new entry @@ -229,9 +265,9 @@ IniFile::addSection(const string §ionName) IniFile::Section * IniFile::findSection(const string §ionName) const { - ConfigTable::const_iterator ci = table.find(sectionName); + SectionTable::const_iterator i = table.find(sectionName); - return (ci == table.end()) ? NULL : ci->second; + return (i == table.end()) ? NULL : i->second; } @@ -248,21 +284,10 @@ IniFile::add(const string &str) string sectionName = str.substr(0, offset); string rest = str.substr(offset + 1); - offset = rest.find('='); - if (offset == string::npos) // no '='found - return false; - - string entryName = rest.substr(0, offset); - string value = rest.substr(offset + 1); - eat_white(sectionName); - eat_white(entryName); - eat_white(value); - Section *s = addSection(sectionName); - s->addEntry(entryName, value); - return true; + return s->add(rest); } bool @@ -294,14 +319,8 @@ IniFile::load(istream &f) if (section == NULL) continue; - string::size_type offset = line.find('='); - string entryName = line.substr(0, offset); - string value = line.substr(offset + 1); - - eat_white(entryName); - eat_white(value); - - section->addEntry(entryName, value); + if (!section->add(line)) + return false; } return true; @@ -387,10 +406,10 @@ IniFile::printUnreferenced() { bool unref = false; - for (ConfigTable::iterator ci = table.begin(); - ci != table.end(); ++ci) { - const string §ionName = ci->first; - Section *section = ci->second; + for (SectionTable::iterator i = table.begin(); + i != table.end(); ++i) { + const string §ionName = i->first; + Section *section = i->second; if (!section->isReferenced()) { if (section->findEntry("unref_section_ok") == NULL) { @@ -416,15 +435,15 @@ IniFile::Section::dump(const string §ionName) for (EntryTable::iterator ei = table.begin(); ei != table.end(); ++ei) { cout << sectionName << ": " << (*ei).first << " => " - << (*ei).second << "\n"; + << (*ei).second->getValue() << "\n"; } } void IniFile::dump() { - for (ConfigTable::iterator ci = table.begin(); - ci != table.end(); ++ci) { - ci->second->dump(ci->first); + for (SectionTable::iterator i = table.begin(); + i != table.end(); ++i) { + i->second->dump(i->first); } } diff --git a/base/inifile.hh b/base/inifile.hh index 919732e1e..3a82f2d4d 100644 --- a/base/inifile.hh +++ b/base/inifile.hh @@ -36,75 +36,171 @@ #include "base/hashmap.hh" +/// +/// @file +/// Declaration of IniFile object. +/// + +/// +/// This class represents the contents of a ".ini" file. +/// +/// It's basically a two level lookup table: a set of named sections, +/// where each section is a set of key/value pairs. Section names, +/// keys, and values are all uninterpreted strings. +/// class IniFile { protected: + + /// + /// A single key/value pair. + /// class Entry { - std::string value; - mutable bool referenced; + std::string value; ///< The entry value. + mutable bool referenced; ///< Has this entry been used? public: + /// Constructor. Entry(const std::string &v) : value(v), referenced(false) { } + /// Has this entry been used? bool isReferenced() { return referenced; } + /// Fetch the value. const std::string &getValue() const; + /// Set the value. void setValue(const std::string &v) { value = v; } + + /// Append the given string to the value. A space is inserted + /// between the existing value and the new value. Since this + /// operation is typically used with values that are + /// space-separated lists of tokens, this keeps the tokens + /// separate. + void appendValue(const std::string &v) { value += " "; value += v; } }; + /// + /// A section. + /// class Section { + /// EntryTable type. Map of strings to Entry object pointers. typedef m5::hash_map EntryTable; - EntryTable table; - mutable bool referenced; + EntryTable table; ///< Table of entries. + mutable bool referenced; ///< Has this section been used? public: + /// Constructor. Section() : table(), referenced(false) { } + /// Has this section been used? bool isReferenced() { return referenced; } - void addEntry(const std::string &entryName, const std::string &value); + /// Add an entry to the table. If an entry with the same name + /// already exists, the 'append' parameter is checked If true, + /// the new value will be appended to the existing entry. If + /// false, the new value will replace the existing entry. + void addEntry(const std::string &entryName, const std::string &value, + bool append); + + /// Add an entry to the table given a string assigment. + /// Assignment should be of the form "param=value" or + /// "param+=value" (for append). This funciton parses the + /// assignment statment and calls addEntry(). + /// @retval True for success, false if parse error. + bool add(const std::string &assignment); + + /// Find the entry with the given name. + /// @retval Pointer to the entry object, or NULL if none. Entry *findEntry(const std::string &entryName) const; + /// Print the unreferenced entries in this section to cerr. + /// Messages can be suppressed using "unref_section_ok" and + /// "unref_entries_ok". + /// @param sectionName Name of this section, for use in output message. + /// @retval True if any entries were printed. bool printUnreferenced(const std::string §ionName); + + /// Print the contents of this section to cout (for debugging). void dump(const std::string §ionName); }; - typedef m5::hash_map ConfigTable; + /// SectionTable type. Map of strings to Section object pointers. + typedef m5::hash_map SectionTable; protected: - ConfigTable table; + /// Hash of section names to Section object pointers. + SectionTable table; + /// Look up section with the given name, creating a new section if + /// not found. + /// @retval Pointer to section object. Section *addSection(const std::string §ionName); + + /// Look up section with the given name. + /// @retval Pointer to section object, or NULL if not found. Section *findSection(const std::string §ionName) const; + /// Load parameter settings from given istream. This is a helper + /// function for load(string) and loadCPP(), which open a file + /// and then pass it here. + /// @retval True if successful, false if errors were encountered. bool load(std::istream &f); public: + /// Constructor. IniFile(); + + /// Destructor. ~IniFile(); + /// Load the specified file, passing it through the C preprocessor. + /// Parameter settings found in the file will be merged with any + /// already defined in this object. + /// @param file The path of the file to load. + /// @param cppFlags Vector of extra flags to pass to cpp. + /// @retval True if successful, false if errors were encountered. bool loadCPP(const std::string &file, std::vector &cppFlags); + + /// Load the specified file. + /// Parameter settings found in the file will be merged with any + /// already defined in this object. + /// @param file The path of the file to load. + /// @retval True if successful, false if errors were encountered. bool load(const std::string &file); + /// Take string of the form "
:=" or + /// "
:+=" and add to database. + /// @retval True if successful, false if parse error. bool add(const std::string &s); + /// Find value corresponding to given section and entry names. + /// Value is returned by reference in 'value' param. + /// @retval True if found, false if not. bool find(const std::string §ion, const std::string &entry, std::string &value) const; + + /// Find value corresponding to given section and entry names, + /// following "default" links to other sections where possible. + /// Value is returned by reference in 'value' param. + /// @retval True if found, false if not. bool findDefault(const std::string §ion, const std::string &entry, std::string &value) const; + /// Print unreferenced entries in object. Iteratively calls + /// printUnreferend() on all the constituent sections. bool printUnreferenced(); + /// Dump contents to cout. For debugging. void dump(); }; -- cgit v1.2.3