diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index 877225617a805a9e7dc0f8c5225cbfe083c0bb8b..c89e989d5612562bd824949cbd8171fe93845a0d 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -17,7 +17,13 @@ build:
     - cd build
     - cmake ..
     - cmake --build .
-    - ctest
+    - ctest -V >& test.log
+    - gzip -9 test.log
+  artifacts:
+    paths:
+      - build/test.log.gz
+    when: on_failure
+
 
 pages:
   stage: build
diff --git a/CMakeLists.txt b/CMakeLists.txt
index f0a9f08988fefa78050cc9b625ebe2149df55a03..d780315d8939bee82fdb79fb09ecff9e4178d608 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -37,10 +37,14 @@ if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES)
 endif()
 
 # enable warnings and disallow non-standard language
-set(CMAKE_CXX_FLAGS "-Wall -pedantic -Wextra -Wno-ignored-qualifiers -Wno-nonportable-include-path")
+set(CMAKE_CXX_FLAGS "-Wall -pedantic -Wextra -Wno-ignored-qualifiers")
 set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS} -O0 -g")
 set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS} -O3")
 
+# clang produces a lot of unecessary warnings without this:
+add_compile_options("$<$<CXX_COMPILER_ID:Clang>:-Wno-nonportable-include-path>") 
+
+  
 # unit testing coverage, does not work yet
 #include (CodeCoverage)
 ##set(COVERAGE_LCOV_EXCLUDES 'Documentation/*')
diff --git a/Documentation/Examples/cascade_example.cc b/Documentation/Examples/cascade_example.cc
index f3ebb57d541bf585312f47083d92b745bf82fc36..a76ce24b4b2dcbb1f70a583a296b00372e90ff96 100644
--- a/Documentation/Examples/cascade_example.cc
+++ b/Documentation/Examples/cascade_example.cc
@@ -80,6 +80,9 @@ public:
       case Code::Electron:
         is_em = true;
         break;
+      case Code::Positron:
+        is_em = true;
+        break;
       case Code::Gamma:
         is_em = true;
         break;
@@ -234,18 +237,27 @@ int main() {
   // setup particle stack, and add primary particle
   setup::Stack stack;
   stack.Clear();
-  const hep::EnergyType E0 = 10_TeV;
+  const hep::EnergyType E0 = 100_TeV;
+  double theta = 0.;
+  double phi = 0.;
   {
     auto particle = stack.NewParticle();
     particle.SetPID(Code::Proton);
     hep::MomentumType P0 = sqrt(E0 * E0 - Proton::GetMass() * Proton::GetMass());
-    auto plab = stack::super_stupid::MomentumVector(rootCS, 0_GeV, 0_GeV, -P0);
+    auto momentumComponents = [](double theta, double phi, MomentumType& ptot) {
+      return std::make_tuple(ptot * sin(theta) * cos(phi), ptot * sin(theta) * sin(phi),
+                             -ptot * cos(theta));
+    };
+    auto const [px, py, pz] =
+        momentumComponents(theta / 180. * M_PI, phi / 180. * M_PI, P0);
+    auto plab = stack::super_stupid::MomentumVector(rootCS, {px, py, pz});
+    cout << "input angles: theta=" << theta << " phi=" << phi << endl;
+    cout << "input momentum: " << plab.GetComponents() / 1_GeV << endl;
     particle.SetEnergy(E0);
     particle.SetMomentum(plab);
     particle.SetTime(0_ns);
     Point p(rootCS, 0_m, 0_m, 0_m);
     particle.SetPosition(p);
-    cout << particle.GetEnergy() / 1_GeV << endl;
   }
 
   // define air shower object, run simulation
diff --git a/Processes/Sibyll/Decay.h b/Processes/Sibyll/Decay.h
index 3d118d2180778de5225910e47d09ed4786c36f67..74bc02418b91f993c0d8ef3017c3dc91b52bd642 100644
--- a/Processes/Sibyll/Decay.h
+++ b/Processes/Sibyll/Decay.h
@@ -21,6 +21,8 @@
 
 #include <corsika/particles/ParticleProperties.h>
 
+#include <fenv.h>
+
 namespace corsika::process {
 
   namespace sibyll {
@@ -143,9 +145,19 @@ namespace corsika::process {
             corsika::particles::GetLifetime(p.GetPID());
         auto const lifetime = gamma * t0;
 
+        const auto mkin =
+            (E * E - p.GetMomentum().squaredNorm()); // delta_mass(p.GetMomentum(), E, m);
         cout << "Decay: code: " << p.GetPID() << endl;
         cout << "Decay: MinStep: t0: " << t0 << endl;
         cout << "Decay: MinStep: energy: " << E / 1_GeV << " GeV" << endl;
+        cout << "Decay: momentum: " << p.GetMomentum().GetComponents() / 1_GeV << " GeV"
+             << endl;
+        cout << "Decay: momentum: shell mass-kin. inv. mass " << mkin / 1_GeV / 1_GeV
+             << " " << m / 1_GeV * m / 1_GeV << endl;
+        // cout << "Decay: sib mass: " << s_mass1_.am2[
+        // process::sibyll::ConvertToSibyllRaw(p.GetPID()) ] << endl;
+        auto sib_id = process::sibyll::ConvertToSibyllRaw(p.GetPID());
+        cout << "Decay: sib mass: " << get_sibyll_mass2(sib_id) << endl;
         cout << "Decay: MinStep: gamma: " << gamma << endl;
         cout << "Decay: MinStep: tau: " << lifetime << endl;
 
@@ -156,6 +168,9 @@ namespace corsika::process {
       void DoDecay(Particle& p, Stack& s) {
         using corsika::geometry::Point;
         using namespace corsika::units::si;
+
+        feenableexcept(FE_INVALID);
+
         fCount++;
         SibStack ss;
         ss.Clear();
@@ -165,6 +180,10 @@ namespace corsika::process {
         pin.SetPID(process::sibyll::ConvertToSibyllRaw(pCode));
         pin.SetEnergy(p.GetEnergy());
         pin.SetMomentum(p.GetMomentum());
+        // setting particle mass with Corsika values, may be inconsistent with sibyll
+        // internal values
+#warning setting particle mass with Corsika values, may be inconsistent with sibyll internal values
+        pin.SetMass(corsika::particles::GetMass(pCode));
         // remember position
         Point decayPoint = p.GetPosition();
         TimeType t0 = p.GetTime();
@@ -181,15 +200,12 @@ namespace corsika::process {
         // print output
         int print_unit = 6;
         sib_list_(print_unit);
+
         // copy particles from sibyll stack to corsika
-        int i = -1;
         for (auto& psib : ss) {
-          ++i;
           // FOR NOW: skip particles that have decayed in Sibyll, move to iterator?
-          if (abs(s_plist_.llist[i]) > 100) continue;
+          if (psib.HasDecayed()) continue;
           // add to corsika stack
-          // cout << "decay product: " << process::sibyll::ConvertFromSibyll(
-          // psib.GetPID() ) << endl;
           auto pnew = s.NewParticle();
           pnew.SetEnergy(psib.GetEnergy());
           pnew.SetPID(process::sibyll::ConvertFromSibyll(psib.GetPID()));
@@ -199,6 +215,8 @@ namespace corsika::process {
         }
         // empty sibyll stack
         ss.Clear();
+
+        fedisableexcept(FE_INVALID);
       }
     };
   } // namespace sibyll
diff --git a/Processes/Sibyll/Interaction.h b/Processes/Sibyll/Interaction.h
index 2b03b26009c1d6f0e4f9c9be75d9452d2715124f..e9edec807e1ba1df784f5a6ecd15517ddbde7425 100644
--- a/Processes/Sibyll/Interaction.h
+++ b/Processes/Sibyll/Interaction.h
@@ -27,10 +27,14 @@ namespace corsika::process::sibyll {
   class Interaction : public corsika::process::InteractionProcess<Interaction> {
 
     mutable int fCount = 0;
+    mutable int fNucCount = 0;
 
   public:
     Interaction() {}
-    ~Interaction() { std::cout << "Sibyll::Interaction n=" << fCount << std::endl; }
+    ~Interaction() {
+      std::cout << "Sibyll::Interaction n=" << fCount << " Nnuc=" << fNucCount
+                << std::endl;
+    }
 
     void Init() {
 
@@ -147,15 +151,28 @@ namespace corsika::process::sibyll {
         // sibyll CS has z along particle momentum
         // FOR NOW: hard coded z-axis for corsika frame
         QuantityVector<length_d> const zAxis{0_m, 0_m, 1_m};
-        QuantityVector<length_d> const xAxis{1_m, 0_m, 0_m};
-        [[maybe_unused]] auto pt = [](MomentumVector p) {
-          return sqrt(p.GetComponents()[0] * p.GetComponents()[0] +
-                      p.GetComponents()[1] * p.GetComponents()[1]);
+        QuantityVector<length_d> const yAxis{0_m, 1_m, 0_m};
+        auto rotation_angles = [](MomentumVector const& pin) {
+          const auto p = pin.GetComponents();
+          const auto th = acos(p[2] / p.norm());
+          const auto ph = atan2(
+              p[1] / 1_GeV, p[0] / 1_GeV); // acos( p[0] / sqrt(p[0]*p[0]+p[1]*p[1] ) );
+          return std::make_tuple(th, ph);
         };
-        double theta = acos(p.GetMomentum().GetComponents()[2] / p.GetMomentum().norm());
-        cout << "ProcessSibyll: polar angle between sibyllCS and rootCS: " << theta
-             << endl;
-        CoordinateSystem sibyllCS = rootCS.rotate(xAxis, theta);
+        // auto pt = []( MomentumVector &p ){
+        // 	    return sqrt(p.GetComponents()[0] * p.GetComponents()[0] +
+        // p.GetComponents()[1] * p.GetComponents()[1]);
+        // 	  };
+        // double theta = acos( p.GetMomentum().GetComponents()[2] /
+        // p.GetMomentum().norm());
+        auto const [theta, phi] = rotation_angles(p.GetMomentum());
+        cout << "ProcessSibyll: zenith angle between sibyllCS and rootCS: "
+             << theta / M_PI * 180. << endl;
+        cout << "ProcessSibyll: azimuth angle between sibyllCS and rootCS: "
+             << phi / M_PI * 180. << endl;
+        // double phi = asin( p.GetMomentum().GetComponents()[0]/pt(p.GetMomentum() ) );
+        const CoordinateSystem tempCS = rootCS.rotate(zAxis, phi);
+        const CoordinateSystem sibyllCS = tempCS.rotate(yAxis, theta);
 
         /*
            the target should be defined by the Environment,
@@ -232,6 +249,7 @@ namespace corsika::process::sibyll {
           // print final state
           int print_unit = 6;
           sib_list_(print_unit);
+          fNucCount += get_nwounded() - 1;
 
           // delete current particle
           p.Delete();
diff --git a/Processes/Sibyll/SibStack.h b/Processes/Sibyll/SibStack.h
index 3364e7deac87ca2fda683d188aeaf3495eaa1fb4..bc3dc4c0fba66fadba4e277c0e305f01b65b1131 100644
--- a/Processes/Sibyll/SibStack.h
+++ b/Processes/Sibyll/SibStack.h
@@ -39,6 +39,11 @@ namespace corsika::process::sibyll {
       using namespace corsika::units::hep;
       s_plist_.p[3][i] = v / 1_GeV;
     }
+    void SetMass(const int i, const corsika::units::hep::MassType v) {
+      using namespace corsika::units::hep;
+      s_plist_.p[4][i] = v / 1_GeV;
+    }
+
     void SetMomentum(const int i, const MomentumVector& v) {
       using namespace corsika::units;
       using namespace corsika::units::hep;
@@ -52,6 +57,10 @@ namespace corsika::process::sibyll {
       using namespace corsika::units::hep;
       return s_plist_.p[3][i] * 1_GeV;
     }
+    corsika::units::hep::EnergyType GetMass(const int i) const {
+      using namespace corsika::units::hep;
+      return s_plist_.p[4][i] * 1_GeV;
+    }
 
     MomentumVector GetMomentum(const int i) const {
       using corsika::geometry::CoordinateSystem;
@@ -91,6 +100,14 @@ namespace corsika::process::sibyll {
     corsika::units::hep::EnergyType GetEnergy() const {
       return GetStackData().GetEnergy(GetIndex());
     }
+
+    void SetMass(const corsika::units::hep::MassType v) {
+      GetStackData().SetMass(GetIndex(), v);
+    }
+    corsika::units::hep::EnergyType GetMass() const {
+      return GetStackData().GetMass(GetIndex());
+    }
+
     bool HasDecayed() const {
       return abs(GetStackData().GetId(GetIndex())) > 100 ? true : false;
     }
diff --git a/Processes/Sibyll/sibyll2.3c.cc b/Processes/Sibyll/sibyll2.3c.cc
index 630b913c66d444bf5311e272b5640759eef9e2f9..f614b596bdcb376fc138af70a1af6539852b54f0 100644
--- a/Processes/Sibyll/sibyll2.3c.cc
+++ b/Processes/Sibyll/sibyll2.3c.cc
@@ -14,6 +14,9 @@
 #include <corsika/random/RNGManager.h>
 #include <random>
 
+int get_nwounded() { return s_chist_.nwd; }
+double get_sibyll_mass2(int& id) { return s_mass1_.am2[id]; }
+
 double s_rndm_(int&) {
   static corsika::random::RNG& rng =
       corsika::random::RNGManager::GetInstance().GetRandomStream("s_rndm");
diff --git a/Processes/Sibyll/sibyll2.3c.h b/Processes/Sibyll/sibyll2.3c.h
index d915e71401ab397b3966f96b615ba0b73e2b7dee..104b55eb0bf2e6fc12277f6d62bf1adcf567acb4 100644
--- a/Processes/Sibyll/sibyll2.3c.h
+++ b/Processes/Sibyll/sibyll2.3c.h
@@ -30,6 +30,10 @@ extern struct {
   int np;
 } s_plist_;
 
+// additional information about interactions.
+// number of wounded nucleons, number of hard and soft scatterings etc.
+extern struct { int nnsof[20], nnjet[20], jdif[20], nwd, njet, nsof; } s_chist_;
+
 extern struct {
   double cbr[223 + 16 + 12 + 8];
   int kdec[1338 + 6 * (16 + 12 + 8)];
@@ -96,6 +100,9 @@ void sib_sigma_hp_(const int&, const double&, double&, double&, double&, double*
 
 double s_rndm_(int&);
 
+int get_nwounded();
+double get_sibyll_mass2(int&);
+
 // phojet random generator setup
 void pho_rndin_(int&, int&, int&, int&);
 }