// Copyright 2016 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "src/builtins/builtins-utils-inl.h" #include "src/builtins/builtins.h" #include "src/codegen/code-factory.h" #include "src/date/date.h" #include "src/date/dateparser-inl.h" #include "src/logging/counters.h" #include "src/numbers/conversions.h" #include "src/objects/objects-inl.h" #ifdef V8_INTL_SUPPORT #include "src/objects/intl-objects.h" #include "src/objects/js-date-time-format.h" #endif #include "src/strings/string-stream.h" namespace v8 { namespace internal { // ----------------------------------------------------------------------------- // ES6 section 20.3 Date Objects namespace { // ES6 section 20.3.1.1 Time Values and Time Range const double kMinYear = -1000000.0; const double kMaxYear = -kMinYear; const double kMinMonth = -10000000.0; const double kMaxMonth = -kMinMonth; // 20.3.1.2 Day Number and Time within Day const double kMsPerDay = 86400000.0; // ES6 section 20.3.1.11 Hours, Minutes, Second, and Milliseconds const double kMsPerSecond = 1000.0; const double kMsPerMinute = 60000.0; const double kMsPerHour = 3600000.0; // ES6 section 20.3.1.14 MakeDate (day, time) double MakeDate(double day, double time) { if (std::isfinite(day) && std::isfinite(time)) { return time + day * kMsPerDay; } return std::numeric_limits::quiet_NaN(); } // ES6 section 20.3.1.13 MakeDay (year, month, date) double MakeDay(double year, double month, double date) { if ((kMinYear <= year && year <= kMaxYear) && (kMinMonth <= month && month <= kMaxMonth) && std::isfinite(date)) { int y = FastD2I(year); int m = FastD2I(month); y += m / 12; m %= 12; if (m < 0) { m += 12; y -= 1; } DCHECK_LE(0, m); DCHECK_LT(m, 12); // kYearDelta is an arbitrary number such that: // a) kYearDelta = -1 (mod 400) // b) year + kYearDelta > 0 for years in the range defined by // ECMA 262 - 15.9.1.1, i.e. upto 100,000,000 days on either side of // Jan 1 1970. This is required so that we don't run into integer // division of negative numbers. // c) there shouldn't be an overflow for 32-bit integers in the following // operations. static const int kYearDelta = 399999; static const int kBaseDay = 365 * (1970 + kYearDelta) + (1970 + kYearDelta) / 4 - (1970 + kYearDelta) / 100 + (1970 + kYearDelta) / 400; int day_from_year = 365 * (y + kYearDelta) + (y + kYearDelta) / 4 - (y + kYearDelta) / 100 + (y + kYearDelta) / 400 - kBaseDay; if ((y % 4 != 0) || (y % 100 == 0 && y % 400 != 0)) { static const int kDayFromMonth[] = {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334}; day_from_year += kDayFromMonth[m]; } else { static const int kDayFromMonth[] = {0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335}; day_from_year += kDayFromMonth[m]; } return static_cast(day_from_year - 1) + DoubleToInteger(date); } return std::numeric_limits::quiet_NaN(); } // ES6 section 20.3.1.12 MakeTime (hour, min, sec, ms) double MakeTime(double hour, double min, double sec, double ms) { if (std::isfinite(hour) && std::isfinite(min) && std::isfinite(sec) && std::isfinite(ms)) { double const h = DoubleToInteger(hour); double const m = DoubleToInteger(min); double const s = DoubleToInteger(sec); double const milli = DoubleToInteger(ms); return h * kMsPerHour + m * kMsPerMinute + s * kMsPerSecond + milli; } return std::numeric_limits::quiet_NaN(); } const char* kShortWeekDays[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; const char* kShortMonths[] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"}; // ES6 section 20.3.1.16 Date Time String Format double ParseDateTimeString(Isolate* isolate, Handle str) { str = String::Flatten(isolate, str); double out[DateParser::OUTPUT_SIZE]; DisallowHeapAllocation no_gc; String::FlatContent str_content = str->GetFlatContent(no_gc); bool result; if (str_content.IsOneByte()) { result = DateParser::Parse(isolate, str_content.ToOneByteVector(), out); } else { result = DateParser::Parse(isolate, str_content.ToUC16Vector(), out); } if (!result) return std::numeric_limits::quiet_NaN(); double const day = MakeDay(out[DateParser::YEAR], out[DateParser::MONTH], out[DateParser::DAY]); double const time = MakeTime(out[DateParser::HOUR], out[DateParser::MINUTE], out[DateParser::SECOND], out[DateParser::MILLISECOND]); double date = MakeDate(day, time); if (std::isnan(out[DateParser::UTC_OFFSET])) { if (date >= -DateCache::kMaxTimeBeforeUTCInMs && date <= DateCache::kMaxTimeBeforeUTCInMs) { date = isolate->date_cache()->ToUTC(static_cast(date)); } else { return std::numeric_limits::quiet_NaN(); } } else { date -= out[DateParser::UTC_OFFSET] * 1000.0; } return DateCache::TimeClip(date); } enum ToDateStringMode { kDateOnly, kTimeOnly, kDateAndTime }; using DateBuffer = base::SmallVector; template DateBuffer FormatDate(const char* format, Args... args) { DateBuffer buffer; SmallStringOptimizedAllocator allocator(&buffer); StringStream sstream(&allocator); sstream.Add(format, args...); buffer.resize_no_init(sstream.length()); return buffer; } // ES6 section 20.3.4.41.1 ToDateString(tv) DateBuffer ToDateString(double time_val, DateCache* date_cache, ToDateStringMode mode = kDateAndTime) { if (std::isnan(time_val)) { return FormatDate("Invalid Date"); } int64_t time_ms = static_cast(time_val); int64_t local_time_ms = date_cache->ToLocal(time_ms); int year, month, day, weekday, hour, min, sec, ms; date_cache->BreakDownTime(local_time_ms, &year, &month, &day, &weekday, &hour, &min, &sec, &ms); int timezone_offset = -date_cache->TimezoneOffset(time_ms); int timezone_hour = std::abs(timezone_offset) / 60; int timezone_min = std::abs(timezone_offset) % 60; const char* local_timezone = date_cache->LocalTimezone(time_ms); switch (mode) { case kDateOnly: return FormatDate((year < 0) ? "%s %s %02d %05d" : "%s %s %02d %04d", kShortWeekDays[weekday], kShortMonths[month], day, year); case kTimeOnly: return FormatDate("%02d:%02d:%02d GMT%c%02d%02d (%s)", hour, min, sec, (timezone_offset < 0) ? '-' : '+', timezone_hour, timezone_min, local_timezone); case kDateAndTime: return FormatDate( (year < 0) ? "%s %s %02d %05d %02d:%02d:%02d GMT%c%02d%02d (%s)" : "%s %s %02d %04d %02d:%02d:%02d GMT%c%02d%02d (%s)", kShortWeekDays[weekday], kShortMonths[month], day, year, hour, min, sec, (timezone_offset < 0) ? '-' : '+', timezone_hour, timezone_min, local_timezone); } UNREACHABLE(); } Object SetLocalDateValue(Isolate* isolate, Handle date, double time_val) { if (time_val >= -DateCache::kMaxTimeBeforeUTCInMs && time_val <= DateCache::kMaxTimeBeforeUTCInMs) { time_val = isolate->date_cache()->ToUTC(static_cast(time_val)); } else { time_val = std::numeric_limits::quiet_NaN(); } return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } } // namespace // ES #sec-date-constructor BUILTIN(DateConstructor) { HandleScope scope(isolate); if (args.new_target()->IsUndefined(isolate)) { double const time_val = JSDate::CurrentTimeValue(isolate); DateBuffer buffer = ToDateString(time_val, isolate->date_cache()); RETURN_RESULT_OR_FAILURE( isolate, isolate->factory()->NewStringFromUtf8(VectorOf(buffer))); } // [Construct] int const argc = args.length() - 1; Handle target = args.target(); Handle new_target = Handle::cast(args.new_target()); double time_val; if (argc == 0) { time_val = JSDate::CurrentTimeValue(isolate); } else if (argc == 1) { Handle value = args.at(1); if (value->IsJSDate()) { time_val = Handle::cast(value)->value().Number(); } else { ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, value, Object::ToPrimitive(value)); if (value->IsString()) { time_val = ParseDateTimeString(isolate, Handle::cast(value)); } else { ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, value, Object::ToNumber(isolate, value)); time_val = value->Number(); } } } else { Handle year_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, year_object, Object::ToNumber(isolate, args.at(1))); Handle month_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, month_object, Object::ToNumber(isolate, args.at(2))); double year = year_object->Number(); double month = month_object->Number(); double date = 1.0, hours = 0.0, minutes = 0.0, seconds = 0.0, ms = 0.0; if (argc >= 3) { Handle date_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, date_object, Object::ToNumber(isolate, args.at(3))); date = date_object->Number(); if (argc >= 4) { Handle hours_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, hours_object, Object::ToNumber(isolate, args.at(4))); hours = hours_object->Number(); if (argc >= 5) { Handle minutes_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, minutes_object, Object::ToNumber(isolate, args.at(5))); minutes = minutes_object->Number(); if (argc >= 6) { Handle seconds_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, seconds_object, Object::ToNumber(isolate, args.at(6))); seconds = seconds_object->Number(); if (argc >= 7) { Handle ms_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, ms_object, Object::ToNumber(isolate, args.at(7))); ms = ms_object->Number(); } } } } } if (!std::isnan(year)) { double const y = DoubleToInteger(year); if (0.0 <= y && y <= 99) year = 1900 + y; } double const day = MakeDay(year, month, date); double const time = MakeTime(hours, minutes, seconds, ms); time_val = MakeDate(day, time); if (time_val >= -DateCache::kMaxTimeBeforeUTCInMs && time_val <= DateCache::kMaxTimeBeforeUTCInMs) { time_val = isolate->date_cache()->ToUTC(static_cast(time_val)); } else { time_val = std::numeric_limits::quiet_NaN(); } } RETURN_RESULT_OR_FAILURE(isolate, JSDate::New(target, new_target, time_val)); } // ES6 section 20.3.3.1 Date.now ( ) BUILTIN(DateNow) { HandleScope scope(isolate); return *isolate->factory()->NewNumber(JSDate::CurrentTimeValue(isolate)); } // ES6 section 20.3.3.2 Date.parse ( string ) BUILTIN(DateParse) { HandleScope scope(isolate); Handle string; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, string, Object::ToString(isolate, args.atOrUndefined(isolate, 1))); return *isolate->factory()->NewNumber(ParseDateTimeString(isolate, string)); } // ES6 section 20.3.3.4 Date.UTC (year,month,date,hours,minutes,seconds,ms) BUILTIN(DateUTC) { HandleScope scope(isolate); int const argc = args.length() - 1; double year = std::numeric_limits::quiet_NaN(); double month = 0.0, date = 1.0, hours = 0.0, minutes = 0.0, seconds = 0.0, ms = 0.0; if (argc >= 1) { Handle year_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, year_object, Object::ToNumber(isolate, args.at(1))); year = year_object->Number(); if (argc >= 2) { Handle month_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, month_object, Object::ToNumber(isolate, args.at(2))); month = month_object->Number(); if (argc >= 3) { Handle date_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, date_object, Object::ToNumber(isolate, args.at(3))); date = date_object->Number(); if (argc >= 4) { Handle hours_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, hours_object, Object::ToNumber(isolate, args.at(4))); hours = hours_object->Number(); if (argc >= 5) { Handle minutes_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, minutes_object, Object::ToNumber(isolate, args.at(5))); minutes = minutes_object->Number(); if (argc >= 6) { Handle seconds_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, seconds_object, Object::ToNumber(isolate, args.at(6))); seconds = seconds_object->Number(); if (argc >= 7) { Handle ms_object; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, ms_object, Object::ToNumber(isolate, args.at(7))); ms = ms_object->Number(); } } } } } } } if (!std::isnan(year)) { double const y = DoubleToInteger(year); if (0.0 <= y && y <= 99) year = 1900 + y; } double const day = MakeDay(year, month, date); double const time = MakeTime(hours, minutes, seconds, ms); return *isolate->factory()->NewNumber( DateCache::TimeClip(MakeDate(day, time))); } // ES6 section 20.3.4.20 Date.prototype.setDate ( date ) BUILTIN(DatePrototypeSetDate) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setDate"); Handle value = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, value, Object::ToNumber(isolate, value)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int const days = isolate->date_cache()->DaysFromTime(local_time_ms); int time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, days); int year, month, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &month, &day); time_val = MakeDate(MakeDay(year, month, value->Number()), time_within_day); } return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.21 Date.prototype.setFullYear (year, month, date) BUILTIN(DatePrototypeSetFullYear) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setFullYear"); int const argc = args.length() - 1; Handle year = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, year, Object::ToNumber(isolate, year)); double y = year->Number(), m = 0.0, dt = 1.0; int time_within_day = 0; if (!std::isnan(date->value().Number())) { int64_t const time_ms = static_cast(date->value().Number()); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int const days = isolate->date_cache()->DaysFromTime(local_time_ms); time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, days); int year, month, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &month, &day); m = month; dt = day; } if (argc >= 2) { Handle month = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, month, Object::ToNumber(isolate, month)); m = month->Number(); if (argc >= 3) { Handle date = args.at(3); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, date, Object::ToNumber(isolate, date)); dt = date->Number(); } } double time_val = MakeDate(MakeDay(y, m, dt), time_within_day); return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.22 Date.prototype.setHours(hour, min, sec, ms) BUILTIN(DatePrototypeSetHours) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setHours"); int const argc = args.length() - 1; Handle hour = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, hour, Object::ToNumber(isolate, hour)); double h = hour->Number(); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int day = isolate->date_cache()->DaysFromTime(local_time_ms); int time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, day); double m = (time_within_day / (60 * 1000)) % 60; double s = (time_within_day / 1000) % 60; double milli = time_within_day % 1000; if (argc >= 2) { Handle min = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, min, Object::ToNumber(isolate, min)); m = min->Number(); if (argc >= 3) { Handle sec = args.at(3); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, sec, Object::ToNumber(isolate, sec)); s = sec->Number(); if (argc >= 4) { Handle ms = args.at(4); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); milli = ms->Number(); } } } time_val = MakeDate(day, MakeTime(h, m, s, milli)); } return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.23 Date.prototype.setMilliseconds(ms) BUILTIN(DatePrototypeSetMilliseconds) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setMilliseconds"); Handle ms = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int day = isolate->date_cache()->DaysFromTime(local_time_ms); int time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, day); int h = time_within_day / (60 * 60 * 1000); int m = (time_within_day / (60 * 1000)) % 60; int s = (time_within_day / 1000) % 60; time_val = MakeDate(day, MakeTime(h, m, s, ms->Number())); } return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.24 Date.prototype.setMinutes ( min, sec, ms ) BUILTIN(DatePrototypeSetMinutes) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setMinutes"); int const argc = args.length() - 1; Handle min = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, min, Object::ToNumber(isolate, min)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int day = isolate->date_cache()->DaysFromTime(local_time_ms); int time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, day); int h = time_within_day / (60 * 60 * 1000); double m = min->Number(); double s = (time_within_day / 1000) % 60; double milli = time_within_day % 1000; if (argc >= 2) { Handle sec = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, sec, Object::ToNumber(isolate, sec)); s = sec->Number(); if (argc >= 3) { Handle ms = args.at(3); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); milli = ms->Number(); } } time_val = MakeDate(day, MakeTime(h, m, s, milli)); } return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.25 Date.prototype.setMonth ( month, date ) BUILTIN(DatePrototypeSetMonth) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setMonth"); int const argc = args.length() - 1; Handle month = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, month, Object::ToNumber(isolate, month)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int days = isolate->date_cache()->DaysFromTime(local_time_ms); int time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, days); int year, unused, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &unused, &day); double m = month->Number(); double dt = day; if (argc >= 2) { Handle date = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, date, Object::ToNumber(isolate, date)); dt = date->Number(); } time_val = MakeDate(MakeDay(year, m, dt), time_within_day); } return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.26 Date.prototype.setSeconds ( sec, ms ) BUILTIN(DatePrototypeSetSeconds) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setSeconds"); int const argc = args.length() - 1; Handle sec = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, sec, Object::ToNumber(isolate, sec)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int day = isolate->date_cache()->DaysFromTime(local_time_ms); int time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, day); int h = time_within_day / (60 * 60 * 1000); double m = (time_within_day / (60 * 1000)) % 60; double s = sec->Number(); double milli = time_within_day % 1000; if (argc >= 2) { Handle ms = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); milli = ms->Number(); } time_val = MakeDate(day, MakeTime(h, m, s, milli)); } return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.27 Date.prototype.setTime ( time ) BUILTIN(DatePrototypeSetTime) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setTime"); Handle value = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, value, Object::ToNumber(isolate, value)); return *JSDate::SetValue(date, DateCache::TimeClip(value->Number())); } // ES6 section 20.3.4.28 Date.prototype.setUTCDate ( date ) BUILTIN(DatePrototypeSetUTCDate) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setUTCDate"); Handle value = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, value, Object::ToNumber(isolate, value)); if (std::isnan(date->value().Number())) return date->value(); int64_t const time_ms = static_cast(date->value().Number()); int const days = isolate->date_cache()->DaysFromTime(time_ms); int const time_within_day = isolate->date_cache()->TimeInDay(time_ms, days); int year, month, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &month, &day); double const time_val = MakeDate(MakeDay(year, month, value->Number()), time_within_day); return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } // ES6 section 20.3.4.29 Date.prototype.setUTCFullYear (year, month, date) BUILTIN(DatePrototypeSetUTCFullYear) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setUTCFullYear"); int const argc = args.length() - 1; Handle year = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, year, Object::ToNumber(isolate, year)); double y = year->Number(), m = 0.0, dt = 1.0; int time_within_day = 0; if (!std::isnan(date->value().Number())) { int64_t const time_ms = static_cast(date->value().Number()); int const days = isolate->date_cache()->DaysFromTime(time_ms); time_within_day = isolate->date_cache()->TimeInDay(time_ms, days); int year, month, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &month, &day); m = month; dt = day; } if (argc >= 2) { Handle month = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, month, Object::ToNumber(isolate, month)); m = month->Number(); if (argc >= 3) { Handle date = args.at(3); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, date, Object::ToNumber(isolate, date)); dt = date->Number(); } } double const time_val = MakeDate(MakeDay(y, m, dt), time_within_day); return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } // ES6 section 20.3.4.30 Date.prototype.setUTCHours(hour, min, sec, ms) BUILTIN(DatePrototypeSetUTCHours) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setUTCHours"); int const argc = args.length() - 1; Handle hour = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, hour, Object::ToNumber(isolate, hour)); double h = hour->Number(); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int day = isolate->date_cache()->DaysFromTime(time_ms); int time_within_day = isolate->date_cache()->TimeInDay(time_ms, day); double m = (time_within_day / (60 * 1000)) % 60; double s = (time_within_day / 1000) % 60; double milli = time_within_day % 1000; if (argc >= 2) { Handle min = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, min, Object::ToNumber(isolate, min)); m = min->Number(); if (argc >= 3) { Handle sec = args.at(3); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, sec, Object::ToNumber(isolate, sec)); s = sec->Number(); if (argc >= 4) { Handle ms = args.at(4); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); milli = ms->Number(); } } } time_val = MakeDate(day, MakeTime(h, m, s, milli)); } return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } // ES6 section 20.3.4.31 Date.prototype.setUTCMilliseconds(ms) BUILTIN(DatePrototypeSetUTCMilliseconds) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setUTCMilliseconds"); Handle ms = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int day = isolate->date_cache()->DaysFromTime(time_ms); int time_within_day = isolate->date_cache()->TimeInDay(time_ms, day); int h = time_within_day / (60 * 60 * 1000); int m = (time_within_day / (60 * 1000)) % 60; int s = (time_within_day / 1000) % 60; time_val = MakeDate(day, MakeTime(h, m, s, ms->Number())); } return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } // ES6 section 20.3.4.32 Date.prototype.setUTCMinutes ( min, sec, ms ) BUILTIN(DatePrototypeSetUTCMinutes) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setUTCMinutes"); int const argc = args.length() - 1; Handle min = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, min, Object::ToNumber(isolate, min)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int day = isolate->date_cache()->DaysFromTime(time_ms); int time_within_day = isolate->date_cache()->TimeInDay(time_ms, day); int h = time_within_day / (60 * 60 * 1000); double m = min->Number(); double s = (time_within_day / 1000) % 60; double milli = time_within_day % 1000; if (argc >= 2) { Handle sec = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, sec, Object::ToNumber(isolate, sec)); s = sec->Number(); if (argc >= 3) { Handle ms = args.at(3); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); milli = ms->Number(); } } time_val = MakeDate(day, MakeTime(h, m, s, milli)); } return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } // ES6 section 20.3.4.31 Date.prototype.setUTCMonth ( month, date ) BUILTIN(DatePrototypeSetUTCMonth) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setUTCMonth"); int const argc = args.length() - 1; Handle month = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, month, Object::ToNumber(isolate, month)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int days = isolate->date_cache()->DaysFromTime(time_ms); int time_within_day = isolate->date_cache()->TimeInDay(time_ms, days); int year, unused, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &unused, &day); double m = month->Number(); double dt = day; if (argc >= 2) { Handle date = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, date, Object::ToNumber(isolate, date)); dt = date->Number(); } time_val = MakeDate(MakeDay(year, m, dt), time_within_day); } return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } // ES6 section 20.3.4.34 Date.prototype.setUTCSeconds ( sec, ms ) BUILTIN(DatePrototypeSetUTCSeconds) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setUTCSeconds"); int const argc = args.length() - 1; Handle sec = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, sec, Object::ToNumber(isolate, sec)); double time_val = date->value().Number(); if (!std::isnan(time_val)) { int64_t const time_ms = static_cast(time_val); int day = isolate->date_cache()->DaysFromTime(time_ms); int time_within_day = isolate->date_cache()->TimeInDay(time_ms, day); int h = time_within_day / (60 * 60 * 1000); double m = (time_within_day / (60 * 1000)) % 60; double s = sec->Number(); double milli = time_within_day % 1000; if (argc >= 2) { Handle ms = args.at(2); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, ms, Object::ToNumber(isolate, ms)); milli = ms->Number(); } time_val = MakeDate(day, MakeTime(h, m, s, milli)); } return *JSDate::SetValue(date, DateCache::TimeClip(time_val)); } // ES6 section 20.3.4.35 Date.prototype.toDateString ( ) BUILTIN(DatePrototypeToDateString) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.toDateString"); DateBuffer buffer = ToDateString(date->value().Number(), isolate->date_cache(), kDateOnly); RETURN_RESULT_OR_FAILURE( isolate, isolate->factory()->NewStringFromUtf8(VectorOf(buffer))); } // ES6 section 20.3.4.36 Date.prototype.toISOString ( ) BUILTIN(DatePrototypeToISOString) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.toISOString"); double const time_val = date->value().Number(); if (std::isnan(time_val)) { THROW_NEW_ERROR_RETURN_FAILURE( isolate, NewRangeError(MessageTemplate::kInvalidTimeValue)); } int64_t const time_ms = static_cast(time_val); int year, month, day, weekday, hour, min, sec, ms; isolate->date_cache()->BreakDownTime(time_ms, &year, &month, &day, &weekday, &hour, &min, &sec, &ms); char buffer[128]; if (year >= 0 && year <= 9999) { SNPrintF(ArrayVector(buffer), "%04d-%02d-%02dT%02d:%02d:%02d.%03dZ", year, month + 1, day, hour, min, sec, ms); } else if (year < 0) { SNPrintF(ArrayVector(buffer), "-%06d-%02d-%02dT%02d:%02d:%02d.%03dZ", -year, month + 1, day, hour, min, sec, ms); } else { SNPrintF(ArrayVector(buffer), "+%06d-%02d-%02dT%02d:%02d:%02d.%03dZ", year, month + 1, day, hour, min, sec, ms); } return *isolate->factory()->NewStringFromAsciiChecked(buffer); } // ES6 section 20.3.4.41 Date.prototype.toString ( ) BUILTIN(DatePrototypeToString) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.toString"); DateBuffer buffer = ToDateString(date->value().Number(), isolate->date_cache()); RETURN_RESULT_OR_FAILURE( isolate, isolate->factory()->NewStringFromUtf8(VectorOf(buffer))); } // ES6 section 20.3.4.42 Date.prototype.toTimeString ( ) BUILTIN(DatePrototypeToTimeString) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.toTimeString"); DateBuffer buffer = ToDateString(date->value().Number(), isolate->date_cache(), kTimeOnly); RETURN_RESULT_OR_FAILURE( isolate, isolate->factory()->NewStringFromUtf8(VectorOf(buffer))); } #ifdef V8_INTL_SUPPORT // ecma402 #sup-date.prototype.tolocaledatestring BUILTIN(DatePrototypeToLocaleDateString) { HandleScope scope(isolate); isolate->CountUsage(v8::Isolate::UseCounterFeature::kDateToLocaleDateString); CHECK_RECEIVER(JSDate, date, "Date.prototype.toLocaleDateString"); RETURN_RESULT_OR_FAILURE( isolate, JSDateTimeFormat::ToLocaleDateTime( isolate, date, // date args.atOrUndefined(isolate, 1), // locales args.atOrUndefined(isolate, 2), // options JSDateTimeFormat::RequiredOption::kDate, // required JSDateTimeFormat::DefaultsOption::kDate)); // defaults } // ecma402 #sup-date.prototype.tolocalestring BUILTIN(DatePrototypeToLocaleString) { HandleScope scope(isolate); isolate->CountUsage(v8::Isolate::UseCounterFeature::kDateToLocaleString); CHECK_RECEIVER(JSDate, date, "Date.prototype.toLocaleString"); RETURN_RESULT_OR_FAILURE( isolate, JSDateTimeFormat::ToLocaleDateTime( isolate, date, // date args.atOrUndefined(isolate, 1), // locales args.atOrUndefined(isolate, 2), // options JSDateTimeFormat::RequiredOption::kAny, // required JSDateTimeFormat::DefaultsOption::kAll)); // defaults } // ecma402 #sup-date.prototype.tolocaletimestring BUILTIN(DatePrototypeToLocaleTimeString) { HandleScope scope(isolate); isolate->CountUsage(v8::Isolate::UseCounterFeature::kDateToLocaleTimeString); CHECK_RECEIVER(JSDate, date, "Date.prototype.toLocaleTimeString"); RETURN_RESULT_OR_FAILURE( isolate, JSDateTimeFormat::ToLocaleDateTime( isolate, date, // date args.atOrUndefined(isolate, 1), // locales args.atOrUndefined(isolate, 2), // options JSDateTimeFormat::RequiredOption::kTime, // required JSDateTimeFormat::DefaultsOption::kTime)); // defaults } #endif // V8_INTL_SUPPORT // ES6 section 20.3.4.43 Date.prototype.toUTCString ( ) BUILTIN(DatePrototypeToUTCString) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.toUTCString"); double const time_val = date->value().Number(); if (std::isnan(time_val)) { return *isolate->factory()->NewStringFromAsciiChecked("Invalid Date"); } char buffer[128]; int64_t time_ms = static_cast(time_val); int year, month, day, weekday, hour, min, sec, ms; isolate->date_cache()->BreakDownTime(time_ms, &year, &month, &day, &weekday, &hour, &min, &sec, &ms); SNPrintF(ArrayVector(buffer), (year < 0) ? "%s, %02d %s %05d %02d:%02d:%02d GMT" : "%s, %02d %s %04d %02d:%02d:%02d GMT", kShortWeekDays[weekday], day, kShortMonths[month], year, hour, min, sec); return *isolate->factory()->NewStringFromAsciiChecked(buffer); } // ES6 section B.2.4.1 Date.prototype.getYear ( ) BUILTIN(DatePrototypeGetYear) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.getYear"); double time_val = date->value().Number(); if (std::isnan(time_val)) return date->value(); int64_t time_ms = static_cast(time_val); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int days = isolate->date_cache()->DaysFromTime(local_time_ms); int year, month, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &month, &day); return Smi::FromInt(year - 1900); } // ES6 section B.2.4.2 Date.prototype.setYear ( year ) BUILTIN(DatePrototypeSetYear) { HandleScope scope(isolate); CHECK_RECEIVER(JSDate, date, "Date.prototype.setYear"); Handle year = args.atOrUndefined(isolate, 1); ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, year, Object::ToNumber(isolate, year)); double m = 0.0, dt = 1.0, y = year->Number(); if (!std::isnan(y)) { double y_int = DoubleToInteger(y); if (0.0 <= y_int && y_int <= 99.0) { y = 1900.0 + y_int; } } int time_within_day = 0; if (!std::isnan(date->value().Number())) { int64_t const time_ms = static_cast(date->value().Number()); int64_t local_time_ms = isolate->date_cache()->ToLocal(time_ms); int const days = isolate->date_cache()->DaysFromTime(local_time_ms); time_within_day = isolate->date_cache()->TimeInDay(local_time_ms, days); int year, month, day; isolate->date_cache()->YearMonthDayFromDays(days, &year, &month, &day); m = month; dt = day; } double time_val = MakeDate(MakeDay(y, m, dt), time_within_day); return SetLocalDateValue(isolate, date, time_val); } // ES6 section 20.3.4.37 Date.prototype.toJSON ( key ) BUILTIN(DatePrototypeToJson) { HandleScope scope(isolate); Handle receiver = args.atOrUndefined(isolate, 0); Handle receiver_obj; ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, receiver_obj, Object::ToObject(isolate, receiver)); Handle primitive; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, primitive, Object::ToPrimitive(receiver_obj, ToPrimitiveHint::kNumber)); if (primitive->IsNumber() && !std::isfinite(primitive->Number())) { return ReadOnlyRoots(isolate).null_value(); } else { Handle name = isolate->factory()->NewStringFromAsciiChecked("toISOString"); Handle function; ASSIGN_RETURN_FAILURE_ON_EXCEPTION( isolate, function, Object::GetProperty(isolate, receiver_obj, name)); if (!function->IsCallable()) { THROW_NEW_ERROR_RETURN_FAILURE( isolate, NewTypeError(MessageTemplate::kCalledNonCallable, name)); } RETURN_RESULT_OR_FAILURE( isolate, Execution::Call(isolate, function, receiver_obj, 0, nullptr)); } } } // namespace internal } // namespace v8