Annotation of embedaddon/php/ext/calendar/julian.c, revision 1.1
1.1 ! misho 1: /* $selId: julian.c,v 2.0 1995/10/24 01:13:06 lees Exp $
! 2: * Copyright 1993-1995, Scott E. Lee, all rights reserved.
! 3: * Permission granted to use, copy, modify, distribute and sell so long as
! 4: * the above copyright and this permission statement are retained in all
! 5: * copies. THERE IS NO WARRANTY - USE AT YOUR OWN RISK.
! 6: */
! 7:
! 8: /**************************************************************************
! 9: *
! 10: * These are the externally visible components of this file:
! 11: *
! 12: * void
! 13: * SdnToJulian(
! 14: * long int sdn,
! 15: * int *pYear,
! 16: * int *pMonth,
! 17: * int *pDay);
! 18: *
! 19: * Convert a SDN to a Julian calendar date. If the input SDN is less than
! 20: * 1, the three output values will all be set to zero, otherwise *pYear
! 21: * will be >= -4713 and != 0; *pMonth will be in the range 1 to 12
! 22: * inclusive; *pDay will be in the range 1 to 31 inclusive.
! 23: *
! 24: * long int
! 25: * JulianToSdn(
! 26: * int inputYear,
! 27: * int inputMonth,
! 28: * int inputDay);
! 29: *
! 30: * Convert a Julian calendar date to a SDN. Zero is returned when the
! 31: * input date is detected as invalid or out of the supported range. The
! 32: * return value will be > 0 for all valid, supported dates, but there are
! 33: * some invalid dates that will return a positive value. To verify that a
! 34: * date is valid, convert it to SDN and then back and compare with the
! 35: * original.
! 36: *
! 37: * VALID RANGE
! 38: *
! 39: * 4713 B.C. to at least 10000 A.D.
! 40: *
! 41: * Although this software can handle dates all the way back to 4713
! 42: * B.C., such use may not be meaningful. The calendar was created in
! 43: * 46 B.C., but the details did not stabilize until at least 8 A.D.,
! 44: * and perhaps as late at the 4th century. Also, the beginning of a
! 45: * year varied from one culture to another - not all accepted January
! 46: * as the first month.
! 47: *
! 48: * CALENDAR OVERVIEW
! 49: *
! 50: * Julias Ceasar created the calendar in 46 B.C. as a modified form of
! 51: * the old Roman republican calendar which was based on lunar cycles.
! 52: * The new Julian calendar set fixed lengths for the months, abandoning
! 53: * the lunar cycle. It also specified that there would be exactly 12
! 54: * months per year and 365.25 days per year with every 4th year being a
! 55: * leap year.
! 56: *
! 57: * Note that the current accepted value for the tropical year is
! 58: * 365.242199 days, not 365.25. This lead to an 11 day shift in the
! 59: * calendar with respect to the seasons by the 16th century when the
! 60: * Gregorian calendar was created to replace the Julian calendar.
! 61: *
! 62: * The difference between the Julian and today's Gregorian calendar is
! 63: * that the Gregorian does not make centennial years leap years unless
! 64: * they are a multiple of 400, which leads to a year of 365.2425 days.
! 65: * In other words, in the Gregorian calendar, 1700, 1800 and 1900 are
! 66: * not leap years, but 2000 is. All centennial years are leap years in
! 67: * the Julian calendar.
! 68: *
! 69: * The details are unknown, but the lengths of the months were adjusted
! 70: * until they finally stablized in 8 A.D. with their current lengths:
! 71: *
! 72: * January 31
! 73: * February 28/29
! 74: * March 31
! 75: * April 30
! 76: * May 31
! 77: * June 30
! 78: * Quintilis/July 31
! 79: * Sextilis/August 31
! 80: * September 30
! 81: * October 31
! 82: * November 30
! 83: * December 31
! 84: *
! 85: * In the early days of the calendar, the days of the month were not
! 86: * numbered as we do today. The numbers ran backwards (decreasing) and
! 87: * were counted from the Ides (15th of the month - which in the old
! 88: * Roman republican lunar calendar would have been the full moon) or
! 89: * from the Nonae (9th day before the Ides) or from the beginning of
! 90: * the next month.
! 91: *
! 92: * In the early years, the beginning of the year varied, sometimes
! 93: * based on the ascension of rulers. It was not always the first of
! 94: * January.
! 95: *
! 96: * Also, today's epoch, 1 A.D. or the birth of Jesus Christ, did not
! 97: * come into use until several centuries later when Christianity became
! 98: * a dominant religion.
! 99: *
! 100: * ALGORITHMS
! 101: *
! 102: * The calculations are based on two different cycles: a 4 year cycle
! 103: * of leap years and a 5 month cycle of month lengths.
! 104: *
! 105: * The 5 month cycle is used to account for the varying lengths of
! 106: * months. You will notice that the lengths alternate between 30 and
! 107: * 31 days, except for three anomalies: both July and August have 31
! 108: * days, both December and January have 31, and February is less than
! 109: * 30. Starting with March, the lengths are in a cycle of 5 months
! 110: * (31, 30, 31, 30, 31):
! 111: *
! 112: * Mar 31 days \
! 113: * Apr 30 days |
! 114: * May 31 days > First cycle
! 115: * Jun 30 days |
! 116: * Jul 31 days /
! 117: *
! 118: * Aug 31 days \
! 119: * Sep 30 days |
! 120: * Oct 31 days > Second cycle
! 121: * Nov 30 days |
! 122: * Dec 31 days /
! 123: *
! 124: * Jan 31 days \
! 125: * Feb 28/9 days |
! 126: * > Third cycle (incomplete)
! 127: *
! 128: * For this reason the calculations (internally) assume that the year
! 129: * starts with March 1.
! 130: *
! 131: * TESTING
! 132: *
! 133: * This algorithm has been tested from the year 4713 B.C. to 10000 A.D.
! 134: * The source code of the verification program is included in this
! 135: * package.
! 136: *
! 137: * REFERENCES
! 138: *
! 139: * Conversions Between Calendar Date and Julian Day Number by Robert J.
! 140: * Tantzen, Communications of the Association for Computing Machinery
! 141: * August 1963. (Also published in Collected Algorithms from CACM,
! 142: * algorithm number 199). [Note: the published algorithm is for the
! 143: * Gregorian calendar, but was adjusted to use the Julian calendar's
! 144: * simpler leap year rule.]
! 145: *
! 146: **************************************************************************/
! 147:
! 148: #include "sdncal.h"
! 149: #include <limits.h>
! 150:
! 151: #define JULIAN_SDN_OFFSET 32083
! 152: #define DAYS_PER_5_MONTHS 153
! 153: #define DAYS_PER_4_YEARS 1461
! 154:
! 155: void SdnToJulian(
! 156: long int sdn,
! 157: int *pYear,
! 158: int *pMonth,
! 159: int *pDay)
! 160: {
! 161: int year;
! 162: int month;
! 163: int day;
! 164: long int temp;
! 165: int dayOfYear;
! 166:
! 167: if (sdn <= 0) {
! 168: goto fail;
! 169: }
! 170: /* Check for overflow */
! 171: if (sdn > (LONG_MAX - JULIAN_SDN_OFFSET * 4 + 1) / 4 || sdn < LONG_MIN / 4) {
! 172: goto fail;
! 173: }
! 174: temp = sdn * 4 + (JULIAN_SDN_OFFSET * 4 - 1);
! 175:
! 176: /* Calculate the year and day of year (1 <= dayOfYear <= 366). */
! 177: {
! 178: long yearl = temp / DAYS_PER_4_YEARS;
! 179: if (yearl > INT_MAX || yearl < INT_MIN) {
! 180: goto fail;
! 181: }
! 182: year = (int) yearl;
! 183: }
! 184: dayOfYear = (temp % DAYS_PER_4_YEARS) / 4 + 1;
! 185:
! 186: /* Calculate the month and day of month. */
! 187: temp = dayOfYear * 5 - 3;
! 188: month = temp / DAYS_PER_5_MONTHS;
! 189: day = (temp % DAYS_PER_5_MONTHS) / 5 + 1;
! 190:
! 191: /* Convert to the normal beginning of the year. */
! 192: if (month < 10) {
! 193: month += 3;
! 194: } else {
! 195: year += 1;
! 196: month -= 9;
! 197: }
! 198:
! 199: /* Adjust to the B.C./A.D. type numbering. */
! 200: year -= 4800;
! 201: if (year <= 0)
! 202: year--;
! 203:
! 204: *pYear = year;
! 205: *pMonth = month;
! 206: *pDay = day;
! 207: return;
! 208:
! 209: fail:
! 210: *pYear = 0;
! 211: *pMonth = 0;
! 212: *pDay = 0;
! 213: }
! 214:
! 215: long int JulianToSdn(
! 216: int inputYear,
! 217: int inputMonth,
! 218: int inputDay)
! 219: {
! 220: int year;
! 221: int month;
! 222:
! 223: /* check for invalid dates */
! 224: if (inputYear == 0 || inputYear < -4713 ||
! 225: inputMonth <= 0 || inputMonth > 12 ||
! 226: inputDay <= 0 || inputDay > 31) {
! 227: return (0);
! 228: }
! 229: /* check for dates before SDN 1 (Jan 2, 4713 B.C.) */
! 230: if (inputYear == -4713) {
! 231: if (inputMonth == 1 && inputDay == 1) {
! 232: return (0);
! 233: }
! 234: }
! 235: /* Make year always a positive number. */
! 236: if (inputYear < 0) {
! 237: year = inputYear + 4801;
! 238: } else {
! 239: year = inputYear + 4800;
! 240: }
! 241:
! 242: /* Adjust the start of the year. */
! 243: if (inputMonth > 2) {
! 244: month = inputMonth - 3;
! 245: } else {
! 246: month = inputMonth + 9;
! 247: year--;
! 248: }
! 249:
! 250: return ((year * DAYS_PER_4_YEARS) / 4
! 251: + (month * DAYS_PER_5_MONTHS + 2) / 5
! 252: + inputDay
! 253: - JULIAN_SDN_OFFSET);
! 254: }
! 255:
! 256: /*
! 257: * Local variables:
! 258: * tab-width: 4
! 259: * c-basic-offset: 4
! 260: * End:
! 261: * vim600: sw=4 ts=4 fdm=marker
! 262: * vim<600: sw=4 ts=4
! 263: */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>