-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmain.cpp
1835 lines (1723 loc) · 75.5 KB
/
main.cpp
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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/*******************************************************************************
main.cpp
scia_retrieval_2d
Two-dimensional trace gas retrieval from SCIAMACHY limb scans
Copyright (c) 2011-2017 Stefan Bender
Copyright (c) 2010-2011 Martin Langowski
Based on the FORTRAN version by Marco Scharringhausen, copyright (c) 2004-2008.
This program is free software: you can redistribute it or modify
it under the terms of the GNU General Public License as published
by the Free Software Foundation, version 2.
See accompanying COPYING.GPL2 file or http://www.gnu.org/licenses/gpl-2.0.html.
Dieses Programm ist eine C++ Version des Retrievalprogramms SCIA2D.f90,
welches von Marco Scharringhausen entwickelt wurde.
Zur Vergleichbarkeit sind die durchgeführten Rechnungen gleich, lediglich
das Layout des Programms soll etwas anders sein.
( Das stimmt schon garnicht mehr....
Streuwinkel werden anders berechnet, SZA auch, Das Raytracing ist 3D, d.h. die
längenänderung wird für die Schritte berücksichtigt
usw.....am Ende kommt aber dasselbe raus, d.h. die kleinen Bugs, die vorher
drin waren waren nicht sehr gravierend
Bei stärkerer Absorption wäre das aber schon der Fall)
Ich benutzte hier als integrierte Entwicklungsumgebung eclipse (kostenlos
erhältlich, für windows und linux) Auf Windows gibt es auch kostenlose
Versionen des Visual Studios (Express Versionen....da ham einen
verständlicheren Debugger) Natürlich kann man auch andere Editoren wie Kate,
oder vi verwenden, je nach Geschmack
Ein Ziel dieses Programms ist es, schnell zu sein. Ein anderes ist
Flexibilität, was in diesem Fall die möglichst schnelle und einfache
Anpassungfähigkeit des Programms an neue Spezies bedeutet. Letzteres soll
dadurch erreicht werden, dass diese wichtigen Veränderung nur im Hauptprogramm
durchgeführt werden müssen. ( Das hat auch geklappt, nur bei den Spezies in
Limbauswertung und Nadirauswertung, müssen einige Teile ergänzt werden, die
aber gleich zu den anderen aussehen) Weitere Teile, die ich oft ändere sind das
Retrievalgitter In Retrievaliteration, kann man die maximal Iterationszahl, bzw
den treshold heruntersetzen, wenns zu lange dauert
Um die kritischen Stellen leicht zu finden, soll der Code im Hauptprogramm
entsprechend kurz sein, das heißt, dass längere Passagen in möglichst klar
benannten Funktionen untergebracht sind, sodass im Hauptprogramm auch die
grobe Struktur der Lösung des Problems schnell nachvollzogen werden kann.
Schnelligkeit ist z.B. dadurch zu erreichen, dass Dateien, die mehrere
Datensätze enthalten, auch wenn dies nicht der intuitiven Abarbeitungslogik
entspricht, trotzdem nur einmal geladen, da dies sehr lange dauert. Das dauert
vor allem lange, wenn die Dateien nicht auf der richtigen Platte liegen, also
lokal speichern
Es sollte auch unbedingt vermieden werden, zu viel mit den Großen Feldern, die
aus den Dateien geladen werden "herumzujonglieren"(d.h. möglichst die großen
Felder nicht kopieren).....Das ist doch nicht so schlimm, das Laden so wie es
jetzt ist, dauert höchstens eine Sekunde, da kann man flexibler sein
Das Programm sollte zudem möglichst parallelisierbar sein, d.h. Gesamtprobleme
sollten so in Teilprobleme aufgeteilt werden, dass zur Lösung der Teilprobleme
die anderen gleichartigen Teilprobleme nicht bekannt sein müssen. Das Programm
wertet nur für einen Orbit aus....... Die einfachste Art der Parallelisierung,
ist mehrere Orbits parallel laufen zu lassen. Dadurch muss am Programm selbst
nicht so viel(eigentlich garnix) geändert werden. Ein umhüllendes Programm kann
dann geschrieben werden, welches die Orbits in gleich große Listen teilt, und
jeweils für teillisten seriell das Programm startet (siehe shell befehle fork,
exec, waitpid, system , für kompliziertere parallelisierungen mpi u.ä.
verwenden (ist aber glaube ich gar nicht nötig)
( Die veränderlichen Parameter in der Konf Datei müssen dem Programm Übergebbar
sein) Die Konf Datei ist ein Relikt aus der Umsetzung von Marcos Programm und
muss immer im Ordner sein, wo auch main.cpp liegt) Einträge dort werden aber
zum Teil überschrieben
Die Geschwindigkeitsoptimierung sollte nur die Flaschenhälse betreffen
Die Dateien auf der eigenen Platte zu haben ist für die Geschwindigkeit
bis jetzt (september 2010) der wichtigste Geschwindigkeitsfaktor damit wurde
auf ein BINÄRDATEIENSYSTEM UMGESTIEGEN (einfache Umwandlungsfunktionen hab ich
auch-> mail)
- unglücklicherweise habe ich Schrägen Säulendichten anfangs Zeilendichten
genannt...ist also hier dasselbe
- Das Raytracing in Teil 3 dauert recht lange. Da die Absorption Quasi keine
Rolle spielt, könnte man auch nur die Lichtwege nehmen. Die kann man
ausrechnen und braucht kein Raytracing
- Die Matrix selbst sollte auch nicht Luftmassenfaktoren Matrix oder Air
mass faktor Matrix heißen, weil das eigentlich ein andere Sachverhalt ist
(Quotient aus Schräger SÄule und Vertikaler Säule, oder so)
Dort, wo sich die Flexibilität mit der Geschwindigkeit beißt, muss abgewogen
werden, was wichtiger ist
//TODO Matrixmultiplikationen überprüfen, ob die Matrizen auch in einer
//günstigen Reihenfolge Multipliziert werden
//TODO Matrixoperationen sind z.t. für große Matrizen langsam, für 200*200
//Matrizen ist das Retrieval aber noch schnell Abhilfe schaffen hier
//Matrizen der LAPACK Bibliothek....da hab ich auch schon ein Programm was
//LAPACK und ATLAS(BLAS)
// benutzt und für eine 4000*4000 Matrix die LU-Zerlegung in 5 sekunden
// schafft (Matlab braucht da 8 für A\x)
//Sonnenzenitwinkel und Streuwinkel werden für jeden Gitterpunkt berechnet,
//und nicht für jeden Messschritt Da die Winkel innerhalb eines Gitterpunktes
//nahezu konstant sind und unterschiede bei der späteren verwendung in sin und
//cos eher noch kontrahiert werden...also vernachlässigbar sind....das
//betrifft Teil 3...wo die Absorption wie gesagt, eh fast zu vernachlässigen
//sit
Für überzeugende Plots, ob das Raytracing auch funktioniert ist es besser TAU
zu plotten als AMF Tau steigt nämlich mit zunehmendem Weg Monoton an...(bereits
implementiert)
Damit das Programm läuft, müssen einige Bibliotheken mit eingebunden werden
(am Besten in der Reihenfolge, also gfortran zuerst):
LimbNadir_IO
lapack
cblas
f77blas
atlas
gfortran
Der Haken ist, dass man ATLAS auf seinem eigenen Rechner installieren muss (was
bei Linux Programmen eine ziemliche Qual ist. Es gibt aber ein Manual
auf deren Seite dem man folgen kann und wenn man Glück hat, dann
funktioniert das sogar), da das auf die eigene Hardware optimiert, das
funktioniert auch nur unter Linux, noch schneller wäre intel mkl, kostet aber
geld
Ein letzter Hinweis,: nicht alle Kommentare in dem Quelltext müssen noch
aktuell sein, da so ein Programm ständigen Änderungen unterworfen ist Falls
irgendetwas nicht funktioniert, mich kontaktieren
*******************************************************************************/
//eingebundene Headerfiles
//standard
#include <string> // string
#include <cstdio> // cout
#include <iostream> // cout
#include <cstdlib> // für atoi
#include <sstream>
#include <iomanip>
#include <vector>
#include <glob.h>
//eigene
#include "Konfiguration.h"
#include "Liniendaten.h"
#include "Messung_Limb.h"
#include "Datei_IO.h"
#include "Speziesfenster.h"
#include "Orbitliste.h"
#include "Sonnenspektrum.h"
#include "Limbauswertung.h"
#include "Nadirauswertung.h"
#include "Nachricht_Schreiben.h"
#include <ctime> //langsames timing...so auf Sekunden genau
#include "Retrievalgitter.h"
#include "MPL_Matrix.h"
#include "Matrizen_Aufbauen.h"
#include "Retrievaliteration.h"
#include "Retrievalfehler_Abschaetzung.h"
#include "Ausdrucke.h" // Plots_Zusammenfassen
#include "Glaetten_2D.h"
#include "NO_emiss.h"
#include "Dateinamensteile_Bestimmen.h"
#include "Ausgewertete_Messung_Limb.h"
#include "Ausgewertete_Messung_Nadir.h"
/* override version with -DVERSION="<version>" cflags or cxxflags */
#ifndef VERSION
#define VERSION "unknown"
#endif
//===========================================================
// eigene externe Bibliotheken-> Müssen mitgeliefert werden,
// falls nicht dabei, mail an martin
// LimbNadir_IO // Bibliothek zum laden und speichern von Scia L1C Daten in
// // Ascii oder martins binärformat
// // wird beim einlesen der Rohdaten in Limbauswertung und
// // Nadirauswertung verwendet
// // und macht so die Funktionenen in Datei_IO.h z.T. überflüssig
//===========================================================
using std::cerr;
using std::cout;
using std::endl;
using std::string;
using std::vector;
///////////////////////////////////////
// START globale Variablen
//Nachrichtenprioritäten
int Prioritylevel = 0;
// ENDE globale Variablen
///////////////////////////////////////
//
// the SCIAMACHY slit function
double slit_func(double fwhm, double x0, double x)
{
const double fwhm2 = fwhm * fwhm;
const double cnorm_inv = fwhm2 * fwhm * 0.25 * M_1_PI * M_SQRT1_2;
// (0.5 * FWHM)^4
const double fwhm2to4 = 0.0625 * fwhm2 * fwhm2;
return cnorm_inv / (fwhm2to4 + std::pow(x0 - x, 4));
}
double slit_func_gauss(double fwhm, double x0, double x)
{
double s = fwhm / (2. * std::sqrt(2. * std::log(2.)));
double n = 1. / (std::sqrt(2. * M_PI) * s);
double dxs = (x - x0) / s;
return n * std::exp(-0.5 * dxs * dxs);
}
/* C++ wrapper for glob(3c) */
inline std::vector<std::string> glob(const std::string& pattern)
{
glob_t glob_result;
std::vector<std::string> ret;
glob(pattern.c_str(), 0, NULL, &glob_result);
for(unsigned int i = 0; i < glob_result.gl_pathc; ++i) {
ret.push_back(std::string(glob_result.gl_pathv[i]));
}
globfree(&glob_result);
return ret;
}
/* We try to determine the orbit solar spectrum by looking at the first line of
* the orbit list and extracting the path name from it. We use this then to
* construct a glob pattern for the solar spectrum of this orbit, which should
* reside in the same directory. */
std::vector<std::string> find_orb_sol_paths(Orbitliste orb_list, bool debug = false)
{
// read first (full) filename
std::string orb_path{orb_list.m_Dateinamen.front()};
// extract orbit number as string
std::string filename{orb_path.substr(orb_path.find_last_of("/\\"))};
std::string orb_num_str{filename.substr(filename.find_first_of(".") - 5, 5)};
// construct solar path glob pattern
std::string sol_glob_str{orb_path.substr(0, orb_path.find_last_of("/\\")) +
"/SCIA_solar_*_" + orb_num_str + ".dat"};
std::vector<std::string> orb_sol_path_list = glob(sol_glob_str);
// debug output
if (debug && !orb_sol_path_list.empty())
std::copy(orb_sol_path_list.begin(), orb_sol_path_list.end(),
std::ostream_iterator<std::string>(std::cerr, "\n"));
return orb_sol_path_list;
}
// argc ist Die Anzahl der Kommandozeilenparameter
// argv[i] enthält den i-ten Kommandozeilenparameter argv[0] ist Programmname
int main(int argc, char *argv[])
{
/* version information first */
std::cout << argv[0] << " version " << VERSION << std::endl;
//////////////////////////////////////////////////////////////////////////
// Übernahme der Kommandozeilenargumente
if (argc < 7) {
cout << "Falscher Programmaufruf von SCIA_RETRIEVAL_2D\n";
cout << "Aufruf: SCIA_RETRIEVAL_2D Orbitlistenpfad "
<< "Pfad_temporäres_Arbeitsverzeichnis "
<< "Pfad_SCIA_Sonnenspektrum "
<< "Pfad_Sonnenrefenzspektrum "
<< "Pfad_multips2pdf Pfad_multips2ps\n";
cout << "Bsp: SCIA_RETRIEVAL_2D /tmp/orbit.list /tmp "
<< "/home/meso/SCIA-DATA/SOLAR "
<< "/home/meso/SCIA-DATA/sao_solar_ref.dat "
<< "/home/martin/Skripts/multips2pdf "
<< "/home/martin/Skripts/multips2ps\n";
cout << "Programm wird abgebrochen\n";
return 1;
}
string Orbitlistenpfad = argv[1]; // um Konf zu überschreiben
string Arbeitsverzeichnis = argv[2]; // für Ausgaben in Datei(5 mal pro Spezies)
string Solarpfad = argv[3]; // um Konf zu überschreiben
string sol_refname = argv[4]; // solar ref for NO emission calculation
string Pfad_multips2pdf = argv[5];
string Pfad_multips2ps = argv[6];
string config_file = "SCIA2D.conf";
if (argc == 8)
config_file = argv[7];
cerr << "Orbitlistenpfad: " << Orbitlistenpfad << "\n";
cerr << "Arbeitsverzeichnis: " << Arbeitsverzeichnis << "\n";
cerr << "Solarpfad: " << Solarpfad << "\n";
cerr << "Solarreferenzpfad: " << sol_refname << "\n";
cerr << "Pfad_multips2pdf: " << Pfad_multips2pdf << "\n";
cerr << "Pfad_multips2ps: " << Pfad_multips2ps << "\n";
cerr << "config_file: " << config_file << "\n";
//vector<string> Zusaetzliche_Orbits;
//for (int i=0;i<Anzahl_zusaetzlicher_Orbits;i++) {
// Zusaetzliche_Orbits.push_back(argv[i+7]);
//}
Konfiguration Konf;
Konf.Konfiguration_einlesen(config_file);
Konf.Konfiguration_anzeigen(); //-> Test erfolgreich
//TODO ACHTUNG ACHUTNG
// Falls ja läuft das Laden der Limbdaten anders ab und
// die Ausschlusskriterien müssen anders angewendet werden;
string untersuche_limb_mesothermo_states = Konf.MLT ? "ja" : "nein";
string mache_Fit_Plots_limb = "nein";
string mache_Fit_Plots_nadir = "nein";
// string mache_Fit_Plots_MgI="ja"; // switches für einzelne Spezies
// Plotterei als Spezieseigenschaft, implementieren volles Retrieval auch
// string mache_Fit_Plots_MgII="nein";
// string mache_Fit_Plots_unknown="nein";
string mache_volles_Retrieval = "ja"; // Falls nicht, nach Teil 2 abbrechen
string mache_volles_Retrieval_MgI = "nein"; // switches für einzelne Spezies
string mache_volles_Retrieval_MgII = "nein";
string mache_volles_Retrieval_unknown = "nein";
string mache_volles_Retrieval_FeI = "nein";
string mache_volles_Retrieval_NO = "ja";
// Zu Arbeitsverzeichnis /////// für jede Spezies sollen 5 Dateien entstehen
//sssss_orbit_xxxxx_yyyymmdd_hhmm_Dichten.txt
//sssss_orbit_xxxxx_yyyymmdd_hhmm_Sx.txt
//sssss_orbit_xxxxx_yyyymmdd_hhmm_AKM.txt
//sssss_orbit_xxxxx_yyyy_mm_dd_hh_mm_nadir_Saeulen.txt
//sssss_orbit_xxxxx_yyyy_mm_dd_hh_mm_0limb_Saeulen.txt
// sssss -Spezies mit führenden 0en z.b. 00MgI
// xxxxx -Orbit_ID kriegt man aus Orbitlistenpfad ......../xxxxx.temp.list
// ...also von hinten zählen und dann substr
//yyyyymmdd_hhmm kriegt man aus dem Namen der ersten Datei in der Orbitliste
////////////////////////////////////////////////////////////////////////////
time_t start_zeit, timer1, deltaT;
time_t Teil1_Start, Teil1_End, T1_Dauer;
time_t Teil2_Start, Teil2_End, T2_Dauer;
time_t Teil3_Start, Teil3_End, T3_Dauer;
time_t Teil4_Start, Teil4_End, T4_Dauer;
time_t Teil5_Start, Teil5_End, T5_Dauer;
time_t Teil6_Start, Teil6_End, T6_Dauer;
time(&start_zeit);
Nachricht_Schreiben("Starte Hauptprogramm...", 3, Prioritylevel);
/***************************************************************************
TEIL 1 VORBEREITUNG
IN DIESEM PROGRAMMTEIL WERDEN DIE DATEN DER ZU UNTERSUCHENEN SPEZIES ANGEGEBEN.
DIE ORBITLISTE WIRD EINGELESEN UND DAS SONNENSPEKTRUM DES MESSTAGES WIRD EINGELESEN.
**************************************************************************/
cerr << "Teil1\n";
time(&Teil1_Start);
double wl; // central peak position wavelength
unsigned int i = 0, l = 0; // Meine Zählvariablen
//Konf mit argv Argumenten
Konf.m_Pfad_Datei_mit_Dateinamen_fuer_Messungen_eines_Orbits = Orbitlistenpfad;
Konf.m_Pfad_Solar_Spektrum = Solarpfad;
////////////////////////////////////////////////////////////////////////////
// Orbitliste Laden
//
// Die Orbitliste ist so zu erstellen, dass die erste Datei eine Limb Datei
// ist Da dieses Verfahren eh nur sinnvoll ist, wenn Limbmessungen
// vorhanden sind, sollte dies kein Problem sein
// TODO Die Orbitliste sollte nach Limb und Nadir sortiert werden. Danach
// muss sie nach Zeit sortiert sein( Wichtig für Einteilung der Boxen
// etc.... falls Ordner nach Dateinamen sortiert, so sollte das immer
// stimmen) Die Frage ist, ob man die Orbitliste so übernimmt, oder ob man
// hier nochmal sortiert. Bis jetzt ist die Orbitliste schon automatisch
// so. Allerdings wär eine Sortierung der Orbitliste ein geringer
// Zeitaufwand
//
////////////////////////////////////////////////////////////////////////////
Nachricht_Schreiben("Lade Orbitliste...", 3, Prioritylevel);
Orbitliste Orbitlist;
if (Orbitlist.Liste_Laden(Konf.m_Pfad_Datei_mit_Dateinamen_fuer_Messungen_eines_Orbits) != 0) {
cout << "Programmabbruch\n";
return -1;
}
// Zur Überprüfung Orbitliste in Datei schreiben
//Orbitlist.In_Datei_Speichern("CHECKDATA/Orbitliste_Ueberpruefung.txt");
// der ordner CHECKDATA muss vorher existieren
//cout<<Orbitlist.m_Dateinamen[Orbitlist.m_Dateinamen.size()-1];//
//TODO Das ist leer...Es muss beim erstellen der Orbitliste
// sichergestellt werden, dass die letzte Zeile kein\n enthält!!!!
// -> Funktion funktioniert... aber Orbitliste Datei muss ordentlich
// erzeugt sein
////////////////////////////////////////////////////////////////////////////
//
// Orbitliste Ist geladen
//
////////////////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////////////////
//
// Sonnenspektrum bestimmen
//
////////////////////////////////////////////////////////////////////////////
std::vector<std::string> orb_sol_paths = find_orb_sol_paths(Orbitlist);
if (Konf.m_Pfad_Solar_Spektrum == "auto" && !orb_sol_paths.empty()) {
Konf.m_Pfad_Solar_Spektrum = orb_sol_paths.front();
std::cerr << "Solarpfad: " << Konf.m_Pfad_Solar_Spektrum << std::endl;
}
Nachricht_Schreiben("Lade Sonnenspektrum...", 3, Prioritylevel);
Sonnenspektrum Solspec;
if (Solspec.Laden_SCIA(Konf.m_Pfad_Solar_Spektrum, Konf.m_Pfad_Solar_Fallback_Spektrum) != 0) {
cout << "Programmabbruch\n";
return -1;
}
//Überprüfen, ob einlesen erfolgreich war
//Solspec.Speichern_was_geladen_wurde("CHECKDATA/Sonne_so_wie_geladen.txt");
//ok -> funktioniert
Nachricht_Schreiben("Lade Referenzsonnenspektrum...", 3, Prioritylevel);
Sonnenspektrum sol_ref;
if (sol_ref.Laden_SCIA(sol_refname, Konf.m_Pfad_Solar_Fallback_Spektrum) != 0) {
cout << "Programmabbruch\n";
return -1;
}
////////////////////////////////////////////////////////////////////////////
//
// Sonnenspektrum ist bestimmt
//
////////////////////////////////////////////////////////////////////////////
// Fitparameter für alle Spezies vorbeiten
Speziesfenster Spez;
Liniendaten Lindat;
vector<Speziesfenster> Spezies_Fenster;
// Linienparameter für alle Spezies einlesen *******************************
// Dies könnte man später aus einer Datei auslesen /////////////////////////
// Die Wahl der Fensterdaten sollte Anhand von ausgewählten Plots von I/F
// um die Linie herum getroffen werden. Das in diesem Fall sehr viel
// einfacher, als eine Automatisierung
//
// TODO Falls das nochmehr spezies werden, dann Vektor von Vektor von
// Speziesfenster Hier werden die Wellenlängen in nm angegeben.(das nm hebt
// sich nacher mit der Multiplikation mit der Intervallbreite weg)
//
Nachricht_Schreiben("Bestimme Speziesdaten...", 3, Prioritylevel);
//Füllen z.B. für Magnesium+ *******************************************
Spez.m_Spezies_Name = "MgII"; //bitte zusammenhängend
Spez.plot_fit = false;
Spez.m_FWHM = Konf.m_FWHM;
//Wellenlaenge 1
wl = 279.553;
Spez.m_Wellenlaengen.push_back(wl);
Spez.m_Basisfenster_links_WLmin.push_back(wl - 3);
Spez.m_Basisfenster_links_WLmax.push_back(wl - 1);
Spez.m_Basisfenster_rechts_WLmin.push_back(wl + 1);
Spez.m_Basisfenster_rechts_WLmax.push_back(wl + 3);
Spez.m_Peakfenster_WLmin.push_back(wl - 3);
Spez.m_Peakfenster_WLmax.push_back(wl + 3);
Lindat.Einlesen(Konf.m_Pfad_Linienparameter_Metalle, wl);
Spez.m_Liniendaten.push_back(Lindat);
/*
//Wellenlänge 2
wl = 280.270;
Spez.m_Wellenlaengen.push_back(wl);
Spez.m_Basisfenster_links_WLmin.push_back(wl - 3);
Spez.m_Basisfenster_links_WLmax.push_back(wl - 1);
Spez.m_Basisfenster_rechts_WLmin.push_back(wl + 1);
Spez.m_Basisfenster_rechts_WLmax.push_back(wl + 3);
Spez.m_Peakfenster_WLmin.push_back(wl - 3);
Spez.m_Peakfenster_WLmax.push_back(wl + 3);
Lindat.Einlesen(Konf.m_Pfad_Linienparameter_Metalle, wl);
Spez.m_Liniendaten.push_back(Lindat);
// */
// weitere Wellenlängen
Spezies_Fenster.push_back(Spez);
// SpezVektoren wieder leeren
// hier könnte man in Spez auch eine Methode für schreiben
Spez.m_Wellenlaengen.resize(0);
Spez.m_Basisfenster_links_WLmin.resize(0);
Spez.m_Basisfenster_links_WLmax.resize(0);
Spez.m_Basisfenster_rechts_WLmin.resize(0);
Spez.m_Basisfenster_rechts_WLmax.resize(0);
Spez.m_Peakfenster_WLmin.resize(0);
Spez.m_Peakfenster_WLmax.resize(0);
Spez.m_Liniendaten.resize(0);
//weitere Spezies
// Magnesium ************************************************************
Spez.m_Spezies_Name = "MgI"; //bitte zusammenhängend
Spez.plot_fit = false;
Spez.m_FWHM = Konf.m_FWHM;
//Wellenlaenge 1
wl = 285.213;
Spez.m_Wellenlaengen.push_back(wl);
Spez.m_Basisfenster_links_WLmin.push_back(wl - 3);
//Spez.m_Basisfenster_links_WLmin.push_back(Spez.m_Wellenlaengen[0]-10);
Spez.m_Basisfenster_links_WLmax.push_back(wl - 1);
Spez.m_Basisfenster_rechts_WLmin.push_back(wl + 1);
Spez.m_Basisfenster_rechts_WLmax.push_back(wl + 3);
//Spez.m_Basisfenster_rechts_WLmax.push_back(Spez.m_Wellenlaengen[0]+4);
Spez.m_Peakfenster_WLmin.push_back(wl - 3);
Spez.m_Peakfenster_WLmax.push_back(wl + 3);
Lindat.Einlesen(Konf.m_Pfad_Linienparameter_Metalle, wl);
Spez.m_Liniendaten.push_back(Lindat);
// weitere Wellenlängen
//**************************************************************************
Spezies_Fenster.push_back(Spez);
//weitere Spezies
// SpezVektoren wieder leeren
// hier könnte man in Spez auch eine Methode für schreiben
Spez.m_Wellenlaengen.resize(0);
Spez.m_Basisfenster_links_WLmin.resize(0);
Spez.m_Basisfenster_links_WLmax.resize(0);
Spez.m_Basisfenster_rechts_WLmin.resize(0);
Spez.m_Basisfenster_rechts_WLmax.resize(0);
Spez.m_Peakfenster_WLmin.resize(0);
Spez.m_Peakfenster_WLmax.resize(0);
Spez.m_Liniendaten.resize(0);
Spez.clear(); // Instanz leeren
// Unbekannte Spezies ******************************************************
Spez.m_Spezies_Name = "unknown"; //bitte zusammenhängend
Spez.plot_fit = false;
Spez.m_FWHM = Konf.m_FWHM;
//Wellenlaenge 1
wl = 288.2;
Spez.m_Wellenlaengen.push_back(wl);
Spez.m_Basisfenster_links_WLmin.push_back(wl - 2);
//Spez.m_Basisfenster_links_WLmin.push_back(Spez.m_Wellenlaengen[0]-10);
Spez.m_Basisfenster_links_WLmax.push_back(wl - 1);
Spez.m_Basisfenster_rechts_WLmin.push_back(wl + 1);
Spez.m_Basisfenster_rechts_WLmax.push_back(wl + 3);
//Spez.m_Basisfenster_rechts_WLmax.push_back(Spez.m_Wellenlaengen[0]+4);
Spez.m_Peakfenster_WLmin.push_back(wl - 2);
Spez.m_Peakfenster_WLmax.push_back(wl + 3);
// Lindat sind unbekannt
Lindat.m_Wellenlaenge = wl;
Lindat.m_rel_Einstein = 1; // stimmt nicht
Lindat.m_f_Wert = 0.162; //0.162; // für Si
Lindat.m_E1 = 0; //0.01; //für Si
Lindat.m_E2 = 1; //0.99; //für Si
// Dies sind beliebige Werte, die auch nicht unbedingt auf I/F führen(prop
// faktoren wie pi r_elektron)...sondern auf beliebige werte
Spez.m_Liniendaten.push_back(Lindat);
// weitere Wellenlängen
//**************************************************************************
Spezies_Fenster.push_back(Spez);
//cout<<Spezies_Fenster.size();
// cout<<Spezies_Fenster[1].m_Liniendaten[0].m_E1<<"\t";
//weitere Spezies
// SpezVektoren wieder leeren
// hier könnte man in Spez auch eine Methode für schreiben
Spez.m_Wellenlaengen.resize(0);
Spez.m_Basisfenster_links_WLmin.resize(0);
Spez.m_Basisfenster_links_WLmax.resize(0);
Spez.m_Basisfenster_rechts_WLmin.resize(0);
Spez.m_Basisfenster_rechts_WLmax.resize(0);
Spez.m_Peakfenster_WLmin.resize(0);
Spez.m_Peakfenster_WLmax.resize(0);
Spez.m_Liniendaten.resize(0);
Spez.clear(); // Instanz leeren
// FeI ************************************************************
Spez.m_Spezies_Name = "FeI"; //bitte zusammenhängend
Spez.plot_fit = false;
Spez.m_FWHM = Konf.m_FWHM;
//Wellenlaenge 1
wl = 248.32707;
Spez.m_Wellenlaengen.push_back(wl);
Spez.m_Basisfenster_links_WLmin.push_back(wl - 1);
Spez.m_Basisfenster_links_WLmax.push_back(wl - 0.5);
Spez.m_Basisfenster_rechts_WLmin.push_back(wl + 1);
Spez.m_Basisfenster_rechts_WLmax.push_back(wl + 2);
Spez.m_Peakfenster_WLmin.push_back(wl - 1);
Spez.m_Peakfenster_WLmax.push_back(wl + 2);
// Liniendaten
Lindat.m_Wellenlaenge = wl;
Lindat.m_rel_Einstein = 0.19; // stimmt nicht
Lindat.m_f_Wert = 0.543;
// ^5D_4 nach ^5F*_5 übergang j=4, dj=1 (Funktion für Matlab geschrieben)
Lindat.m_E1 = 0.1733;
Lindat.m_E2 = 0.8267;
// Dies sind beliebige Werte, die auch nicht unbedingt auf I/F führen(prop
// faktoren wie pi r_elektron)...sondern auf beliebige werte
Spez.m_Liniendaten.push_back(Lindat);
// weitere Wellenlängen
//**************************************************************************
Spezies_Fenster.push_back(Spez);
// SpezVektoren wieder leeren
// hier könnte man in Spez auch eine Methode für schreiben
Spez.m_Wellenlaengen.resize(0);
Spez.m_Basisfenster_links_WLmin.resize(0);
Spez.m_Basisfenster_links_WLmax.resize(0);
Spez.m_Basisfenster_rechts_WLmin.resize(0);
Spez.m_Basisfenster_rechts_WLmax.resize(0);
Spez.m_Peakfenster_WLmin.resize(0);
Spez.m_Peakfenster_WLmax.resize(0);
Spez.m_Liniendaten.resize(0);
Spez.clear(); // Instanz leeren
//weitere Spezies
// NO stuff
// from config file
for (i = 0; i < Konf.no_NO_transitions; i++) {
Spez.m_Spezies_Name = "NO";
Spez.plot_fit = true;
NO_emiss NO(Konf.NO_v_u.at(i), Konf.NO_v_l.at(i),
Konf.NO_v_l_abs.at(i), Konf.atmo_Temp);
NO.get_solar_data(sol_ref);
NO.read_luque_data_from_file(Konf.m_Pfad_NO_parameters);
NO.calc_excitation();
NO.calc_line_emissivities();
std::cout << "NO transition " << i
<< ": v_u = " << NO.get_vu()
<< ", v_l = " << NO.get_vl()
<< ", v_l_abs = " << NO.get_vl_abs()
<< " at (initial) " << Konf.atmo_Temp << " K" << std::endl;
NO.print_line_emissivities();
//
Spez.NO_vec.push_back(NO);
wl = 246.9; // dummy, will be set later more accurately
Spez.m_Wellenlaengen.push_back(wl);
Spez.m_Basisfenster_links_WLmin.push_back(wl - 1);
Spez.m_Basisfenster_links_WLmax.push_back(wl - 0.5);
Spez.m_Basisfenster_rechts_WLmin.push_back(wl + 0.5);
Spez.m_Basisfenster_rechts_WLmax.push_back(wl + 1);
Spez.m_Peakfenster_WLmin.push_back(wl - 1);
Spez.m_Peakfenster_WLmax.push_back(wl + 1);
Spez.m_FWHM = Konf.m_FWHM;
// Liniendaten
Lindat.m_Wellenlaenge = wl;
Lindat.m_rel_Einstein = 1;
Lindat.m_f_Wert = 0.162;
// isotropic scattering
Lindat.m_E1 = 0;
Lindat.m_E2 = 1;
// half-isotropic scattering
//Lindat.m_E1 = 0.5;
//Lindat.m_E2 = 0.5;
// Rayleigh scattering with depolarization rho = 0.0295
// [Rozanov 2001, Scharringhausen 2008]
//Lindat.m_E1 = 0.9563932002956393;
//Lindat.m_E2 = 4.3606799704360676e-2;
Spez.m_Liniendaten.push_back(Lindat);
}
Spezies_Fenster.push_back(Spez);
// SpezVektoren wieder leeren
// hier könnte man in Spez auch eine Methode für schreiben
Spez.m_Wellenlaengen.resize(0);
Spez.m_Basisfenster_links_WLmin.resize(0);
Spez.m_Basisfenster_links_WLmax.resize(0);
Spez.m_Basisfenster_rechts_WLmin.resize(0);
Spez.m_Basisfenster_rechts_WLmax.resize(0);
Spez.m_Peakfenster_WLmin.resize(0);
Spez.m_Peakfenster_WLmax.resize(0);
Spez.m_Liniendaten.resize(0);
Spez.clear(); // Instanz leeren
//**************************************************************************
////////////////////////////////////////////////////////////////////////////
// Linienparameter für alle Spezies einlesen ende *************************
Nachricht_Schreiben("Speziesdaten bestimmt...", 3, Prioritylevel);
// Für jede Spezies einen Vector mit Messergebnissen erstellen
// Mg und Mg+
vector<Ausgewertete_Messung_Limb> Ausgewertete_Limbmessung_MgI;
vector<Ausgewertete_Messung_Limb> Ausgewertete_Limbmessung_MgII;
vector<Ausgewertete_Messung_Nadir> Ausgewertete_Nadirmessung_MgI;
vector<Ausgewertete_Messung_Nadir> Ausgewertete_Nadirmessung_MgII;
vector<Ausgewertete_Messung_Limb> Ausgewertete_Limbmessung_unknown;
vector<Ausgewertete_Messung_Nadir> Ausgewertete_Nadirmessung_unknown;
vector<Ausgewertete_Messung_Limb> Ausgewertete_Limbmessung_FeI;
vector<Ausgewertete_Messung_Nadir> Ausgewertete_Nadirmessung_FeI;
vector<Ausgewertete_Messung_Limb> Ausgewertete_Limbmessung_NO;
vector<Ausgewertete_Messung_Nadir> Ausgewertete_Nadirmessung_NO;
// Convolve high resolution solar spectra with the Sciamachy resolution
// function if the median wavelength spacing is below 0.05 nm.
// The typical wavelength spacing for Sciamachy spectra is about 0.1 nm.
std::vector<double> solwl_diffs;
std::adjacent_difference(Solspec.m_Wellenlaengen.begin(),
Solspec.m_Wellenlaengen.end(), std::back_inserter(solwl_diffs));
std::nth_element(solwl_diffs.begin(),
solwl_diffs.begin() + solwl_diffs.size() / 2, solwl_diffs.end());
if (solwl_diffs.at(solwl_diffs.size() / 2) < 0.05) {
std::cerr << "Binning high resolution solar spectrum." << std::endl;
Solspec.saoref_to_sciamachy();
}
time(&Teil1_End);
T1_Dauer = Teil1_End - Teil1_Start;
std::cerr << T1_Dauer << " s." << std::endl;
/***************************************************************************
ENDE TEIL 1
**************************************************************************/
/***************************************************************************
TEIL 2 BESTIMMUNG DER SÄULENDICHTEN
FÜR ALLE LIMB- UND NADIRMESSUNGEN EINES ORBITS WERDEN DIE SÄULENDICHTEN FÜR
JEDE LINIE BESTIMMT UND IN EINE EINFACHERE STRUKTUR(Ausgewertete_Messungen)
GESTECKT.
***************************************************************************/
cerr << "Teil2\n";
time(&Teil2_Start);
////////////////////////////////////////////////////////////////////////////
//
// Säulendichtenbestimmung
//
////////////////////////////////////////////////////////////////////////////
Nachricht_Schreiben("Bestimme Säulendichten...", 3, Prioritylevel);
// Aus jeder Messung egal ob Limb oder Nadir die Säulendichte für jede
// Linie jeder Spezies bestimmen
//counter für Qualitätscheck der Messung initialisieren
int counter_Nachtmessungen = 0;
//TODO ausgabe der counter später implementieren
int counter_NLC_detektiert = 0;
int counter_Richtungsvektor_nicht_ok = 0;
int counter_Nachtmessungen_Nadir = 0;
int counter_Nadir_Nacht_Dateien = 0;
/* After setting up the NO parameters, we don't need the highly
* resolved solar spectrum anymore. Instead, we use it to obtain
* the scaling factor from comparing it to the measured solar
* spectrum. Therefore, we degrade its resolution first. */
sol_ref.saoref_to_sciamachy();
// Aus jeder Messung egal ob Limb oder Nadir die Säulendichte für jede
// Linie jeder Spezies bestimmen
for (l = 0; l < Orbitlist.m_Dateinamen.size(); l++) {
string L1CDatei = Orbitlist.m_Dateinamen[l];
//cout<<L1CDatei<<"\n";
//Falls Limb-> Limbauslesung
if (Orbitlist.Ist_Messung_Limbmessung(l) == true) {
/////////////////////////////////////
//Limbauswertung
////////////////////////////////////
// Die erste Limbdatei ist wichtig für die Interpolation des
// Sonnenspektrums... es ist also wichtig, dass für l=0 eine
// Limbmessung vorliegt
//cerr<<"limbauswertung start\n";
//NotTODO Die Säulendichtebestimmung kann deutlich schneller
//geschehen, Verbesserungen hier sind aber irrelevant
Limb_Auswertung(Orbitlist, l, Solspec, sol_ref, Spezies_Fenster,
counter_Nachtmessungen, counter_NLC_detektiert,
counter_Richtungsvektor_nicht_ok,
Arbeitsverzeichnis, mache_Fit_Plots_limb,
untersuche_limb_mesothermo_states,
Ausgewertete_Limbmessung_MgI,
Ausgewertete_Limbmessung_MgII,
Ausgewertete_Limbmessung_unknown,
Ausgewertete_Limbmessung_FeI,
Ausgewertete_Limbmessung_NO,
Konf);
//cerr<<"limbauswertung Ende\n";
// Die Zwischenergebnisse stehen nun in
// Ausgewertete_Limbmessung_MgI und Ausgewertete_Limbmessung_MgII
/////////////////////////////////////
//Limbauswertung Ende
////////////////////////////////////
}//ende if (Orbitlist.Ist_Messung_Limbmessung(i)==0)
//Falls Nadir-> Nadirauslesung
//TODO Ladeunterroutine überprüfen
if (Orbitlist.Ist_Messung_Nadirmessung(l) == true) {
/////////////////////////////////////
//Nadirauswertung
/////////////////////////////////////
// TODO Die Zeilendichtebestimmung kann deutlich schneller
// geschehen...da werden viel zu viele unnötig große Schleifen
// verwendet
//cerr<<"Nadirauswertung start\n";
Nadir_Auswertung(Orbitlist, l, Solspec, Spezies_Fenster,
counter_Nachtmessungen_Nadir,
counter_Nadir_Nacht_Dateien,
Arbeitsverzeichnis, mache_Fit_Plots_nadir,
Ausgewertete_Nadirmessung_MgI,
Ausgewertete_Nadirmessung_MgII,
Ausgewertete_Nadirmessung_unknown,
Ausgewertete_Nadirmessung_FeI,
Ausgewertete_Nadirmessung_NO,
Konf);
/////////////////////////////////////
//Nadirauswertung Ende
////////////////////////////////////
}//ENDE if(Orbitlist.Ist_Messung_Nadirmessung(l)==0)
}//ende Schleife über l
//////// TEIL 2B/////////////////////
// Glätten der Ausgewerteten Messunge
/*
for (int i = 0; i < 3; i++) {
SCD_Glaettung(Ausgewertete_Limbmessung_MgI, 1, untersuche_limb_mesothermo_states);
SCD_Glaettung(Ausgewertete_Limbmessung_unknown, 1, untersuche_limb_mesothermo_states);
SCD_Glaettung(Ausgewertete_Limbmessung_MgII, 2, untersuche_limb_mesothermo_states);
SCD_Glaettung(Ausgewertete_Limbmessung_FeI, 1, untersuche_limb_mesothermo_states);
}
// */
//////// TEIL 2B/////////////////////
////////////////////////////////////////////////////////////////////////////
//
// Zeilendichtenbestimmung ist abgeschlossen
// Bis hierhin braucht das Programm keine Sekunde, wenn die ROHDATEN auf
// der LOKALEN Platte liegen.
//
////////////////////////////////////////////////////////////////////////////
Nachricht_Schreiben("Zeilendichten sind bestimmt....", 3, Prioritylevel);
////////////////////////////////////////////////////////////////////////////
//Ausgabe der Vektoren mit den Zwischenergebnissen in Dateien
// TODO sieht Ok aus...evtl später mal ein par fits angucken
string xxxxx = xxxxx_Bestimmen(Orbitlistenpfad);
string yyyymmdd_hhmm = yyyymmdd_hhmm_Bestimmen(Orbitlist.m_Dateinamen[0]);
string sssss_MgI = "00MgI";
string sssss_MgII = "0MgII";
string sssss_unknown = "unkno";
string sssss_FeI = "00FeI";
string sssss_NO = "000NO";
string Endung_Limb = "_0limb_Saeulen.txt";
string Endung_Limb_back = "_1limb_Saeulen.txt";
string Endung_Nadir = "_nadir_Saeulen.txt";
string Dateiout_Mittelteil = "_orbit_" + xxxxx + "_" + yyyymmdd_hhmm;
//sssss_orbit_xxxxx_yyyy_mm_dd_hh_mm_0limb_Saeulen.txt
string Pfad_Saeulen_Limb_MgI = Arbeitsverzeichnis + "/" + sssss_MgI + Dateiout_Mittelteil + Endung_Limb;
string Pfad_Saeulen_Limb_MgII = Arbeitsverzeichnis + "/" + sssss_MgII + Dateiout_Mittelteil + Endung_Limb;
string Pfad_Saeulen_Limb_unknown = Arbeitsverzeichnis + "/" + sssss_unknown + Dateiout_Mittelteil + Endung_Limb;
string Pfad_Saeulen_Limb_FeI = Arbeitsverzeichnis + "/" + sssss_FeI + Dateiout_Mittelteil + Endung_Limb;
string Pfad_Saeulen_Limb_NO = Arbeitsverzeichnis + "/" + sssss_NO + Dateiout_Mittelteil + Endung_Limb;
string Pfad_Saeulen_Limb_NO_back = Arbeitsverzeichnis + "/" + sssss_NO
+ Dateiout_Mittelteil + Endung_Limb_back;
string Pfad_Saeulen_Nadir_MgI = Arbeitsverzeichnis + "/" + sssss_MgI + Dateiout_Mittelteil + Endung_Nadir;
string Pfad_Saeulen_Nadir_MgII = Arbeitsverzeichnis + "/" + sssss_MgII + Dateiout_Mittelteil + Endung_Nadir;
string Pfad_Saeulen_Nadir_unknown = Arbeitsverzeichnis + "/" + sssss_unknown + Dateiout_Mittelteil + Endung_Nadir;
string Pfad_Saeulen_Nadir_FeI = Arbeitsverzeichnis + "/" + sssss_FeI + Dateiout_Mittelteil + Endung_Nadir;
string Pfad_Saeulen_Nadir_NO = Arbeitsverzeichnis + "/" + sssss_NO + Dateiout_Mittelteil + Endung_Nadir;
//cerr<<"Ausgabe_Saeulendichten\n";
Ausgabe_Saeulendichten(Pfad_Saeulen_Limb_MgI.c_str(), Ausgewertete_Limbmessung_MgI);
Ausgabe_Saeulendichten(Pfad_Saeulen_Limb_MgII.c_str(), Ausgewertete_Limbmessung_MgII);
Ausgabe_Saeulendichten(Pfad_Saeulen_Limb_unknown.c_str(), Ausgewertete_Limbmessung_unknown);
Ausgabe_Saeulendichten(Pfad_Saeulen_Limb_FeI.c_str(), Ausgewertete_Limbmessung_FeI);
Ausgabe_Saeulendichten(Pfad_Saeulen_Limb_NO.c_str(), Ausgewertete_Limbmessung_NO);
Ausgabe_Saeulendichten(Pfad_Saeulen_Nadir_MgI.c_str(), Ausgewertete_Nadirmessung_MgI);
Ausgabe_Saeulendichten(Pfad_Saeulen_Nadir_MgII.c_str(), Ausgewertete_Nadirmessung_MgII);
Ausgabe_Saeulendichten(Pfad_Saeulen_Nadir_unknown.c_str(), Ausgewertete_Nadirmessung_unknown);
Ausgabe_Saeulendichten(Pfad_Saeulen_Nadir_FeI.c_str(), Ausgewertete_Nadirmessung_FeI);
Ausgabe_Saeulendichten(Pfad_Saeulen_Nadir_NO.c_str(), Ausgewertete_Nadirmessung_NO);
//cerr<<"Plotdateien zusammenfassen\n";
//Plotdateien zusammenfassen
if ((mache_Fit_Plots_limb == "ja") || (mache_Fit_Plots_nadir == "ja")) {
vector<Speziesfenster>::iterator sfit;
for (sfit = Spezies_Fenster.begin(); sfit != Spezies_Fenster.end(); ++sfit) {
// Namen der Ausgabedatei zusammenschustern
string pdf_Datei = Arbeitsverzeichnis + "/Plots/Orbit_" + xxxxx
+ "_" + sfit->m_Spezies_Name + "Fits.pdf";
Plots_Zusammenfassen(Pfad_multips2pdf, Pfad_multips2ps, pdf_Datei,
sfit->m_Liste_der_Plot_Dateinamen);
}
}
////////////////////////////////////////////////////////////////////////////
time(&Teil2_End);
T2_Dauer = Teil2_End - Teil2_Start;
std::cerr << T2_Dauer << " s." << std::endl;
if (mache_volles_Retrieval != "ja") { // Falls nicht, nach Teil 2 abbrechen
// Hier könnt man echtmal ne sprunganweisung machen...ich verlass mich
// erstmal auf den garbage collector von c++
cerr << "Programm wird vorzeitig nach Säulendichtenbestimmung beendet\n";
return 0;
}
/***************************************************************************
ENDE TEIL 2
***************************************************************************/
// check for successfully analysed limb scans
// and return early if there are none.
if (Ausgewertete_Limbmessung_MgI.size() == 0) {
std::cout << "No usable limb scans found in this orbit, exiting." << std::endl;
return -1;
}
/***************************************************************************
TEIL 3 AUFBAU DER FÜR DAS RETRIEVAL BENÖTIGTEN MATRIZZEN
MEHRERE EINFACHE MATRIZZEN WERDEN BENÖTIGT.
DIE WICHTIGSTE UND KOMPLIZIERTESTE MATRIX IST DIE MATRIX FÜR DIE
LUFTMASSENFAKTOREN AMF.
***************************************************************************/
cerr << "Teil3\n";
time(&Teil3_Start);
////////////////////////////////////////////////////////////////////////////
// Matrizen und Vektoren aufbauen
////////////////////////////////////////////////////////////////////////////
/***************************************************************************
// Wir benötigen das Gitter, auf dem wir die Dichte ausrechnen
// Ab nun Vektoren und Matrizen für jede Spezies bestimmen
// einen Vektor der Dichte_n
// einen Vektor x_a für die a-priori-Lösung der Dichte (oder Startwert usw)
// einen Vektor der Zeilendichten
// Die Gesamtwichtungsfaktoren Lambda_H und Lambda_PHI
// zwei Matrizen S_H und S_LAT mit den Unrterschieden der
// Nachbarpunkte(zur Glättung) die Matrix K, in der das Absorptionsgesetz
// für die Lichtwege drinsteckt
// Kovarianzmatrizen S_y und S_a...die man aber als Diagonalvektor benutzt
***************************************************************************/
//das hier weg( das erlaubt nur Limbmessungen)
Ausgewertete_Nadirmessung_MgI.resize(0);
//das hier weg( das erlaubt nur Limbmessungen)
Ausgewertete_Nadirmessung_MgII.resize(0);
Ausgewertete_Nadirmessung_unknown.resize(0);
Ausgewertete_Nadirmessung_FeI.resize(0);
Retrievalgitter Grid;
//5.0 ist vernünftig; 2.0 ist gut für TAU_LOS_plot
double Mindestabstand_Lat_in_Grad = 4.0;
Grid.Retrievalgitter_erzeugen(Ausgewertete_Limbmessung_MgI,
Mindestabstand_Lat_in_Grad, Konf);
//Folgende Ausgabe sieht ok aus 28.9.2010 (Durchstoßpunkte werden erst
//später ermittelt)
string Pfad_Grid = Arbeitsverzeichnis + "/" + "Gitter.txt";
Grid.In_Datei_Ausgeben(Pfad_Grid);
// ein Vektor der totalen Anzahldichte
MPL_Matrix Dichte_n_tot(Grid.m_Anzahl_Punkte, 1); //Spaltenvektor
prepare_total_density(Grid, Dichte_n_tot, Ausgewertete_Limbmessung_NO);
////////////////////////////////////////////////////////////////////////////
// Spezies Mg I //
////////////////////////////////////////////////////////////////////////////
// einen Vektor der Dichte_n
MPL_Matrix Dichte_n_MgI; //Spaltenvektor
//Dichte_n_MgI.in_Datei_speichern("/tmp/mlangowski/0/Dichte_n_MgI");
// einen Vektor x_a für die a-priori-Lösung der Dichte (oder Startwert usw)
MPL_Matrix Dichte_apriori_MgI; //Spaltenvektor
// Vektor mit Zeilendichten für alle Messungen einer Spezies
MPL_Matrix Saeulendichten_MgI;
MPL_Matrix Saeulendichten_Fehler_MgI;
// verwendete Konfiguration bis 20.1.2011
//double MgI_Lambda_Hoehe= 5E-7;//5E-6;//5E-6;//5E-6; gute Werte 5 E-6
// TODO das laden der Parameter eindeutiger machen
// Konf.m_Retrieval_Kovarianzen[0+1*m_Anzahl_der_Emitter];
//double MgI_Lambda_Breite= 5E-6;//1E-6;//1E-7;//1E-7; gute Werte 1 E-6
//double MgI_Lambda_apriori= 5E-5;//5E-6 oder -5;
//double MgI_Lambda_Hoehe= 5E-7; // Gut für 3 km Schirtte mit 30 Höhen
//double MgI_Lambda_Breite= 5E-6; // Gut für 3 km Schirtte mit 30 Höhen
//double MgI_Lambda_apriori=5E-5; // Gut für 3 km Schirtte mit 30 Höhen
//double MgI_Lambda_Hoehe= 1E-5; // Gut für 1 km Schirtte mit 82 Höhen
//double MgI_Lambda_Breite= 1E-5;
//double MgI_Lambda_apriori=5E-5;
//double MgI_Lambda_letzte_Hoehe=100;
// double MgI_Lambda_Hoehe= 1E-5;
// double MgI_Lambda_Breite= 1E-5;
// double MgI_Lambda_apriori=5E-5;
// double MgI_Lambda_letzte_Hoehe=1E-32; //100 für ein
// Spezies Index for MgI
int spez_index = 1;
double MgI_Lambda_letzte_Hoehe = 1E-32; //100 für ein
// take the lambdas from the config file, no need to re-compile
double MgI_Lambda_apriori
= Konf.m_Retrieval_Kovarianzen[spez_index + 0 * Konf.m_Anzahl_der_Emitter];
double MgI_Lambda_Breite
= Konf.m_Retrieval_Kovarianzen[spez_index + 1 * Konf.m_Anzahl_der_Emitter];
double MgI_Lambda_Hoehe
= Konf.m_Retrieval_Kovarianzen[spez_index + 2 * Konf.m_Anzahl_der_Emitter];
cout << "Retrieval Kovarianzen MgI:" << endl;
cout << "L_apriori = " << MgI_Lambda_apriori << ", "
<< "L_Breite = " << MgI_Lambda_Breite << ", "
<< "L_Höhe = " << MgI_Lambda_Hoehe << endl;
/* Memory allocation happens in Matrizen_Aufbauen()
* for the species to be retrieved. */
// Speziesunabhängige Matrizen, alle sind quadratisch
MPL_Matrix S_Breite;
MPL_Matrix S_Hoehe;
MPL_Matrix S_letzte_Hoehe_MgI;
// Messfehlermatrix S_y y*y
// quadratische matrix
MPL_Matrix S_y_MgI;
MPL_Matrix S_apriori_MgI;
// AMF_Matrix erstellen // Zeilen wie y spalten wie x
MPL_Matrix AMF_MgI;
// S_Breite, S_Hoehe, S_Apriori, S_y, AMF aufbauen
//cerr<<"MgI Matrizen aufbaun\n";
if (mache_volles_Retrieval_MgI == "ja") {
int IERR = 0;
Matrizen_Aufbauen(S_Breite, S_Hoehe, S_letzte_Hoehe_MgI,
MgI_Lambda_letzte_Hoehe,
S_apriori_MgI, S_y_MgI,
AMF_MgI, MgI_Lambda_apriori,
// TODO 1 ist MgI, 0 ist MgII hier wieder korrigieren
Spezies_Fenster[spez_index],
Grid,
Ausgewertete_Limbmessung_MgI,
Ausgewertete_Nadirmessung_MgI,
Konf, IERR);
if (IERR != 0) {
cerr << "Fehler bei Matrizen_Aufbauen\n";
return -1; //Hauptprogramm beenden
}
/* memory allocation for the retrieval */
Dichte_n_MgI = MPL_Matrix(Grid.m_Anzahl_Punkte, 1);
Dichte_apriori_MgI = MPL_Matrix(Grid.m_Anzahl_Punkte, 1);
// Säulendichten und Fehler auffüllen (Fehler für Wichtungsmatrixberechnung)
Saeulendichten_MgI = MPL_Matrix(Ausgewertete_Limbmessung_MgI.size()