blob: e37e8a9262739f74e1733b5d86be1bad92d4ccb1 [file] [log] [blame]
// Copyright 2022 The Pigweed Authors
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy of
// the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations under
// the License.
#pragma once
#include <algorithm>
#include <cstddef>
#include <string_view>
#include "pw_containers/vector.h"
#include "pw_status/status.h"
#include "pw_status/status_with_size.h"
#include "pw_string/string.h"
#include "pw_string/util.h"
namespace pw {
namespace string {
// Copies the source string to the dest, truncating if the full string does not
// fit. dest will not be null terminated, instead the length is reflected in
// the vector size.
//
// Returns the number of characters written. If the string is truncated, the
// status is ResourceExhausted.
//
// Precondition: The destination and source shall not overlap.
// Precondition: The source shall be a valid pointer.
inline StatusWithSize Copy(const std::string_view& source,
pw::Vector<char>& dest) {
if (dest.capacity() == 0) {
return StatusWithSize::ResourceExhausted();
}
dest.resize(source.size());
const size_t copied = source.copy(dest.data(), dest.size());
return StatusWithSize(
copied == source.size() ? OkStatus() : Status::ResourceExhausted(),
copied);
}
inline StatusWithSize Copy(const pw::Vector<char>& source, span<char> dest) {
if (dest.empty()) {
return StatusWithSize::ResourceExhausted();
}
const size_t copied = std::string_view(source.data(), source.size())
.copy(dest.data(), dest.size() - 1);
dest[copied] = '\0';
return StatusWithSize(
copied == source.size() ? OkStatus() : Status::ResourceExhausted(),
copied);
}
inline StatusWithSize Copy(const char* source, pw::Vector<char>& dest) {
PW_DASSERT(source != nullptr);
return Copy(ClampedCString(source, dest.capacity()), dest);
}
// Overload for InlineString to simplify the transition from using Vector<char>
// for protobuf string fields. An external Copy() function is not necessary for
// InlineString; use assign() or the assignment operator instead.
//
// This function will be removed once projects switch to pw::InlineString<>.
inline StatusWithSize Copy(const std::string_view& source,
pw::InlineString<>& dest) {
size_t copied = std::min(source.size(), static_cast<size_t>(dest.capacity()));
dest.assign(source, 0, copied);
return StatusWithSize(
copied == source.size() ? OkStatus() : Status::ResourceExhausted(),
copied);
}
inline StatusWithSize Copy(const char* source, pw::InlineString<>& dest) {
PW_DASSERT(source != nullptr);
// Clamp to capacity + 1 so strings larger than the capacity yield an error.
return Copy(ClampedCString(source, dest.capacity() + 1), dest);
}
} // namespace string
} // namespace pw