// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/capture/video/chromeos/camera_metadata_utils.h"
#include <unordered_set>
#include "base/containers/span.h"
#include "base/ranges/algorithm.h"
namespace media {
template <>
const cros::mojom::EntryType entry_type_of<bool>::value =
cros::mojom::EntryType::TYPE_BYTE;
template <>
const cros::mojom::EntryType entry_type_of<uint8_t>::value =
cros::mojom::EntryType::TYPE_BYTE;
template <>
const cros::mojom::EntryType entry_type_of<int32_t>::value =
cros::mojom::EntryType::TYPE_INT32;
template <>
const cros::mojom::EntryType entry_type_of<float>::value =
cros::mojom::EntryType::TYPE_FLOAT;
template <>
const cros::mojom::EntryType entry_type_of<int64_t>::value =
cros::mojom::EntryType::TYPE_INT64;
template <>
const cros::mojom::EntryType entry_type_of<double>::value =
cros::mojom::EntryType::TYPE_DOUBLE;
template <>
const cros::mojom::EntryType entry_type_of<Rational>::value =
cros::mojom::EntryType::TYPE_RATIONAL;
// TODO(shik): support TYPE_RATIONAL
cros::mojom::CameraMetadataEntryPtr* GetMetadataEntry(
const cros::mojom::CameraMetadataPtr& camera_metadata,
cros::mojom::CameraMetadataTag tag) {
if (!camera_metadata || !camera_metadata->entries.has_value()) {
return nullptr;
}
// We assume the metadata entries are sorted.
auto iter = base::ranges::find(camera_metadata->entries.value(), tag,
&cros::mojom::CameraMetadataEntry::tag);
if (iter == camera_metadata->entries.value().end()) {
return nullptr;
}
auto* entry_ptr = &(camera_metadata->entries.value()[(*iter)->index]);
if ((*entry_ptr)->data.empty()) {
// Metadata tag found with no valid data.
LOG(WARNING) << "Found tag " << static_cast<int>(tag)
<< " but with invalid data";
return nullptr;
}
return entry_ptr;
}
void AddOrUpdateMetadataEntry(cros::mojom::CameraMetadataPtr* to,
cros::mojom::CameraMetadataEntryPtr entry) {
auto* e = GetMetadataEntry(*to, entry->tag);
if (e) {
(*to)->data_count += entry->data.size() - (*e)->data.size();
(*to)->data_capacity = std::max((*to)->data_capacity, (*to)->data_count);
(*e)->count = entry->count;
(*e)->data = std::move(entry->data);
} else {
(*to)->entry_count += 1;
(*to)->entry_capacity = std::max((*to)->entry_capacity, (*to)->entry_count);
(*to)->data_count += entry->data.size();
(*to)->data_capacity = std::max((*to)->data_capacity, (*to)->data_count);
if (!(*to)->entries) {
(*to)->entries = std::vector<cros::mojom::CameraMetadataEntryPtr>();
}
entry->index = (*to)->entries->size();
(*to)->entries->push_back(std::move(entry));
SortCameraMetadata(to);
}
}
void SortCameraMetadata(cros::mojom::CameraMetadataPtr* camera_metadata) {
if (!camera_metadata || !(*camera_metadata) ||
!(*camera_metadata)->entries.has_value()) {
return;
}
std::sort((*camera_metadata)->entries.value().begin(),
(*camera_metadata)->entries.value().end(),
[](const cros::mojom::CameraMetadataEntryPtr& a,
const cros::mojom::CameraMetadataEntryPtr& b) {
return a->tag < b->tag;
});
for (size_t i = 0; i < (*camera_metadata)->entries.value().size(); ++i) {
(*camera_metadata)->entries.value()[i]->index = i;
}
}
void MergeMetadata(cros::mojom::CameraMetadataPtr* to,
const cros::mojom::CameraMetadataPtr& from) {
DCHECK(to);
(*to)->entry_count += from->entry_count;
(*to)->entry_capacity += from->entry_count;
(*to)->data_count += from->data_count;
(*to)->data_capacity += from->data_count;
if (!from->entries) {
return;
}
std::unordered_set<cros::mojom::CameraMetadataTag> tags;
if ((*to)->entries) {
for (const auto& entry : (*to)->entries.value()) {
tags.insert(entry->tag);
}
} else {
(*to)->entries = std::vector<cros::mojom::CameraMetadataEntryPtr>();
}
for (const auto& entry : from->entries.value()) {
if (tags.find(entry->tag) != tags.end()) {
(*to)->entry_count -= 1;
LOG(ERROR) << "Found duplicated entries for tag " << entry->tag;
continue;
}
tags.insert(entry->tag);
(*to)->entries->push_back(entry->Clone());
}
SortCameraMetadata(to);
}
} // namespace media