Binary resource inclusion (since C23)
#embed is a preprocessor directive to include (binary) resources in the build, where a resource is defined as a source of data accessible from the translation environment.
Contents
[edit] Syntax
#embed < h-char-sequence > embed-parameter-sequence (optional) new-line
 (1)
 
#embed " q-char-sequence " embed-parameter-sequence (optional) new-line
 (2)
 
#embed pp-tokens new-line
 (3)
 
__has_embed ( " q-char-sequence " embed-parameter-sequence (optional) )__has_embed ( < h-char-sequence > embed-parameter-sequence (optional) )
 (4)
 
__has_embed ( string-literal pp-balanced-token-sequence (optional) )__has_embed ( < h-pp-tokens > pp-balanced-token-sequence (optional) )
 (5)
 
- the character '
- the character "
- the character \
- the character sequence //
- the character sequence /*
- the character '
- the character \
- the character sequence //
- the character sequence /*
[edit] Explanation
embed in the directive are processed just as in normal text (i.e., each identifier currently defined as a macro name is replaced by its replacement list of preprocessing tokens). The directive resulting after all replacements shall match one of the two previous forms. The method by which a sequence of preprocessing tokens between < and > preprocessing token pair or a pair of " characters is combined into a single header name preprocessing token is implementation-defined.In the case the resource is not found or one of the parameters is not supported by the implementation, the program is ill-formed.
__has_embed can be expanded in the expression of #if and #elif. It is treated as a defined macro by #ifdef, #ifndef, #elifdef, #elifndef and defined but cannot be used anywhere else.
A resource has an implementation resource width which is the implementation-defined size in bits of the located resource. Its resource width is the implementation resource width unless modified by a limit parameter. If the resource width is 0, the resource is considered empty. The embed element width is equal to CHAR_BIT  unless modified by an implementation defined parameter. The resource width must be divisible by the embed element width.
The expansion of a #embed directive is a token sequence formed from the list of integer constant expressions described below. The group of tokens for each integer constant expression in the list is separated in the token sequence from the group of tokens for the previous integer constant expression in the list by a comma. The sequence neither begins nor ends in a comma. If the list of integer constant expressions is empty, the token sequence is empty. The directive is replaced by its expansion and, with the presence of certain embed parameters, additional or replacement token sequences.
The values of the integer constant expressions in the expanded sequence are determined by an implementation-defined mapping of the resource’s data. Each integer constant expression’s value is in the range [0, 2embed element width). If:
- The list of integer constant expressions is used to initialize an array of a type compatible with unsigned char, or compatible with char if char cannot hold negative values, and
- The embed element width is equal to CHAR_BIT ,
then the contents of the initialized elements of the array are as-if the resource’s binary data is fread into the array at translation time.
Implementations are encouraged to take into account translation-time bit and byte orders as well as execution-time bit and byte orders to more appropriately represent the resource’s binary data from the directive. This maximizes the chance that, if the resource referenced at translation time through the #embed directive is the same one accessed through execution-time means, the data that is e.g. fread or similar into contiguous storage will compare bit-for-bit equal to an array of character type initialized from an #embed directive’s expanded contents.
[edit] Parameters
The standard defines the parameters limit, prefix, suffix and if_empty. Any other parameter that appears in the directive must be implementation-defined, or the program is ill-formed. Implementation-defined embed parameters may change the semantics of the directive.
[edit] limit
limit( constant-expression )
 (1)
 
__limit__( constant-expression )
 (2)
 
The limit embed parameter can appear at most once in the embed parameter sequence. It must have an argument, which must be an integer (preprocessor) constant expression that evaluates to a non negative number and does not contain the token defined. The resource width is set to the minimum of the integer constant expression multiplied by the embed element width and the implementation resource width.
[edit] suffix
suffix( pp-balanced-token-sequence (optional) )
 (1)
 
__suffix__( pp-balanced-token-sequence (optional) )
 (2)
 
The suffix embed parameter can appear at most once in the embed parameter sequence. It must have a (possibly empty) preprocessor argument clause. If the resource is non empty, the contents of the parameter clause are placed immediately after the expansion of the directive. Otherwise, it has no effect.
[edit] prefix
prefix( pp-balanced-token-sequence (optional) )
 (1)
 
__prefix__( pp-balanced-token-sequence (optional) )
 (2)
 
The prefix embed parameter can appear at most once in the embed parameter sequence. It must have a (possibly empty) preprocessor argument clause. If the resource is non empty, the contents of the parameter clause are placed immediately before the expansion of the directive. Otherwise, it has no effect.
[edit] if_empty
if_empty( pp-balanced-token-sequence (optional) )
 (1)
 
__if_empty__( pp-balanced-token-sequence (optional) )
 (2)
 
The if_empty embed parameter can appear at most once in the embed parameter sequence. It must have a (possibly empty) preprocessor argument clause. If the resource is empty, the contents of the parameter clause replace the directive. Otherwise, it has no effect.
[edit] Example
#include <stdint.h> #include <stdio.h> const uint8_t image_data[] = { #embed "image.png" }; const char message[] = { #embed "message.txt" if_empty('M', 'i', 's', 's', 'i', 'n', 'g', '\n') ,'0円' // null terminator }; void dump(const uint8_t arr[], size_t size) { for (size_t i = 0; i != size; ++i) printf ("%02X%c", arr[i], (i + 1) % 16 ? ' ' : '\n'); puts (""); } int main() { puts ("image_data[]:"); dump(image_data, sizeof image_data); puts ("message[]:"); dump((const uint8_t *)message, sizeof message); }
Possible output:
image_data[]: 89 50 4E 47 0D 0A 1A 0A 00 00 00 0D 49 48 44 52 00 00 00 01 00 00 00 01 01 03 00 00 00 25 DB 56 ... message[]: 4D 69 73 73 69 6E 67 0A 00
[edit] References
- C23 standard (ISO/IEC 9899:2024):
- 6.4.7 Header names (p: 69)
 
- 6.10.1 Conditional inclusion (p: 165-169)
 
- 6.10.2 Binary resource inclusion (p: 170-177)