// // Author: Joshua Cohen // Copyright 2016 // #include #include #include #include #include #include "Constants.h" #include "Orbit.h" using std::getline; using std::ifstream; using std::ofstream; using std::showpos; using std::string; using std::vector; // Default constructor Orbit::Orbit() : position(0), velocity(0), UTCtime(0) { nVectors = 0; basis = 0; } // Unfortunately due to the way the algorithm works it will set up an Orbit and resize the internal // vectors later void Orbit::setOrbit(int nvec, int bs) { position.resize(nvec * 3); velocity.resize(nvec * 3); UTCtime.resize(nvec); nVectors = nvec; basis = bs; } // Can set the Orbit by reading in a CSV HDR file (size=(nVec x 7)) void Orbit::setOrbit(const char *filename, int bs) { string line; vector pos(3), vel(3); double t; int count = 0; nVectors = 0; basis = bs; ifstream fs(filename); if (!fs.is_open()) { printf("Error in Orbit::Orbit - Unable to open HDR file: %s\n", filename); exit(1); } // Rapid iterator to count number of lines safely while (getline(fs,line)) ++nVectors; position.resize(nVectors * 3); velocity.resize(nVectors*3); UTCtime.resize(nVectors); // Reset filestream before reading lines fs.clear(); fs.seekg(0); // Take advantage of filestream overridden >>/<< operators while (fs >> t >> pos[0] >> pos[1] >> pos[2] >> vel[0] >> vel[1] >> vel[2]) { setStateVector(count,t,pos,vel); count++; } fs.close(); printf("Read in %d State Vectors from %s\n", nVectors, filename); } void Orbit::getPositionVelocity(double tintp, vector &pos, vector &vel) { if (basis == WGS84_ORBIT) interpolateWGS84Orbit(tintp, pos, vel); else interpolateSCHOrbit(tintp, pos, vel); } void Orbit::setStateVector(int idx, double t, vector &pos, vector &vel) { if ((idx >= nVectors) || (idx < 0)) { printf("Error in Orbit::setStateVector - Trying to set state vector %d out of %d\n", idx, nVectors); exit(1); } UTCtime[idx] = t; for (int i=0; i<3; i++) { position[(3*idx)+i] = pos[i]; velocity[(3*idx)+i] = vel[i]; } } void Orbit::getStateVector(int idx, double &t, vector &pos, vector &vel) { if ((idx >= nVectors) || (idx < 0)) { printf("Error in Orbit::getStateVector - Trying to get state vector %d out of %d\n", idx, nVectors); exit(1); } t = UTCtime[idx]; for (int i=0; i<3; i++) { pos[i] = position[(3*idx)+i]; vel[i] = velocity[(3*idx)+i]; } } // Common interface for orbit interpolation (avoid setting function pointers in main controller int Orbit::interpolateOrbit(double tintp, vector &opos, vector &ovel, int method) { int ret; if (method == HERMITE_METHOD) ret = interpolateWGS84Orbit(tintp,opos,ovel); else if (method == SCH_METHOD) ret = interpolateSCHOrbit(tintp,opos,ovel); else if (method == LEGENDRE_METHOD) ret = interpolateLegendreOrbit(tintp,opos,ovel); else { printf("Error in Orbit::interpolateOrbit - Invalid orbit interpolation method.\n"); exit(1); } return ret; } int Orbit::interpolateSCHOrbit(double tintp, vector &opos, vector &ovel) { vector > pos(2,vector(3)), vel(2,vector(3)); vector t(2); double frac,num,den; if (nVectors < 2) { printf("Error in Orbit::interpolateSCHOrbit - Need at least 2 state vectors for SCH orbit interpolation.\n"); exit(1); } if ((tintp < UTCtime[0]) || (tintp > UTCtime[nVectors-1])) { printf("Error in Orbit::interpolateSCHOrbit - Requested epoch outside orbit state vector span.\n"); exit(1); } for (int i=0; i<3; i++) { opos[i] = 0.0; ovel[i] = 0.0; } for (int i=0; i &opos, vector &ovel) { vector > pos(4,vector(3)), vel(4,vector(3)); vector t(4); int ii; if (nVectors < 4) return 1; for (int i=0; i= tintp) break; } ii = ii - 2; if (ii < 0) ii = 0; if (ii > (nVectors - 4)) ii = (nVectors - 4); for (int j=0; j<4; j++) getStateVector((ii+j),t[j],pos[j],vel[j]); orbitHermite(pos,vel,t,tintp,opos,ovel); if ((tintp < UTCtime[0]) || (tintp > UTCtime[(nVectors-1)])) return 1; else return 0; } int Orbit::interpolateLegendreOrbit(double tintp, vector &opos, vector &ovel) { vector > pos(9,vector(3)), vel(9,vector(3)); vector t(9); double nmr[] = {40320.0, -5040.0, 1440.0, -720.0, 576.0, -720.0, 1440.0, -5040.0, 40320.0}; vector noemer(nmr,nmr+9); double trel, coeff, teller; int ii; for (int i=0; i<3; i++) { opos[i] = 0.0; ovel[i] = 0.0; } if (nVectors < 9) return 1; for (int i=0; i= tintp) break; } ii = ii - 5; if (ii < 0) ii = 0; if (ii > (nVectors - 9)) ii = (nVectors - 9); for (int j=0; j<9; j++) getStateVector((ii+j),t[j],pos[j],vel[j]); trel = (8.0 * (tintp - t[0])) / (t[8] - t[0]); teller = 1.0; for (int j=0; j<9; j++) teller = teller * (trel - j); if (teller == 0.0) { int i = int(trel); for (int j=0; j<3; j++) { opos[j] = pos[i][j]; ovel[j] = vel[i][j]; } } else { for (int i=0; i<9; i++) { coeff = (teller / noemer[i]) / (trel - i); for (int j=0; j<3; j++) { opos[j] = opos[j] + (coeff * pos[i][j]); ovel[j] = ovel[j] + (coeff * vel[i][j]); } } } if ((tintp < UTCtime[0]) || (tintp > UTCtime[(nVectors-1)])) return 1; else return 0; } int Orbit::computeAcceleration(double tintp, vector &acc) { vector xbef(3), vbef(3), xaft(3), vaft(3); double temp; int stat; for (int i=0; i<3; i++) acc[i] = 0.0; temp = tintp - 0.01; stat = interpolateWGS84Orbit(temp, xbef, vbef); if (stat != 0) return 1; temp = tintp + 0.01; stat = interpolateWGS84Orbit(temp, xaft, vaft); if (stat != 0) return 1; for (int i=0; i<3; i++) acc[i] = (vaft[i] - vbef[i]) / 0.02; return 0; } void Orbit::orbitHermite(vector > &x, vector > &v, vector &t, double time, vector &xx, vector &vv) { vector h(4,0.), hdot(4,0.), f0(4,0.), f1(4,0.), g0(4,0.), g1(4,0.); double sum, product; int n1 = 4; int n2 = 3; for (int i=0; i