mirror of
https://github.com/genodelabs/genode.git
synced 2025-01-03 20:44:11 +00:00
b5bd6e0114
Fix #3161
461 lines
11 KiB
C++
461 lines
11 KiB
C++
/*
|
|
* \brief Utility for generating XML
|
|
* \author Norman Feske
|
|
* \date 2014-01-07
|
|
*/
|
|
|
|
/*
|
|
* Copyright (C) 2014-2017 Genode Labs GmbH
|
|
*
|
|
* This file is part of the Genode OS framework, which is distributed
|
|
* under the terms of the GNU Affero General Public License version 3.
|
|
*/
|
|
|
|
#ifndef _INCLUDE__UTIL__XML_GENERATOR_H_
|
|
#define _INCLUDE__UTIL__XML_GENERATOR_H_
|
|
|
|
#include <util/string.h>
|
|
#include <util/print_lines.h>
|
|
#include <base/snprintf.h>
|
|
|
|
namespace Genode { class Xml_generator; }
|
|
|
|
|
|
class Genode::Xml_generator
|
|
{
|
|
public:
|
|
|
|
/**
|
|
* Exception type
|
|
*/
|
|
class Buffer_exceeded { };
|
|
|
|
private:
|
|
|
|
/**
|
|
* Buffer descriptor where the XML output goes to
|
|
*
|
|
* All 'append' methods may throw a 'Buffer_exceeded' exception.
|
|
*/
|
|
class Out_buffer
|
|
{
|
|
private:
|
|
|
|
char *_dst;
|
|
size_t _capacity;
|
|
size_t _used = 0;
|
|
|
|
void _check_advance(size_t const len) const {
|
|
if (_used + len > _capacity)
|
|
throw Buffer_exceeded(); }
|
|
|
|
public:
|
|
|
|
Out_buffer(char *dst, size_t capacity)
|
|
: _dst(dst), _capacity(capacity) { }
|
|
|
|
void advance(size_t const len)
|
|
{
|
|
_check_advance(len);
|
|
_used += len;
|
|
}
|
|
|
|
void undo_append(size_t const len) {
|
|
_used = len < _used ? _used - len : 0; }
|
|
|
|
/**
|
|
* Append character
|
|
*/
|
|
void append(char const c)
|
|
{
|
|
_check_advance(1);
|
|
_dst[_used] = c;
|
|
advance(1);
|
|
}
|
|
|
|
/**
|
|
* Append character 'n' times
|
|
*/
|
|
void append(char const c, size_t n) {
|
|
for (; n--; append(c)); }
|
|
|
|
/**
|
|
* Append character buffer
|
|
*/
|
|
void append(char const *src, size_t len) {
|
|
for (; len--; append(*src++)); }
|
|
|
|
/**
|
|
* Append null-terminated string
|
|
*/
|
|
void append(char const *src) { append(src, strlen(src)); }
|
|
|
|
/**
|
|
* Append character, sanitize it if needed
|
|
*/
|
|
void append_sanitized(char const c)
|
|
{
|
|
switch (c) {
|
|
case 0: append("�"); break;
|
|
case '>': append(">"); break;
|
|
case '<': append("<"); break;
|
|
case '&': append("&"); break;
|
|
case '"': append("""); break;
|
|
case '\'': append("'"); break;
|
|
default: append(c); break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Append character buffer, sanitize characters if needed
|
|
*/
|
|
void append_sanitized(char const *src, size_t len)
|
|
{
|
|
for (; len--; append_sanitized(*src++));
|
|
}
|
|
|
|
/**
|
|
* Return unused part of the buffer
|
|
*/
|
|
Out_buffer remainder() const {
|
|
return Out_buffer(_dst + _used, _capacity - _used); }
|
|
|
|
/**
|
|
* Insert gap into already populated part of the buffer
|
|
*
|
|
* \return empty buffer spanning the gap
|
|
*/
|
|
Out_buffer insert_gap(unsigned const at, size_t const len)
|
|
{
|
|
/* don't allow the insertion into non-populated part */
|
|
if (at > _used)
|
|
return Out_buffer(_dst + at, 0);
|
|
|
|
_check_advance(len);
|
|
memmove(_dst + at + len, _dst + at, _used - at);
|
|
advance(len);
|
|
|
|
return Out_buffer(_dst + at, len);
|
|
}
|
|
|
|
bool has_trailing_newline() const
|
|
{
|
|
return (_used > 1) && (_dst[_used - 1] == '\n');
|
|
}
|
|
|
|
/**
|
|
* Return number of unused bytes of the buffer
|
|
*/
|
|
size_t used() const { return _used; }
|
|
|
|
void discard_trailing_whitespace()
|
|
{
|
|
for (; _used > 0 && is_whitespace(_dst[_used - 1]); _used--);
|
|
}
|
|
};
|
|
|
|
|
|
class Node
|
|
{
|
|
private:
|
|
|
|
/**
|
|
* Indentation level of node
|
|
*/
|
|
unsigned const _indent_level;
|
|
|
|
Node * const _parent_node = 0;
|
|
bool const _parent_was_indented;
|
|
bool const _parent_had_content;
|
|
|
|
Out_buffer _out_buffer;
|
|
|
|
bool _has_content = false;
|
|
bool _is_indented = false;
|
|
bool _has_attributes = false;
|
|
|
|
/**
|
|
* Cursor position of next attribute to insert
|
|
*/
|
|
unsigned _attr_offset = 0;
|
|
|
|
/**
|
|
* Called by sub node
|
|
*
|
|
* \param indented if true, the returned buffer and the
|
|
* end tag start at a fresh line. This is the
|
|
* case if the content buffer contains sub
|
|
* nodes. But when inserting raw content,
|
|
* such additional whitespaces are not desired.
|
|
*/
|
|
Out_buffer _content_buffer(bool indented)
|
|
{
|
|
if (!_has_content)
|
|
_out_buffer.append(">");
|
|
|
|
if (indented)
|
|
_out_buffer.append("\n");
|
|
|
|
_has_content = true;
|
|
_is_indented = indented;
|
|
|
|
return _out_buffer.remainder();
|
|
}
|
|
|
|
void _undo_content_buffer(bool indented,
|
|
bool was_indented,
|
|
bool had_content)
|
|
{
|
|
_is_indented = was_indented;
|
|
_has_content = had_content;
|
|
|
|
if (indented)
|
|
_out_buffer.undo_append(1);
|
|
|
|
if (!_has_content)
|
|
_out_buffer.undo_append(1);
|
|
}
|
|
|
|
/**
|
|
* Called by sub node
|
|
*/
|
|
void _commit_content(Out_buffer content_buffer)
|
|
{
|
|
_out_buffer.advance(content_buffer.used());
|
|
}
|
|
|
|
public:
|
|
|
|
void insert_attribute(char const *name, char const *value)
|
|
{
|
|
/* ' ' + name + '=' + '"' + value + '"' */
|
|
size_t const gap = 1 + strlen(name) + 1 + 1 + strlen(value) + 1;
|
|
|
|
Out_buffer dst = _out_buffer.insert_gap(_attr_offset, gap);
|
|
dst.append(' ');
|
|
dst.append(name);
|
|
dst.append("=\"");
|
|
dst.append(value, strlen(value));
|
|
dst.append("\"");
|
|
|
|
_attr_offset += gap;
|
|
}
|
|
|
|
void append(char const *src, size_t src_len)
|
|
{
|
|
Out_buffer content_buffer = _content_buffer(false);
|
|
content_buffer.append(src, src_len);
|
|
_commit_content(content_buffer);
|
|
}
|
|
|
|
/**
|
|
* Append character, sanitize it if needed
|
|
*/
|
|
void append_sanitized(char const c)
|
|
{
|
|
Out_buffer content_buffer = _content_buffer(false);
|
|
content_buffer.append_sanitized(c);
|
|
_commit_content(content_buffer);
|
|
}
|
|
|
|
void append_sanitized(char const *src, size_t src_len)
|
|
{
|
|
Out_buffer content_buffer = _content_buffer(false);
|
|
content_buffer.append_sanitized(src, src_len);
|
|
_commit_content(content_buffer);
|
|
}
|
|
|
|
template <typename FUNC>
|
|
Node(Xml_generator &xml, char const *name, FUNC const &func)
|
|
:
|
|
_indent_level(xml._curr_indent),
|
|
_parent_node(xml._curr_node),
|
|
_parent_was_indented(_parent_node ? _parent_node->is_indented() : false),
|
|
_parent_had_content (_parent_node ? _parent_node->has_content() : false),
|
|
_out_buffer(_parent_node ? _parent_node->_content_buffer(true)
|
|
: xml._out_buffer)
|
|
{
|
|
_out_buffer.append('\t', _indent_level);
|
|
_out_buffer.append("<");
|
|
_out_buffer.append(name);
|
|
_attr_offset = _out_buffer.used();
|
|
|
|
xml._curr_node = this;
|
|
xml._curr_indent++;
|
|
|
|
try {
|
|
/*
|
|
* Process attributes and sub nodes
|
|
*/
|
|
func();
|
|
} catch (...) {
|
|
/* reset and drop changes by not committing it */
|
|
xml._curr_node = _parent_node;
|
|
xml._curr_indent--;
|
|
if (_parent_node) {
|
|
_parent_node->_undo_content_buffer(true, _parent_was_indented, _parent_had_content); }
|
|
throw;
|
|
}
|
|
|
|
xml._curr_node = _parent_node;
|
|
xml._curr_indent--;
|
|
|
|
if (_is_indented) {
|
|
_out_buffer.append("\n");
|
|
_out_buffer.append('\t', _indent_level);
|
|
}
|
|
|
|
if (_has_content) {
|
|
_out_buffer.append("</");
|
|
_out_buffer.append(name);
|
|
_out_buffer.append(">");
|
|
} else
|
|
_out_buffer.append("/>");
|
|
|
|
if (_parent_node)
|
|
_parent_node->_commit_content(_out_buffer);
|
|
else
|
|
xml._out_buffer = _out_buffer;
|
|
|
|
_out_buffer.append('\0');
|
|
}
|
|
|
|
bool has_content() { return _has_content; }
|
|
bool is_indented() { return _is_indented; }
|
|
};
|
|
|
|
Out_buffer _out_buffer;
|
|
Node *_curr_node = 0;
|
|
unsigned _curr_indent = 0;
|
|
|
|
public:
|
|
|
|
template <typename FUNC>
|
|
Xml_generator(char *dst, size_t dst_len,
|
|
char const *name, FUNC const &func)
|
|
:
|
|
_out_buffer(dst, dst_len)
|
|
{
|
|
if (dst) {
|
|
node(name, func);
|
|
_out_buffer.append('\n');
|
|
_out_buffer.append('\0');
|
|
}
|
|
}
|
|
|
|
template <typename FUNC>
|
|
void node(char const *name, FUNC const &func = [] () { } )
|
|
{
|
|
Node(*this, name, func);
|
|
}
|
|
|
|
void node(char const *name) { Node(*this, name, [] () { }); }
|
|
|
|
void attribute(char const *name, char const *str)
|
|
{
|
|
_curr_node->insert_attribute(name, str);
|
|
}
|
|
|
|
template <size_t N>
|
|
void attribute(char const *name, String<N> const &str)
|
|
{
|
|
_curr_node->insert_attribute(name, str.string());
|
|
}
|
|
|
|
void attribute(char const *name, bool value)
|
|
{
|
|
_curr_node->insert_attribute(name, value ? "true" : "false");
|
|
}
|
|
|
|
void attribute(char const *name, long long value)
|
|
{
|
|
char buf[64];
|
|
Genode::snprintf(buf, sizeof(buf), "%lld", value);
|
|
_curr_node->insert_attribute(name, buf);
|
|
}
|
|
|
|
void attribute(char const *name, long value)
|
|
{
|
|
attribute(name, static_cast<long long>(value));
|
|
}
|
|
|
|
void attribute(char const *name, int value)
|
|
{
|
|
attribute(name, static_cast<long long>(value));
|
|
}
|
|
|
|
void attribute(char const *name, unsigned long long value)
|
|
{
|
|
char buf[64];
|
|
Genode::snprintf(buf, sizeof(buf), "%llu", value);
|
|
_curr_node->insert_attribute(name, buf);
|
|
}
|
|
|
|
void attribute(char const *name, unsigned long value)
|
|
{
|
|
attribute(name, static_cast<unsigned long long>(value));
|
|
}
|
|
|
|
void attribute(char const *name, unsigned value)
|
|
{
|
|
attribute(name, static_cast<unsigned long long>(value));
|
|
}
|
|
|
|
void attribute(char const *name, double value)
|
|
{
|
|
String<64> buf(value);
|
|
_curr_node->insert_attribute(name, buf.string());
|
|
}
|
|
|
|
/**
|
|
* Append content to XML node
|
|
*
|
|
* This method must not be followed by calls of 'attribute'.
|
|
*/
|
|
void append(char const *str, size_t str_len = ~0UL)
|
|
{
|
|
_curr_node->append(str, str_len == ~0UL ? strlen(str) : str_len);
|
|
}
|
|
|
|
/**
|
|
* Append sanitized content to XML node
|
|
*
|
|
* This method must not be followed by calls of 'attribute'.
|
|
*/
|
|
void append_sanitized(char const *str, size_t str_len = ~0UL)
|
|
{
|
|
_curr_node->append_sanitized(str, str_len == ~0UL ? strlen(str) : str_len);
|
|
}
|
|
|
|
/**
|
|
* Append printable objects to XML node as sanitized content
|
|
*
|
|
* This method must not be followed by calls of 'attribute'.
|
|
*/
|
|
template <typename... ARGS>
|
|
void append_content(ARGS &&... args)
|
|
{
|
|
struct Node_output : Genode::Output
|
|
{
|
|
Node &node; Node_output(Node &n) : node(n) { }
|
|
|
|
/******************************
|
|
** Genode::Output interface **
|
|
******************************/
|
|
|
|
void out_char(char c) override {
|
|
node.append_sanitized(c); }
|
|
|
|
void out_string(char const *str, size_t n) override {
|
|
node.append_sanitized(str, n); }
|
|
|
|
} output { *_curr_node };
|
|
|
|
Output::out_args(output, args...);
|
|
}
|
|
|
|
size_t used() const { return _out_buffer.used(); }
|
|
};
|
|
|
|
#endif /* _INCLUDE__UTIL__XML_GENERATOR_H_ */
|