"Привет, κόσμος!" 
 ― anonymous user 
1. Introduction
A new I/O-agnostic text formatting library was introduced in C++20 ([FORMAT]). This paper proposes integrating it with standard I/O facilities via a simple and intuitive API achieving the following goals:
- 
     Usability 
- 
     Unicode support 
- 
     Good performance 
- 
     Small binary footprint 
2. Revision history
Changes since R4:
- 
     Added SG16 Unicode poll results. 
- 
     Added a list of candidate headers formatted output functions can be added to. 
- 
     Moved the non- ostream 
Changes since R3:
- 
     Replaced _isatty ( _fileno ( stream )) GetConsoleMode ( _get_osfhandle ( _fileno ( stream )), ...) 
Changes since R2:
- 
     Added better compatibility with other formatted I/O facilities as another advantage of using stdout 
- 
     Clarified that [P1885] can be used for literal encoding detection per SG16 feedback. 
- 
     Added comparison of Unicode handling in various languages in Appendix A: Unicode tests and a summary in § 7 Unicode per SG16 request. 
- 
     Removed incorrect "exposition-only" in § 7 Unicode. 
- 
     Replaced "both source and literal encodings are UTF-8, which is enabled by the /utf-8compiler flag" with "the literal (execution) encoding is UTF-8, which is enabled by the/execution-charset:utf-8compiler flag" since the source encoding is irrelevant there.
- 
     Rephrased Effects of vprint_unicode 
Changes since R1:
- 
     Added missing println FILE * ostream & 
- 
     Moved the print functions that take ostream & < ostream > < format > < ostream > 
- 
     Clarified why it is useful to provide vprint * 
- 
     Rebased the wording onto the latest working draft, N4861, in particular updating the Throws clauses to match existing wording. 
- 
     Replaced std :: system_error system_error 
- 
     Added paragraph numbers to the wording. 
Changes since R0:
- 
     Clarified that adding wchar_t 
3. SG16 polls (R3)
Forward P2093R3 to LEWG.
SF WF N WA SA 4 2 2 0 1 
Consensus? Yes
4. LEWG polls (R1)
We prefer 
SF WF N WA SA 3 2 6 6 5 
Add a member function on 
Consensus against.SF WF N WA SA 0 2 5 17 3 
Remove 
No consensus for change.SF WF N WA SA 1 10 7 4 4 
We are happy with the design with regards to UTF-8 output.
Unanimous consent.
Attendance: 35
5. Motivating examples
Consider a common task of printing formatted text to 
| C++20 | Proposed | 
|---|---|
|  |  | 
The proposed 
Existing alternatives in C++20:
| Code | Comments | 
|---|---|
|  | Requires even more formatted I/O function calls; message is interleaved with parameters; can result in interleaved output. | 
|  | Only works if is a null-terminated character string. | 
|  | Constructs a temporary string; requires a call to and a separate
    I/O function call, although potentially cheaper than. | 
Another problem is formatting of Unicode text:
If the source and execution encoding is UTF-8 this will produce the expected output on most GNU/Linux and macOS systems. Unfortunately on Windows it is almost guaranteed to produce mojibake despite the fact that the system is fully capable of printing Unicode, for examplestd :: cout << "Привет, κόσμος!" ; 
Приветeven when compiled with, κόσμος! 
/utf-8 using Visual C++
([MSVC-UTF8]). This happens because the terminal assumes code page 437 in this
case independently of the execution encoding. 
   With the proposed paper
will printstd :: ( "Привет, κόσμος!" ); 
"Привет, κόσμος!" >>> ( "Привет, κόσμος!" ) Привет, κόσμος! 
This problem is independent of formatting 
6. API and naming
Many programming languages provide functions for printing text to standard output, often combined with formatting:
| Language | Function(s) | 
|---|---|
| C | [N2176] | 
| C#/.NET | [DOTNET-WRITE] | 
| COBOL | statement [N0147] | 
| Fortran | andstatements [N2162] | 
| Go | [GO-FMT] | 
| Java | ,,[JAVA-PRINT] | 
| JavaScript | [WHATWG-CONSOLE] | 
| Perl | [PERL-PRINTF] | 
| PHP | [PHP-PRINTF] | 
| Python | statement or function [PY-FUNC] | 
| R | [R-PRINT] | 
| Ruby | and[RUBY-PRINT] | 
| Rust | [RUST-PRINT] | 
| Swift | [SWIFT-PRINT] | 
Variations of 
We propose adding a free function called 
- 
     stdout 
- 
     Better compatibilty with other formatted I/O facilities compared to std :: cout std :: streambuf 
- 
     print ostream 
Since 
Another option is to make 
A free function can also be overloaded to takestd :: cout . ( "Hello, {}!" , name ); 
FILE * printf There are multiple approaches to appending a trailing newline:
- 
     Don’t append a newline automatically: printf 
- 
     Append a newline but don’t format arguments: puts fputs 
- 
     Have two formatting functions/macros, one that appends newline and another that doesn’t: print println print ! println ! Printf Println Write WriteLine 
- 
     Let the user choose a terminating string defaulting to " \n " print 
We propose not appending a newline automatically for consistency with 
std :: ( "Hello, {}!" , name ); // doesn’t print a newline std :: ( "Hello, {}! \n " , name ); // prints a newline 
Additionally we can provide a function that appends a newline:
std :: println ( "Hello, {}!" , name ); // prints a newline 
Although 
Another question is which header non-
- 
     < io > 
- 
     < print > 
- 
     < format > 
- 
     < ostream > 
- 
     < utility > 
The current paper proposes 
% echo '#include < ostream > '| clang ++ - E - x c ++ - | wc - l 42491 
It also pulls in a lot of unrelated symbols such as 
7. Unicode
We can prevent mojibake in the Unicode example by detecting if the string literal encoding is UTF-8 and dispatching to a different function that correctly handles Unicode, for example:
where theconstexpr bool is_utf8 () { const unsigned char micro [] = " \u00B5 " ; return sizeof ( micro ) == 3 && micro [ 0 ] == 0xC2 && micro [ 1 ] == 0xB5 ; } template < typename ... Args > void ( string_view fmt , const Args & ... args ) { if ( is_utf8 ()) vprint_unicode ( fmt , make_format_args ( args ...)); else vprint_nonunicode ( fmt , make_format_args ( args ...)); } 
vprint_unicode vprint_nonunicode print In Visual C++ true if the literal (execution) encoding
is UTF-8, which is enabled by the /execution-charset:utf-8 compiler flags or other means, and false otherwise. Literal encoding detection
can be implemented in a more elegant way using [P1885].
This approach has been implemented in the fmt library ([FMT]) and successfully tested on a variety of platforms.
Here’s an example output on Windows:
At the same time interoperability with legacy code is preserved when literal
encoding is not UTF-8. In particular, in case of EBCDIC, Shift JIS or a
non-Unicode Windows code page, 
The following table summarizes the behavior of formatted output facilities in different programming languages:
| Linux | macOS | Windows | ||||
|---|---|---|---|---|---|---|
| Language | Terminal | Redirect | Terminal | Redirect | Terminal | Redirect | 
| C | Correct | UTF-8 | Correct | UTF-8 | Wrong | UTF-8 | 
| Go | Correct | UTF-8 | Correct | UTF-8 | Correct | UTF-8 | 
| Java | Correct | UTF-8* | Correct | UTF-8* | Wrong | CP1251 (lossy) | 
| JavaScript | Correct | UTF-8* | Correct | UTF-8* | Correct | UTF-8* | 
| Python | Correct | UTF-8* | Correct | UTF-8* | Correct | Error | 
| Rust | Correct | UTF-8 | Correct | UTF-8 | Correct | UTF-8 | 
* - the output is transcoded from a different UTF representation.
Correct means that the test message "Привет, κόσμος!" was fully readable in the
terminal output. None of the tested language facilities were able to produce
readable output when piped through the standard 
The current paper proposes following C, Go, JavaScript and Rust and preserve
the original encoding (modulo UTF conversion). The only difference compared to 
- 
     There is a silent data loss for valid Unicode code points when the output is redirected to a file. 
- 
     It is more expensive because of transcoding. 
- 
     It may give an unusable result when piped through standard Windows commands like findstr 
- 
     It transcodes into legacy encodings that are rarely used in practice nowadays. For example, usage of CP1251 dropped from 4.3% to 0.9% in the last 10 years ([ENCODING-TRENDS]). 
The full listings of test programs are given in Appendix A: Unicode tests.
8. Performance
All the performance benefits of 
The following benchmark compares the reference implementation of 
#include <cstdio>#include <iostream>#include <benchmark/benchmark.h>#include <fmt/ostream.h>void printf ( benchmark :: State & s ) { while ( s . KeepRunning ()) std :: printf ( "The answer is %d. \n " , 42 ); } BENCHMARK ( printf ); void ostream ( benchmark :: State & s ) { std :: ios :: sync_with_stdio ( false); while ( s . KeepRunning ()) std :: cout << "The answer is " << 42 << ". \n " ; } BENCHMARK ( ostream ); void ( benchmark :: State & s ) { while ( s . KeepRunning ()) fmt :: ( "The answer is {}. \n " , 42 ); } BENCHMARK ( ); void print_cout ( benchmark :: State & s ) { std :: ios :: sync_with_stdio ( false); while ( s . KeepRunning ()) fmt :: ( std :: cout , "The answer is {}. \n " , 42 ); } BENCHMARK ( print_cout ); void print_cout_sync ( benchmark :: State & s ) { std :: ios :: sync_with_stdio ( true); while ( s . KeepRunning ()) fmt :: ( std :: cout , "The answer is {}. \n " , 42 ); } BENCHMARK ( print_cout_sync ); BENCHMARK_MAIN (); 
The benchmark was compiled with Apple clang version 11.0.0 (clang-1100.0.33.17)
with 
Run on (8 X 2800 MHz CPU s) CPU Caches: L1 Data 32K (x4) L1 Instruction 32K (x4) L2 Unified 262K (x4) L3 Unified 8388K (x1) Load Average: 1.83, 1.88, 1.82 ---------------------------------------------------------- Benchmark Time CPU Iterations ---------------------------------------------------------- printf 87.0 ns 86.9 ns 7834009 ostream 255 ns 255 ns 2746434 print 78.4 ns 78.3 ns 9095989 print_cout 89.4 ns 89.4 ns 7702973 print_cout_sync 91.5 ns 91.4 ns 7903889
Both 
On Windows 10 with Visual C++ 2019 the results are similar althought the
difference between 
Run on (1 X 2808 MHz CPU ) CPU Caches: L1 Data 32K (x1) L1 Instruction 32K (x1) L2 Unified 262K (x1) L3 Unified 8388K (x1) ---------------------------------------------------------- Benchmark Time CPU Iterations ---------------------------------------------------------- printf 835 ns 816 ns 746667 ostream 2410 ns 2400 ns 280000 print 580 ns 572 ns 1120000 print_cout 623 ns 614 ns 1120000 print_cout_sync 615 ns 614 ns 1120000
9. Binary code
We propose minimizing per-call binary code size by applying the type erasure
mechanism from [P0645]. In this approach all the formatting and printing logic
is implemented in a non-variadic function 
void vprint ( string_view fmt , format_args args ); template < class ... Args > inline void ( string_view fmt , const Args & ... args ) { return vprint ( fmt , make_format_args ( args ...)); } 
We provide 
void vlog ( log_level level , string_view fmt , format_args args ) { // Print the log level and use vprint* overloads to format and print the // message. } template < class ... Args > inline void log ( log_level level , string_view fmt , const Args & ... args ) { return vlog ( level , fmt , make_format_args ( args ...)); } 
Here 
Below we compare the reference implementation of -O3 -DNDEBUG -c -std=c++17  and the resulting binaries are disassembled
with objdump -S:
void printf_test ( const char * name ) { printf ( "Hello, %s!" , name ); } 
__Z11printf_testPKc:
       0:       55      pushq   %rbp
       1:       48 89 e5        movq    %rsp, %rbp
       4:       48 89 fe        movq    %rdi, %rsi
       7:       48 8d 3d 08 00 00 00    leaq    8(%rip), %rdi
       e:       31 c0   xorl    %eax, %eax
      10:       5d      popq    %rbp
      11:       e9 00 00 00 00  jmp     0 <__Z11printf_testPKc+0x16>
void ostream_test ( const char * name ) { std :: cout << "Hello, " << name << "!" ; } 
__Z12ostream_testPKc:
       0:       55      pushq   %rbp
       1:       48 89 e5        movq    %rsp, %rbp
       4:       41 56   pushq   %r14
       6:       53      pushq   %rbx
       7:       48 89 fb        movq    %rdi, %rbx
       a:       48 8b 3d 00 00 00 00    movq    (%rip), %rdi
      11:       48 8d 35 6c 03 00 00    leaq    876(%rip), %rsi
      18:       ba 07 00 00 00  movl    $7, %edx
      1d:       e8 00 00 00 00  callq   0 <__Z12ostream_testPKc+0x22>
      22:       49 89 c6        movq    %rax, %r14
      25:       48 89 df        movq    %rbx, %rdi
      28:       e8 00 00 00 00  callq   0 <__Z12ostream_testPKc+0x2d>
      2d:       4c 89 f7        movq    %r14, %rdi
      30:       48 89 de        movq    %rbx, %rsi
      33:       48 89 c2        movq    %rax, %rdx
      36:       e8 00 00 00 00  callq   0 <__Z12ostream_testPKc+0x3b>
      3b:       48 8d 35 4a 03 00 00    leaq    842(%rip), %rsi
      42:       ba 01 00 00 00  movl    $1, %edx
      47:       48 89 c7        movq    %rax, %rdi
      4a:       5b      popq    %rbx
      4b:       41 5e   popq    %r14
      4d:       5d      popq    %rbp
      4e:       e9 00 00 00 00  jmp     0 <__Z12ostream_testPKc+0x53>
      53:       66 2e 0f 1f 84 00 00 00 00 00   nopw    %cs:(%rax,%rax)
      5d:       0f 1f 00        nopl    (%rax)
void print_test ( const char * name ) { ( "Hello, {}!" , name ); } 
__Z10print_testPKc:
       0:	55 	pushq	%rbp
       1:	48 89 e5 	movq	%rsp, %rbp
       4:	48 83 ec 10 	subq	$16, %rsp
       8:	48 89 7d f0 	movq	%rdi, -16(%rbp)
       c:	48 8d 3d 19 00 00 00 	leaq	25(%rip), %rdi
      13:	48 8d 4d f0 	leaq	-16(%rbp), %rcx
      17:	be 0a 00 00 00 	movl	$10, %esi
      1c:	ba 0d 00 00 00 	movl	$13, %edx
      21:	e8 00 00 00 00 	callq	0 <__Z10print_testPKc+0x26>
      26:	48 83 c4 10 	addq	$16, %rsp
      2a:	5d 	popq	%rbp
      2b:	c3 	retq
   The code generated for the 
The following factors contribute to the difference in binary code size between 
- 
     Passing format string as string_view const char * 
- 
     Capturing and passing argument type information. 
- 
     Preparing the array of formatting arguments. 
10. Impact on existing code
The current proposal adds new functions to the headers 
11. Implementation
The proposed 
12. Wording
Add an entry for 
#define __cpp_lib_print 202005L **placeholder** // also in <format> 
Add the header 
< io > namespace std { template < class ... Args > void ( string_view fmt , const Args & ... args ); template < class ... Args > void ( FILE * stream , string_view fmt , const Args & ... args ); template < class ... Args > void println ( string_view fmt , const Args & ... args ); template < class ... Args > void println ( FILE * stream , string_view fmt , const Args & ... args ); void vprint_unicode ( string_view fmt , format_args args ); void vprint_unicode ( FILE * stream , string_view fmt , format_args args ); void vprint_nonunicode ( string_view fmt , format_args args ); void vprint_nonunicode ( FILE * stream , string_view fmt , format_args args ); } 
Modify section "Header 
29.7.? Print functions [print.fun].... template < class charT , class traits , class T > basic_ostream < charT , traits >& operator << ( basic_ostream < charT , traits >&& os , const T & x ); template < class ... Args > void ( ostream & os , string_view fmt , const Args & ... args ); template < class ... Args > void println ( ostream & os , string_view fmt , const Args & ... args ); void vprint_unicode ( ostream & os , string_view fmt , format_args args ); void vprint_nonunicode ( ostream & os , string_view fmt , format_args args ); 
template < class ... Args > void ( string_view fmt , const Args & ... args ); 
26 Effects: Equivalent to:
( stdout , fmt , make_format_args ( args ...)); 
template < class ... Args > void ( FILE * stream , string_view fmt , const Args & ... args ); 
27 Effects: If string literal encoding is UTF-8, equivalent to:
Otherwise, equivalent to:vprint_unicode ( stream , fmt , make_format_args ( args ...)); 
vprint_nonunicode ( stream , fmt , make_format_args ( args ...)); 
template < class ... Args > void println ( string_view fmt , const Args & ... args ); 
28 Effects: Equivalent to:
( "{} \n " , format ( fmt , args ...)); 
template < class ... Args > void println ( FILE * stream , string_view fmt , const Args & ... args ); 
29 Effects: Equivalent to:
( stream , "{} \n " , format ( fmt , args ...)); 
void vprint_unicode ( string_view fmt , format_args args ); 
30 Effects: Equivalent to:
vprint_unicode ( stdout , fmt , args )); 
31 Effects: Letvoid vprint_unicode ( FILE * stream , string_view fmt , format_args args ); 
out  =  vformat ( fmt ,  args ) stream isatty ( fileno ( stream )) GetConsoleMode ( _get_osfhandle ( _fileno ( stream )),  ...) out WriteConsoleW out stream 
     Throws: As specified in [format.err.report]
or 
void vprint_nonunicode ( string_view fmt , format_args args ); 
32 Effects: Equivalent to:
vprint_nonunicode ( stdout , fmt , args )); 
33 Effects: Writes the result ofvoid vprint_nonunicode ( FILE * stream , string_view fmt , format_args args ); 
vformat ( fmt ,  args ) stream 
     Throws: As specified in [format.err.report]
or 
Add subsection "Print [ostream.formatted.print]" to "Formatted output functions [ostream.formatted]":
template < class ... Args > void ( ostream & os , string_view fmt , const Args & ... args ); 
1 Effects: If string literal encoding is UTF-8, equivalent to:
Otherwise, equivalent to:vprint_unicode ( os , fmt , make_format_args ( args ...)); 
vprint_nonunicode ( os , fmt , make_format_args ( args ...)); 
2 Effects: Letvoid vprint_unicode ( ostream & os , string_view fmt , format_args args ); 
out  =  vformat ( os . getloc (),  fmt ,  args ) os basic_filebuf out out stream 
     Throws: As specified in [format.err.report]
or 
3 Effects: Writes the result ofvoid vprint_nonunicode ( ostream & os , string_view fmt , format_args args ); 
vformat ( os . getloc (),  fmt ,  args ) os 
     Throws: As specified in [format.err.report]
or 
Appendix A: Unicode tests
This appendix gives full listings of programs for testing Unicode handling in various formatting facilities as well as test commands and their output on different platforms. The code contains additional sanity checks to ensure that the strings are encoded in some form of UTF as opposed to a legacy encoding.
C (
#include <stdio.h>#include <stdlib.h>int main () { const char * message = "Привет, κόσμος! \n " ; if (( unsigned char ) message [ 0 ] != 0xD0 && ( unsigned char ) message [ 1 ] != 0x9F ) abort (); printf ( message ); } 
Go (
package mainimport "fmt" import "log" func main() { var message= "Привет, κόσμος!" if message[ 0 ] != 0xD0 && message[ 1 ] != 0x9F { log. Fatal( "wrong encoding" ) } fmt. Println( message) } 
Java (
class Test { public static void main ( String [] args ) { String message = "Привет, κόσμος!\n" ; if ( message . charAt ( 0 ) != 0x41F ) throw new RuntimeException (); System . out . ( message ); } } 
JavaScript / Node.js (
message= "Привет, κόσμος!" ; if ( message. charCodeAt( 0 ) != 0x41F ) throw "wrong encoding" ; console. log( message); 
Python (
message = "Привет, κόσμος!" if ord( message [ 0 ]) != 0x41F : raise Exception () print( message ) 
Rust (
fn main () { if "Привет, κόσμος!" . chars (). nth ( 0 ). unwrap () as u32 != 0x41F { panic ! (); } println ! ( "Привет, κόσμος!" ); } 
Linux:
$cc test . c - o c - test $. / c - test Привет, κόσμος! $. / c - test > out - c - linux . txt $go build - o go - test test . go $. / go - test Привет, κόσμος! $. / go - test > out - go - linux . txt $java Test Привет, κόσμος! $java Test > out - java - linux . txt $node test . js Привет, κόσμος! $node test . js > out - js - linux . txt $python3 test . py Привет, κόσμος! $python3 test . py > out - py - linux . txt $rustc test . rs - o rust - test $. / rust - test Привет, κόσμος! $. / rust - test > out - rust - linux . txt 
All output files are in UTF-8:
Linux configuration:
- 
     Ubuntu Focal 20.04 with the ru_RU.UTF-8 locale 
- 
     cc: gcc 9.3.0 
- 
     go: go1.13.8 
- 
     java: openjdk 11.0.9.1 
- 
     node: v14.5.0 
- 
     python3: 3.7.5 
- 
     rustc: 1.47.0 
macOS:
% cc test . c - o c - test % . / c - test Привет, κόσμος! % . / c - test > out - c - macos . txt % go build - o test - go test . go % . / test - go Привет, κόσμος! % . / test - go > out - go - macos . txt % java Test Привет, κόσμος! % java Test > out - java - macos . txt % node test . js Привет, κόσμος! % node test . js > out - js - macos . txt % python3 test . py Привет, κόσμος! % python3 test . py > out - py - macos . txt % rustc test . rs - o rust - test % . / rust - test Привет, κόσμος! % . / rust - test > out - rust - macos . txt 
All output files are in UTF-8:
macOS configuration:
- 
     macOS Catalina 10.15.7 with the ru_RU.UTF-8 locale which is the default for Russian 
- 
     cc: Apple clang version 12.0.0 (clang-1200.0.32.27) 
- 
     go: go1.15.5 
- 
     java: openjdk 14.0.1 
- 
     node: v14.5.0 
- 
     python3: 3.7.5 
- 
     rustc: 1.47.0 
Windows:
> cl / Fe : c - test . exe test . c ... > c - test ╨Я╤А╨╕╨▓╨╡╤В, ╬║╧М╧Г╬╝╬┐╧В! > c - test > out - c - windows . txt > c - test | findstr , ╨Я╤А╨╕╨▓╨╡╤В, ╬║╧М╧Г╬╝╬┐╧В! > go build - o go - test . exe test . go > go - test Привет, κόσμος! > go - test > out - go - windows . txt > go - test | findstr , ╨Я╤А╨╕╨▓╨╡╤В, ╬║╧М╧Г╬╝╬┐╧В! > java Test Привет, ??????! > java Test > out - java - windows . txt > java Test | findstr , ╧ЁштхЄ, ??????! > node test . js Привет, κόσμος! > node test . js > out - js - windows . txt > node test . js | findstr , ╨Я╤А╨╕╨▓╨╡╤В, ╬║╧М╧Г╬╝╬┐╧В! > python test . py Привет, κόσμος! > python test . py > out - py - windows . txt Traceback ( most recent call last ) : File "... \t est.py" , line 4 , in < module > ( message ) File "...\Python39\lib\encodings\cp1251.py" , line 19 , in encode return codecs . charmap_encode ( input , self . errors , encoding_table )[ 0 ] UnicodeEncodeError : 'charmap 'codec can ’t encode characters in position 8 - 13 : character maps to < undefined > > python test . py | findstr , Traceback ( most recent call last ) : File "... \t est.py" , line 4 , in < module > ( message ) File "...\Python39\lib\encodings\cp1251.py" , line 19 , in encode return codecs . charmap_encode ( input , self . errors , encoding_table )[ 0 ] UnicodeEncodeError : 'charmap 'codec can ’t encode characters in position 8 - 13 : character maps to < undefined > > rustc test . rs - o rust - test . exe > rust - test Привет, κόσμος! > rust - test > out - rust - windows . txt > rust - test | findstr , ╨Я╤А╨╕╨▓╨╡╤В, ╬║╧М╧Г╬╝╬┐╧В! 
C, JavaScript (node.js), Rust and Go produced valid UTF-8 when the output was
redirected to files. Java produced a file in the legacy CP1251 encoding with 
Windows configuration:
- 
     Windows 10 10.0.19041 with Russian Region and Language settings 
- 
     cl: Microsoft (R) C/C++ Optimizing Compiler Version 19.28.29335 
- 
     go: go1.15.6 
- 
     java: Java HotSpot(TM) 64-Bit Server VM (build 15.0.1+9-18) 
- 
     node: v14.15.3 
- 
     python: 3.9.1 
- 
     rustc: v14.15.3 
13. Acknowledgements
Thanks to Corentin Jabot for his work on text encodings in C++ and in particular [P1885] that will simplify implementation of the current proposal.
Thanks to Roger Orr, Peter Brett, the BSI C++ panel and Tom Honermann for their feedback, support, constructive criticism and contributions to the proposal.