]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
libstdc++: Remove unnecessary "& 1" from year_month_day_last::day()
authorCassio Neri <cassio.neri@gmail.com>
Sat, 11 Nov 2023 16:44:58 +0000 (16:44 +0000)
committerJonathan Wakely <jwakely@redhat.com>
Tue, 14 Nov 2023 22:32:08 +0000 (22:32 +0000)
When year_month_day_last::day() was implemented, Dr. Matthias Kretz realised
that the operation "& 1" wasn't necessary but we did not patch it at that
time. This patch removes the unnecessary operation.

libstdc++-v3/ChangeLog:

* include/std/chrono (year_month_day_last::day): Remove &1.

libstdc++-v3/include/std/chrono

index 10e868e5a036924177cf952a81f4bce10c497106..a826982803b7855bfdd6d8ce15b2be677c621efa 100644 (file)
@@ -1800,22 +1800,26 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
       {
        const auto __m = static_cast<unsigned>(month());
 
-       // Excluding February, the last day of month __m is either 30 or 31 or,
-       // in another words, it is 30 + b = 30 | b, where b is in {0, 1}.
+       // The result is unspecified if __m < 1 or __m > 12.  Hence, assume
+       // 1 <= __m <= 12.  For __m != 2, day() == 30 or day() == 31 or, in
+       // other words, day () == 30 | b, where b is in {0, 1}.
 
-       // If __m in {1, 3, 4, 5, 6, 7}, then b is 1 if, and only if __m is odd.
-       // Hence, b = __m & 1 = (__m ^ 0) & 1.
+       // If __m in {1, 3, 4, 5, 6, 7}, then b is 1 if, and only if, __m is
+       // odd.  Hence, b = __m & 1 = (__m ^ 0) & 1.
 
-       // If __m in {8, 9, 10, 11, 12}, then b is 1 if, and only if __m is even.
-       // Hence, b = (__m ^ 1) & 1.
+       // If __m in {8, 9, 10, 11, 12}, then b is 1 if, and only if, __m is
+       // even.  Hence, b = (__m ^ 1) & 1.
 
        // Therefore, b = (__m ^ c) & 1, where c = 0, if __m < 8, or c = 1 if
        // __m >= 8, that is, c = __m >> 3.
 
-       // The above mathematically justifies this implementation whose
-       // performance does not depend on look-up tables being on the L1 cache.
-       return chrono::day{__m != 2 ? ((__m ^ (__m >> 3)) & 1) | 30
-                                   : _M_y.is_leap() ? 29 : 28};
+       // Since 30 = (11110)_2 and __m <= 31 = (11111)_2, the "& 1" in b's
+       // calculation is unnecessary.
+
+       // The performance of this implementation does not depend on look-up
+       // tables being on the L1 cache.
+       return chrono::day{__m != 2 ? (__m ^ (__m >> 3)) | 30
+         : _M_y.is_leap() ? 29 : 28};
       }
 
       constexpr