// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef MOJO_PUBLIC_CPP_BINDINGS_ARRAY_TRAITS_STL_H_
#define MOJO_PUBLIC_CPP_BINDINGS_ARRAY_TRAITS_STL_H_
#include <map>
#include <set>
#include <vector>
#include "mojo/public/cpp/bindings/array_traits.h"
namespace mojo {
template <typename T>
struct ArrayTraits<std::vector<T>> {
using Element = T;
static bool IsNull(const std::vector<T>& input) {
// std::vector<> is always converted to non-null mojom array.
return false;
}
static void SetToNull(std::vector<T>* output) {
// std::vector<> doesn't support null state. Set it to empty instead.
output->clear();
}
static size_t GetSize(const std::vector<T>& input) { return input.size(); }
static T* GetData(std::vector<T>& input) { return input.data(); }
static const T* GetData(const std::vector<T>& input) { return input.data(); }
static typename std::vector<T>::reference GetAt(std::vector<T>& input,
size_t index) {
return input[index];
}
static typename std::vector<T>::const_reference GetAt(
const std::vector<T>& input,
size_t index) {
return input[index];
}
static inline bool Resize(std::vector<T>& input, size_t size) {
// Instead of calling std::vector<T>::resize() directly, this is a hack to
// make compilers happy. Some compilers (e.g., Mac, Android, Linux MSan)
// currently don't allow resizing types like
// std::vector<std::vector<MoveOnlyType>>.
// Because the deserialization code doesn't care about the original contents
// of |input|, we discard them directly.
//
// The "inline" keyword of this method matters. Without it, we have observed
// significant perf regression with some tests on Mac. crbug.com/631415
if (input.size() != size) {
std::vector<T> temp(size);
input.swap(temp);
}
return true;
}
};
// This ArrayTraits specialization is used only for serialization.
template <typename T>
struct ArrayTraits<std::set<T>> {
using Element = T;
using ConstIterator = typename std::set<T>::const_iterator;
static bool IsNull(const std::set<T>& input) {
// std::set<> is always converted to non-null mojom array.
return false;
}
static size_t GetSize(const std::set<T>& input) { return input.size(); }
static ConstIterator GetBegin(const std::set<T>& input) {
return input.begin();
}
static void AdvanceIterator(ConstIterator& iterator) {
++iterator;
}
static const T& GetValue(ConstIterator& iterator) {
return *iterator;
}
};
template <typename K, typename V>
struct MapValuesArrayView {
explicit MapValuesArrayView(const std::map<K, V>& map) : map(map) {}
const std::map<K, V>& map;
};
// Convenience function to create a MapValuesArrayView<> that infers the
// template arguments from its argument type.
template <typename K, typename V>
MapValuesArrayView<K, V> MapValuesToArray(const std::map<K, V>& map) {
return MapValuesArrayView<K, V>(map);
}
// This ArrayTraits specialization is used only for serialization and converts
// a map<K, V> into an array<V>, discarding the keys.
template <typename K, typename V>
struct ArrayTraits<MapValuesArrayView<K, V>> {
using Element = V;
using ConstIterator = typename std::map<K, V>::const_iterator;
static bool IsNull(const MapValuesArrayView<K, V>& input) {
// std::map<> is always converted to non-null mojom array.
return false;
}
static size_t GetSize(const MapValuesArrayView<K, V>& input) {
return input.map.size();
}
static ConstIterator GetBegin(const MapValuesArrayView<K, V>& input) {
return input.map.begin();
}
static void AdvanceIterator(ConstIterator& iterator) { ++iterator; }
static const V& GetValue(ConstIterator& iterator) { return iterator->second; }
};
} // namespace mojo
#endif // MOJO_PUBLIC_CPP_BINDINGS_ARRAY_TRAITS_STL_H_