File: simulateStarCameraSentinel1.cpp

package info (click to toggle)
groops 0%2Bgit20250907%2Bds-1
  • links: PTS, VCS
  • area: non-free
  • in suites: forky, sid
  • size: 11,140 kB
  • sloc: cpp: 135,607; fortran: 1,603; makefile: 20
file content (124 lines) | stat: -rw-r--r-- 4,319 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/***********************************************/
/**
* @file simulateStarCameraSentinel1.cpp
*
* @brief simulate star camera data for Sentinel 1A.
*
* @author Norbert Zehentner
* @date 2016-03-23
*/
/***********************************************/

// Latex documentation
#define DOCSTRING docstring
static const char *docstring = R"(
This program simulates \file{star camera}{instrument} measurements at each satellite's position for the Sentinel 1A satellite.
The \configFile{inputfileOrbit}{instrument} must contain positions and velocities (see \program{OrbitAddVelocityAndAcceleration}).
The resulting rotation matrices rotate from satellite frame to inertial frame.
)";

/***********************************************/

#include "programs/program.h"
#include "files/fileInstrument.h"
#ifndef GROOPS_DISABLE_ERFA
#include <erfa.h>
#endif

/***** CLASS ***********************************/

/** @brief simulate star camera data for Sentinel 1A.
* @ingroup programsGroup */
class SimulateStarCameraSentinel1
{
public:
  void run(Config &config, Parallel::CommunicatorPtr comm);
};

GROOPS_REGISTER_PROGRAM(SimulateStarCameraSentinel1, PARALLEL, "simulate star camera data for Sentinel 1", Simulation, Instrument)

/***********************************************/

void SimulateStarCameraSentinel1::run(Config &config, Parallel::CommunicatorPtr comm)
{
  try
  {
    FileName orbitName, starCameraName;

    readConfig(config, "outputfileStarCamera", starCameraName, Config::MUSTSET,  "", "");
    readConfig(config, "inputfileOrbit",       orbitName,      Config::MUSTSET, "", "position and velocity defines the orientation of the satellite at each epoch");
    if(isCreateSchema(config)) return;

    // StarCamera-Daten erzeugen
    // -----------------------
    logStatus<<"read orbit and generate star camera data"<<Log::endl;
    Ellipsoid      ellipsoid(DEFAULT_GRS80_a, DEFAULT_GRS80_f);
    InstrumentFile orbitFile(orbitName);

    std::vector<Arc> arcList(orbitFile.arcCount());
    Parallel::forEach(arcList, [&](UInt arcNo)
    {
      OrbitArc orbit = orbitFile.readArc(arcNo);
      StarCameraArc arc;
      for(UInt i=0; i<orbit.size(); i++)
      {
        // get rotation form inertial to TOD frame
        Double rc2i[3][3];
#ifdef GROOPS_DISABLE_ERFA
        throw(Exception("compiled without ERFA library"));
#else
        const Time timeTT = timeGPS2TT(orbit.at(i).time);
        Double X=0, Y=0, S=0;
        eraXys00b(2400000.5+timeTT.mjdInt(), timeTT.mjdMod(), &X, &Y, &S);
        eraC2ixys(X, Y, S, rc2i);
#endif
        Matrix R(3,3);
        R(0,0) = rc2i[0][0]; R(0,1) = rc2i[1][0]; R(0,2) = rc2i[2][0];
        R(1,0) = rc2i[0][1]; R(1,1) = rc2i[1][1]; R(1,2) = rc2i[2][1];
        R(2,0) = rc2i[0][2]; R(2,1) = rc2i[1][2]; R(2,2) = rc2i[2][2];
        Rotary3d rot(R);

        Vector3d pos = orbit.at(i).position;
        Vector3d vel = orbit.at(i).velocity;
        const Double omega = 0.729211585e-4;
        vel.x() +=  omega*pos.y(); // Correct velocity for earth rotation
        vel.y() += -omega*pos.x();
        // correct position
        // compute subsatellite point
        Angle  L, B;
        Double h;
        ellipsoid(pos, L, B, h); // FIXME rot.rotate(pos)???
        Vector3d posGround = ellipsoid(L, B, 0);
        Vector3d x         = normalize(vel);
        Rotary3d rotC2ZD   = Rotary3d(x, crossProduct(x, pos-posGround)) * rot;

        // part 2: rotation from zero-Doppler reference frame to sentinel 1 reference frame (body)
        ellipsoid(rot.rotate(pos), L, B, h); // compute height
        const Double   rThetaRef  = 0.513999;  // rad
        const Double   rAlphaRoll = 9.878564e-7; // rad/meter
        const Double   rHref      = 711700;
        const Rotary3d rot2       = rotaryX(Angle(rThetaRef - rAlphaRoll*(h-rHref)));

        StarCameraEpoch epoch;
        epoch.time   = orbit.at(i).time;
        epoch.rotary = rotC2ZD*rot2;
        arc.push_back(epoch);
      }
      return arc;
    }, comm); // forEach

    // write
    // -----
    if(Parallel::isMaster(comm))
    {
      logStatus<<"write star camera data to file <"<<starCameraName<<">"<<Log::endl;
      InstrumentFile::write(starCameraName, arcList);
    }
  }
  catch(std::exception &e)
  {
    GROOPS_RETHROW(e)
  }
}

/***********************************************/