Initial commit
This commit is contained in:
394
common/formats/xmlparser.h
Normal file
394
common/formats/xmlparser.h
Normal file
@@ -0,0 +1,394 @@
|
||||
/* ScummVM - Graphic Adventure Engine
|
||||
*
|
||||
* ScummVM is the legal property of its developers, whose names
|
||||
* are too numerous to list here. Please refer to the COPYRIGHT
|
||||
* file distributed with this source distribution.
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef XML_PARSER_H
|
||||
#define XML_PARSER_H
|
||||
|
||||
#include "common/scummsys.h"
|
||||
#include "common/types.h"
|
||||
|
||||
#include "common/fs.h"
|
||||
#include "common/list.h"
|
||||
#include "common/hashmap.h"
|
||||
#include "common/hash-str.h"
|
||||
#include "common/stack.h"
|
||||
#include "common/memorypool.h"
|
||||
|
||||
|
||||
namespace Common {
|
||||
|
||||
/**
|
||||
* @defgroup common_xmlparser XML parser
|
||||
* @ingroup common
|
||||
*
|
||||
* @brief The XML parser allows for parsing XML-like files.
|
||||
*
|
||||
* @{
|
||||
*/
|
||||
|
||||
class SeekableReadStream;
|
||||
|
||||
#define MAX_XML_DEPTH 8
|
||||
|
||||
#define XML_KEY(keyName) {\
|
||||
lay = new CustomXMLKeyLayout;\
|
||||
lay->callback = (&kLocalParserName::parserCallback_##keyName);\
|
||||
layout.top()->children[#keyName] = lay;\
|
||||
layout.push(lay); \
|
||||
_layoutList.push_back(lay);
|
||||
|
||||
#define XML_KEY_RECURSIVE(keyName) {\
|
||||
layout.top()->children[#keyName] = layout.top();\
|
||||
layout.push(layout.top());\
|
||||
}
|
||||
|
||||
#define KEY_END() layout.pop(); }
|
||||
|
||||
#define XML_PROP(propName, req) {\
|
||||
prop.name = #propName; \
|
||||
prop.required = req; \
|
||||
layout.top()->properties.push_back(prop); }
|
||||
|
||||
|
||||
|
||||
#define CUSTOM_XML_PARSER(parserName) \
|
||||
protected: \
|
||||
typedef parserName kLocalParserName; \
|
||||
bool keyCallback(ParserNode *node) override {return node->layout->doCallback(this, node); }\
|
||||
struct CustomXMLKeyLayout : public XMLKeyLayout {\
|
||||
typedef bool (parserName::*ParserCallback)(ParserNode *node);\
|
||||
ParserCallback callback;\
|
||||
bool doCallback(XMLParser *parent, ParserNode *node) {return ((kLocalParserName *)parent->*callback)(node);} };\
|
||||
void buildLayout() override { \
|
||||
Common::Stack<XMLKeyLayout *> layout; \
|
||||
CustomXMLKeyLayout *lay = 0; \
|
||||
XMLKeyLayout::XMLKeyProperty prop; \
|
||||
_XMLkeys = new CustomXMLKeyLayout; \
|
||||
layout.push(_XMLkeys);
|
||||
|
||||
#define PARSER_END() layout.clear(); }
|
||||
|
||||
/**
|
||||
* The base XMLParser class implements generic functionality for parsing
|
||||
* XML-like files.
|
||||
*
|
||||
* In order to use it, it must be inherited with a child class that implements
|
||||
* the XMLParser::keyCallback() function.
|
||||
*
|
||||
* @see XMLParser::keyCallback()
|
||||
*/
|
||||
class XMLParser {
|
||||
public:
|
||||
/**
|
||||
* Parser constructor.
|
||||
*/
|
||||
XMLParser() : _XMLkeys(nullptr), _stream(nullptr), _allowText(false), _char(0) {}
|
||||
|
||||
virtual ~XMLParser();
|
||||
|
||||
/** Active state for the parser */
|
||||
enum ParserState {
|
||||
kParserNeedHeader,
|
||||
kParserNeedKey,
|
||||
kParserNeedKeyName,
|
||||
|
||||
kParserNeedPropertyName,
|
||||
kParserNeedPropertyOperator,
|
||||
kParserNeedPropertyValue,
|
||||
|
||||
kParserError
|
||||
};
|
||||
|
||||
struct XMLKeyLayout;
|
||||
struct ParserNode;
|
||||
|
||||
typedef HashMap<String, XMLParser::XMLKeyLayout*, IgnoreCase_Hash, IgnoreCase_EqualTo> ChildMap;
|
||||
|
||||
/** nested struct representing the layout of the XML file */
|
||||
struct XMLKeyLayout {
|
||||
struct XMLKeyProperty {
|
||||
String name;
|
||||
bool required;
|
||||
};
|
||||
|
||||
List<XMLKeyProperty> properties;
|
||||
ChildMap children;
|
||||
|
||||
virtual bool doCallback(XMLParser *parent, ParserNode *node) = 0;
|
||||
|
||||
virtual ~XMLKeyLayout() {
|
||||
properties.clear();
|
||||
}
|
||||
};
|
||||
|
||||
XMLKeyLayout *_XMLkeys;
|
||||
|
||||
/** Struct representing a parsed node */
|
||||
struct ParserNode {
|
||||
String name;
|
||||
StringMap values;
|
||||
bool ignore;
|
||||
bool header;
|
||||
int depth;
|
||||
XMLKeyLayout *layout;
|
||||
};
|
||||
|
||||
ObjectPool<ParserNode, MAX_XML_DEPTH> _nodePool;
|
||||
|
||||
ParserNode *allocNode() {
|
||||
return new (_nodePool) ParserNode;
|
||||
}
|
||||
|
||||
void freeNode(ParserNode *node) {
|
||||
_nodePool.deleteChunk(node);
|
||||
}
|
||||
|
||||
/**
|
||||
* Loads a file into the parser.
|
||||
* Used for the loading of Theme Description files
|
||||
* straight from the filesystem.
|
||||
*
|
||||
* @param filename Name of the file to load.
|
||||
*/
|
||||
bool loadFile(const Path &filename);
|
||||
|
||||
bool loadFile(const FSNode &node);
|
||||
|
||||
/**
|
||||
* Loads a memory buffer into the parser.
|
||||
* Used for loading the default theme fallback directly
|
||||
* from memory if no themes can be found.
|
||||
*
|
||||
* @param buffer Pointer to the buffer.
|
||||
* @param size Size of the buffer
|
||||
* @param disposable Sets if the XMLParser owns the buffer,
|
||||
* i.e. if it can be freed safely after it's
|
||||
* no longer needed by the parser.
|
||||
*/
|
||||
bool loadBuffer(const byte *buffer, uint32 size, DisposeAfterUse::Flag disposable = DisposeAfterUse::NO);
|
||||
|
||||
bool loadStream(SeekableReadStream *stream, const String &name = "File Stream");
|
||||
|
||||
void close();
|
||||
|
||||
/**
|
||||
* The actual parsing function.
|
||||
* Parses the loaded data stream, returns true if successful.
|
||||
*/
|
||||
bool parse();
|
||||
|
||||
/**
|
||||
* Returns the active node being parsed (the one on top of
|
||||
* the node stack).
|
||||
*/
|
||||
ParserNode *getActiveNode() {
|
||||
if (!_activeKey.empty())
|
||||
return _activeKey.top();
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the parent of a given node in the stack.
|
||||
*/
|
||||
ParserNode *getParentNode(ParserNode *child) {
|
||||
return child->depth > 0 ? _activeKey[child->depth - 1] : 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Allow text nodes (eg <tag>this is a text node</tag>) to appear in the
|
||||
* document.
|
||||
*
|
||||
* By default this parser does not allow text nodes and expects all data
|
||||
* to appear in attributes.
|
||||
*/
|
||||
void setAllowText() {
|
||||
_allowText = true;
|
||||
}
|
||||
|
||||
protected:
|
||||
|
||||
/**
|
||||
* The buildLayout function builds the layout for the parser to use
|
||||
* based on a series of helper macros. This function is automatically
|
||||
* generated by the CUSTOM_XML_PARSER() macro on custom parsers.
|
||||
*
|
||||
* See the documentation regarding XML layouts.
|
||||
*/
|
||||
virtual void buildLayout() = 0;
|
||||
|
||||
/**
|
||||
* The keycallback function is automatically overloaded on custom parsers
|
||||
* when using the CUSTOM_XML_PARSER() macro.
|
||||
*
|
||||
* Its job is to call the corresponding Callback function for the given node.
|
||||
* A function for each key type must be declared separately. See the custom
|
||||
* parser creation instructions.
|
||||
*
|
||||
* When parsing a key in such function, one may chose to skip it, e.g. because it's not needed
|
||||
* on the current configuration. In order to ignore a key, you must set
|
||||
* the "ignore" field of its KeyNode struct to "true": The key and all its children
|
||||
* will then be automatically ignored by the parser.
|
||||
*
|
||||
* The callback function must return true if the key was properly handled (this includes the case when the
|
||||
* key is being ignored). False otherwise. The return of keyCallback() is the same as
|
||||
* the callback function's.
|
||||
* See the sample implementation in GUI::ThemeParser.
|
||||
*/
|
||||
virtual bool keyCallback(ParserNode *node) = 0;
|
||||
|
||||
/**
|
||||
* The closed key callback function MAY be overloaded by inheriting classes to
|
||||
* implement parser-specific functions.
|
||||
*
|
||||
* The closedKeyCallback is issued once a key has been finished parsing, to let
|
||||
* the parser verify that all the required subkeys, etc, were included.
|
||||
*
|
||||
* Unlike the keyCallbacks(), there's just a closedKeyCallback() for all keys.
|
||||
* Use "node->name" to distinguish between each key type.
|
||||
*
|
||||
* Returns true if the key was properly closed, false otherwise.
|
||||
* By default, all keys are properly closed.
|
||||
*/
|
||||
virtual bool closedKeyCallback(ParserNode *node) {
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when a text node is found. This will only be called if
|
||||
* setAllowText() has been called, otherwise text nodes are considered
|
||||
* parse errors.
|
||||
*/
|
||||
virtual bool textCallback(const String &val) {
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when a node is closed. Manages its cleanup and calls the
|
||||
* closing callback function if needed.
|
||||
*/
|
||||
bool closeKey();
|
||||
|
||||
/**
|
||||
* Parses the value of a given key. There's no reason to overload this.
|
||||
*/
|
||||
bool parseKeyValue(String keyName);
|
||||
|
||||
/**
|
||||
* Called once a key has been parsed. It handles the closing/cleanup of the
|
||||
* node stack and calls the keyCallback.
|
||||
*/
|
||||
bool parseActiveKey(bool closed);
|
||||
|
||||
/**
|
||||
* Prints an error message when parsing fails and stops the parser.
|
||||
* Parser error always returns "false" so we can pass the return value
|
||||
* directly and break down the parsing.
|
||||
*/
|
||||
bool parserError(const String &errStr);
|
||||
|
||||
/**
|
||||
* Skips spaces/whitelines etc.
|
||||
* @return true if any spaces were skipped.
|
||||
*/
|
||||
bool skipSpaces();
|
||||
|
||||
/**
|
||||
* Skips comment blocks and comment lines.
|
||||
* @return true if any comments were skipped.
|
||||
*/
|
||||
bool skipComments();
|
||||
|
||||
/**
|
||||
* Check if a given character can be part of a KEY or VALUE name.
|
||||
* Overload this if you want to support keys with strange characters
|
||||
* in their name.
|
||||
*/
|
||||
virtual inline bool isValidNameChar(char c) {
|
||||
return isAlnum(c) || c == '_';
|
||||
}
|
||||
|
||||
/**
|
||||
* Parses a the first textual token found.
|
||||
*/
|
||||
bool parseToken();
|
||||
|
||||
/**
|
||||
* Parses the values inside an integer key.
|
||||
* The count parameter specifies the number of values inside
|
||||
* the key, which are expected to be separated with commas.
|
||||
*
|
||||
* Sample usage:
|
||||
* parseIntegerKey("255, 255, 255", 3, &red, &green, &blue);
|
||||
* [will parse each field into its own integer]
|
||||
*
|
||||
* parseIntegerKey("1234", 1, &number);
|
||||
* [will parse the single number into the variable]
|
||||
*
|
||||
* @param key String containing the integers to be parsed.
|
||||
* @param count Number of comma-separated ints in the string.
|
||||
* @param ... Integer variables to store the parsed ints, passed
|
||||
* by reference.
|
||||
* @returns True if the parsing succeeded.
|
||||
*/
|
||||
bool parseIntegerKey(const char *key, int count, ...);
|
||||
bool parseIntegerKey(const String &keyStr, int count, ...);
|
||||
bool vparseIntegerKey(const char *key, int count, va_list args);
|
||||
|
||||
bool parseXMLHeader(ParserNode *node);
|
||||
|
||||
/**
|
||||
* Overload if your parser needs to support parsing the same file
|
||||
* several times, so you can clean up the internal state of the
|
||||
* parser before each parse.
|
||||
*/
|
||||
virtual void cleanup() {}
|
||||
|
||||
/**
|
||||
* Overload if your parser wants to be notified of keys which haven't
|
||||
* been explicitly declared.
|
||||
*
|
||||
* The functions should return true if the key was handled and parsing should
|
||||
* continue, or false (default) to raise a parsing error.
|
||||
*/
|
||||
virtual bool handleUnknownKey(ParserNode *node) { return false; }
|
||||
|
||||
List<XMLKeyLayout *> _layoutList;
|
||||
|
||||
private:
|
||||
char _char;
|
||||
bool _allowText; /** Allow text nodes in the doc (default false) */
|
||||
SeekableReadStream *_stream;
|
||||
Path _fileName;
|
||||
|
||||
ParserState _state; /** Internal state of the parser */
|
||||
|
||||
String _error; /** Current error message */
|
||||
String _token; /** Current text token */
|
||||
|
||||
Stack<ParserNode *> _activeKey; /** Node stack of the parsed keys */
|
||||
};
|
||||
|
||||
/** @} */
|
||||
|
||||
} // End of namespace Common
|
||||
|
||||
#endif
|
||||
Reference in New Issue
Block a user