#include <ctype.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>


/* MS-DOS time/date conversion routines derived from: */

/*
*  linux/fs/msdos/misc.c
*
*  Written 1992,1993 by Werner Almesberger
*/

/* Linear day numbers of the respective 1sts in non-leap years. */

static int day_n[] = { 0,31,59,90,120,151,181,212,243,273,304,334,0,0,0,0 };
                 /* JanFebMarApr May Jun Jul Aug Sep Oct Nov Dec */


/* Convert a MS-DOS time/date pair to a UNIX date (seconds since 1 1 70). */

int date_dos2unix(unsigned short time,unsigned short date)
{
       int month,year,secs;

       month = ((date >> 5) & 15)-1;
       year = date >> 9;
       secs = (time & 31)*2+60*((time >> 5) & 63)+(time >> 11)*3600+86400*
           ((date & 31)-1+day_n[month]+(year/4)+year*365-((year & 3) == 0 &&
           month < 2 ? 1 : 0)+3653);
                       /* days since 1.1.70 plus 80's leap day */
       return secs;
}


/* Convert linear UNIX date to a MS-DOS time/date pair. */

void date_unix2dos(int unix_date,unsigned short *time,
   unsigned short *date)
{
       int day,year,nl_day,month;

       *time = (unix_date % 60)/2+(((unix_date/60) % 60) << 5)+
           (((unix_date/3600) % 24) << 11);
       day = unix_date/86400-3652;
       year = day/365;
       if ((year+3)/4+365*year > day) year--;
       day -= (year+3)/4+365*year;
       if (day == 59 && !(year & 3)) {
               nl_day = day;
               month = 2;
       }
       else {
               nl_day = (year & 3) || day <= 59 ? day : day-1;
               for (month = 0; month < 12; month++)
                       if (day_n[month] > nl_day) break;
       }
       *date = nl_day-day_n[month-1]+1+(month << 5)+(year << 9);
}

/* Convert yapp format 8 hex characters into Unix time */

int yapp2unix(char * ytime)
{
       int i;
       unsigned short time,date;
       if(strlen(ytime)!=8) return 0;
       for(i=0;i<8;i++) if(!isxdigit(ytime[i])) return 0;
       time = strtoul(ytime+4,(char **)NULL,16);
       ytime[4]=0;
       date = strtoul(ytime,(char **)NULL,16);
       return(date_dos2unix(time,date));

}

/* Convert unix time to 8 character yapp hex format */

void unix2yapp(int unix_date, char * buffer)
{
       unsigned short time,date;
       date_unix2dos(unix_date,&time,&date);
       sprintf(buffer,"%04X%04X",date,time);
}