@@ -1002,12 +1002,12 @@ pub mod span;
1002
1002
pub mod __macro_support {
1003
1003
pub use crate :: callsite:: { Callsite , Registration } ;
1004
1004
use crate :: { collect:: Interest , Metadata } ;
1005
- use core:: fmt;
1006
1005
use core:: sync:: atomic:: { AtomicU8 , Ordering } ;
1006
+ use core:: { fmt, mem} ;
1007
1007
// Re-export the `core` functions that are used in macros. This allows
1008
1008
// a crate to be named `core` and avoid name clashes.
1009
1009
// See here: https://github.com/tokio-rs/tracing/issues/2761
1010
- pub use core:: { concat, file, format_args, iter:: Iterator , line, option:: Option } ;
1010
+ pub use core:: { concat, file, format_args, iter:: Iterator , line, option:: Option , stringify } ;
1011
1011
1012
1012
/// Callsite implementation used by macro-generated code.
1013
1013
///
@@ -1195,6 +1195,66 @@ pub mod __macro_support {
1195
1195
. finish ( )
1196
1196
}
1197
1197
}
1198
+
1199
+ /// Implementation detail used for constructing FieldSet names from raw
1200
+ /// identifiers. In `info!(..., r#type = "...")` the macro would end up
1201
+ /// constructing a name equivalent to `FieldName(*b"type")`.
1202
+ pub struct FieldName < const N : usize > ( [ u8 ; N ] ) ;
1203
+
1204
+ impl < const N : usize > FieldName < N > {
1205
+ /// Convert `"prefix.r#keyword.suffix"` to `b"prefix.keyword.suffix"`.
1206
+ pub const fn new ( input : & str ) -> Self {
1207
+ let input = input. as_bytes ( ) ;
1208
+ let mut output = [ 0u8 ; N ] ;
1209
+ let mut read = 0 ;
1210
+ let mut write = 0 ;
1211
+ while read < input. len ( ) {
1212
+ if read + 1 < input. len ( ) && input[ read] == b'r' && input[ read + 1 ] == b'#' {
1213
+ read += 2 ;
1214
+ }
1215
+ output[ write] = input[ read] ;
1216
+ read += 1 ;
1217
+ write += 1 ;
1218
+ }
1219
+ assert ! ( write == N ) ;
1220
+ Self ( output)
1221
+ }
1222
+
1223
+ pub const fn as_str ( & self ) -> & str {
1224
+ // SAFETY: Because of the private visibility of self.0, it must have
1225
+ // been computed by Self::new. So these bytes are all of the bytes
1226
+ // of some original valid UTF-8 string, but with "r#" substrings
1227
+ // removed, which cannot have produced invalid UTF-8.
1228
+ unsafe { mem:: transmute ( self . 0 . as_slice ( ) ) }
1229
+ }
1230
+ }
1231
+
1232
+ impl FieldName < 0 > {
1233
+ /// For `"prefix.r#keyword.suffix"` compute `"prefix.keyword.suffix".len()`.
1234
+ pub const fn len ( input : & str ) -> usize {
1235
+ // Count occurrences of "r#"
1236
+ let mut raw = 0 ;
1237
+
1238
+ let mut i = 0 ;
1239
+ while i < input. len ( ) {
1240
+ if input. as_bytes ( ) [ i] == b'#' {
1241
+ raw += 1 ;
1242
+ }
1243
+ i += 1 ;
1244
+ }
1245
+
1246
+ input. len ( ) - 2 * raw
1247
+ }
1248
+ }
1249
+
1250
+ impl < const N : usize > fmt:: Debug for FieldName < N > {
1251
+ fn fmt ( & self , formatter : & mut fmt:: Formatter < ' _ > ) -> fmt:: Result {
1252
+ formatter
1253
+ . debug_tuple ( "FieldName" )
1254
+ . field ( & self . as_str ( ) )
1255
+ . finish ( )
1256
+ }
1257
+ }
1198
1258
}
1199
1259
1200
1260
#[ cfg( feature = "log" ) ]
0 commit comments