2018-01-23 16:25:53 +00:00
|
|
|
/*
|
|
|
|
* \brief List-based data model created and updated from XML
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2017-08-09
|
|
|
|
*
|
|
|
|
* The 'List_model' stores a component-internal representation of XML-node
|
2023-10-27 14:01:23 +00:00
|
|
|
* content. The internal representation 'ELEM' carries two methods 'matches'
|
|
|
|
* and 'type_matches' that define the relation of the elements to XML nodes.
|
|
|
|
* E.g.,
|
|
|
|
*
|
|
|
|
* ! struct Item : List_model<Item>::Element
|
|
|
|
* ! {
|
|
|
|
* ! static bool type_matches(Xml_node const &);
|
|
|
|
* !
|
|
|
|
* ! bool matches(Xml_node const &) const;
|
|
|
|
* ! ...
|
|
|
|
* ! };
|
|
|
|
*
|
|
|
|
* The class function 'type_matches' returns true if the specified XML node
|
|
|
|
* matches the 'Item' type. It can thereby be used to control the creation
|
|
|
|
* of 'ELEM' nodes by responding to specific XML tags while ignoring unrelated
|
|
|
|
* XML tags.
|
|
|
|
*
|
|
|
|
* The 'matches' method returns true if the concrete element instance matches
|
|
|
|
* the given XML node. It is used to correlate existing 'ELEM' objects with
|
|
|
|
* new versions of XML nodes to update the 'ELEM' objects.
|
|
|
|
*
|
|
|
|
* The functor arguments 'create_fn', 'destroy_fn', and 'update_fn' for the
|
|
|
|
* 'update_from_xml' method define how objects are created, destructed, and
|
|
|
|
* updated. E.g.,
|
|
|
|
*
|
|
|
|
* ! _list_model.update_from_xml(node,
|
|
|
|
* !
|
|
|
|
* ! [&] (Xml_node const &node) -> Item & {
|
|
|
|
* ! return *new (alloc) Item(node); },
|
|
|
|
* !
|
|
|
|
* ! [&] (Item &item) { destroy(alloc, &item); },
|
|
|
|
* !
|
|
|
|
* ! [&] (Item &item, Xml_node const &node) { item.update(node); }
|
|
|
|
* ! );
|
|
|
|
*
|
|
|
|
* The elements are ordered according to the order of XML nodes.
|
|
|
|
*
|
|
|
|
* The list model is a container owning the elements. Before destructing a
|
|
|
|
* list model, its elements must be removed by calling 'update_from_xml'
|
|
|
|
* with an 'Xml_node("<empty/>")' as argument, which results in the call
|
|
|
|
* of 'destroy_fn' for each element.
|
2018-01-23 16:25:53 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2023-10-27 14:01:23 +00:00
|
|
|
* Copyright (C) 2017-2023 Genode Labs GmbH
|
2018-01-23 16:25:53 +00:00
|
|
|
*
|
|
|
|
* 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__LIST_MODEL_H_
|
|
|
|
#define _INCLUDE__UTIL__LIST_MODEL_H_
|
|
|
|
|
2018-02-11 13:46:26 +00:00
|
|
|
/* Genode includes */
|
2023-10-27 14:01:23 +00:00
|
|
|
#include <util/noncopyable.h>
|
2018-01-23 16:25:53 +00:00
|
|
|
#include <util/xml_node.h>
|
2018-02-11 13:46:26 +00:00
|
|
|
#include <util/list.h>
|
2018-12-23 15:37:50 +00:00
|
|
|
#include <base/log.h>
|
2018-01-23 16:25:53 +00:00
|
|
|
|
2023-10-27 14:01:23 +00:00
|
|
|
namespace Genode { template <typename> class List_model; }
|
2018-01-23 16:25:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
template <typename ELEM>
|
2023-10-27 14:01:23 +00:00
|
|
|
class Genode::List_model : Noncopyable
|
2018-01-23 16:25:53 +00:00
|
|
|
{
|
|
|
|
private:
|
|
|
|
|
|
|
|
List<ELEM> _elements { };
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
struct Update_policy;
|
|
|
|
|
|
|
|
class Element : private List<ELEM>::Element
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
|
|
|
|
/* used by 'List_model::update_from_xml' only */
|
|
|
|
friend class List_model;
|
|
|
|
friend class List<ELEM>;
|
|
|
|
|
|
|
|
ELEM *_next() const { return List<ELEM>::Element::next(); }
|
|
|
|
|
2023-10-27 14:01:23 +00:00
|
|
|
/**
|
|
|
|
* Noncopyable
|
|
|
|
*/
|
|
|
|
Element(Element const &) = delete;
|
|
|
|
Element & operator = (Element const &) = delete;
|
|
|
|
|
2018-01-23 16:25:53 +00:00
|
|
|
public:
|
|
|
|
|
2023-10-27 14:01:23 +00:00
|
|
|
Element() { };
|
|
|
|
|
2018-01-23 16:25:53 +00:00
|
|
|
/**
|
|
|
|
* Return the element's neighbor if present, otherwise nullptr
|
|
|
|
*/
|
|
|
|
ELEM const *next() const { return List<ELEM>::Element::next(); }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Unknown_element_type : Exception { };
|
|
|
|
|
|
|
|
~List_model()
|
|
|
|
{
|
|
|
|
if (_elements.first())
|
|
|
|
warning("list model not empty at destruction time");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-27 14:01:23 +00:00
|
|
|
* Update data model according to the given XML node
|
2018-01-23 16:25:53 +00:00
|
|
|
*/
|
2023-10-27 14:01:23 +00:00
|
|
|
inline void update_from_xml(Xml_node const &,
|
|
|
|
auto const &create_fn,
|
|
|
|
auto const &destroy_fn,
|
|
|
|
auto const &update_fn);
|
2018-01-23 16:25:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Call functor 'fn' for each const element
|
|
|
|
*/
|
|
|
|
template <typename FN>
|
|
|
|
void for_each(FN const &fn) const
|
|
|
|
{
|
|
|
|
for (Element const *e = _elements.first(); e; e = e->next())
|
|
|
|
fn(static_cast<ELEM const &>(*e));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call functor 'fn' for each non-const element
|
|
|
|
*/
|
|
|
|
template <typename FN>
|
|
|
|
void for_each(FN const &fn)
|
|
|
|
{
|
|
|
|
Element *next = nullptr;
|
|
|
|
for (Element *e = _elements.first(); e; e = next) {
|
|
|
|
next = e->_next();
|
|
|
|
fn(static_cast<ELEM &>(*e));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-23 15:37:50 +00:00
|
|
|
/**
|
2023-10-28 10:30:40 +00:00
|
|
|
* Call functor 'fn' with the first element of the list model
|
2018-12-23 15:37:50 +00:00
|
|
|
*
|
2023-10-28 10:30:40 +00:00
|
|
|
* Using this method combined with 'Element::next', the list model
|
|
|
|
* can be traversed manually. This is handy in situations where the
|
|
|
|
* list-model elements are visited via recursive function calls
|
2018-12-23 15:37:50 +00:00
|
|
|
* instead of a 'for_each' loop.
|
|
|
|
*/
|
|
|
|
template <typename FN>
|
2023-10-28 10:30:40 +00:00
|
|
|
void with_first(FN const &fn) const
|
2018-12-23 15:37:50 +00:00
|
|
|
{
|
|
|
|
if (Element const *e = _elements.first())
|
|
|
|
fn(static_cast<ELEM const &>(*e));
|
|
|
|
}
|
2018-01-23 16:25:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template <typename ELEM>
|
2023-10-27 14:01:23 +00:00
|
|
|
void Genode::List_model<ELEM>::update_from_xml(Xml_node const &node,
|
|
|
|
auto const &create_fn,
|
|
|
|
auto const &destroy_fn,
|
|
|
|
auto const &update_fn)
|
2018-01-23 16:25:53 +00:00
|
|
|
{
|
2023-10-27 14:01:23 +00:00
|
|
|
List<ELEM> updated_list;
|
2018-01-23 16:25:53 +00:00
|
|
|
|
2023-10-27 14:01:23 +00:00
|
|
|
ELEM *last_updated = nullptr; /* used for appending to 'updated_list' */
|
2018-01-23 16:25:53 +00:00
|
|
|
|
|
|
|
node.for_each_sub_node([&] (Xml_node sub_node) {
|
|
|
|
|
|
|
|
/* skip XML nodes that are unrelated to the data model */
|
2023-10-27 14:01:23 +00:00
|
|
|
if (!ELEM::type_matches(sub_node))
|
2018-01-23 16:25:53 +00:00
|
|
|
return;
|
|
|
|
|
2018-05-09 17:01:43 +00:00
|
|
|
/* check for duplicates, which must not exist in the list model */
|
2023-10-27 14:01:23 +00:00
|
|
|
for (ELEM *dup = updated_list.first(); dup; dup = dup->_next()) {
|
2018-05-09 17:01:43 +00:00
|
|
|
|
|
|
|
/* update existing element with information from later node */
|
2023-10-27 14:01:23 +00:00
|
|
|
if (dup->matches(sub_node)) {
|
|
|
|
update_fn(*dup, sub_node);
|
2018-05-09 17:01:43 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-23 16:25:53 +00:00
|
|
|
/* look up corresponding element in original list */
|
2023-10-27 14:01:23 +00:00
|
|
|
ELEM *curr = _elements.first();
|
|
|
|
while (curr && !curr->matches(sub_node))
|
2018-01-23 16:25:53 +00:00
|
|
|
curr = curr->_next();
|
|
|
|
|
|
|
|
/* consume existing element or create new one */
|
2023-10-27 14:01:23 +00:00
|
|
|
if (curr)
|
2018-01-23 16:25:53 +00:00
|
|
|
_elements.remove(curr);
|
2023-10-27 14:01:23 +00:00
|
|
|
else
|
|
|
|
curr = &create_fn(sub_node);
|
2018-01-23 16:25:53 +00:00
|
|
|
|
|
|
|
/* append current element to 'updated_list' */
|
|
|
|
updated_list.insert(curr, last_updated);
|
|
|
|
last_updated = curr;
|
|
|
|
|
2023-10-27 14:01:23 +00:00
|
|
|
update_fn(*curr, sub_node);
|
2018-01-23 16:25:53 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
/* remove stale elements */
|
2023-10-27 14:01:23 +00:00
|
|
|
ELEM *next = nullptr;
|
|
|
|
for (ELEM *e = _elements.first(); e; e = next) {
|
2018-01-23 16:25:53 +00:00
|
|
|
next = e->_next();
|
2023-10-27 14:01:23 +00:00
|
|
|
destroy_fn(*e);
|
2018-01-23 16:25:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* use 'updated_list' list new data model */
|
|
|
|
_elements = updated_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* _INCLUDE__UTIL__LIST_MODEL_H_ */
|