519 lines
16 KiB
C
519 lines
16 KiB
C
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <math.h>
|
|
|
|
#define RADS 0.0174532925199433
|
|
#define DEGS 57.2957795130823
|
|
#define TPI 6.28318530717959
|
|
#define PI 3.1415927
|
|
|
|
/* ratio of earth radius to astronomical unit */
|
|
#define ER_OVER_AU 0.0000426352325194252
|
|
|
|
/* all prototypes here */
|
|
|
|
double getcoord(int coord);
|
|
void getargs(int argc, char *argv[], int *y, int *m, double *tz, double *glong, double *glat);
|
|
double range(double y);
|
|
double rangerad(double y);
|
|
double days(int y, int m, int dn, double hour);
|
|
double days_(int *y, int *m, int *dn, double *hour);
|
|
void moonpos(double, double *, double *, double *);
|
|
void sunpos(double , double *, double *, double *);
|
|
double moontransit(int y, int m, int d, double timezone, double glat, double glong, int *nt);
|
|
double atan22(double y, double x);
|
|
double epsilon(double d);
|
|
void equatorial(double d, double *lon, double *lat, double *r);
|
|
void ecliptic(double d, double *lon, double *lat, double *r);
|
|
double gst(double d);
|
|
void topo(double lst, double glat, double *alp, double *dec, double *r);
|
|
double alt(double glat, double ha, double dec);
|
|
void libration(double day, double lambda, double beta, double alpha, double *l, double *b, double *p);
|
|
void illumination(double day, double lra, double ldec, double dr, double sra, double sdec, double *pabl, double *ill);
|
|
int daysinmonth(int y, int m);
|
|
int isleap(int y);
|
|
void tmoonsub_(double *day, double *glat, double *glong, double *moonalt,
|
|
double *mrv, double *l, double *b, double *paxis);
|
|
|
|
static const char
|
|
usage[] = " Usage: tmoon date[yyyymm] timz[+/-h.hh] long[+/-dddmm] lat[+/-ddmm]\n"
|
|
"example: tmoon 200009 0 -00155 5230\n";
|
|
|
|
/*
|
|
getargs() gets the arguments from the command line, does some basic error
|
|
checking, and converts arguments into numerical form. Arguments are passed
|
|
back in pointers. Error messages print to stderr so re-direction of output
|
|
to file won't leave users blind. Error checking prints list of all errors
|
|
in a command line before quitting.
|
|
*/
|
|
void getargs(int argc, char *argv[], int *y, int *m, double *tz,
|
|
double *glong, double *glat) {
|
|
|
|
int date, latitude, longitude;
|
|
int mflag = 0, yflag = 0, longflag = 0, latflag = 0, tzflag = 0;
|
|
int longminflag = 0, latminflag = 0, dflag = 0;
|
|
|
|
/* if not right number of arguments, then print example command line */
|
|
|
|
if (argc !=5) {
|
|
fprintf(stderr, usage);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
date = atoi(argv[1]);
|
|
*y = date / 100;
|
|
*m = date - *y * 100;
|
|
*tz = (double) atof(argv[2]);
|
|
longitude = atoi(argv[3]);
|
|
latitude = atoi(argv[4]);
|
|
*glong = RADS * getcoord(longitude);
|
|
*glat = RADS * getcoord(latitude);
|
|
|
|
/* set a flag for each error found */
|
|
|
|
if (*m > 12 || *m < 1) mflag = 1;
|
|
if (*y > 2500) yflag = 1;
|
|
if (date < 150001) dflag = 1;
|
|
if (fabs((float) *glong) > 180 * RADS) longflag = 1;
|
|
if (abs(longitude) % 100 > 59) longminflag = 1;
|
|
if (fabs((float) *glat) > 90 * RADS) latflag = 1;
|
|
if (abs(latitude) % 100 > 59) latminflag = 1;
|
|
if (fabs((float) *tz) > 12) tzflag = 1;
|
|
|
|
/* print all the errors found */
|
|
|
|
if (dflag == 1) {
|
|
fprintf(stderr, "date: dates must be in form yyyymm, gregorian, and later than 1500 AD\n");
|
|
}
|
|
if (yflag == 1) {
|
|
fprintf(stderr, "date: too far in future - accurate from 1500 to 2500\n");
|
|
}
|
|
if (mflag == 1) {
|
|
fprintf(stderr, "date: month must be in range 0 to 12, eg - August 2000 is entered as 200008\n");
|
|
}
|
|
if (tzflag == 1) {
|
|
fprintf(stderr, "timz: must be in range +/- 12 hours, eg -6 for Chicago\n");
|
|
}
|
|
if (longflag == 1) {
|
|
fprintf(stderr, "long: must be in range +/- 180 degrees\n");
|
|
}
|
|
if (longminflag == 1) {
|
|
fprintf(stderr, "long: last two digits are arcmin - max 59\n");
|
|
}
|
|
if (latflag == 1) {
|
|
fprintf(stderr, " lat: must be in range +/- 90 degrees\n");
|
|
}
|
|
if (latminflag == 1) {
|
|
fprintf(stderr, " lat: last two digits are arcmin - max 59\n");
|
|
}
|
|
|
|
/* quits if one or more flags set */
|
|
|
|
if (dflag + mflag + yflag + longflag + latflag + tzflag + longminflag + latminflag > 0) {
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
returns coordinates in decimal degrees given the
|
|
coord as a ddmm value stored in an integer.
|
|
*/
|
|
double getcoord(int coord) {
|
|
int west = 1;
|
|
double glg, deg;
|
|
if (coord < 0) west = -1;
|
|
glg = fabs((double) coord/100);
|
|
deg = floor(glg);
|
|
glg = west* (deg + (glg - deg)*100 / 60);
|
|
return(glg);
|
|
}
|
|
|
|
/*
|
|
days() takes the year, month, day in the month and decimal hours
|
|
in the day and returns the number of days since J2000.0.
|
|
Assumes Gregorian calendar.
|
|
*/
|
|
double days(int y, int m, int d, double h) {
|
|
int a, b;
|
|
double day;
|
|
|
|
/*
|
|
The lines below work from 1900 march to feb 2100
|
|
a = 367 * y - 7 * (y + (m + 9) / 12) / 4 + 275 * m / 9 + d;
|
|
day = (double)a - 730531.5 + hour / 24;
|
|
*/
|
|
|
|
/* These lines work for any Gregorian date since 0 AD */
|
|
if (m ==1 || m==2) {
|
|
m +=12;
|
|
y -= 1;
|
|
}
|
|
a = y / 100;
|
|
b = 2 - a + a/4;
|
|
day = floor(365.25*(y + 4716)) + floor(30.6001*(m + 1))
|
|
+ d + b - 1524.5 - 2451545 + h/24;
|
|
return(day);
|
|
}
|
|
double days_(int *y0, int *m0, int *d0, double *h0)
|
|
{
|
|
return days(*y0,*m0,*d0,*h0);
|
|
}
|
|
|
|
/*
|
|
Returns 1 if y a leap year, and 0 otherwise, according
|
|
to the Gregorian calendar
|
|
*/
|
|
int isleap(int y) {
|
|
int a = 0;
|
|
if(y % 4 == 0) a = 1;
|
|
if(y % 100 == 0) a = 0;
|
|
if(y % 400 == 0) a = 1;
|
|
return(a);
|
|
}
|
|
|
|
/*
|
|
Given the year and the month, function returns the
|
|
number of days in the month. Valid for Gregorian
|
|
calendar.
|
|
*/
|
|
int daysinmonth(int y, int m) {
|
|
int b = 31;
|
|
if(m == 2) {
|
|
if(isleap(y) == 1) b= 29;
|
|
else b = 28;
|
|
}
|
|
if(m == 4 || m == 6 || m == 9 || m == 11) b = 30;
|
|
return(b);
|
|
}
|
|
|
|
/*
|
|
moonpos() takes days from J2000.0 and returns ecliptic coordinates
|
|
of moon in the pointers. Note call by reference.
|
|
This function is within a couple of arcminutes most of the time,
|
|
and is truncated from the Meeus Ch45 series, themselves truncations of
|
|
ELP-2000. Returns moon distance in earth radii.
|
|
Terms have been written out explicitly rather than using the
|
|
table based method as only a small number of terms is
|
|
retained.
|
|
*/
|
|
void moonpos(double d, double *lambda, double *beta, double *rvec) {
|
|
double dl, dB, dR, L, D, M, M1, F, e, lm, bm, rm, t;
|
|
|
|
t = d / 36525;
|
|
|
|
L = range(218.3164591 + 481267.88134236 * t) * RADS;
|
|
D = range(297.8502042 + 445267.1115168 * t) * RADS;
|
|
M = range(357.5291092 + 35999.0502909 * t) * RADS;
|
|
M1 = range(134.9634114 + 477198.8676313 * t - .008997 * t * t) * RADS;
|
|
F = range(93.27209929999999 + 483202.0175273 * t - .0034029*t*t)*RADS;
|
|
e = 1 - .002516 * t;
|
|
|
|
dl = 6288774 * sin(M1);
|
|
dl += 1274027 * sin(2 * D - M1);
|
|
dl += 658314 * sin(2 * D);
|
|
dl += 213618 * sin(2 * M1);
|
|
dl -= e * 185116 * sin(M);
|
|
dl -= 114332 * sin(2 * F) ;
|
|
dl += 58793 * sin(2 * D - 2 * M1);
|
|
dl += e * 57066 * sin(2 * D - M - M1) ;
|
|
dl += 53322 * sin(2 * D + M1);
|
|
dl += e * 45758 * sin(2 * D - M);
|
|
dl -= e * 40923 * sin(M - M1);
|
|
dl -= 34720 * sin(D) ;
|
|
dl -= e * 30383 * sin(M + M1) ;
|
|
dl += 15327 * sin(2 * D - 2 * F) ;
|
|
dl -= 12528 * sin(M1 + 2 * F);
|
|
dl += 10980 * sin(M1 - 2 * F);
|
|
lm = rangerad(L + dl / 1000000 * RADS);
|
|
|
|
dB = 5128122 * sin(F);
|
|
dB += 280602 * sin(M1 + F);
|
|
dB += 277693 * sin(M1 - F);
|
|
dB += 173237 * sin(2 * D - F);
|
|
dB += 55413 * sin(2 * D - M1 + F);
|
|
dB += 46271 * sin(2 * D - M1 - F);
|
|
dB += 32573 * sin(2 * D + F);
|
|
dB += 17198 * sin(2 * M1 + F);
|
|
dB += 9266 * sin(2 * D + M1 - F);
|
|
dB += 8822 * sin(2 * M1 - F);
|
|
dB += e * 8216 * sin(2 * D - M - F);
|
|
dB += 4324 * sin(2 * D - 2 * M1 - F);
|
|
bm = dB / 1000000 * RADS;
|
|
|
|
dR = -20905355 * cos(M1);
|
|
dR -= 3699111 * cos(2 * D - M1);
|
|
dR -= 2955968 * cos(2 * D);
|
|
dR -= 569925 * cos(2 * M1);
|
|
dR += e * 48888 * cos(M);
|
|
dR -= 3149 * cos(2 * F);
|
|
dR += 246158 * cos(2 * D - 2 * M1);
|
|
dR -= e * 152138 * cos(2 * D - M - M1) ;
|
|
dR -= 170733 * cos(2 * D + M1);
|
|
dR -= e * 204586 * cos(2 * D - M);
|
|
dR -= e * 129620 * cos(M - M1);
|
|
dR += 108743 * cos(D);
|
|
dR += e * 104755 * cos(M + M1);
|
|
dR += 79661 * cos(M1 - 2 * F);
|
|
rm = 385000.56 + dR / 1000;
|
|
|
|
*lambda = lm;
|
|
*beta = bm;
|
|
/* distance to Moon must be in Earth radii */
|
|
*rvec = rm / 6378.14;
|
|
}
|
|
|
|
/*
|
|
topomoon() takes the local siderial time, the geographical
|
|
latitude of the observer, and pointers to the geocentric
|
|
equatorial coordinates. The function overwrites the geocentric
|
|
coordinates with topocentric coordinates on a simple spherical
|
|
earth model (no polar flattening). Expects Moon-Earth distance in
|
|
Earth radii. Formulas scavenged from Astronomical Almanac 'low
|
|
precision formulae for Moon position' page D46.
|
|
*/
|
|
|
|
void topo(double lst, double glat, double *alp, double *dec, double *r) {
|
|
double x, y, z, r1;
|
|
x = *r * cos(*dec) * cos(*alp) - cos(glat) * cos(lst);
|
|
y = *r * cos(*dec) * sin(*alp) - cos(glat) * sin(lst);
|
|
z = *r * sin(*dec) - sin(glat);
|
|
r1 = sqrt(x*x + y*y + z*z);
|
|
*alp = atan22(y, x);
|
|
*dec = asin(z / r1);
|
|
*r = r1;
|
|
}
|
|
|
|
/*
|
|
moontransit() takes date, the time zone and geographic longitude
|
|
of observer and returns the time (decimal hours) of lunar transit
|
|
on that day if there is one, and sets the notransit flag if there
|
|
isn't. See Explanatory Supplement to Astronomical Almanac
|
|
section 9.32 and 9.31 for the method.
|
|
*/
|
|
|
|
double moontransit(int y, int m, int d, double tz, double glat, double glong, int *notransit) {
|
|
double hm, ht, ht1, lon, lat, rv, dnew, lst;
|
|
int itcount;
|
|
|
|
ht1 = 180 * RADS;
|
|
ht = 0;
|
|
itcount = 0;
|
|
*notransit = 0;
|
|
do {
|
|
ht = ht1;
|
|
itcount++;
|
|
dnew = days(y, m, d, ht * DEGS/15) - tz/24;
|
|
lst = gst(dnew) + glong;
|
|
/* find the topocentric Moon ra (hence hour angle) and dec */
|
|
moonpos(dnew, &lon, &lat, &rv);
|
|
equatorial(dnew, &lon, &lat, &rv);
|
|
topo(lst, glat, &lon, &lat, &rv);
|
|
hm = rangerad(lst - lon);
|
|
ht1 = rangerad(ht - hm);
|
|
/* if no convergence, then no transit on that day */
|
|
if (itcount > 30) {
|
|
*notransit = 1;
|
|
break;
|
|
}
|
|
}
|
|
while (fabs(ht - ht1) > 0.04 * RADS);
|
|
return(ht1);
|
|
}
|
|
|
|
/*
|
|
Calculates the selenographic coordinates of either the sub Earth point
|
|
(optical libration) or the sub-solar point (selen. coords of centre of
|
|
bright hemisphere). Based on Meeus chapter 51 but neglects physical
|
|
libration and nutation, with some simplification of the formulas.
|
|
*/
|
|
void libration(double day, double lambda, double beta, double alpha, double *l, double *b, double *p) {
|
|
double i, f, omega, w, y, x, a, t, eps;
|
|
t = day / 36525;
|
|
i = 1.54242 * RADS;
|
|
eps = epsilon(day);
|
|
f = range(93.2720993 + 483202.0175273 * t - .0034029 * t * t) * RADS;
|
|
omega = range(125.044555 - 1934.1361849 * t + .0020762 * t * t) * RADS;
|
|
w = lambda - omega;
|
|
y = sin(w) * cos(beta) * cos(i) - sin(beta) * sin(i);
|
|
x = cos(w) * cos(beta);
|
|
a = atan22(y, x);
|
|
*l = a - f;
|
|
|
|
/* kludge to catch cases of 'round the back' angles */
|
|
if (*l < -90 * RADS) *l += TPI;
|
|
if (*l > 90 * RADS) *l -= TPI;
|
|
*b = asin(-sin(w) * cos(beta) * sin(i) - sin(beta) * cos(i));
|
|
|
|
/* pa pole axis - not used for Sun stuff */
|
|
x = sin(i) * sin(omega);
|
|
y = sin(i) * cos(omega) * cos(eps) - cos(i) * sin(eps);
|
|
w = atan22(x, y);
|
|
*p = rangerad(asin(sqrt(x*x + y*y) * cos(alpha - w) / cos(*b)));
|
|
}
|
|
|
|
/*
|
|
Takes: days since J2000.0, eq coords Moon, ratio of moon to sun distance,
|
|
eq coords Sun
|
|
Returns: position angle of bright limb wrt NCP, percentage illumination
|
|
of Sun
|
|
*/
|
|
void illumination(double day , double lra, double ldec, double dr, double sra, double sdec, double *pabl, double *ill) {
|
|
double x, y, phi, i;
|
|
(void)day;
|
|
y = cos(sdec) * sin(sra - lra);
|
|
x = sin(sdec) * cos(ldec) - cos(sdec) * sin(ldec) * cos (sra - lra);
|
|
*pabl = atan22(y, x);
|
|
phi = acos(sin(sdec) * sin(ldec) + cos(sdec) * cos(ldec) * cos(sra-lra));
|
|
i = atan22(sin(phi) , (dr - cos(phi)));
|
|
*ill = 0.5*(1 + cos(i));
|
|
}
|
|
|
|
/*
|
|
sunpos() takes days from J2000.0 and returns ecliptic longitude
|
|
of Sun in the pointers. Latitude is zero at this level of precision,
|
|
but pointer left in for consistency in number of arguments.
|
|
This function is within 0.01 degree (1 arcmin) almost all the time
|
|
for a century either side of J2000.0. This is from the 'low precision
|
|
fomulas for the Sun' from C24 of Astronomical Alamanac
|
|
*/
|
|
void sunpos(double d, double *lambda, double *beta, double *rvec) {
|
|
double L, g, ls, bs, rs;
|
|
|
|
L = range(280.461 + .9856474 * d) * RADS;
|
|
g = range(357.528 + .9856003 * d) * RADS;
|
|
ls = L + (1.915 * sin(g) + .02 * sin(2 * g)) * RADS;
|
|
bs = 0;
|
|
rs = 1.00014 - .01671 * cos(g) - .00014 * cos(2 * g);
|
|
*lambda = ls;
|
|
*beta = bs;
|
|
*rvec = rs;
|
|
}
|
|
|
|
/*
|
|
this routine returns the altitude given the days since J2000.0
|
|
the hour angle and declination of the object and the latitude
|
|
of the observer. Used to find the Sun's altitude to put a letter
|
|
code on the transit time, and to find the Moon's altitude at
|
|
transit just to make sure that the Moon is visible.
|
|
*/
|
|
double alt(double glat, double ha, double dec) {
|
|
return(asin(sin(dec) * sin(glat) + cos(dec) * cos(glat) * cos(ha)));
|
|
}
|
|
|
|
/* returns an angle in degrees in the range 0 to 360 */
|
|
double range(double x) {
|
|
double a, b;
|
|
b = x / 360;
|
|
a = 360 * (b - floor(b));
|
|
if (a < 0)
|
|
a = 360 + a;
|
|
return(a);
|
|
}
|
|
|
|
/* returns an angle in rads in the range 0 to two pi */
|
|
double rangerad(double x) {
|
|
double a, b;
|
|
b = x / TPI;
|
|
a = TPI * (b - floor(b));
|
|
if (a < 0)
|
|
a = TPI + a;
|
|
return(a);
|
|
}
|
|
|
|
/*
|
|
gets the atan2 function returning angles in the right
|
|
order and range
|
|
*/
|
|
double atan22(double y, double x) {
|
|
double a;
|
|
|
|
a = atan2(y, x);
|
|
if (a < 0) a += TPI;
|
|
return(a);
|
|
}
|
|
|
|
/*
|
|
returns mean obliquity of ecliptic in radians given days since
|
|
J2000.0.
|
|
*/
|
|
double epsilon(double d) {
|
|
double t = d/ 36525;
|
|
return((23.4392911111111 - (t* (46.8150 + 0.00059*t)/3600)) *RADS);
|
|
}
|
|
|
|
/*
|
|
replaces ecliptic coordinates with equatorial coordinates
|
|
note: call by reference destroys original values
|
|
R is unchanged.
|
|
*/
|
|
void equatorial(double d, double *lon, double *lat, double * r) {
|
|
double eps, ceps, seps, l, b;
|
|
(void)r;
|
|
|
|
l = *lon;
|
|
b = * lat;
|
|
eps = epsilon(d);
|
|
ceps = cos(eps);
|
|
seps = sin(eps);
|
|
*lon = atan22(sin(l)*ceps - tan(b)*seps, cos(l));
|
|
*lat = asin(sin(b)*ceps + cos(b)*seps*sin(l));
|
|
}
|
|
|
|
/*
|
|
replaces equatorial coordinates with ecliptic ones. Inverse
|
|
of above, but used to find topocentric ecliptic coords.
|
|
*/
|
|
void ecliptic(double d, double *lon, double *lat, double * r) {
|
|
double eps, ceps, seps, alp, dec;
|
|
(void)r;
|
|
|
|
alp = *lon;
|
|
dec = *lat;
|
|
eps = epsilon(d);
|
|
ceps = cos(eps);
|
|
seps = sin(eps);
|
|
*lon = atan22(sin(alp)*ceps + tan(dec)*seps, cos(alp));
|
|
*lat = asin(sin(dec)*ceps - cos(dec)*seps*sin(alp));
|
|
}
|
|
|
|
/*
|
|
returns the siderial time at greenwich meridian as
|
|
an angle in radians given the days since J2000.0
|
|
*/
|
|
double gst( double d) {
|
|
double t = d / 36525;
|
|
double theta;
|
|
theta = range(280.46061837 + 360.98564736629 * d + 0.000387933 * t * t);
|
|
return(theta * RADS);
|
|
}
|
|
|
|
void tmoonsub_(double *day, double *glat, double *glong, double *moonalt,
|
|
double *mrv, double *l, double *b, double *paxis)
|
|
{
|
|
double mlambda, mbeta;
|
|
double malpha, mdelta;
|
|
double lst, mhr;
|
|
double tlambda, tbeta, trv;
|
|
|
|
lst = gst(*day) + *glong;
|
|
|
|
/* find Moon topocentric coordinates for libration calculations */
|
|
|
|
moonpos(*day, &mlambda, &mbeta, mrv);
|
|
malpha = mlambda;
|
|
mdelta = mbeta;
|
|
equatorial(*day, &malpha, &mdelta, mrv);
|
|
topo(lst, *glat, &malpha, &mdelta, mrv);
|
|
mhr = rangerad(lst - malpha);
|
|
*moonalt = alt(*glat, mhr, mdelta);
|
|
|
|
/* Optical libration and Position angle of the Pole */
|
|
|
|
tlambda = malpha;
|
|
tbeta = mdelta;
|
|
trv = *mrv;
|
|
ecliptic(*day, &tlambda, &tbeta, &trv);
|
|
libration(*day, tlambda, tbeta, malpha, l, b, paxis);
|
|
}
|