Project

General

Profile

Feature #4068 » stolen_base.patch

tadf (tadayoshi funaba), 11/27/2010 09:05 AM

View differences:

time.c (working copy)
rb_define_virtual_variable("$find_time_numguess", find_time_numguess_getter, NULL);
#endif
}
#if 1 /* stolen_base - written by tadayoshi funaba 2010 */
VALUE rb_cDate, rb_cDateTime;
static const int monthtab[2][13] = {
{ 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },
{ 0, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }
};
static int
last_day_of_month(VALUE y, int m)
{
return monthtab[leap_year_p(NUM2LONG(mod(y, INT2FIX(400)))) ? 0 : 1][m];
}
static VALUE
date_s_valid_civil_p(int argc, VALUE *argv, VALUE klass)
{
VALUE v[4];
VALUE y;
int m, d;
rb_scan_args(argc, argv, "31", &v[0],&v[1],&v[2],&v[3]);
y = obj2vint(v[0]);
m = obj2int(v[1]);
d = obj2int(v[2]);
{
int last = last_day_of_month(y, m);
if (m < 0)
m = 13 + m;
if (d < 0)
d = last_day_of_month(y, m) + d + 1;
return (!(m < 0 || m > 12 ||
d < 1 || d > last)) ? Qtrue : Qfalse;
}
}
static void
date_validate_vtm(struct vtm *vtm)
{
int last = last_day_of_month(vtm->year, vtm->mon);
if ( vtm->mon < 1 || vtm->mon > 12
|| vtm->mday < 1 || vtm->mday > last
|| vtm->hour < 0 || vtm->hour > 24
|| (vtm->hour == 24 && (vtm->min > 0 || vtm->sec > 0))
|| vtm->min < 0 || vtm->min > 59
|| vtm->sec < 0 || vtm->sec > 59
|| lt(vtm->subsecx, INT2FIX(0)) || ge(vtm->subsecx, INT2FIX(TIME_SCALE))
|| (!NIL_P(vtm->utc_offset) && (validate_utc_offset(vtm->utc_offset), 0)))
rb_raise(rb_eArgError, "invalid date");
}
static VALUE
datetime_init_1(int argc, VALUE *argv, VALUE time)
{
struct vtm vtm;
VALUE v[8];
struct time_object *tobj;
vtm.wday = -1;
vtm.yday = 0;
vtm.zone = "";
/* year mon mday hour min sec off sg */
rb_scan_args(argc, argv, "08", &v[0],&v[1],&v[2],&v[3],&v[4],&v[5],&v[6],&v[7]);
vtm.year = NIL_P(v[0]) ? INT2FIX(-4713) : obj2vint(v[0]);
if (NIL_P(v[0])) {
vtm.mon = NIL_P(v[1]) ? 11 : obj2int(v[1]);
vtm.mday = NIL_P(v[2]) ? 24 : obj2int(v[2]);
}
else {
vtm.mon = NIL_P(v[1]) ? 1 : obj2int(v[1]);
vtm.mday = NIL_P(v[2]) ? 1 : obj2int(v[2]);
}
vtm.hour = NIL_P(v[3]) ? 0 : obj2int(v[3]);
vtm.min = NIL_P(v[4]) ? 0 : obj2int(v[4]);
vtm.sec = 0;
vtm.subsecx = INT2FIX(0);
if (!NIL_P(v[5])) {
VALUE sec = num_exact(v[5]);
VALUE subsec;
divmodv(sec, INT2FIX(1), &sec, &subsec);
vtm.sec = NUM2INT(sec);
vtm.subsecx = w2v(rb_time_magnify(v2w(subsec)));
}
vtm.isdst = -1;
vtm.utc_offset = INT2FIX(0);
if (!NIL_P(v[6])) {
VALUE arg = v[6];
vtm.utc_offset = utc_offset_arg(arg);
}
{
if (vtm.mon < 0)
vtm.mon = 13 + vtm.mon;
if (vtm.mday < 0)
vtm.mday = last_day_of_month(vtm.year, vtm.mon) + vtm.mday + 1;
if (vtm.hour < 0)
vtm.hour = 24 + vtm.hour;
if (vtm.min < 0)
vtm.min = 60 + vtm.min;
if (vtm.sec < 0)
vtm.sec = 60 + vtm.sec;
if (vtm.sec == 60)
vtm.sec = 59;
}
date_validate_vtm(&vtm);
time_modify(time);
GetTimeval(time, tobj);
tobj->tm_got=0;
tobj->timew = WINT2FIXWV(0);
{
VALUE off = vtm.utc_offset;
vtm_add_offset(&vtm, neg(off));
vtm.utc_offset = Qnil;
tobj->timew = timegmw_noleapsecond(&vtm);
return time_set_utc_offset(time, off);
}
}
static VALUE
datetime_init(int argc, VALUE *argv, VALUE time)
{
if (argc >= 7 && TYPE(argv[6]) == T_RATIONAL)
argv[6] = rb_funcall(argv[6], '*', 1, INT2FIX(86400));
return datetime_init_1(argc, argv, time);
}
static VALUE
date_init_1(int argc, VALUE *argv, VALUE time)
{
VALUE v[8];
/* year mon mday sg */
rb_scan_args(argc, argv, "04", &v[0],&v[1],&v[2],&v[3]);
v[7] = v[3];
v[3] = Qnil;
v[4] = Qnil;
v[5] = Qnil;
v[6] = Qnil;
return datetime_init_1(argc, v, time);
}
static VALUE
date_init(int argc, VALUE *argv, VALUE time)
{
return date_init_1(argc, argv, time);
}
static VALUE
date_s_civil(int argc, VALUE *argv, VALUE klass)
{
VALUE time = time_s_alloc(klass);
return date_init(argc, argv, time);
}
static VALUE
date_s_today_or_now(VALUE klass, int today)
{
struct timespec ts;
#ifdef HAVE_CLOCK_GETTIME
if (clock_gettime(CLOCK_REALTIME, &ts) == -1) {
rb_sys_fail("clock_gettime");
}
#else
{
struct timeval tv;
if (gettimeofday(&tv, 0) < 0) {
rb_sys_fail("gettimeofday");
}
ts.tv_sec = tv.tv_sec;
ts.tv_nsec = tv.tv_usec * 1000;
}
#endif
{
wideval_t timew = timespec2timew(&ts);
VALUE time = time_s_alloc(klass);
struct vtm vtm;
VALUE v[8];
if (!localtimew(timew, &vtm))
rb_raise(rb_eArgError, "localtime error");
v[0] = vtm.year;
v[1] = INT2FIX(vtm.mon);
v[2] = INT2FIX(vtm.mday);
if (today) {
return date_init(3, v, time);
}
else {
v[3] = INT2FIX(vtm.hour);
v[4] = INT2FIX(vtm.min);
v[5] = rb_funcall(INT2FIX(vtm.sec), '+', 1,
rb_funcall(vtm.subsecx, rb_intern("quo"), 1,
INT2FIX(TIME_SCALE)));
v[6] = vtm.utc_offset;
return datetime_init(7, v, time);
}
}
}
static VALUE
date_s_today(VALUE klass)
{
return date_s_today_or_now(klass, 1);
}
static VALUE
datetime_s_now(VALUE klass)
{
return date_s_today_or_now(klass, 0);
}
static VALUE
datetime_to_s(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
return strftimev("%Y-%m-%dT%H:%M:%S%:z", time);
}
static VALUE
datetime_inspect(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
return strftimev("#<DateTime (sb): %Y-%m-%dT%H:%M:%S%:z (%s.%N)>", time);
}
static VALUE
date_to_s(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
return strftimev("%Y-%m-%d", time);
}
static VALUE
date_inspect(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
return strftimev("#<Date (sb): %Y-%m-%d (%s.%N)>", time);
}
static VALUE
date_add(struct time_object *tobj, VALUE offset, int sign, VALUE klass)
{
VALUE result;
offset = num_exact(offset);
if (sign < 0)
result = time_new_timew(klass, wsub(tobj->timew, rb_time_magnify(v2w(offset))));
else
result = time_new_timew(klass, wadd(tobj->timew, rb_time_magnify(v2w(offset))));
if (TIME_UTC_P(tobj)) {
GetTimeval(result, tobj);
TIME_SET_UTC(tobj);
}
else if (TIME_FIXOFF_P(tobj)) {
VALUE off = tobj->vtm.utc_offset;
GetTimeval(result, tobj);
TIME_SET_FIXOFF(tobj, off);
}
return result;
}
static VALUE
date_plus(VALUE time1, VALUE time2)
{
struct time_object *tobj;
GetTimeval(time1, tobj);
if (IsTimeval(time2)) {
rb_raise(rb_eTypeError, "date + date?");
}
return date_add(tobj, rb_funcall
(time2, '*', 1, INT2FIX(86400)), 1, CLASS_OF(time1));
}
static VALUE
date_minus(VALUE time1, VALUE time2)
{
struct time_object *tobj;
GetTimeval(time1, tobj);
if (IsTimeval(time2)) {
struct time_object *tobj2;
GetTimeval(time2, tobj2);
return rb_Float(rb_funcall(rb_time_unmagnify_to_float
(wsub(tobj->timew,
tobj2->timew)),
rb_intern("quo"), 1, INT2FIX(86400)));
}
return date_add(tobj, rb_funcall
(time2, '*', 1, INT2FIX(86400)), -1, CLASS_OF(time1));
}
static VALUE
date_succ(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
return date_add(tobj, INT2FIX(86400), 1, CLASS_OF(time));
}
static VALUE
date_subsec(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
return rb_funcall(quo(w2v(wmod(tobj->timew, WINT2FIXWV(TIME_SCALE))),
INT2FIX(TIME_SCALE)), rb_intern("to_f"), 0);
}
static VALUE
date_astro_num(VALUE time, double epoch)
{
struct time_object *tobj;
VALUE f;
GetTimeval(time, tobj);
f = rb_Float(rb_time_unmagnify_to_float(tobj->timew));
return rb_funcall(rb_funcall(f, rb_intern("fdiv"), 1, INT2FIX(86400)),
'+', 1, DBL2NUM(epoch));
}
static VALUE
date_ajd(VALUE time)
{
return date_astro_num(time, 2440587.5);
}
static VALUE
date_amjd(VALUE time)
{
return date_astro_num(time, 40587.0);
}
static VALUE
date_chrono_num(VALUE time, int epoch)
{
struct time_object *tobj;
VALUE s;
GetTimeval(time, tobj);
s = w2v(wdiv(tobj->timew, WINT2FIXWV(TIME_SCALE)));
s = rb_funcall(s, '+', 1, tobj->vtm.utc_offset);
return rb_funcall(rb_funcall(s, rb_intern("div"), 1,
INT2FIX(86400)), '+', 1,
INT2FIX(epoch));
}
static VALUE
date_jd(VALUE time)
{
return date_chrono_num(time, 2440588);
}
static VALUE
date_mjd(VALUE time)
{
return date_chrono_num(time, 40587);
}
static VALUE
date_ld(VALUE time)
{
return date_chrono_num(time, 141428);
}
static VALUE
date_zone(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
return strftimev("%:z", time);
}
static VALUE
date_utc_offset(VALUE time)
{
struct time_object *tobj;
GetTimeval(time, tobj);
MAKE_TM(time, tobj);
if (TIME_UTC_P(tobj)) {
return INT2FIX(0);
}
else {
return rb_funcall(tobj->vtm.utc_offset,
rb_intern("fdiv"), 1, INT2FIX(86400));
}
}
void
Init_Date(void)
{
#undef rb_intern
#define rb_intern(str) rb_intern_const(str)
rb_cDate = rb_define_class("Date", rb_cObject);
rb_include_module(rb_cDate, rb_mComparable);
rb_define_alloc_func(rb_cDate, time_s_alloc);
rb_define_singleton_method(rb_cDate, "valid_civil?", date_s_valid_civil_p, -1);
rb_define_singleton_method(rb_cDate, "valid_date?", date_s_valid_civil_p, -1);
rb_define_singleton_method(rb_cDate, "civil", date_s_civil, -1);
rb_define_singleton_method(rb_cDate, "today", date_s_today, 0);
rb_define_method(rb_cDate, "<=>", time_cmp, 1);
rb_define_method(rb_cDate, "eql?", time_eql, 1);
rb_define_method(rb_cDate, "hash", time_hash, 0);
rb_define_method(rb_cDate, "initialize", date_init, -1);
rb_define_method(rb_cDate, "initialize_copy", time_init_copy, 1);
rb_define_method(rb_cDate, "ctime", time_asctime, 0);
rb_define_method(rb_cDate, "asctime", time_asctime, 0);
rb_define_method(rb_cDate, "to_s", date_to_s, 0);
rb_define_method(rb_cDate, "inspect", date_inspect, 0);
rb_define_method(rb_cDate, "+", date_plus, 1);
rb_define_method(rb_cDate, "-", date_minus, 1);
rb_define_method(rb_cDate, "succ", date_succ, 0);
rb_define_method(rb_cDate, "next", date_succ, 0);
rb_define_method(rb_cDate, "mday", time_mday, 0);
rb_define_method(rb_cDate, "day", time_mday, 0);
rb_define_method(rb_cDate, "mon", time_mon, 0);
rb_define_method(rb_cDate, "month", time_mon, 0);
rb_define_method(rb_cDate, "year", time_year, 0);
rb_define_method(rb_cDate, "wday", time_wday, 0);
rb_define_method(rb_cDate, "yday", time_yday, 0);
rb_define_method(rb_cDate, "ajd", date_ajd, 0);
rb_define_method(rb_cDate, "amjd", date_amjd, 0);
rb_define_method(rb_cDate, "jd", date_jd, 0);
rb_define_method(rb_cDate, "mjd", date_mjd, 0);
rb_define_method(rb_cDate, "ld", date_ld, 0);
rb_define_method(rb_cDate, "sunday?", time_sunday, 0);
rb_define_method(rb_cDate, "monday?", time_monday, 0);
rb_define_method(rb_cDate, "tuesday?", time_tuesday, 0);
rb_define_method(rb_cDate, "wednesday?", time_wednesday, 0);
rb_define_method(rb_cDate, "thursday?", time_thursday, 0);
rb_define_method(rb_cDate, "friday?", time_friday, 0);
rb_define_method(rb_cDate, "saturday?", time_saturday, 0);
rb_define_method(rb_cDate, "strftime", time_strftime, 1);
/* methods for marshaling */
rb_define_method(rb_cDate, "_dump", time_dump, -1);
rb_define_singleton_method(rb_cDate, "_load", time_load, 1);
}
void
Init_DateTime(void)
{
rb_cDateTime = rb_define_class("DateTime", rb_cDate);
rb_undef_method(CLASS_OF(rb_cDateTime), "today");
rb_define_singleton_method(rb_cDateTime, "now", datetime_s_now, 0);
rb_define_method(rb_cDateTime, "initialize", datetime_init, -1);
rb_define_method(rb_cDateTime, "initialize_copy", time_init_copy, 1);
rb_define_method(rb_cDateTime, "to_s", datetime_to_s, 0);
rb_define_method(rb_cDateTime, "inspect", datetime_inspect, 0);
rb_define_method(rb_cDateTime, "sec", time_sec, 0);
rb_define_method(rb_cDateTime, "second", time_sec, 0);
rb_define_method(rb_cDateTime, "sec_fraction", date_subsec, 0);
rb_define_method(rb_cDateTime, "second_fraction", date_subsec, 0);
rb_define_method(rb_cDateTime, "min", time_min, 0);
rb_define_method(rb_cDateTime, "minute", time_min, 0);
rb_define_method(rb_cDateTime, "hour", time_hour, 0);
rb_define_method(rb_cDateTime, "zone", date_zone, 0);
rb_define_method(rb_cDateTime, "offset", date_utc_offset, 0);
}
#endif /* stolen_base */
inits.c (working copy)
CALL(IO);
CALL(Dir);
CALL(Time);
CALL(Date);
CALL(DateTime);
CALL(Random);
CALL(signal);
CALL(process);
(1-1/7)