porffor 0.2.0-b9abe0d → 0.2.0-c1ed50d
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -1
- package/compiler/builtins/date.ts +728 -3
- package/compiler/builtins.js +0 -13
- package/compiler/codegen.js +20 -19
- package/compiler/generated_builtins.js +259 -7
- package/compiler/index.js +2 -1
- package/compiler/precompile.js +1 -1
- package/compiler/wasmSpec.js +1 -0
- package/compiler/wrap.js +7 -4
- package/package.json +1 -1
- package/.vscode/launch.json +0 -18
- package/test262_changes_from_1afe9b87d2_to_04-09.md +0 -270
@@ -1,9 +1,734 @@
|
|
1
1
|
// @porf -funsafe-no-unlikely-proto-checks
|
2
2
|
|
3
|
-
|
3
|
+
// 21.4.1.3 Day (t)
|
4
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-day
|
5
|
+
// 1. Return 𝔽(floor(ℝ(t / msPerDay))).
|
6
|
+
export const __ecma262_Day = (t: number): number => Math.floor(t / 86400000);
|
7
|
+
|
8
|
+
// 21.4.1.4 TimeWithinDay (t)
|
9
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-day
|
10
|
+
// 1. Return 𝔽(ℝ(t) modulo ℝ(msPerDay)).
|
11
|
+
export const __ecma262_TimeWithinDay = (t: number): number => t % 86400000;
|
12
|
+
|
13
|
+
// 21.4.1.5 DaysInYear (y)
|
14
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-daysinyear
|
15
|
+
export const __ecma262_DaysInYear = (y: number): number => {
|
16
|
+
// 1. Let ry be ℝ(y).
|
17
|
+
|
18
|
+
// 2. If (ry modulo 400) = 0, return 366𝔽.
|
19
|
+
if (y % 400 == 0) return 366;
|
20
|
+
|
21
|
+
// 3. If (ry modulo 100) = 0, return 365𝔽.
|
22
|
+
if (y % 100 == 0) return 365;
|
23
|
+
|
24
|
+
// 4. If (ry modulo 4) = 0, return 366𝔽.
|
25
|
+
if (y % 4 == 0) return 366;
|
26
|
+
|
27
|
+
// 5. Return 365𝔽.
|
28
|
+
return 365;
|
29
|
+
};
|
30
|
+
|
31
|
+
// 21.4.1.6 DayFromYear (y)
|
32
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-dayfromyear
|
33
|
+
export const __ecma262_DayFromYear = (y: number): number => {
|
34
|
+
// 1. Let ry be ℝ(y).
|
35
|
+
// 2. NOTE: In the following steps, numYears1, numYears4, numYears100, and numYears400
|
36
|
+
// represent the number of years divisible by 1, 4, 100, and 400, respectively,
|
37
|
+
// that occur between the epoch and the start of year y.
|
38
|
+
// The number is negative if y is before the epoch.
|
39
|
+
|
40
|
+
// 3. Let numYears1 be (ry - 1970).
|
41
|
+
const numYears1: number = y - 1970;
|
42
|
+
|
43
|
+
// 4. Let numYears4 be floor((ry - 1969) / 4).
|
44
|
+
const numYears4: number = Math.floor((y - 1969) / 4);
|
45
|
+
|
46
|
+
// 5. Let numYears100 be floor((ry - 1901) / 100).
|
47
|
+
const numYears100: number = Math.floor((y - 1901) / 100);
|
48
|
+
|
49
|
+
// 6. Let numYears400 be floor((ry - 1601) / 400).
|
50
|
+
const numYears400: number = Math.floor((y - 1601) / 400);
|
51
|
+
|
52
|
+
// 7. Return 𝔽(365 × numYears1 + numYears4 - numYears100 + numYears400).
|
53
|
+
return 365 * numYears1 + numYears4 - numYears100 + numYears400;
|
54
|
+
};
|
55
|
+
|
56
|
+
// 21.4.1.7 TimeFromYear (y)
|
57
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-timefromyear
|
58
|
+
// 1. Return msPerDay × DayFromYear(y).
|
59
|
+
export const __ecma262_TimeFromYear = (y: number): number => 86400000 * __ecma262_DayFromYear(y);
|
60
|
+
|
61
|
+
// 21.4.1.8 YearFromTime (t)
|
62
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-yearfromtime
|
63
|
+
export const __ecma262_YearFromTime = (t: number): number => {
|
64
|
+
// 1. Return the largest integral Number y (closest to +∞) such that TimeFromYear(y) ≤ t.
|
65
|
+
|
66
|
+
// guess year with floor(t / (365.2425 * msPerDay)) + 1970
|
67
|
+
const y: number = Math.floor(t / 31556952000) + 1970;
|
68
|
+
|
69
|
+
// get timestamp for guessed year
|
70
|
+
const t2: number = __ecma262_TimeFromYear(y);
|
71
|
+
|
72
|
+
// if timestamp is higher, we guessed too high
|
73
|
+
if (t2 > t) return y - 1;
|
74
|
+
|
75
|
+
// if timestamp + days in year is lower, we guessed too low
|
76
|
+
if ((t2 + __ecma262_DaysInYear(y) * 86400000) <= t) return y + 1;
|
77
|
+
|
78
|
+
// we guessed correct
|
79
|
+
return y;
|
80
|
+
};
|
81
|
+
|
82
|
+
// 21.4.1.9 DayWithinYear (t)
|
83
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-daywithinyear
|
84
|
+
// 1. Return Day(t) - DayFromYear(YearFromTime(t)).
|
85
|
+
export const __ecma262_DayWithinYear = (t: number): number => __ecma262_Day(t) - __ecma262_DayFromYear(__ecma262_YearFromTime(t));
|
86
|
+
|
87
|
+
// 21.4.1.10 InLeapYear (t)
|
88
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-inleapyear
|
89
|
+
// 1. If DaysInYear(YearFromTime(t)) is 366𝔽, return 1𝔽; else return +0𝔽.
|
90
|
+
export const __ecma262_InLeapYear = (t: number): number => __ecma262_DaysInYear(__ecma262_YearFromTime(t)) == 366 ? 1 : 0;
|
91
|
+
|
92
|
+
// 21.4.1.11 MonthFromTime (t)
|
93
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-monthfromtime
|
94
|
+
export const __ecma262_MonthFromTime = (t: number): number => {
|
95
|
+
// 1. Let inLeapYear be InLeapYear(t).
|
96
|
+
const inLeapYear: number = __ecma262_InLeapYear(t);
|
97
|
+
|
98
|
+
// 2. Let dayWithinYear be DayWithinYear(t).
|
99
|
+
const dayWithinYear: number = __ecma262_DayWithinYear(t);
|
100
|
+
|
101
|
+
// 3. If dayWithinYear < 31𝔽, return +0𝔽.
|
102
|
+
if (dayWithinYear < 31) return 0;
|
103
|
+
|
104
|
+
// 4. If dayWithinYear < 59𝔽 + inLeapYear, return 1𝔽.
|
105
|
+
if (dayWithinYear < 59 + inLeapYear) return 1;
|
106
|
+
|
107
|
+
// 5. If dayWithinYear < 90𝔽 + inLeapYear, return 2𝔽.
|
108
|
+
if (dayWithinYear < 90 + inLeapYear) return 2;
|
109
|
+
|
110
|
+
// 6. If dayWithinYear < 120𝔽 + inLeapYear, return 3𝔽.
|
111
|
+
if (dayWithinYear < 120 + inLeapYear) return 3;
|
112
|
+
|
113
|
+
// 7. If dayWithinYear < 151𝔽 + inLeapYear, return 4𝔽.
|
114
|
+
if (dayWithinYear < 151 + inLeapYear) return 4;
|
115
|
+
|
116
|
+
// 8. If dayWithinYear < 181𝔽 + inLeapYear, return 5𝔽.
|
117
|
+
if (dayWithinYear < 181 + inLeapYear) return 5;
|
118
|
+
|
119
|
+
// 9. If dayWithinYear < 212𝔽 + inLeapYear, return 6𝔽.
|
120
|
+
if (dayWithinYear < 212 + inLeapYear) return 6;
|
121
|
+
|
122
|
+
// 10. If dayWithinYear < 243𝔽 + inLeapYear, return 7𝔽.
|
123
|
+
if (dayWithinYear < 243 + inLeapYear) return 7;
|
124
|
+
|
125
|
+
// 11. If dayWithinYear < 273𝔽 + inLeapYear, return 8𝔽.
|
126
|
+
if (dayWithinYear < 273 + inLeapYear) return 8;
|
127
|
+
|
128
|
+
// 12. If dayWithinYear < 304𝔽 + inLeapYear, return 9𝔽.
|
129
|
+
if (dayWithinYear < 304 + inLeapYear) return 9;
|
130
|
+
|
131
|
+
// 13. If dayWithinYear < 334𝔽 + inLeapYear, return 10𝔽.
|
132
|
+
if (dayWithinYear < 334 + inLeapYear) return 10;
|
133
|
+
|
134
|
+
// 14. Assert: dayWithinYear < 365𝔽 + inLeapYear.
|
135
|
+
|
136
|
+
// 15. Return 11𝔽.
|
137
|
+
return 11;
|
138
|
+
};
|
139
|
+
|
140
|
+
// 21.4.1.12 DateFromTime (t)
|
141
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-datefromtime
|
142
|
+
export const __ecma262_DateFromTime = (t: number): number => {
|
143
|
+
// 1. Let inLeapYear be InLeapYear(t).
|
144
|
+
const inLeapYear: number = __ecma262_InLeapYear(t);
|
145
|
+
|
146
|
+
// 2. Let dayWithinYear be DayWithinYear(t).
|
147
|
+
const dayWithinYear: number = __ecma262_DayWithinYear(t);
|
148
|
+
|
149
|
+
// 3. Let month be MonthFromTime(t).
|
150
|
+
const month = __ecma262_MonthFromTime(t);
|
151
|
+
|
152
|
+
// 4. If month is +0𝔽, return dayWithinYear + 1𝔽.
|
153
|
+
if (month == 0) return dayWithinYear + 1;
|
154
|
+
|
155
|
+
// 5. If month is 1𝔽, return dayWithinYear - 30𝔽.
|
156
|
+
if (month == 1) return dayWithinYear - 30;
|
157
|
+
|
158
|
+
// 6. If month is 2𝔽, return dayWithinYear - 58𝔽 - inLeapYear.
|
159
|
+
if (month == 2) return dayWithinYear - 58 - inLeapYear;
|
160
|
+
|
161
|
+
// 7. If month is 3𝔽, return dayWithinYear - 89𝔽 - inLeapYear.
|
162
|
+
if (month == 3) return dayWithinYear - 89 - inLeapYear;
|
163
|
+
|
164
|
+
// 8. If month is 4𝔽, return dayWithinYear - 119𝔽 - inLeapYear.
|
165
|
+
if (month == 4) return dayWithinYear - 119 - inLeapYear;
|
166
|
+
|
167
|
+
// 9. If month is 5𝔽, return dayWithinYear - 150𝔽 - inLeapYear.
|
168
|
+
if (month == 5) return dayWithinYear - 150 - inLeapYear;
|
169
|
+
|
170
|
+
// 10. If month is 6𝔽, return dayWithinYear - 180𝔽 - inLeapYear.
|
171
|
+
if (month == 6) return dayWithinYear - 180 - inLeapYear;
|
172
|
+
|
173
|
+
// 11. If month is 7𝔽, return dayWithinYear - 211𝔽 - inLeapYear.
|
174
|
+
if (month == 7) return dayWithinYear - 211 - inLeapYear;
|
175
|
+
|
176
|
+
// 12. If month is 8𝔽, return dayWithinYear - 242𝔽 - inLeapYear.
|
177
|
+
if (month == 8) return dayWithinYear - 242 - inLeapYear;
|
178
|
+
|
179
|
+
// 13. If month is 9𝔽, return dayWithinYear - 272𝔽 - inLeapYear.
|
180
|
+
if (month == 9) return dayWithinYear - 272 - inLeapYear;
|
181
|
+
|
182
|
+
// 14. If month is 10𝔽, return dayWithinYear - 303𝔽 - inLeapYear.
|
183
|
+
if (month == 10) return dayWithinYear - 303 - inLeapYear;
|
184
|
+
|
185
|
+
// 15. Assert: month is 11𝔽.
|
186
|
+
|
187
|
+
// 16. Return dayWithinYear - 333𝔽 - inLeapYear.
|
188
|
+
return dayWithinYear - 333 - inLeapYear;
|
189
|
+
};
|
190
|
+
|
191
|
+
// 21.4.1.13 WeekDay (t)
|
192
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-weekday
|
193
|
+
// 1. Return 𝔽(ℝ(Day(t) + 4𝔽) modulo 7).
|
194
|
+
export const __ecma262_WeekDay = (t: number): number => (__ecma262_Day(t) + 4) % 7;
|
195
|
+
|
196
|
+
// 21.4.1.14 HourFromTime (t)
|
197
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-hourfromtime
|
198
|
+
// 1. Return 𝔽(floor(ℝ(t / msPerHour)) modulo HoursPerDay).
|
199
|
+
export const __ecma262_HourFromTime = (t: number): number => Math.floor(t / 3600000) % 24;
|
200
|
+
|
201
|
+
// 21.4.1.15 MinFromTime (t)
|
202
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-minfromtime
|
203
|
+
// 1. Return 𝔽(floor(ℝ(t / msPerMinute)) modulo MinutesPerHour).
|
204
|
+
export const __ecma262_MinFromTime = (t: number): number => Math.floor(t / 60000) % 60;
|
205
|
+
|
206
|
+
// 21.4.1.16 SecFromTime (t)
|
207
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-secfromtime
|
208
|
+
// 1. Return 𝔽(floor(ℝ(t / msPerSecond)) modulo SecondsPerMinute).
|
209
|
+
export const __ecma262_SecFromTime = (t: number): number => Math.floor(t / 1000) % 60;
|
210
|
+
|
211
|
+
// 21.4.1.17 msFromTime (t)
|
212
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-msfromtime
|
213
|
+
// 1. Return 𝔽(ℝ(t) modulo ℝ(msPerSecond)).
|
214
|
+
export const __ecma262_msFromTime = (t: number): number => t % 1000;
|
215
|
+
|
216
|
+
|
217
|
+
// // 21.4.1.21 GetNamedTimeZoneOffsetNanoseconds (timeZoneIdentifier, epochNanoseconds)
|
218
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-getnamedtimezoneoffsetnanoseconds
|
219
|
+
// export const __ecma262_GetNamedTimeZoneOffsetNanoseconds = (timeZoneIdentifier: bytestring, epochNanoseconds: number /* BigInt (unused) */): number => {
|
220
|
+
// // 1. Assert: timeZoneIdentifier is "UTC".
|
221
|
+
|
222
|
+
// // 2. Return 0.
|
223
|
+
// return 0;
|
224
|
+
// };
|
225
|
+
|
226
|
+
// // 21.4.1.23 AvailableNamedTimeZoneIdentifiers ()
|
227
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-availablenamedtimezoneidentifiers
|
228
|
+
// export const __ecma262_AvailableNamedTimeZoneIdentifiers = (): bytestring[] => {
|
229
|
+
// // 1. If the implementation does not include local political rules for any time zones, then
|
230
|
+
// // a. Return « the Time Zone Identifier Record { [[Identifier]]: "UTC", [[PrimaryIdentifier]]: "UTC" } ».
|
231
|
+
// return [ 'UTC' ];
|
232
|
+
// };
|
233
|
+
|
234
|
+
// // 21.4.1.24 SystemTimeZoneIdentifier ()
|
235
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-systemtimezoneidentifier
|
236
|
+
// export const __ecma262_SystemTimeZoneIdentifier = (): bytestring => {
|
237
|
+
// // 1. If the implementation only supports the UTC time zone, return "UTC".
|
238
|
+
// return 'UTC';
|
239
|
+
// };
|
240
|
+
|
241
|
+
// 21.4.1.25 LocalTime (t)
|
242
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-localtime
|
243
|
+
// slightly break spec here by just simplifying the abstraction for if implementation does not include local political rules for any time zones
|
244
|
+
export const __ecma262_LocalTime = (t: number): number => t;
|
245
|
+
|
246
|
+
// 21.4.1.26 UTC (t)
|
247
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-utc-t
|
248
|
+
// slightly break spec here by just simplifying the abstraction for if implementation does not include local political rules for any time zones
|
249
|
+
export const __ecma262_UTC = (t: number): number => {
|
250
|
+
// 1. If t is not finite, return NaN.
|
251
|
+
if (!Number.isFinite(t)) return NaN;
|
252
|
+
|
253
|
+
return t;
|
254
|
+
};
|
255
|
+
|
256
|
+
|
257
|
+
// todo: move this somewhere generic?
|
258
|
+
// 7.1.5 ToIntegerOrInfinity (argument)
|
259
|
+
// https://tc39.es/ecma262/multipage/abstract-operations.html#sec-tointegerorinfinity
|
260
|
+
export const __ecma262_ToIntegerOrInfinity = (argument: unknown): number => {
|
261
|
+
// 1. Let number be ? ToNumber(argument).
|
262
|
+
const number: number = Number(argument);
|
263
|
+
|
264
|
+
// 2. If number is one of NaN, +0𝔽, or -0𝔽, return 0.
|
265
|
+
if (Number.isNaN(number)) return 0;
|
266
|
+
|
267
|
+
// 3. If number is +∞𝔽, return +∞.
|
268
|
+
// 4. If number is -∞𝔽, return -∞.
|
269
|
+
// if (!Number.isFinite(number)) return number;
|
270
|
+
|
271
|
+
// 5. Return truncate(ℝ(number)).
|
272
|
+
return Math.trunc(number);
|
273
|
+
};
|
274
|
+
|
275
|
+
// 21.4.1.27 MakeTime (hour, min, sec, ms)
|
276
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-maketime
|
277
|
+
export const __ecma262_MakeTime = (hour: number, min: number, sec: number, ms: number): number => {
|
278
|
+
// 1. If hour is not finite, min is not finite, sec is not finite, or ms is not finite, return NaN.
|
279
|
+
if (Porffor.fastOr(!Number.isFinite(hour), !Number.isFinite(min), !Number.isFinite(sec), !Number.isFinite(ms))) return NaN;
|
280
|
+
|
281
|
+
// 2. Let h be 𝔽(! ToIntegerOrInfinity(hour)).
|
282
|
+
const h: number = __ecma262_ToIntegerOrInfinity(hour);
|
283
|
+
// 3. Let m be 𝔽(! ToIntegerOrInfinity(min)).
|
284
|
+
const m: number = __ecma262_ToIntegerOrInfinity(min);
|
285
|
+
// 4. Let s be 𝔽(! ToIntegerOrInfinity(sec)).
|
286
|
+
const s: number = __ecma262_ToIntegerOrInfinity(sec);
|
287
|
+
// 5. Let milli be 𝔽(! ToIntegerOrInfinity(ms)).
|
288
|
+
const milli: number = __ecma262_ToIntegerOrInfinity(ms);
|
289
|
+
|
290
|
+
// 6. Return ((h × msPerHour + m × msPerMinute) + s × msPerSecond) + milli.
|
291
|
+
return ((h * 3600000 + m * 60000) + s * 1000) + milli;
|
292
|
+
};
|
293
|
+
|
294
|
+
// 21.4.1.28 MakeDay (year, month, date)
|
295
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-makeday
|
296
|
+
export const __ecma262_MakeDay = (year: number, month: number, date: number): number => {
|
297
|
+
// 1. If year is not finite, month is not finite, or date is not finite, return NaN.
|
298
|
+
if (Porffor.fastOr(!Number.isFinite(year), !Number.isFinite(month), !Number.isFinite(date))) return NaN;
|
299
|
+
|
300
|
+
// 2. Let y be 𝔽(! ToIntegerOrInfinity(year)).
|
301
|
+
const y: number = __ecma262_ToIntegerOrInfinity(year);
|
302
|
+
// 3. Let m be 𝔽(! ToIntegerOrInfinity(month)).
|
303
|
+
const m: number = __ecma262_ToIntegerOrInfinity(month);
|
304
|
+
// 4. Let dt be 𝔽(! ToIntegerOrInfinity(date)).
|
305
|
+
const dt: number = __ecma262_ToIntegerOrInfinity(date);
|
306
|
+
|
307
|
+
// 5. Let ym be y + 𝔽(floor(ℝ(m) / 12)).
|
308
|
+
let ym: number = y + Math.floor(m / 12);
|
309
|
+
|
310
|
+
// 6. If ym is not finite, return NaN.
|
311
|
+
if (!Number.isFinite(ym)) return NaN;
|
312
|
+
|
313
|
+
// 7. Let mn be 𝔽(ℝ(m) modulo 12).
|
314
|
+
const mn: number = m % 12;
|
315
|
+
|
316
|
+
// 8. Find a finite time value t such that YearFromTime(t) is ym, MonthFromTime(t) is mn, and DateFromTime(t) is 1𝔽; but if this is not possible (because some argument is out of range), return NaN.
|
317
|
+
|
318
|
+
// https://howardhinnant.github.io/date_algorithms.html#days_from_civil
|
319
|
+
if (mn <= 1) ym -= 1;
|
320
|
+
|
321
|
+
const era: number = Math.trunc((ym >= 0 ? ym : (ym - 399)) / 400);
|
322
|
+
const yoe: number = ym - era * 400;
|
323
|
+
const doy: number = Math.trunc((153 * (mn + (mn > 1 ? -2 : 10)) + 2) / 5);
|
324
|
+
const doe: number = yoe * 365 + Math.trunc(yoe / 4) - Math.trunc(yoe / 100) + doy;
|
325
|
+
const day: number = era * 146097 + doe - 719468;
|
326
|
+
|
327
|
+
// 9. Return Day(t) + dt - 1𝔽.
|
328
|
+
// day = Day(t) (our day calculated is already as day)
|
329
|
+
return day + dt - 1;
|
330
|
+
};
|
331
|
+
|
332
|
+
// 21.4.1.29 MakeDate (day, time)
|
333
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-makedate
|
334
|
+
export const __ecma262_MakeDate = (day: number, time: number): number => {
|
335
|
+
// 1. If day is not finite or time is not finite, return NaN.
|
336
|
+
if (Porffor.fastOr(!Number.isFinite(day), !Number.isFinite(time))) return NaN;
|
337
|
+
|
338
|
+
// 2. Let tv be day × msPerDay + time.
|
339
|
+
const tv: number = day * 86400000 + time;
|
340
|
+
|
341
|
+
// 3. If tv is not finite, return NaN.
|
342
|
+
if (!Number.isFinite(tv)) return NaN;
|
343
|
+
|
344
|
+
// 4. Return tv.
|
345
|
+
return tv;
|
346
|
+
};
|
347
|
+
|
348
|
+
// 21.4.1.30 MakeFullYear (year)
|
349
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-makefullyear
|
350
|
+
export const __ecma262_MakeFullYear = (year: number): number => {
|
351
|
+
// 1. If year is NaN, return NaN.
|
352
|
+
if (Number.isNaN(year)) return NaN;
|
353
|
+
|
354
|
+
// 2. Let truncated be ! ToIntegerOrInfinity(year).
|
355
|
+
const truncated: number = __ecma262_ToIntegerOrInfinity(year);
|
356
|
+
|
357
|
+
// 3. If truncated is in the inclusive interval from 0 to 99, return 1900𝔽 + 𝔽(truncated).
|
358
|
+
if (truncated >= 0 && truncated <= 99) return 1900 + truncated;
|
359
|
+
|
360
|
+
// 4. Return 𝔽(truncated).
|
361
|
+
return truncated;
|
362
|
+
};
|
363
|
+
|
364
|
+
|
365
|
+
// 21.4.1.31 TimeClip (time)
|
366
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-timeclip
|
367
|
+
export const __ecma262_TimeClip = (time: number): number => {
|
368
|
+
// 1. If time is not finite, return NaN.
|
369
|
+
if (!Number.isFinite(time)) return NaN;
|
370
|
+
|
371
|
+
// 2. If abs(ℝ(time)) > 8.64 × 10**15, return NaN.
|
372
|
+
if (Math.abs(time) > 8.64e+15) return NaN;
|
4
373
|
|
374
|
+
// 3. Return 𝔽(! ToIntegerOrInfinity(time)).
|
375
|
+
return __ecma262_ToIntegerOrInfinity(time);
|
376
|
+
};
|
377
|
+
|
378
|
+
|
379
|
+
// 21.4.2.1 Date (...values)
|
380
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-date
|
5
381
|
export const Date = (): bytestring => {
|
6
|
-
//
|
382
|
+
// 1. If NewTarget is undefined, then
|
383
|
+
// a. Let now be the time value (UTC) identifying the current time.
|
384
|
+
// b. Return ToDateString(now).
|
385
|
+
// return Date$constructor().toString();
|
386
|
+
return '';
|
387
|
+
};
|
388
|
+
|
389
|
+
// dark wasm magic for a basic allocator, sorry.
|
390
|
+
export const __Porffor_date_allocate = (): Date => {
|
391
|
+
const hack: bytestring = '';
|
392
|
+
|
393
|
+
if (hack.length == 0) {
|
394
|
+
hack.length = Porffor.wasm`i32.const 1
|
395
|
+
memory.grow 0
|
396
|
+
drop
|
397
|
+
memory.size 0
|
398
|
+
i32.const 1
|
399
|
+
i32.sub
|
400
|
+
i32.const 65536
|
401
|
+
i32.mul
|
402
|
+
i32.from_u`;
|
403
|
+
}
|
404
|
+
|
405
|
+
const ptr: number = hack.length;
|
406
|
+
hack.length = ptr + 8;
|
407
|
+
|
408
|
+
return ptr;
|
409
|
+
};
|
410
|
+
|
411
|
+
export const __Porffor_date_read = (ptr: Date): number => Porffor.wasm.f64.load(ptr, 0, 0);
|
412
|
+
export const __Porffor_date_write = (ptr: Date, val: number) => {
|
413
|
+
Porffor.wasm.f64.store(ptr, val, 0, 0);
|
7
414
|
};
|
8
415
|
|
9
|
-
|
416
|
+
|
417
|
+
export const Date$constructor = (v0: unknown, v1: unknown, v2: unknown, v3: unknown, v4: unknown, v5: unknown, v6: unknown): Date => {
|
418
|
+
// todo: passing undefined to params should not act like no arg was passed
|
419
|
+
|
420
|
+
// 2. Let numberOfArgs be the number of elements in values.
|
421
|
+
// sorry.
|
422
|
+
const numberOfArgs: i32 =
|
423
|
+
(Porffor.rawType(v0) != Porffor.TYPES.undefined) +
|
424
|
+
(Porffor.rawType(v1) != Porffor.TYPES.undefined) +
|
425
|
+
(Porffor.rawType(v2) != Porffor.TYPES.undefined) +
|
426
|
+
(Porffor.rawType(v3) != Porffor.TYPES.undefined) +
|
427
|
+
(Porffor.rawType(v4) != Porffor.TYPES.undefined) +
|
428
|
+
(Porffor.rawType(v5) != Porffor.TYPES.undefined) +
|
429
|
+
(Porffor.rawType(v6) != Porffor.TYPES.undefined);
|
430
|
+
|
431
|
+
let dv: number = 0;
|
432
|
+
|
433
|
+
// 3. If numberOfArgs = 0, then
|
434
|
+
if (numberOfArgs == 0) {
|
435
|
+
// a. Let dv be the time value (UTC) identifying the current time.
|
436
|
+
dv = __Date_now();
|
437
|
+
} else if (numberOfArgs == 1) {
|
438
|
+
// 4. Else if numberOfArgs = 1, the n
|
439
|
+
// a. Let value be values[0].
|
440
|
+
const value: any = v0;
|
441
|
+
|
442
|
+
const valueType: i32 = Porffor.rawType(v0);
|
443
|
+
|
444
|
+
let tv: number = 0;
|
445
|
+
|
446
|
+
// b. If value is an Object and value has a [[DateValue]] internal slot, then
|
447
|
+
if (valueType == Porffor.TYPES._date) {
|
448
|
+
// i. Let tv be value.[[DateValue]].
|
449
|
+
tv = __Porffor_date_read(value);
|
450
|
+
} else {
|
451
|
+
// c. Else,
|
452
|
+
// ii. If v is a String, then
|
453
|
+
if (valueType == Porffor.TYPES.string || valueType == Porffor.TYPES._bytestring) {
|
454
|
+
// 1. Assert: The next step never returns an abrupt completion because v is a String.
|
455
|
+
|
456
|
+
// 2. Let tv be the result of parsing v as a date, in exactly the same manner as for the parse method (21.4.3.2).
|
457
|
+
// todo
|
458
|
+
} else {
|
459
|
+
// iii. Else,
|
460
|
+
// 1. Let tv be ? ToNumber(v).
|
461
|
+
tv = Number(value);
|
462
|
+
}
|
463
|
+
}
|
464
|
+
|
465
|
+
// d. Let dv be TimeClip(tv).
|
466
|
+
dv = __ecma262_TimeClip(tv);
|
467
|
+
} else {
|
468
|
+
// 5. Else,
|
469
|
+
// a. Assert: numberOfArgs ≥ 2.
|
470
|
+
|
471
|
+
// b. Let y be ? ToNumber(values[0]).
|
472
|
+
const y: number = Number(v0);
|
473
|
+
|
474
|
+
// c. Let m be ? ToNumber(values[1]).
|
475
|
+
const m: number = Number(v1);
|
476
|
+
|
477
|
+
// d. If numberOfArgs > 2, let dt be ? ToNumber(values[2]); else let dt be 1𝔽.
|
478
|
+
let dt: number = 1;
|
479
|
+
if (numberOfArgs > 2) dt = Number(v2);
|
480
|
+
|
481
|
+
// e. If numberOfArgs > 3, let h be ? ToNumber(values[3]); else let h be +0𝔽.
|
482
|
+
let h: number = 0;
|
483
|
+
if (numberOfArgs > 3) h = Number(v3);
|
484
|
+
|
485
|
+
// f. If numberOfArgs > 4, let min be ? ToNumber(values[4]); else let min be +0𝔽.
|
486
|
+
let min: number = 0;
|
487
|
+
if (numberOfArgs > 4) min = Number(v4);
|
488
|
+
|
489
|
+
// g. If numberOfArgs > 5, let s be ? ToNumber(values[5]); else let s be +0𝔽.
|
490
|
+
let s: number = 0;
|
491
|
+
if (numberOfArgs > 5) s = Number(v5);
|
492
|
+
|
493
|
+
// h. If numberOfArgs > 6, let milli be ? ToNumber(values[6]); else let milli be +0𝔽.
|
494
|
+
let milli: number = 0;
|
495
|
+
if (numberOfArgs > 6) milli = Number(v6);
|
496
|
+
|
497
|
+
// i. Let yr be MakeFullYear(y).
|
498
|
+
const yr: number = __ecma262_MakeFullYear(y);
|
499
|
+
|
500
|
+
// j. Let finalDate be MakeDate(MakeDay(yr, m, dt), MakeTime(h, min, s, milli)).
|
501
|
+
const finalDate: number = __ecma262_MakeDate(__ecma262_MakeDay(yr, m, dt), __ecma262_MakeTime(h, min, s, milli));
|
502
|
+
|
503
|
+
// k. Let dv be TimeClip(UTC(finalDate)).
|
504
|
+
dv = __ecma262_TimeClip(__ecma262_UTC(finalDate));
|
505
|
+
}
|
506
|
+
|
507
|
+
// 6. Let O be ? OrdinaryCreateFromConstructor(NewTarget, "%Date.prototype%", « [[DateValue]] »).
|
508
|
+
const O: Date = __Porffor_date_allocate();
|
509
|
+
|
510
|
+
// 7. Set O.[[DateValue]] to dv.
|
511
|
+
__Porffor_date_write(O, dv);
|
512
|
+
|
513
|
+
// 8. Return O.
|
514
|
+
return O;
|
515
|
+
};
|
516
|
+
|
517
|
+
|
518
|
+
// 21.4.3.1 Date.now ()
|
519
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-date.now
|
520
|
+
// This function returns the time value designating the UTC date and time of the occurrence of the call to it.
|
521
|
+
export const __Date_now = (): number => Math.trunc(performance.timeOrigin + performance.now());
|
522
|
+
|
523
|
+
// 21.4.3.2 Date.parse (string)
|
524
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-date.parse
|
525
|
+
// todo
|
526
|
+
|
527
|
+
// 21.4.3.4 Date.UTC (year [, month [, date [, hours [, minutes [, seconds [, ms ]]]]]])
|
528
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-date.utc
|
529
|
+
export const __Date_UTC = (year: unknown, month: unknown, date: unknown, hours: unknown, minutes: unknown, seconds: unknown, ms: unknown): number => {
|
530
|
+
// todo: passing undefined to params should not act like no arg was passed
|
531
|
+
|
532
|
+
// 1. Let y be ? ToNumber(year).
|
533
|
+
const y: number = Number(year);
|
534
|
+
|
535
|
+
// 2. If month is present, let m be ? ToNumber(month); else let m be +0𝔽.
|
536
|
+
let m: number = 0;
|
537
|
+
if (Porffor.rawType(month) != Porffor.TYPES.undefined) m = Number(month);
|
538
|
+
|
539
|
+
// 3. If date is present, let dt be ? ToNumber(date); else let dt be 1𝔽.
|
540
|
+
let dt: number = 1;
|
541
|
+
if (Porffor.rawType(date) != Porffor.TYPES.undefined) dt = Number(date);
|
542
|
+
|
543
|
+
// 4. If hours is present, let h be ? ToNumber(hours); else let h be +0𝔽.
|
544
|
+
let h: number = 0;
|
545
|
+
if (Porffor.rawType(hours) != Porffor.TYPES.undefined) h = Number(hours);
|
546
|
+
|
547
|
+
// 5. If minutes is present, let min be ? ToNumber(minutes); else let min be +0𝔽.
|
548
|
+
let min: number = 0;
|
549
|
+
if (Porffor.rawType(minutes) != Porffor.TYPES.undefined) min = Number(minutes);
|
550
|
+
|
551
|
+
// 6. If seconds is present, let s be ? ToNumber(seconds); else let s be +0𝔽.
|
552
|
+
let s: number = 0;
|
553
|
+
if (Porffor.rawType(seconds) != Porffor.TYPES.undefined) s = Number(seconds);
|
554
|
+
|
555
|
+
// 7. If ms is present, let milli be ? ToNumber(ms); else let milli be +0𝔽.
|
556
|
+
let milli: number = 0;
|
557
|
+
if (Porffor.rawType(ms) != Porffor.TYPES.undefined) h = Number(ms);
|
558
|
+
|
559
|
+
// 8. Let yr be MakeFullYear(y).
|
560
|
+
const yr: number = __ecma262_MakeFullYear(y);
|
561
|
+
|
562
|
+
// 9. Return TimeClip(MakeDate(MakeDay(yr, m, dt), MakeTime(h, min, s, milli))).
|
563
|
+
return __ecma262_TimeClip(__ecma262_MakeDate(__ecma262_MakeDay(yr, m, dt), __ecma262_MakeTime(h, min, s, milli)));
|
564
|
+
};
|
565
|
+
|
566
|
+
// 21.4.4 Properties of the Date Prototype Object
|
567
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-properties-of-the-date-prototype-object
|
568
|
+
|
569
|
+
// 21.4.4.2 Date.prototype.getDate ()
|
570
|
+
// https://tc39.es/ecma262/multipage/numbers-and-dates.html#sec-date.prototype.getdate
|
571
|
+
export const ___date_prototype_getDate = (_this: Date) => {
|
572
|
+
// 1. Let dateObject be the this value.
|
573
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
574
|
+
// 3. Let t be dateObject.[[DateValue]].
|
575
|
+
const t: number = __Porffor_date_read(_this);
|
576
|
+
|
577
|
+
// 4. If t is NaN, return NaN.
|
578
|
+
if (Number.isNaN(t)) return NaN;
|
579
|
+
|
580
|
+
// 5. Return DateFromTime(LocalTime(t)).
|
581
|
+
return __ecma262_DateFromTime(__ecma262_LocalTime(t));
|
582
|
+
};
|
583
|
+
|
584
|
+
// 21.4.4.3 Date.prototype.getDay ()
|
585
|
+
|
586
|
+
|
587
|
+
// 1. Let dateObject be the this value.
|
588
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
589
|
+
// 3. Let t be dateObject.[[DateValue]].
|
590
|
+
// 4. If t is NaN, return NaN.
|
591
|
+
// 5. Return WeekDay(LocalTime(t)).
|
592
|
+
|
593
|
+
// 21.4.4.4 Date.prototype.getFullYear ()
|
594
|
+
|
595
|
+
|
596
|
+
// 1. Let dateObject be the this value.
|
597
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
598
|
+
// 3. Let t be dateObject.[[DateValue]].
|
599
|
+
// 4. If t is NaN, return NaN.
|
600
|
+
// 5. Return YearFromTime(LocalTime(t)).
|
601
|
+
|
602
|
+
// 21.4.4.5 Date.prototype.getHours ()
|
603
|
+
|
604
|
+
|
605
|
+
// 1. Let dateObject be the this value.
|
606
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
607
|
+
// 3. Let t be dateObject.[[DateValue]].
|
608
|
+
// 4. If t is NaN, return NaN.
|
609
|
+
// 5. Return HourFromTime(LocalTime(t)).
|
610
|
+
|
611
|
+
// 21.4.4.6 Date.prototype.getMilliseconds ()
|
612
|
+
|
613
|
+
|
614
|
+
// 1. Let dateObject be the this value.
|
615
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
616
|
+
// 3. Let t be dateObject.[[DateValue]].
|
617
|
+
// 4. If t is NaN, return NaN.
|
618
|
+
// 5. Return msFromTime(LocalTime(t)).
|
619
|
+
|
620
|
+
// 21.4.4.7 Date.prototype.getMinutes ()
|
621
|
+
|
622
|
+
|
623
|
+
// 1. Let dateObject be the this value.
|
624
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
625
|
+
// 3. Let t be dateObject.[[DateValue]].
|
626
|
+
// 4. If t is NaN, return NaN.
|
627
|
+
// 5. Return MinFromTime(LocalTime(t)).
|
628
|
+
|
629
|
+
// 21.4.4.8 Date.prototype.getMonth ()
|
630
|
+
|
631
|
+
|
632
|
+
// 1. Let dateObject be the this value.
|
633
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
634
|
+
// 3. Let t be dateObject.[[DateValue]].
|
635
|
+
// 4. If t is NaN, return NaN.
|
636
|
+
// 5. Return MonthFromTime(LocalTime(t)).
|
637
|
+
|
638
|
+
// 21.4.4.9 Date.prototype.getSeconds ()
|
639
|
+
|
640
|
+
|
641
|
+
// 1. Let dateObject be the this value.
|
642
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
643
|
+
// 3. Let t be dateObject.[[DateValue]].
|
644
|
+
// 4. If t is NaN, return NaN.
|
645
|
+
// 5. Return SecFromTime(LocalTime(t)).
|
646
|
+
|
647
|
+
// 21.4.4.10 Date.prototype.getTime ()
|
648
|
+
|
649
|
+
|
650
|
+
// 1. Let dateObject be the this value.
|
651
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
652
|
+
// 3. Return dateObject.[[DateValue]].
|
653
|
+
|
654
|
+
// 21.4.4.11 Date.prototype.getTimezoneOffset ()
|
655
|
+
|
656
|
+
|
657
|
+
// 1. Let dateObject be the this value.
|
658
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
659
|
+
// 3. Let t be dateObject.[[DateValue]].
|
660
|
+
// 4. If t is NaN, return NaN.
|
661
|
+
// 5. Return (t - LocalTime(t)) / msPerMinute.
|
662
|
+
|
663
|
+
// 21.4.4.12 Date.prototype.getUTCDate ()
|
664
|
+
|
665
|
+
|
666
|
+
// 1. Let dateObject be the this value.
|
667
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
668
|
+
// 3. Let t be dateObject.[[DateValue]].
|
669
|
+
// 4. If t is NaN, return NaN.
|
670
|
+
// 5. Return DateFromTime(t).
|
671
|
+
|
672
|
+
// 21.4.4.13 Date.prototype.getUTCDay ()
|
673
|
+
|
674
|
+
|
675
|
+
// 1. Let dateObject be the this value.
|
676
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
677
|
+
// 3. Let t be dateObject.[[DateValue]].
|
678
|
+
// 4. If t is NaN, return NaN.
|
679
|
+
// 5. Return WeekDay(t).
|
680
|
+
|
681
|
+
// 21.4.4.14 Date.prototype.getUTCFullYear ()
|
682
|
+
|
683
|
+
|
684
|
+
// 1. Let dateObject be the this value.
|
685
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
686
|
+
// 3. Let t be dateObject.[[DateValue]].
|
687
|
+
// 4. If t is NaN, return NaN.
|
688
|
+
// 5. Return YearFromTime(t).
|
689
|
+
|
690
|
+
// 21.4.4.15 Date.prototype.getUTCHours ()
|
691
|
+
|
692
|
+
|
693
|
+
// 1. Let dateObject be the this value.
|
694
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
695
|
+
// 3. Let t be dateObject.[[DateValue]].
|
696
|
+
// 4. If t is NaN, return NaN.
|
697
|
+
// 5. Return HourFromTime(t).
|
698
|
+
|
699
|
+
// 21.4.4.16 Date.prototype.getUTCMilliseconds ()
|
700
|
+
|
701
|
+
|
702
|
+
// 1. Let dateObject be the this value.
|
703
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
704
|
+
// 3. Let t be dateObject.[[DateValue]].
|
705
|
+
// 4. If t is NaN, return NaN.
|
706
|
+
// 5. Return msFromTime(t).
|
707
|
+
|
708
|
+
// 21.4.4.17 Date.prototype.getUTCMinutes ()
|
709
|
+
|
710
|
+
|
711
|
+
// 1. Let dateObject be the this value.
|
712
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
713
|
+
// 3. Let t be dateObject.[[DateValue]].
|
714
|
+
// 4. If t is NaN, return NaN.
|
715
|
+
// 5. Return MinFromTime(t).
|
716
|
+
|
717
|
+
// 21.4.4.18 Date.prototype.getUTCMonth ()
|
718
|
+
|
719
|
+
|
720
|
+
// 1. Let dateObject be the this value.
|
721
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
722
|
+
// 3. Let t be dateObject.[[DateValue]].
|
723
|
+
// 4. If t is NaN, return NaN.
|
724
|
+
// 5. Return MonthFromTime(t).
|
725
|
+
|
726
|
+
// 21.4.4.19 Date.prototype.getUTCSeconds ()
|
727
|
+
|
728
|
+
|
729
|
+
// 1. Let dateObject be the this value.
|
730
|
+
// 2. Perform ? RequireInternalSlot(dateObject, [[DateValue]]).
|
731
|
+
// 3. Let t be dateObject.[[DateValue]].
|
732
|
+
// 4. If t is NaN, return NaN.
|
733
|
+
// 5. Return SecFromTime(t).
|
734
|
+
|