2022-09-14 13:35:13 +00:00
|
|
|
/*
|
|
|
|
* \brief Utility for accessing objects by name
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2022-09-14
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2022 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__DICTIONARY_H_
|
|
|
|
#define _INCLUDE__UTIL__DICTIONARY_H_
|
|
|
|
|
|
|
|
#include <util/meta.h>
|
|
|
|
#include <util/string.h>
|
|
|
|
#include <util/avl_tree.h>
|
|
|
|
#include <util/noncopyable.h>
|
|
|
|
#include <base/log.h>
|
|
|
|
|
|
|
|
namespace Genode { template <typename, typename> class Dictionary; }
|
|
|
|
|
|
|
|
|
|
|
|
template <typename T, typename NAME>
|
|
|
|
class Genode::Dictionary : Noncopyable
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
|
|
|
|
Avl_tree<T> _tree { };
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
class Element : private Avl_node<T>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
NAME const name;
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
using This = Dictionary<T, NAME>::Element;
|
|
|
|
|
|
|
|
Dictionary<T, NAME> &_dictionary;
|
|
|
|
|
2023-01-10 15:46:54 +00:00
|
|
|
bool higher(T const *other) const { return other->This::name > name; }
|
2022-09-14 13:35:13 +00:00
|
|
|
|
|
|
|
friend class Avl_tree<T>;
|
|
|
|
friend class Avl_node<T>;
|
|
|
|
friend class Dictionary<T, NAME>;
|
|
|
|
|
|
|
|
static T *_matching_sub_tree(T &curr, NAME const &name)
|
|
|
|
{
|
2023-01-10 15:46:54 +00:00
|
|
|
typename Avl_node<T>::Side side = (name > curr.This::name);
|
2022-09-14 13:35:13 +00:00
|
|
|
return curr.Avl_node<T>::child(side);
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
Element(Dictionary &dictionary, NAME const &name)
|
|
|
|
:
|
|
|
|
name(name), _dictionary(dictionary)
|
|
|
|
{
|
|
|
|
if (_dictionary.exists(name))
|
|
|
|
warning("dictionary entry '", name, "' is not unique");
|
|
|
|
|
|
|
|
_dictionary._tree.insert(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
~Element()
|
|
|
|
{
|
|
|
|
_dictionary._tree.remove(this);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call 'match_fn' with named constant dictionary element
|
|
|
|
*
|
|
|
|
* The 'match_fn' functor is called with a const reference to the
|
|
|
|
* matching dictionary element. If no maching element exists,
|
|
|
|
* 'no_match_fn' is called without argument.
|
|
|
|
*/
|
2024-05-22 14:35:38 +00:00
|
|
|
template <typename FN>
|
|
|
|
auto with_element(NAME const &name, FN const &match_fn, auto const &no_match_fn)
|
|
|
|
-> typename Trait::Functor<decltype(&FN::operator())>::Return_type
|
2022-09-14 13:35:13 +00:00
|
|
|
{
|
|
|
|
T *curr_ptr = _tree.first();
|
|
|
|
for (;;) {
|
|
|
|
if (!curr_ptr)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (curr_ptr->Element::name == name) {
|
|
|
|
return match_fn(*curr_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
curr_ptr = Element::_matching_sub_tree(*curr_ptr, name);
|
|
|
|
}
|
|
|
|
return no_match_fn();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call 'match_fn' with named mutable dictionary element
|
|
|
|
*
|
|
|
|
* The 'match_fn' functor is called with a non-const reference to the
|
|
|
|
* matching dictionary element. If no maching element exists,
|
|
|
|
* 'no_match_fn' is called without argument.
|
|
|
|
*/
|
2024-05-22 14:35:38 +00:00
|
|
|
template <typename FN>
|
|
|
|
auto with_element(NAME const &name, FN const &match_fn, auto const &no_match_fn) const
|
|
|
|
-> typename Trait::Functor<decltype(&FN::operator())>::Return_type
|
2022-09-14 13:35:13 +00:00
|
|
|
{
|
|
|
|
auto const_match_fn = [&] (T const &e) { return match_fn(e); };
|
|
|
|
auto non_const_this = const_cast<Dictionary *>(this);
|
|
|
|
return non_const_this->with_element(name, const_match_fn, no_match_fn);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call 'fn' with a non-const reference to any dictionary element
|
|
|
|
*
|
|
|
|
* \return true if 'fn' was called, or
|
|
|
|
* false if the dictionary is empty
|
|
|
|
*
|
|
|
|
* This method is intended for the orderly destruction of a dictionary.
|
|
|
|
* It allows for the consecutive destruction of all elements.
|
|
|
|
*/
|
2024-05-22 14:35:38 +00:00
|
|
|
bool with_any_element(auto const &fn)
|
2022-09-14 13:35:13 +00:00
|
|
|
{
|
|
|
|
T *curr_ptr = _tree.first();
|
|
|
|
if (!curr_ptr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
fn(*curr_ptr);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-05-22 14:35:38 +00:00
|
|
|
void for_each(auto const &fn) const { _tree.for_each(fn); }
|
2022-09-14 13:35:13 +00:00
|
|
|
|
|
|
|
bool exists(NAME const &name) const
|
|
|
|
{
|
|
|
|
return with_element(name, [] (T const &) { return true; },
|
|
|
|
[] { return false; });
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* _INCLUDE__UTIL__DICTIONARY_H_ */
|