| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2023-04-13 15:41:29 +02:00
										 |  |  |  |  * Copyright (c) 2021-2023, Linus Groh <linusg@serenityos.org> | 
					
						
							| 
									
										
										
										
											2021-11-23 23:09:01 +00:00
										 |  |  |  |  * Copyright (c) 2021, Luke Wilde <lukew@serenityos.org> | 
					
						
							| 
									
										
										
										
											2024-01-05 13:28:39 +13:00
										 |  |  |  |  * Copyright (c) 2024, Shannon Booth <shannon@serenityos.org> | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |  * Copyright (c) 2024, Tim Flynn <trflynn89@ladybird.org> | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 12:36:06 -04:00
										 |  |  |  | #include <AK/Math.h>
 | 
					
						
							| 
									
										
										
										
											2024-11-16 12:51:53 -05:00
										 |  |  |  | #include <AK/NumericLimits.h>
 | 
					
						
							|  |  |  |  | #include <LibCrypto/BigInt/SignedBigInteger.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  | #include <LibJS/Runtime/Temporal/Duration.h>
 | 
					
						
							| 
									
										
										
										
											2024-11-16 12:51:53 -05:00
										 |  |  |  | #include <LibJS/Runtime/Value.h>
 | 
					
						
							|  |  |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | namespace JS::Temporal { | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  | // 7.5.16 IsValidDuration ( years, months, weeks, days, hours, minutes, seconds, milliseconds, microseconds, nanoseconds ), https://tc39.es/proposal-temporal/#sec-temporal-isvalidduration
 | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  | bool is_valid_duration(double years, double months, double weeks, double days, double hours, double minutes, double seconds, double milliseconds, double microseconds, double nanoseconds) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |     // 1. Let sign be 0.
 | 
					
						
							|  |  |  |  |     auto sign = 0; | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 2. For each value v of « years, months, weeks, days, hours, minutes, seconds, milliseconds, microseconds, nanoseconds », do
 | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |     for (auto value : { years, months, weeks, days, hours, minutes, seconds, milliseconds, microseconds, nanoseconds }) { | 
					
						
							| 
									
										
										
										
											2022-05-07 16:26:32 +02:00
										 |  |  |  |         // a. If 𝔽(v) is not finite, return false.
 | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |         if (!isfinite(value)) | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  |             return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |         // b. If v < 0, then
 | 
					
						
							|  |  |  |  |         if (value < 0) { | 
					
						
							|  |  |  |  |             // i. If sign > 0, return false.
 | 
					
						
							|  |  |  |  |             if (sign > 0) | 
					
						
							|  |  |  |  |                 return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // ii. Set sign to -1.
 | 
					
						
							|  |  |  |  |             sign = -1; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         // c. Else if v > 0, then
 | 
					
						
							|  |  |  |  |         else if (value > 0) { | 
					
						
							|  |  |  |  |             // i. If sign < 0, return false.
 | 
					
						
							|  |  |  |  |             if (sign < 0) | 
					
						
							|  |  |  |  |                 return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // ii. Set sign to 1.
 | 
					
						
							|  |  |  |  |             sign = 1; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 12:36:06 -04:00
										 |  |  |  |     // 3. If abs(years) ≥ 2**32, return false.
 | 
					
						
							|  |  |  |  |     if (AK::fabs(years) > NumericLimits<u32>::max()) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 4. If abs(months) ≥ 2**32, return false.
 | 
					
						
							|  |  |  |  |     if (AK::fabs(months) > NumericLimits<u32>::max()) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 5. If abs(weeks) ≥ 2**32, return false.
 | 
					
						
							|  |  |  |  |     if (AK::fabs(weeks) > NumericLimits<u32>::max()) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |     // 6. Let totalFractionalSeconds be days × 86,400 + hours × 3600 + minutes × 60 + seconds + ℝ(𝔽(milliseconds)) × 10**-3 + ℝ(𝔽(microseconds)) × 10**-6 + ℝ(𝔽(nanoseconds)) × 10**-9.
 | 
					
						
							|  |  |  |  |     // 7. NOTE: The above step cannot be implemented directly using floating-point arithmetic. Multiplying by 10**-3,
 | 
					
						
							|  |  |  |  |     //          10**-6, and 10**-9 respectively may be imprecise when milliseconds, microseconds, or nanoseconds is an
 | 
					
						
							|  |  |  |  |     //          unsafe integer. This multiplication can be implemented in C++ with an implementation of std::remquo()
 | 
					
						
							|  |  |  |  |     //          with sufficient bits in the quotient. String manipulation will also give an exact result, since the
 | 
					
						
							|  |  |  |  |     //          multiplication is by a power of 10.
 | 
					
						
							| 
									
										
										
										
											2024-08-13 12:36:06 -04:00
										 |  |  |  |     static Crypto::SignedBigInteger days_to_nanoseconds { 8.64e13 }; | 
					
						
							|  |  |  |  |     static Crypto::SignedBigInteger hours_to_nanoseconds { 3.6e12 }; | 
					
						
							|  |  |  |  |     static Crypto::SignedBigInteger minutes_to_nanoseconds { 6e10 }; | 
					
						
							|  |  |  |  |     static Crypto::SignedBigInteger seconds_to_nanoseconds { 1e9 }; | 
					
						
							|  |  |  |  |     static Crypto::SignedBigInteger milliseconds_to_nanoseconds { 1e6 }; | 
					
						
							|  |  |  |  |     static Crypto::SignedBigInteger microseconds_to_nanoseconds { 1e3 }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     auto normalized_nanoseconds = Crypto::SignedBigInteger { days }.multiplied_by(days_to_nanoseconds); | 
					
						
							|  |  |  |  |     normalized_nanoseconds = normalized_nanoseconds.plus(Crypto::SignedBigInteger { hours }.multiplied_by(hours_to_nanoseconds)); | 
					
						
							|  |  |  |  |     normalized_nanoseconds = normalized_nanoseconds.plus(Crypto::SignedBigInteger { minutes }.multiplied_by(minutes_to_nanoseconds)); | 
					
						
							|  |  |  |  |     normalized_nanoseconds = normalized_nanoseconds.plus(Crypto::SignedBigInteger { seconds }.multiplied_by(seconds_to_nanoseconds)); | 
					
						
							|  |  |  |  |     normalized_nanoseconds = normalized_nanoseconds.plus(Crypto::SignedBigInteger { milliseconds }.multiplied_by(milliseconds_to_nanoseconds)); | 
					
						
							|  |  |  |  |     normalized_nanoseconds = normalized_nanoseconds.plus(Crypto::SignedBigInteger { microseconds }.multiplied_by(microseconds_to_nanoseconds)); | 
					
						
							|  |  |  |  |     normalized_nanoseconds = normalized_nanoseconds.plus(Crypto::SignedBigInteger { nanoseconds }); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 8. If abs(normalizedSeconds) ≥ 2**53, return false.
 | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |     static auto maximum_time = Crypto::SignedBigInteger { MAX_ARRAY_LIKE_INDEX }.plus(1_bigint).multiplied_by(seconds_to_nanoseconds); | 
					
						
							| 
									
										
										
										
											2024-08-13 12:36:06 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (normalized_nanoseconds.is_negative()) | 
					
						
							|  |  |  |  |         normalized_nanoseconds.negate(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (normalized_nanoseconds >= maximum_time) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-16 13:14:14 -05:00
										 |  |  |  |     // 9. Return true.
 | 
					
						
							| 
									
										
										
										
											2021-07-15 23:20:43 +01:00
										 |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | } |