#include #include double longitude = 2.0 + 20.0 / 60.0; const double eccentricity = 0.0167; const double obliquity = 23.44; const double deg2rad = 2 * PI / 360; /* Adjusts a time structure by delta min */ void correct(Tm *t, double delta) { int hours, min, sec; hours = delta / 60; min = delta - hours * 60; sec = (delta - (int)delta) * 60; t->sec += sec; if(t->sec >= 60) t->sec -= 60, t->min++; if(t->sec < 0) t->sec += 60, t->min--; t->min += min; if(t->min >= 60) t->min -= 60, t->hour++; if(t->min < 0) t->min += 60, t->hour--; t->hour += hours; if(t->hour >= 24) t->hour -= 24; if(t->hour < 0) t->hour += 24; } double dsin(double x) { return sin(x * deg2rad); } double dcos(double x) { return cos(x * deg2rad); } double dtan(double x) { return tan(x * deg2rad); } double datan(double x) { return atan(x) / deg2rad; } /* Hex formatter */ #pragma varargck type "H" Tm* int hexfmt(Fmt *f) { int hours, min, sec; uint time; Tm *t; t = va_arg(f->args, Tm *); time = 0.5+(float)0x10000 * ((float)t->hour + (float)t->min / 60.0 + (float)t->sec / (60.0 * 60.0)) / 24.0; hours = time / 0x1000; min = time % 0x1000 / 0x10; sec = time % 0x10; return fmtprint(f, "%01X_%02X_%01X", hours, min, sec); } /* Duodecimal formatter */ #pragma varargck type "D" Tm* int duofmt(Fmt *f) { Tm *t; t = va_arg(f->args, Tm*); return fmtprint(f, "%02d:%02d:%02d", t->hour, t->min, t->sec); } void usage(void) { fprint(2, "%s: [-l longitude] [-x] [-f]", argv0); exits("usage"); } void main(int argc, char *argv[]) { Tm *t; int hex, follow; char *fmt; fmt = "%D\n"; hex = 0; follow = 0; double w, a, b, c, eot; ARGBEGIN{ case 'l': longitude = atof(EARGF(usage())); break; case 'x': fmt = "%H\n"; hex++; break; case 'f': follow++; break; case 'h': default: usage(); }ARGEND; if(argc != 0) usage(); fmtinstall('H', hexfmt); fmtinstall('D', duofmt); beginning: /* Get local mean time */ t = gmtime(time(0)); correct(t, longitude * 4.0); /* Calculate equation of time */ w = 360 / 365.24; // Mean angular orbit velocity a = w * (t->yday + 10); // angular orbital distance from solstice (10 days befor jan 1) b = a + (360 / PI) * eccentricity * dsin(w * (t->yday - 2)); // angular deflection of earth c = (a - datan(dtan(b) / dcos(obliquity))) / 180; // Difference between mean and corrected angles eot = 720 * (c - floor(c + 0.5)); /* Alternative formulation */ /* w = 2 * PI * (t->yday - 1) / 365.242; eot = 0.258 * cos(w) - 7.416 * sin(w) - 3.648 * cos(2 * w) - 9.228 * sin(2 * w);*/ //print("%f\n", eot); correct(t, -eot); print(fmt, t); if(follow){ sleep(hex ? (1000 / 0.76) : 1000); goto beginning; } exits(0); }