@@ -23,6 +23,16 @@ macro_rules! unlikely {
23
23
} ;
24
24
}
25
25
26
+ // Use this when the generated code should differ between signed and unsigned types.
27
+ macro_rules! sign_dependent_expr {
28
+ ( signed ? if signed { $signed_case: expr } if unsigned { $unsigned_case: expr } ) => {
29
+ $signed_case
30
+ } ;
31
+ ( unsigned ? if signed { $signed_case: expr } if unsigned { $unsigned_case: expr } ) => {
32
+ $unsigned_case
33
+ } ;
34
+ }
35
+
26
36
// All these modules are technically private and only exposed for coretests:
27
37
#[ cfg( not( no_fp_fmt_parse) ) ]
28
38
pub mod bignum;
@@ -1410,15 +1420,25 @@ const fn from_str_radix_panic(radix: u32) {
1410
1420
}
1411
1421
1412
1422
macro_rules! from_str_radix {
1413
- ( $( $int_ty: ty) +) => { $(
1423
+ ( $signedness : ident $ ( $int_ty: ty) +) => { $(
1414
1424
impl $int_ty {
1415
1425
/// Converts a string slice in a given base to an integer.
1416
1426
///
1417
- /// The string is expected to be an optional `+` sign
1418
- /// followed by digits.
1419
- /// Leading and trailing whitespace represent an error.
1420
- /// Digits are a subset of these characters, depending on `radix`:
1427
+ /// The string is expected to be an optional
1428
+ #[ doc = sign_dependent_expr!{
1429
+ $signedness ?
1430
+ if signed {
1431
+ " `+` or `-` "
1432
+ }
1433
+ if unsigned {
1434
+ " `+` "
1435
+ }
1436
+ } ]
1437
+ /// sign followed by only digits. Leading and trailing non-digit characters (including
1438
+ /// whitespace) represent an error. Underscores (which are accepted in rust literals)
1439
+ /// also represent an error.
1421
1440
///
1441
+ /// Digits are a subset of these characters, depending on `radix`:
1422
1442
/// * `0-9`
1423
1443
/// * `a-z`
1424
1444
/// * `A-Z`
@@ -1430,10 +1450,13 @@ macro_rules! from_str_radix {
1430
1450
/// # Examples
1431
1451
///
1432
1452
/// Basic usage:
1433
- ///
1434
1453
/// ```
1435
1454
#[ doc = concat!( "assert_eq!(" , stringify!( $int_ty) , "::from_str_radix(\" A\" , 16), Ok(10));" ) ]
1436
1455
/// ```
1456
+ /// Trailing space returns error:
1457
+ /// ```
1458
+ #[ doc = concat!( "assert!(" , stringify!( $int_ty) , "::from_str_radix(\" 1 \" , 10).is_err());" ) ]
1459
+ /// ```
1437
1460
#[ stable( feature = "rust1" , since = "1.0.0" ) ]
1438
1461
#[ rustc_const_stable( feature = "const_int_from_str" , since = "1.82.0" ) ]
1439
1462
pub const fn from_str_radix( src: & str , radix: u32 ) -> Result <$int_ty, ParseIntError > {
@@ -1535,20 +1558,31 @@ macro_rules! from_str_radix {
1535
1558
) +}
1536
1559
}
1537
1560
1538
- from_str_radix ! { i8 u8 i16 u16 i32 u32 i64 u64 i128 u128 }
1561
+ from_str_radix ! { unsigned u8 u16 u32 u64 u128 }
1562
+ from_str_radix ! { signed i8 i16 i32 i64 i128 }
1539
1563
1540
1564
// Re-use the relevant implementation of from_str_radix for isize and usize to avoid outputting two
1541
1565
// identical functions.
1542
1566
macro_rules! from_str_radix_size_impl {
1543
- ( $( $t: ident $size: ty) ,* ) => { $(
1567
+ ( $( $signedness : ident $ t: ident $size: ty) ,* ) => { $(
1544
1568
impl $size {
1545
1569
/// Converts a string slice in a given base to an integer.
1546
1570
///
1547
- /// The string is expected to be an optional `+` sign
1548
- /// followed by digits.
1549
- /// Leading and trailing whitespace represent an error.
1550
- /// Digits are a subset of these characters, depending on `radix`:
1571
+ /// The string is expected to be an optional
1572
+ #[ doc = sign_dependent_expr!{
1573
+ $signedness ?
1574
+ if signed {
1575
+ " `+` or `-` "
1576
+ }
1577
+ if unsigned {
1578
+ " `+` "
1579
+ }
1580
+ } ]
1581
+ /// sign followed by only digits. Leading and trailing non-digit characters (including
1582
+ /// whitespace) represent an error. Underscores (which are accepted in rust literals)
1583
+ /// also represent an error.
1551
1584
///
1585
+ /// Digits are a subset of these characters, depending on `radix`:
1552
1586
/// * `0-9`
1553
1587
/// * `a-z`
1554
1588
/// * `A-Z`
@@ -1560,10 +1594,13 @@ macro_rules! from_str_radix_size_impl {
1560
1594
/// # Examples
1561
1595
///
1562
1596
/// Basic usage:
1563
- ///
1564
1597
/// ```
1565
1598
#[ doc = concat!( "assert_eq!(" , stringify!( $size) , "::from_str_radix(\" A\" , 16), Ok(10));" ) ]
1566
1599
/// ```
1600
+ /// Trailing space returns error:
1601
+ /// ```
1602
+ #[ doc = concat!( "assert!(" , stringify!( $size) , "::from_str_radix(\" 1 \" , 10).is_err());" ) ]
1603
+ /// ```
1567
1604
#[ stable( feature = "rust1" , since = "1.0.0" ) ]
1568
1605
#[ rustc_const_stable( feature = "const_int_from_str" , since = "1.82.0" ) ]
1569
1606
pub const fn from_str_radix( src: & str , radix: u32 ) -> Result <$size, ParseIntError > {
@@ -1576,8 +1613,8 @@ macro_rules! from_str_radix_size_impl {
1576
1613
}
1577
1614
1578
1615
#[ cfg( target_pointer_width = "16" ) ]
1579
- from_str_radix_size_impl ! { i16 isize , u16 usize }
1616
+ from_str_radix_size_impl ! { signed i16 isize , unsigned u16 usize }
1580
1617
#[ cfg( target_pointer_width = "32" ) ]
1581
- from_str_radix_size_impl ! { i32 isize , u32 usize }
1618
+ from_str_radix_size_impl ! { signed i32 isize , unsigned u32 usize }
1582
1619
#[ cfg( target_pointer_width = "64" ) ]
1583
- from_str_radix_size_impl ! { i64 isize , u64 usize }
1620
+ from_str_radix_size_impl ! { signed i64 isize , unsigned u64 usize }
0 commit comments