2023-03-06 14:42:04 +01:00
|
|
|
// SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
// Copyright CM4all GmbH
|
|
|
|
// author: Max Kellermann <mk@cm4all.com>
|
2018-07-26 17:29:59 +02:00
|
|
|
|
2022-03-14 12:08:06 +01:00
|
|
|
#pragma once
|
2018-07-26 17:29:59 +02:00
|
|
|
|
2022-02-18 15:03:22 +01:00
|
|
|
#include <array>
|
2020-03-13 01:08:53 +01:00
|
|
|
#include <cstdint>
|
2022-06-29 17:22:11 +02:00
|
|
|
#include <span>
|
2018-07-26 17:29:59 +02:00
|
|
|
|
2022-03-14 12:08:06 +01:00
|
|
|
constexpr char hex_digits[] = "0123456789abcdef";
|
|
|
|
|
|
|
|
[[gnu::always_inline]]
|
|
|
|
static constexpr char *
|
|
|
|
HexFormatUint8Fixed(char dest[2], uint8_t number) noexcept
|
|
|
|
{
|
|
|
|
dest[0] = hex_digits[(number >> 4) & 0xf];
|
|
|
|
dest[1] = hex_digits[number & 0xf];
|
|
|
|
return dest + 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
[[gnu::always_inline]]
|
|
|
|
static constexpr char *
|
|
|
|
HexFormatUint16Fixed(char dest[4], uint16_t number) noexcept
|
|
|
|
{
|
|
|
|
dest[0] = hex_digits[(number >> 12) & 0xf];
|
|
|
|
dest[1] = hex_digits[(number >> 8) & 0xf];
|
|
|
|
dest[2] = hex_digits[(number >> 4) & 0xf];
|
|
|
|
dest[3] = hex_digits[number & 0xf];
|
|
|
|
return dest + 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
[[gnu::always_inline]]
|
|
|
|
static constexpr char *
|
|
|
|
HexFormatUint32Fixed(char dest[8], uint32_t number) noexcept
|
|
|
|
{
|
|
|
|
dest[0] = hex_digits[(number >> 28) & 0xf];
|
|
|
|
dest[1] = hex_digits[(number >> 24) & 0xf];
|
|
|
|
dest[2] = hex_digits[(number >> 20) & 0xf];
|
|
|
|
dest[3] = hex_digits[(number >> 16) & 0xf];
|
|
|
|
dest[4] = hex_digits[(number >> 12) & 0xf];
|
|
|
|
dest[5] = hex_digits[(number >> 8) & 0xf];
|
|
|
|
dest[6] = hex_digits[(number >> 4) & 0xf];
|
|
|
|
dest[7] = hex_digits[number & 0xf];
|
|
|
|
return dest + 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
[[gnu::always_inline]]
|
|
|
|
static constexpr char *
|
|
|
|
HexFormatUint64Fixed(char dest[16], uint64_t number) noexcept
|
|
|
|
{
|
|
|
|
dest = HexFormatUint32Fixed(dest, number >> 32);
|
|
|
|
dest = HexFormatUint32Fixed(dest, number);
|
|
|
|
return dest;
|
|
|
|
}
|
|
|
|
|
2018-07-26 17:29:59 +02:00
|
|
|
/**
|
2022-03-14 12:08:06 +01:00
|
|
|
* Format the given input buffer of bytes to hex. The caller ensures
|
|
|
|
* that the output buffer is at least twice as large as the input.
|
|
|
|
* Does not null-terminate the output buffer.
|
2018-07-26 17:29:59 +02:00
|
|
|
*
|
2022-03-14 12:08:06 +01:00
|
|
|
* @return a pointer to one after the last written character
|
2018-07-26 17:29:59 +02:00
|
|
|
*/
|
2022-03-14 12:08:06 +01:00
|
|
|
constexpr char *
|
2022-02-18 15:03:22 +01:00
|
|
|
HexFormat(char *output, std::span<const std::byte> input) noexcept
|
2022-03-14 12:08:06 +01:00
|
|
|
{
|
|
|
|
for (const auto &i : input)
|
2022-02-18 15:03:22 +01:00
|
|
|
output = HexFormatUint8Fixed(output, (uint8_t)i);
|
2022-03-14 12:08:06 +01:00
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
2018-07-26 17:29:59 +02:00
|
|
|
|
|
|
|
/**
|
2022-02-18 15:03:22 +01:00
|
|
|
* Return a std::array<char> (not null-terminated) containing a hex
|
|
|
|
* dump of the given fixed-size input.
|
2018-07-26 17:29:59 +02:00
|
|
|
*/
|
2022-02-18 15:03:22 +01:00
|
|
|
template<std::size_t size>
|
2022-03-14 12:08:06 +01:00
|
|
|
constexpr auto
|
2022-02-18 15:03:22 +01:00
|
|
|
HexFormat(std::span<const std::byte, size> input) noexcept
|
2018-07-26 17:29:59 +02:00
|
|
|
{
|
2022-02-18 15:03:22 +01:00
|
|
|
std::array<char, size * 2> output;
|
|
|
|
HexFormat(output.data(), input);
|
|
|
|
return output;
|
2018-07-26 17:29:59 +02:00
|
|
|
}
|
2023-08-07 12:31:24 +02:00
|
|
|
|
|
|
|
template<std::size_t size>
|
|
|
|
constexpr auto
|
|
|
|
HexFormat(std::span<std::byte, size> input) noexcept
|
|
|
|
{
|
|
|
|
std::span<const std::byte, size> const_input{input};
|
|
|
|
return HexFormat(const_input);
|
|
|
|
}
|