blob: 193427cc1576d9c3fe775221e038334c3ebc9ed3 [file] [log] [blame]
// Copyright 2020 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 <cstddef>
#include <span>
#include <type_traits>
#include "pw_status/status_with_size.h"
namespace pw {
namespace internal {
template <typename T>
struct FunctionTraits;
template <typename T, typename ReturnType, typename... Args>
struct FunctionTraits<ReturnType (T::*)(Args...)> {
using Class = T;
using Return = ReturnType;
};
} // namespace internal
// Writes bytes to an unspecified output. Provides a Write function that takes a
// std::span of bytes and returns a Status.
class Output {
public:
StatusWithSize Write(std::span<const std::byte> data) {
return DoWrite(data);
}
// Convenience wrapper for writing data from a pointer and length.
StatusWithSize Write(const void* data, size_t size_bytes) {
return Write(std::span<const std::byte>(static_cast<const std::byte*>(data),
size_bytes));
}
protected:
~Output() = default;
private:
virtual StatusWithSize DoWrite(std::span<const std::byte> data) = 0;
};
class Input {
public:
StatusWithSize Read(std::span<std::byte> data) { return DoRead(data); }
// Convenience wrapper for reading data from a pointer and length.
StatusWithSize Read(void* data, size_t size_bytes) {
return Read(
std::span<std::byte>(static_cast<std::byte*>(data), size_bytes));
}
protected:
~Input() = default;
private:
virtual StatusWithSize DoRead(std::span<std::byte> data) = 0;
};
// Output adapter that calls a method on a class with a std::span of bytes. If
// the method returns void instead of the expected Status, Write always returns
// OkStatus().
template <typename T, T kMethod>
class OutputToMethod final : public Output {
using Class = typename internal::FunctionTraits<decltype(kMethod)>::Class;
public:
constexpr OutputToMethod(Class* object) : object_(*object) {}
private:
using Return = typename internal::FunctionTraits<decltype(kMethod)>::Return;
template <T kMethodImpl = kMethod>
typename std::enable_if<std::is_void<typename internal::FunctionTraits<
decltype(kMethodImpl)>::Return>::value,
StatusWithSize>::type
DoWriteImpl(std::span<const std::byte> data) {
(object_.*kMethod)(data);
return StatusWithSize(data.size());
}
template <T kMethodImpl = kMethod>
typename std::enable_if<!std::is_void<typename internal::FunctionTraits<
decltype(kMethodImpl)>::Return>::value,
StatusWithSize>::type
DoWriteImpl(std::span<const std::byte> data) {
return (object_.*kMethod)(data);
}
StatusWithSize DoWrite(std::span<const std::byte> data) override {
return DoWriteImpl(data);
}
private:
Class& object_;
};
// Output adapter that calls a free function.
class OutputToFunction final : public Output {
public:
OutputToFunction(StatusWithSize (*function)(std::span<const std::byte>))
: function_(function) {}
private:
StatusWithSize DoWrite(std::span<const std::byte> data) override {
return function_(data);
}
StatusWithSize (*function_)(std::span<const std::byte>);
};
} // namespace pw