blob: a45e69a2b9cee3f38d4da0d1af29d0a8a61e9c40 [file] [log] [blame]
/*
* Copyright (c) 2023 Yonatan Schachter
*
* SPDX-License-Identifier: Apache-2.0
*/
#ifndef ZEPHYR_INCLUDE_ZEPHYR_BINDESC_H_
#define ZEPHYR_INCLUDE_ZEPHYR_BINDESC_H_
#include <zephyr/sys/util_macro.h>
#ifdef __cplusplus
extern "C" {
#endif /* __cplusplus */
/*
* Corresponds to the definitions in scripts/west_commands/bindesc.py.
* Do not change without syncing the definitions in both files!
*/
#define BINDESC_MAGIC 0xb9863e5a7ea46046
#define BINDESC_ALIGNMENT 4
#define BINDESC_TYPE_UINT 0x0
#define BINDESC_TYPE_STR 0x1
#define BINDESC_TYPE_BYTES 0x2
#define BINDESC_TYPE_DESCRIPTORS_END 0xf
/**
* @brief Binary Descriptor Definition
* @defgroup bindesc_define Bindesc Define
* @{
*/
/*
* Corresponds to the definitions in scripts/west_commands/bindesc.py.
* Do not change without syncing the definitions in both files!
*/
/** The app version string such as "1.2.3" */
#define BINDESC_ID_APP_VERSION_STRING 0x800
/** The app version major such as 1 */
#define BINDESC_ID_APP_VERSION_MAJOR 0x801
/** The app version minor such as 2 */
#define BINDESC_ID_APP_VERSION_MINOR 0x802
/** The app version patchlevel such as 3 */
#define BINDESC_ID_APP_VERSION_PATCHLEVEL 0x803
/** The app version number such as 0x10203 */
#define BINDESC_ID_APP_VERSION_NUMBER 0x804
/** The kernel version string such as "3.4.0" */
#define BINDESC_ID_KERNEL_VERSION_STRING 0x900
/** The kernel version major such as 3 */
#define BINDESC_ID_KERNEL_VERSION_MAJOR 0x901
/** The kernel version minor such as 4 */
#define BINDESC_ID_KERNEL_VERSION_MINOR 0x902
/** The kernel version patchlevel such as 0 */
#define BINDESC_ID_KERNEL_VERSION_PATCHLEVEL 0x903
/** The kernel version number such as 0x30400 */
#define BINDESC_ID_KERNEL_VERSION_NUMBER 0x904
/** The year the image was compiled in */
#define BINDESC_ID_BUILD_TIME_YEAR 0xa00
/** The month of the year the image was compiled in */
#define BINDESC_ID_BUILD_TIME_MONTH 0xa01
/** The day of the month the image was compiled in */
#define BINDESC_ID_BUILD_TIME_DAY 0xa02
/** The hour of the day the image was compiled in */
#define BINDESC_ID_BUILD_TIME_HOUR 0xa03
/** The minute the image was compiled in */
#define BINDESC_ID_BUILD_TIME_MINUTE 0xa04
/** The second the image was compiled in */
#define BINDESC_ID_BUILD_TIME_SECOND 0xa05
/** The UNIX time (seconds since midnight of 1970/01/01) the image was compiled in */
#define BINDESC_ID_BUILD_TIME_UNIX 0xa06
/** The date and time of compilation such as "2023/02/05 00:07:04" */
#define BINDESC_ID_BUILD_DATE_TIME_STRING 0xa07
/** The date of compilation such as "2023/02/05" */
#define BINDESC_ID_BUILD_DATE_STRING 0xa08
/** The time of compilation such as "00:07:04" */
#define BINDESC_ID_BUILD_TIME_STRING 0xa09
/** The name of the host that compiled the image */
#define BINDESC_ID_HOST_NAME 0xb00
/** The C compiler name */
#define BINDESC_ID_C_COMPILER_NAME 0xb01
/** The C compiler version */
#define BINDESC_ID_C_COMPILER_VERSION 0xb02
/** The C++ compiler name */
#define BINDESC_ID_CXX_COMPILER_NAME 0xb03
/** The C++ compiler version */
#define BINDESC_ID_CXX_COMPILER_VERSION 0xb04
#define BINDESC_TAG_DESCRIPTORS_END BINDESC_TAG(DESCRIPTORS_END, 0x0fff)
/**
* @cond INTERNAL_HIDDEN
*/
/*
* Utility macro to generate a tag from a type and an ID
*
* type - Type of the descriptor, UINT, STR or BYTES
* id - Unique ID for the descriptor, must fit in 12 bits
*/
#define BINDESC_TAG(type, id) ((BINDESC_TYPE_##type & 0xf) << 12 | (id & 0x0fff))
/**
* @endcond
*/
#if !IS_ENABLED(_LINKER)
#include <zephyr/sys/byteorder.h>
/**
* @cond INTERNAL_HIDDEN
*/
/*
* Utility macro to get the name of a bindesc entry
*/
#define BINDESC_NAME(name) bindesc_entry_##name
/* Convenience helper for declaring a binary descriptor entry. */
#define __BINDESC_ENTRY_DEFINE(name) \
__aligned(BINDESC_ALIGNMENT) const struct bindesc_entry BINDESC_NAME(name) \
__in_section(_bindesc_entry, static, name) __used __noasan
/**
* @endcond
*/
/**
* @brief Define a binary descriptor of type string.
*
* @details
* Define a string that is registered in the binary descriptor header.
* The defined string can be accessed using @ref BINDESC_GET_STR
*
* @note The defined string is not static, so its name must not collide with
* any other symbol in the executable.
*
* @param name Name of the descriptor
* @param id Unique ID of the descriptor
* @param value A string value for the descriptor
*/
#define BINDESC_STR_DEFINE(name, id, value) \
__BINDESC_ENTRY_DEFINE(name) = { \
.tag = BINDESC_TAG(STR, id), \
.len = (uint16_t)sizeof(value), \
.data = value, \
}
/**
* @brief Define a binary descriptor of type uint.
*
* @details
* Define an integer that is registered in the binary descriptor header.
* The defined integer can be accessed using @ref BINDESC_GET_UINT
*
* @note The defined integer is not static, so its name must not collide with
* any other symbol in the executable.
*
* @param name Name of the descriptor
* @param id Unique ID of the descriptor
* @param value An integer value for the descriptor
*/
#define BINDESC_UINT_DEFINE(name, id, value) \
__BINDESC_ENTRY_DEFINE(name) = { \
.tag = BINDESC_TAG(UINT, id), \
.len = (uint16_t)sizeof(uint32_t), \
.data = sys_uint32_to_array(value), \
}
/**
* @brief Define a binary descriptor of type bytes.
*
* @details
* Define a uint8_t array that is registered in the binary descriptor header.
* The defined array can be accessed using @ref BINDESC_GET_BYTES.
* The value should be given as an array literal, wrapped in parentheses, for
* example:
*
* BINDESC_BYTES_DEFINE(name, id, ({1, 2, 3, 4}));
*
* @note The defined array is not static, so its name must not collide with
* any other symbol in the executable.
*
* @param name Name of the descriptor
* @param id Unique ID of the descriptor
* @param value A uint8_t array as data for the descriptor
*/
#define BINDESC_BYTES_DEFINE(name, id, value) \
__BINDESC_ENTRY_DEFINE(name) = { \
.tag = BINDESC_TAG(BYTES, id), \
.len = (uint16_t)sizeof((uint8_t [])__DEBRACKET value), \
.data = __DEBRACKET value, \
}
/**
* @brief Get the value of a string binary descriptor
*
* @details
* Get the value of a string binary descriptor, previously defined by
* BINDESC_STR_DEFINE.
*
* @param name Name of the descriptor
*/
#define BINDESC_GET_STR(name) BINDESC_NAME(name).data
/**
* @brief Get the value of a uint binary descriptor
*
* @details
* Get the value of a uint binary descriptor, previously defined by
* BINDESC_UINT_DEFINE.
*
* @param name Name of the descriptor
*/
#define BINDESC_GET_UINT(name) *(uint32_t *)&(BINDESC_NAME(name).data)
/**
* @brief Get the value of a bytes binary descriptor
*
* @details
* Get the value of a string binary descriptor, previously defined by
* BINDESC_BYTES_DEFINE. The returned value can be accessed as an array:
*
* for (size_t i = 0; i < BINDESC_GET_SIZE(name); i++)
* BINDESC_GET_BYTES(name)[i];
*
* @param name Name of the descriptor
*/
#define BINDESC_GET_BYTES(name) BINDESC_NAME(name).data
/**
* @brief Get the size of a binary descriptor
*
* @details
* Get the size of a binary descriptor. This is particularly useful for
* bytes binary descriptors where there's no null terminator.
*
* @param name Name of the descriptor
*/
#define BINDESC_GET_SIZE(name) BINDESC_NAME(name).len
/**
* @}
*/
/*
* An entry of the binary descriptor header. Each descriptor is
* described by one of these entries.
*/
struct bindesc_entry {
/** Tag of the entry */
uint16_t tag;
/** Length of the descriptor data */
uint16_t len;
/** Value of the entry. This is either an integer or a string */
uint8_t data[];
} __packed;
/*
* We're assuming that `struct bindesc_entry` has a specific layout in
* memory, so it's worth making sure that the layout is really what we
* think it is. If these assertions fail for your toolchain/platform,
* please open a bug report.
*/
BUILD_ASSERT(offsetof(struct bindesc_entry, tag) == 0, "Incorrect memory layout");
BUILD_ASSERT(offsetof(struct bindesc_entry, len) == 2, "Incorrect memory layout");
BUILD_ASSERT(offsetof(struct bindesc_entry, data) == 4, "Incorrect memory layout");
#if IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_STRING)
extern const struct bindesc_entry BINDESC_NAME(kernel_version_string);
#endif /* IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_STRING) */
#if IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_MAJOR)
extern const struct bindesc_entry BINDESC_NAME(kernel_version_major);
#endif /* IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_MAJOR) */
#if IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_MINOR)
extern const struct bindesc_entry BINDESC_NAME(kernel_version_minor);
#endif /* IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_MINOR) */
#if IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_PATCHLEVEL)
extern const struct bindesc_entry BINDESC_NAME(kernel_version_patchlevel);
#endif /* IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_PATCHLEVEL) */
#if IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_NUMBER)
extern const struct bindesc_entry BINDESC_NAME(kernel_version_number);
#endif /* IS_ENABLED(CONFIG_BINDESC_KERNEL_VERSION_NUMBER) */
#if IS_ENABLED(CONFIG_BINDESC_APP_VERSION_STRING)
extern const struct bindesc_entry BINDESC_NAME(app_version_string);
#endif /* IS_ENABLED(CONFIG_BINDESC_APP_VERSION_STRING) */
#if IS_ENABLED(CONFIG_BINDESC_APP_VERSION_MAJOR)
extern const struct bindesc_entry BINDESC_NAME(app_version_major);
#endif /* IS_ENABLED(CONFIG_BINDESC_APP_VERSION_MAJOR) */
#if IS_ENABLED(CONFIG_BINDESC_APP_VERSION_MINOR)
extern const struct bindesc_entry BINDESC_NAME(app_version_minor);
#endif /* IS_ENABLED(CONFIG_BINDESC_APP_VERSION_MINOR) */
#if IS_ENABLED(CONFIG_BINDESC_APP_VERSION_PATCHLEVEL)
extern const struct bindesc_entry BINDESC_NAME(app_version_patchlevel);
#endif /* IS_ENABLED(CONFIG_BINDESC_APP_VERSION_PATCHLEVEL) */
#if IS_ENABLED(CONFIG_BINDESC_APP_VERSION_NUMBER)
extern const struct bindesc_entry BINDESC_NAME(app_version_number);
#endif /* IS_ENABLED(CONFIG_BINDESC_APP_VERSION_NUMBER) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_YEAR)
extern const struct bindesc_entry BINDESC_NAME(build_time_year);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_YEAR) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_MONTH)
extern const struct bindesc_entry BINDESC_NAME(build_time_month);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_MONTH) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_DAY)
extern const struct bindesc_entry BINDESC_NAME(build_time_day);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_DAY) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_HOUR)
extern const struct bindesc_entry BINDESC_NAME(build_time_hour);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_HOUR) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_MINUTE)
extern const struct bindesc_entry BINDESC_NAME(build_time_minute);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_MINUTE) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_SECOND)
extern const struct bindesc_entry BINDESC_NAME(build_time_second);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_SECOND) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_UNIX)
extern const struct bindesc_entry BINDESC_NAME(build_time_unix);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_UNIX) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_DATE_TIME_STRING)
extern const struct bindesc_entry BINDESC_NAME(build_date_time_string);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_DATE_TIME_STRING) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_DATE_STRING)
extern const struct bindesc_entry BINDESC_NAME(build_date_string);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_DATE_STRING) */
#if IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_STRING)
extern const struct bindesc_entry BINDESC_NAME(build_time_string);
#endif /* IS_ENABLED(CONFIG_BINDESC_BUILD_TIME_STRING) */
#if IS_ENABLED(CONFIG_BINDESC_HOST_NAME)
extern const struct bindesc_entry BINDESC_NAME(host_name);
#endif /* IS_ENABLED(CONFIG_BINDESC_HOST_NAME) */
#if IS_ENABLED(CONFIG_BINDESC_C_COMPILER_NAME)
extern const struct bindesc_entry BINDESC_NAME(c_compiler_name);
#endif /* IS_ENABLED(CONFIG_BINDESC_C_COMPILER_NAME) */
#if IS_ENABLED(CONFIG_BINDESC_C_COMPILER_VERSION)
extern const struct bindesc_entry BINDESC_NAME(c_compiler_version);
#endif /* IS_ENABLED(CONFIG_BINDESC_C_COMPILER_VERSION) */
#if IS_ENABLED(CONFIG_BINDESC_CXX_COMPILER_NAME)
extern const struct bindesc_entry BINDESC_NAME(cxx_compiler_name);
#endif /* IS_ENABLED(CONFIG_BINDESC_CXX_COMPILER_NAME) */
#if IS_ENABLED(CONFIG_BINDESC_CXX_COMPILER_VERSION)
extern const struct bindesc_entry BINDESC_NAME(cxx_compiler_version);
#endif /* IS_ENABLED(CONFIG_BINDESC_CXX_COMPILER_VERSION) */
#endif /* !IS_ENABLED(_LINKER) */
#ifdef __cplusplus
}
#endif
#endif /* ZEPHYR_INCLUDE_ZEPHYR_BINDESC_H_ */