magnetic-tesseroids/toolkits/tessutil_magnetize_model.c

1319 lines
43 KiB
C
Executable File

/*
Tesseroid magnetizer
This program sets inducing magnetic field to the tesseroid model. Inducing field is calculated from SH model, such as IGRF12.
Program is based on Geomag 7.0 by Stefan Maus.
https://www.ngdc.noaa.gov/IAGA/vmod/geomag70_license.html
The Geomag 7.0 software code is in the public domain and not licensed or under copyright.
U.S. Government material is incorporated in this work and that material is not subject to copyright protection.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <math.h>
#include "../lib/constants.h"
int my_isnan(double d)
{
return (d != d); /* IEEE: only NaN is not equal to itself */
}
#if defined(_MSC_VER) /* Added for windows by Yi Zhang on 2021-08-26*/
#include <BaseTsd.h>
typedef SSIZE_T ssize_t;
#endif
#define NaN log(-1.0)
#define FT2KM (1.0/0.0003048)
#define RAD2DEG (180.0/PI)
#ifndef SEEK_SET
#define SEEK_SET 0
#define SEEK_CUR 1
#define SEEK_END 2
#endif
#define IEXT 0
#define FALSE 0
#define TRUE 1 /* constants */
#define RECL 81
#define MAXINBUFF RECL+14
/** Max size of in buffer **/
#define MAXREAD MAXINBUFF-2
/** Max to read 2 less than total size (just to be safe) **/
#define MAXMOD 30
/** Max number of models in a file **/
#define PATH MAXREAD
/** Max path and filename length **/
#define EXT_COEFF1 (double)0
#define EXT_COEFF2 (double)0
#define EXT_COEFF3 (double)0
#define MAXDEG 13
#define MAXCOEFF (MAXDEG*(MAXDEG+2)+1) /* index starts with 1!, (from old Fortran?) */
double gh1[MAXCOEFF];
double gh2[MAXCOEFF];
double gha[MAXCOEFF]; /* Geomag global variables */
double ghb[MAXCOEFF];
double d=0,f=0,h=0,i=0;
double dtemp,ftemp,htemp,itemp;
double x=0,y=0,z=0;
double xtemp,ytemp,ztemp;
FILE *stream = NULL; /* Pointer to specified model data file */
int main(int argc, char**argv)
{
/* Variable declaration */
/* Variables related to tesseroids */
FILE * tessfp;
FILE * tessoutfp;
char * line = NULL;
size_t len = 0;
ssize_t read;
float W, E, S, N;
float HOT, HOB;
float DENSITY;
float SUSCEPT, OBX, OBY, OBZ;
float lon_c, lat_c, alt_c;
int i, count;
char tessfilename[PATH];
char tessoutfilename[PATH];
/* Control variables */
int again = 1;
int decyears = 3;
int units = 4;
int decdeg = 3;
int range = -1;
int counter = 0;
int warn_H, warn_H_strong, warn_P;
int modelI; /* Which model (Index) */
int nmodel; /* Number of models in file */
int max1[MAXMOD];
int max2[MAXMOD];
int max3[MAXMOD];
int nmax;
int igdgc=3;
int isyear=-1;
int ismonth=-1;
int isday=-1;
int ieyear=-1;
int iemonth=-1;
int ieday=-1;
int ilat_deg=200;
int ilat_min=200;
int ilat_sec=200;
int ilon_deg=200;
int ilon_min=200;
int ilon_sec=200;
int fileline;
long irec_pos[MAXMOD];
int coords_from_file = 0;
int arg_err = 0;
int need_to_read_model = 1;
char mdfile[PATH];
char inbuff[MAXINBUFF];
char model[MAXMOD][9];
char *begin;
char *rest;
char args[7][MAXREAD];
int iarg;
char coord_fname[PATH];
char out_fname[PATH];
FILE *coordfile,*outfile;
int iline=0;
int read_flag;
double epoch[MAXMOD];
double yrmin[MAXMOD];
double yrmax[MAXMOD];
double minyr;
double maxyr;
double altmin[MAXMOD];
double altmax[MAXMOD];
double minalt;
double maxalt;
double alt=-999999;
double sdate=-1;
double step=-1;
double syr;
double edate=-1;
double latitude=200;
double longitude=200;
double ddot;
double fdot;
double hdot;
double idot;
double xdot;
double ydot;
double zdot;
double warn_H_val, warn_H_strong_val;
/* Subroutines used */
double degrees_to_decimal();
double julday();
int interpsh();
int extrapsh();
int shval3();
int dihf();
int safegets(char *buffer,int n);
int getshc();
/* Initializations. */
inbuff[MAXREAD+1]='\0'; /* Just to protect mem. */
inbuff[MAXINBUFF-1]='\0'; /* Just to protect mem. */
for (iarg=0; iarg<argc; iarg++)
if (argv[iarg] != NULL)
strncpy(args[iarg],argv[iarg],MAXREAD);
/* printing out version number and header */
printf("Tesseroid magnetizer (based on Geomag v7.0) - Eldar Baykiev, Feb, 2017\n");
if ( argc <= 2)
{
printf("Usage: %s [SH coeff file] [input tess file] [day] [month] [year] [output tess file]\n", args[0]);
exit(1);
}
else if (argc == 7)
{
strcpy(mdfile, args[1]); //SH coeff
strcpy(tessfilename, args[2]); //input tesseroid model filename
sscanf(args[3], "%d", &isday);
sscanf(args[4], "%d", &ismonth);
sscanf(args[5], "%d", &isyear);
strcpy(tessoutfilename, args[6]);
}
else
{
printf("ERROR: Wrong input!\n");
exit(1);
}
/* Obtain the desired model file and read the data */
if (!(stream = fopen(mdfile, "rt")))
printf("ERROR: Can not open file %s.", mdfile);
rewind(stream);
fileline = 0; /* First line will be 1 */
modelI = -1; /* First model will be 0 */
while (fgets(inbuff,MAXREAD,stream)) /* While not end of file
* read to end of line or buffer */
{
fileline++; /* On new line */
if (strlen(inbuff) != RECL) /* IF incorrect record size */
{
printf("ERROR: Corrupt record in file %s on line %d.\n", mdfile, fileline);
fclose(stream);
exit(5);
}
/* old statement Dec 1999 */
/* if (!strncmp(inbuff," ",4)){ // If 1st 4 chars are spaces */
/* New statement Dec 1999 changed by wmd required by year 2000 models */
if (!strncmp(inbuff," ",3)) /* If 1st 3 chars are spaces */
{
modelI++; /* New model */
if (modelI > MAXMOD) /* If too many headers */
{
printf("ERROR: Too many models in file %s on line %d.", mdfile, fileline);
fclose(stream);
exit(6);
}
irec_pos[modelI]=ftell(stream);
/* Get fields from buffer into individual vars. */
sscanf(inbuff, "%s%lg%d%d%d%lg%lg%lg%lg", model[modelI], &epoch[modelI], &max1[modelI], &max2[modelI], &max3[modelI], &yrmin[modelI], &yrmax[modelI], &altmin[modelI], &altmax[modelI]);
/* Compute date range for all models */
if (modelI == 0) /*If first model */
{
minyr=yrmin[0];
maxyr=yrmax[0];
}
else
{
if (yrmin[modelI]<minyr){
minyr=yrmin[modelI];
}
if (yrmax[modelI]>maxyr){
maxyr=yrmax[modelI];
}
} /* if modelI != 0 */
} /* If 1st 3 chars are spaces */
} /* While not end of model file */
nmodel = modelI + 1;
fclose(stream);
decdeg=1;
decyears=2;
range=1;
igdgc=1;
units = 1;
sdate = julday(ismonth,isday,isyear);
/* Pick model */
for (modelI=0; modelI<nmodel; modelI++)
if (sdate<yrmax[modelI]) break;
if (modelI == nmodel) modelI--; /* if beyond end of last model use last model */
/* Get altitude min and max for selected model. */
minalt=altmin[modelI];
maxalt=altmax[modelI];
/** This will compute everything needed for 1 point in time. **/
if (max2[modelI] == 0)
{
getshc(mdfile, 1, irec_pos[modelI], max1[modelI], 1);
getshc(mdfile, 1, irec_pos[modelI+1], max1[modelI+1], 2);
nmax = interpsh(sdate, yrmin[modelI], max1[modelI], yrmin[modelI+1], max1[modelI+1], 3);
nmax = interpsh(sdate+1, yrmin[modelI] , max1[modelI], yrmin[modelI+1], max1[modelI+1],4);
}
else
{
getshc(mdfile, 1, irec_pos[modelI], max1[modelI], 1);
getshc(mdfile, 0, irec_pos[modelI], max2[modelI], 2);
nmax = extrapsh(sdate, epoch[modelI], max1[modelI], max2[modelI], 3);
nmax = extrapsh(sdate+1, epoch[modelI], max1[modelI], max2[modelI], 4);
}
/* ELDAR: open tesseroid file */
tessfp = fopen(tessfilename, "r");
if (tessfp == NULL)
exit(EXIT_FAILURE);
tessoutfp = fopen(tessoutfilename, "w");
if (tessoutfp == NULL)
exit(EXIT_FAILURE);
fprintf(tessoutfp, "#Tesseroid magnetizer\n");
fprintf(tessoutfp, "#SH model: %s\n", mdfile);
fprintf(tessoutfp, "#New date: %d-%d-%d\n", isday, ismonth, isyear);
if ( igdgc == 1)
fprintf(tessoutfp, "#Geodetic (change in source code)\n");
else
fprintf(tessoutfp, "#Geocentric (change in source code)\n");
while ((read = getline(&line, &len, tessfp)) != -1) {
//printf("Retrieved line of length %zu :\n", read);
//printf("%s\n", line);
count = 0;
for (i = 0;i<read;i++)
{
if (line[i] == ' ')
count++;
}
count++;
if (line[0] != '#')
{
SUSCEPT = 1;
if ( count == 7)
{
sscanf(line, "%f %f %f %f %f %f %f", &W, &E, &S, &N, &HOT, &HOB, &DENSITY);
}
else if (count == 8)
{
sscanf(line, "%f %f %f %f %f %f %f %f", &W, &E, &S, &N, &HOT, &HOB, &DENSITY, &SUSCEPT);
}
else if (count == 11)
{
sscanf(line, "%f %f %f %f %f %f %f %f %f %f %f", &W, &E, &S, &N, &HOT, &HOB, &DENSITY, &SUSCEPT, &OBX, &OBY, &OBZ);
}
alt_c=0.5*(HOT + HOB);
latitude=0.5*(S + N);
longitude=0.5*(W + E);
alt=(alt_c + MEAN_EARTH_RADIUS - EARTH_RADIUS_IGRF_KM * 1000.0)/1000.0;
/* Do the first calculations */
shval3(igdgc, latitude, longitude, alt, nmax, 3, IEXT, EXT_COEFF1, EXT_COEFF2, EXT_COEFF3);
dihf(3);
shval3(igdgc, latitude, longitude, alt, nmax, 4, IEXT, EXT_COEFF1, EXT_COEFF2, EXT_COEFF3);
dihf(4);
if (90.0-fabs(latitude) <= 0.001) /* at geographic poles */
{
x = NaN;
y = NaN;
d = NaN;
}
//if ( count == 7)
//{
// fprintf(tessoutfp, "%s 1.0 %f %f %f\n", line, x, y, -z);
//}
//else if (count == 8)
//{
// fprintf(tessoutfp, "%s %f %f %f\n", line, x, y, -z);
//}
//else if (count == 11)
//{
fprintf(tessoutfp, "%.2f %.2f %.2f %.2f %.3f %.3f %f %f %f %f %f\n", W, E, S, N, HOT, HOB, DENSITY, SUSCEPT, x, y, -z);
//}
}
else
{
fprintf(tessoutfp, "%s\n", line);
}
}
fclose(tessfp);
fclose(tessoutfp);
//if (line)
// free(line);
exit(EXIT_SUCCESS);
return 0;
}
/* Subroutines below is directly taken from Geomag 7.0 */
/****************************************************************************/
/* */
/* Subroutine safegets */
/* */
/****************************************************************************/
/* */
/* Gets characters from stdin untill it has reached n characters or \n, */
/* whichever comes first. \n is converted to \0. */
/* */
/* Input: n - Integer number of chars */
/* *buffer - Character array ptr which can contain n+1 characters */
/* */
/* Output: size - integer size of sting in buffer */
/* */
/* Note: All strings will be null terminated. */
/* */
/* By: David Owens */
/* dio@ngdc.noaa.gov */
/****************************************************************************/
int safegets(char *buffer,int n){
char *ptr; /** ptr used for finding '\n' **/
buffer[0] = '\0';
fgets(buffer,n,stdin); /** Get n chars **/
buffer[n+1]='\0'; /** Set last char to null **/
ptr=strchr(buffer,'\n'); /** If string contains '\n' **/
if (ptr!=NULL){ /** If string contains '\n' **/
ptr[0]='\0'; /** Change char to '\0' **/
if (buffer[0] == '\0') printf("\n ... no entry ...\n");
}
return strlen(buffer); /** Return the length **/
}
/****************************************************************************/
/* */
/* Subroutine degrees_to_decimal */
/* */
/****************************************************************************/
/* */
/* Converts degrees,minutes, seconds to decimal degrees. */
/* */
/* Input: */
/* degrees - Integer degrees */
/* minutes - Integer minutes */
/* seconds - Integer seconds */
/* */
/* Output: */
/* decimal - degrees in decimal degrees */
/* */
/* C */
/* C. H. Shaffer */
/* Lockheed Missiles and Space Company, Sunnyvale CA */
/* August 12, 1988 */
/* */
/****************************************************************************/
double degrees_to_decimal(int degrees,int minutes,int seconds)
{
double deg;
double min;
double sec;
double decimal;
deg = degrees;
min = minutes/60.0;
sec = seconds/3600.0;
decimal = fabs(sec) + fabs(min) + fabs(deg);
if (deg < 0) {
decimal = -decimal;
} else if (deg == 0){
if (min < 0){
decimal = -decimal;
} else if (min == 0){
if (sec<0){
decimal = -decimal;
}
}
}
return(decimal);
}
/****************************************************************************/
/* */
/* Subroutine julday */
/* */
/****************************************************************************/
/* */
/* Computes the decimal day of year from month, day, year. */
/* Supplied by Daniel Bergstrom */
/* */
/* References: */
/* */
/* 1. Nachum Dershowitz and Edward M. Reingold, Calendrical Calculations, */
/* Cambridge University Press, 3rd edition, ISBN 978-0-521-88540-9. */
/* */
/* 2. Claus Tøndering, Frequently Asked Questions about Calendars, */
/* Version 2.9, http://www.tondering.dk/claus/calendar.html */
/* */
/****************************************************************************/
double julday(month, day, year)
int month;
int day;
int year;
{
int days[12] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334};
int leap_year = (((year % 4) == 0) &&
(((year % 100) != 0) || ((year % 400) == 0)));
double day_in_year = (days[month - 1] + day + (month > 2 ? leap_year : 0));
return ((double)year + (day_in_year / (365.0 + leap_year)));
}
/****************************************************************************/
/* */
/* Subroutine getshc */
/* */
/****************************************************************************/
/* */
/* Reads spherical harmonic coefficients from the specified */
/* model into an array. */
/* */
/* Input: */
/* stream - Logical unit number */
/* iflag - Flag for SV equal to ) or not equal to 0 */
/* for designated read statements */
/* strec - Starting record number to read from model */
/* nmax_of_gh - Maximum degree and order of model */
/* */
/* Output: */
/* gh1 or 2 - Schmidt quasi-normal internal spherical */
/* harmonic coefficients */
/* */
/* FORTRAN */
/* Bill Flanagan */
/* NOAA CORPS, DESDIS, NGDC, 325 Broadway, Boulder CO. 80301 */
/* */
/* C */
/* C. H. Shaffer */
/* Lockheed Missiles and Space Company, Sunnyvale CA */
/* August 15, 1988 */
/* */
/****************************************************************************/
int getshc(file, iflag, strec, nmax_of_gh, gh)
char file[PATH];
int iflag;
long int strec;
int nmax_of_gh;
int gh;
{
char inbuff[MAXINBUFF];
char irat[9];
int ii,m,n,mm,nn;
int ios;
int line_num;
double g,hh;
double trash;
stream = fopen(file, "rt");
if (stream == NULL)
{
printf("\nError on opening file %s", file);
}
else
{
ii = 0;
ios = 0;
fseek(stream,strec,SEEK_SET);
for ( nn = 1; nn <= nmax_of_gh; ++nn)
{
for (mm = 0; mm <= nn; ++mm)
{
if (iflag == 1)
{
fgets(inbuff, MAXREAD, stream);
sscanf(inbuff, "%d%d%lg%lg%lg%lg%s%d",
&n, &m, &g, &hh, &trash, &trash, irat, &line_num);
}
else
{
fgets(inbuff, MAXREAD, stream);
sscanf(inbuff, "%d%d%lg%lg%lg%lg%s%d",
&n, &m, &trash, &trash, &g, &hh, irat, &line_num);
}
if ((nn != n) || (mm != m))
{
ios = -2;
fclose(stream);
return(ios);
}
ii = ii + 1;
switch(gh)
{
case 1: gh1[ii] = g;
break;
case 2: gh2[ii] = g;
break;
default: printf("\nError in subroutine getshc");
break;
}
if (m != 0)
{
ii = ii+ 1;
switch(gh)
{
case 1: gh1[ii] = hh;
break;
case 2: gh2[ii] = hh;
break;
default: printf("\nError in subroutine getshc");
break;
}
}
}
}
}
fclose(stream);
return(ios);
}
/****************************************************************************/
/* */
/* Subroutine extrapsh */
/* */
/****************************************************************************/
/* */
/* Extrapolates linearly a spherical harmonic model with a */
/* rate-of-change model. */
/* */
/* Input: */
/* date - date of resulting model (in decimal year) */
/* dte1 - date of base model */
/* nmax1 - maximum degree and order of base model */
/* gh1 - Schmidt quasi-normal internal spherical */
/* harmonic coefficients of base model */
/* nmax2 - maximum degree and order of rate-of-change model */
/* gh2 - Schmidt quasi-normal internal spherical */
/* harmonic coefficients of rate-of-change model */
/* */
/* Output: */
/* gha or b - Schmidt quasi-normal internal spherical */
/* harmonic coefficients */
/* nmax - maximum degree and order of resulting model */
/* */
/* FORTRAN */
/* A. Zunde */
/* USGS, MS 964, box 25046 Federal Center, Denver, CO. 80225 */
/* */
/* C */
/* C. H. Shaffer */
/* Lockheed Missiles and Space Company, Sunnyvale CA */
/* August 16, 1988 */
/* */
/****************************************************************************/
int extrapsh(date, dte1, nmax1, nmax2, gh)
double date;
double dte1;
int nmax1;
int nmax2;
int gh;
{
int nmax;
int k, l;
int ii;
double factor;
factor = date - dte1;
if (nmax1 == nmax2)
{
k = nmax1 * (nmax1 + 2);
nmax = nmax1;
}
else
{
if (nmax1 > nmax2)
{
k = nmax2 * (nmax2 + 2);
l = nmax1 * (nmax1 + 2);
switch(gh)
{
case 3: for ( ii = k + 1; ii <= l; ++ii)
{
gha[ii] = gh1[ii];
}
break;
case 4: for ( ii = k + 1; ii <= l; ++ii)
{
ghb[ii] = gh1[ii];
}
break;
default: printf("\nError in subroutine extrapsh");
break;
}
nmax = nmax1;
}
else
{
k = nmax1 * (nmax1 + 2);
l = nmax2 * (nmax2 + 2);
switch(gh)
{
case 3: for ( ii = k + 1; ii <= l; ++ii)
{
gha[ii] = factor * gh2[ii];
}
break;
case 4: for ( ii = k + 1; ii <= l; ++ii)
{
ghb[ii] = factor * gh2[ii];
}
break;
default: printf("\nError in subroutine extrapsh");
break;
}
nmax = nmax2;
}
}
switch(gh)
{
case 3: for ( ii = 1; ii <= k; ++ii)
{
gha[ii] = gh1[ii] + factor * gh2[ii];
}
break;
case 4: for ( ii = 1; ii <= k; ++ii)
{
ghb[ii] = gh1[ii] + factor * gh2[ii];
}
break;
default: printf("\nError in subroutine extrapsh");
break;
}
return(nmax);
}
/****************************************************************************/
/* */
/* Subroutine interpsh */
/* */
/****************************************************************************/
/* */
/* Interpolates linearly, in time, between two spherical harmonic */
/* models. */
/* */
/* Input: */
/* date - date of resulting model (in decimal year) */
/* dte1 - date of earlier model */
/* nmax1 - maximum degree and order of earlier model */
/* gh1 - Schmidt quasi-normal internal spherical */
/* harmonic coefficients of earlier model */
/* dte2 - date of later model */
/* nmax2 - maximum degree and order of later model */
/* gh2 - Schmidt quasi-normal internal spherical */
/* harmonic coefficients of internal model */
/* */
/* Output: */
/* gha or b - coefficients of resulting model */
/* nmax - maximum degree and order of resulting model */
/* */
/* FORTRAN */
/* A. Zunde */
/* USGS, MS 964, box 25046 Federal Center, Denver, CO. 80225 */
/* */
/* C */
/* C. H. Shaffer */
/* Lockheed Missiles and Space Company, Sunnyvale CA */
/* August 17, 1988 */
/* */
/****************************************************************************/
int interpsh(date, dte1, nmax1, dte2, nmax2, gh)
double date;
double dte1;
int nmax1;
double dte2;
int nmax2;
int gh;
{
int nmax;
int k, l;
int ii;
double factor;
factor = (date - dte1) / (dte2 - dte1);
if (nmax1 == nmax2)
{
k = nmax1 * (nmax1 + 2);
nmax = nmax1;
}
else
{
if (nmax1 > nmax2)
{
k = nmax2 * (nmax2 + 2);
l = nmax1 * (nmax1 + 2);
switch(gh)
{
case 3: for ( ii = k + 1; ii <= l; ++ii)
{
gha[ii] = gh1[ii] + factor * (-gh1[ii]);
}
break;
case 4: for ( ii = k + 1; ii <= l; ++ii)
{
ghb[ii] = gh1[ii] + factor * (-gh1[ii]);
}
break;
default: printf("\nError in subroutine extrapsh");
break;
}
nmax = nmax1;
}
else
{
k = nmax1 * (nmax1 + 2);
l = nmax2 * (nmax2 + 2);
switch(gh)
{
case 3: for ( ii = k + 1; ii <= l; ++ii)
{
gha[ii] = factor * gh2[ii];
}
break;
case 4: for ( ii = k + 1; ii <= l; ++ii)
{
ghb[ii] = factor * gh2[ii];
}
break;
default: printf("\nError in subroutine extrapsh");
break;
}
nmax = nmax2;
}
}
switch(gh)
{
case 3: for ( ii = 1; ii <= k; ++ii)
{
gha[ii] = gh1[ii] + factor * (gh2[ii] - gh1[ii]);
}
break;
case 4: for ( ii = 1; ii <= k; ++ii)
{
ghb[ii] = gh1[ii] + factor * (gh2[ii] - gh1[ii]);
}
break;
default: printf("\nError in subroutine extrapsh");
break;
}
return(nmax);
}
/****************************************************************************/
/* */
/* Subroutine shval3 */
/* */
/****************************************************************************/
/* */
/* Calculates field components from spherical harmonic (sh) */
/* models. */
/* */
/* Input: */
/* igdgc - indicates coordinate system used; set equal */
/* to 1 if geodetic, 2 if geocentric */
/* latitude - north latitude, in degrees */
/* longitude - east longitude, in degrees */
/* elev - WGS84 altitude above ellipsoid (igdgc=1), or */
/* radial distance from earth's center (igdgc=2) */
/* a2,b2 - squares of semi-major and semi-minor axes of */
/* the reference spheroid used for transforming */
/* between geodetic and geocentric coordinates */
/* or components */
/* nmax - maximum degree and order of coefficients */
/* iext - external coefficients flag (=0 if none) */
/* ext1,2,3 - the three 1st-degree external coefficients */
/* (not used if iext = 0) */
/* */
/* Output: */
/* x - northward component */
/* y - eastward component */
/* z - vertically-downward component */
/* */
/* based on subroutine 'igrf' by D. R. Barraclough and S. R. C. Malin, */
/* report no. 71/1, institute of geological sciences, U.K. */
/* */
/* FORTRAN */
/* Norman W. Peddie */
/* USGS, MS 964, box 25046 Federal Center, Denver, CO. 80225 */
/* */
/* C */
/* C. H. Shaffer */
/* Lockheed Missiles and Space Company, Sunnyvale CA */
/* August 17, 1988 */
/* */
/****************************************************************************/
int shval3(igdgc, flat, flon, elev, nmax, gh, iext, ext1, ext2, ext3)
int igdgc;
double flat;
double flon;
double elev;
int nmax;
int gh;
int iext;
double ext1;
double ext2;
double ext3;
{
double earths_radius = 6371.2;
double dtr = 0.01745329;
double slat;
double clat;
double ratio;
double aa, bb, cc, dd;
double sd;
double cd;
double r;
double a2;
double b2;
double rr;
double fm,fn;
double sl[14];
double cl[14];
double p[119];
double q[119];
int ii,j,k,l,m,n;
int npq;
int ios;
double argument;
double power;
a2 = 40680631.59; /* WGS84 */
b2 = 40408299.98; /* WGS84 */
ios = 0;
r = elev;
argument = flat * dtr;
slat = sin( argument );
if ((90.0 - flat) < 0.001)
{
aa = 89.999; /* 300 ft. from North pole */
}
else
{
if ((90.0 + flat) < 0.001)
{
aa = -89.999; /* 300 ft. from South pole */
}
else
{
aa = flat;
}
}
argument = aa * dtr;
clat = cos( argument );
argument = flon * dtr;
sl[1] = sin( argument );
cl[1] = cos( argument );
switch(gh)
{
case 3: x = 0;
y = 0;
z = 0;
break;
case 4: xtemp = 0;
ytemp = 0;
ztemp = 0;
break;
default: printf("\nError in subroutine shval3");
break;
}
sd = 0.0;
cd = 1.0;
l = 1;
n = 0;
m = 1;
npq = (nmax * (nmax + 3)) / 2;
if (igdgc == 1)
{
aa = a2 * clat * clat;
bb = b2 * slat * slat;
cc = aa + bb;
argument = cc;
dd = sqrt( argument );
argument = elev * (elev + 2.0 * dd) + (a2 * aa + b2 * bb) / cc;
r = sqrt( argument );
cd = (elev + dd) / r;
sd = (a2 - b2) / dd * slat * clat / r;
aa = slat;
slat = slat * cd - clat * sd;
clat = clat * cd + aa * sd;
}
ratio = earths_radius / r;
argument = 3.0;
aa = sqrt( argument );
p[1] = 2.0 * slat;
p[2] = 2.0 * clat;
p[3] = 4.5 * slat * slat - 1.5;
p[4] = 3.0 * aa * clat * slat;
q[1] = -clat;
q[2] = slat;
q[3] = -3.0 * clat * slat;
q[4] = aa * (slat * slat - clat * clat);
for ( k = 1; k <= npq; ++k)
{
if (n < m)
{
m = 0;
n = n + 1;
argument = ratio;
power = n + 2;
rr = pow(argument,power);
fn = n;
}
fm = m;
if (k >= 5)
{
if (m == n)
{
argument = (1.0 - 0.5/fm);
aa = sqrt( argument );
j = k - n - 1;
p[k] = (1.0 + 1.0/fm) * aa * clat * p[j];
q[k] = aa * (clat * q[j] + slat/fm * p[j]);
sl[m] = sl[m-1] * cl[1] + cl[m-1] * sl[1];
cl[m] = cl[m-1] * cl[1] - sl[m-1] * sl[1];
}
else
{
argument = fn*fn - fm*fm;
aa = sqrt( argument );
argument = ((fn - 1.0)*(fn-1.0)) - (fm * fm);
bb = sqrt( argument )/aa;
cc = (2.0 * fn - 1.0)/aa;
ii = k - n;
j = k - 2 * n + 1;
p[k] = (fn + 1.0) * (cc * slat/fn * p[ii] - bb/(fn - 1.0) * p[j]);
q[k] = cc * (slat * q[ii] - clat/fn * p[ii]) - bb * q[j];
}
}
switch(gh)
{
case 3: aa = rr * gha[l];
break;
case 4: aa = rr * ghb[l];
break;
default: printf("\nError in subroutine shval3");
break;
}
if (m == 0)
{
switch(gh)
{
case 3: x = x + aa * q[k];
z = z - aa * p[k];
break;
case 4: xtemp = xtemp + aa * q[k];
ztemp = ztemp - aa * p[k];
break;
default: printf("\nError in subroutine shval3");
break;
}
l = l + 1;
}
else
{
switch(gh)
{
case 3: bb = rr * gha[l+1];
cc = aa * cl[m] + bb * sl[m];
x = x + cc * q[k];
z = z - cc * p[k];
if (clat > 0)
{
y = y + (aa * sl[m] - bb * cl[m]) *
fm * p[k]/((fn + 1.0) * clat);
}
else
{
y = y + (aa * sl[m] - bb * cl[m]) * q[k] * slat;
}
l = l + 2;
break;
case 4: bb = rr * ghb[l+1];
cc = aa * cl[m] + bb * sl[m];
xtemp = xtemp + cc * q[k];
ztemp = ztemp - cc * p[k];
if (clat > 0)
{
ytemp = ytemp + (aa * sl[m] - bb * cl[m]) *
fm * p[k]/((fn + 1.0) * clat);
}
else
{
ytemp = ytemp + (aa * sl[m] - bb * cl[m]) *
q[k] * slat;
}
l = l + 2;
break;
default: printf("\nError in subroutine shval3");
break;
}
}
m = m + 1;
}
if (iext != 0)
{
aa = ext2 * cl[1] + ext3 * sl[1];
switch(gh)
{
case 3: x = x - ext1 * clat + aa * slat;
y = y + ext2 * sl[1] - ext3 * cl[1];
z = z + ext1 * slat + aa * clat;
break;
case 4: xtemp = xtemp - ext1 * clat + aa * slat;
ytemp = ytemp + ext2 * sl[1] - ext3 * cl[1];
ztemp = ztemp + ext1 * slat + aa * clat;
break;
default: printf("\nError in subroutine shval3");
break;
}
}
switch(gh)
{
case 3: aa = x;
x = x * cd + z * sd;
z = z * cd - aa * sd;
break;
case 4: aa = xtemp;
xtemp = xtemp * cd + ztemp * sd;
ztemp = ztemp * cd - aa * sd;
break;
default: printf("\nError in subroutine shval3");
break;
}
return(ios);
}
/****************************************************************************/
/* */
/* Subroutine dihf */
/* */
/****************************************************************************/
/* */
/* Computes the geomagnetic d, i, h, and f from x, y, and z. */
/* */
/* Input: */
/* x - northward component */
/* y - eastward component */
/* z - vertically-downward component */
/* */
/* Output: */
/* d - declination */
/* i - inclination */
/* h - horizontal intensity */
/* f - total intensity */
/* */
/* FORTRAN */
/* A. Zunde */
/* USGS, MS 964, box 25046 Federal Center, Denver, CO. 80225 */
/* */
/* C */
/* C. H. Shaffer */
/* Lockheed Missiles and Space Company, Sunnyvale CA */
/* August 22, 1988 */
/* */
/****************************************************************************/
int dihf (gh)
int gh;
{
int ios;
int j;
double sn;
double h2;
double hpx;
double argument, argument2;
ios = gh;
sn = 0.0001;
switch(gh)
{
case 3: for (j = 1; j <= 1; ++j)
{
h2 = x*x + y*y;
argument = h2;
h = sqrt(argument); /* calculate horizontal intensity */
argument = h2 + z*z;
f = sqrt(argument); /* calculate total intensity */
if (f < sn)
{
d = NaN; /* If d and i cannot be determined, */
i = NaN; /* set equal to NaN */
}
else
{
argument = z;
argument2 = h;
i = atan2(argument,argument2);
if (h < sn)
{
d = NaN;
}
else
{
hpx = h + x;
if (hpx < sn)
{
d = PI;
}
else
{
argument = y;
argument2 = hpx;
d = 2.0 * atan2(argument,argument2);
}
}
}
}
break;
case 4: for (j = 1; j <= 1; ++j)
{
h2 = xtemp*xtemp + ytemp*ytemp;
argument = h2;
htemp = sqrt(argument);
argument = h2 + ztemp*ztemp;
ftemp = sqrt(argument);
if (ftemp < sn)
{
dtemp = NaN; /* If d and i cannot be determined, */
itemp = NaN; /* set equal to 999.0 */
}
else
{
argument = ztemp;
argument2 = htemp;
itemp = atan2(argument,argument2);
if (htemp < sn)
{
dtemp = NaN;
}
else
{
hpx = htemp + xtemp;
if (hpx < sn)
{
dtemp = PI;
}
else
{
argument = ytemp;
argument2 = hpx;
dtemp = 2.0 * atan2(argument,argument2);
}
}
}
}
break;
default: printf("\nError in subroutine dihf");
break;
}
return(ios);
}