From 642679b1ae75b01eb1aefb81cf9f67c0ae9d1ac3 Mon Sep 17 00:00:00 2001
From: Luc Maisonobe <luc@orekit.org>
Date: Wed, 26 Mar 2014 15:28:04 +0100
Subject: [PATCH] Fixed a numerical issue at tile exit.

The low point at tile minimum elevation was computed with a tiny
positive error, meaning the point really was above min elevation (at
micrometer level). The line segment between entry and exit stopped just
before traversing the Digital Elevation Model, and an error was
triggered.

The solution was to take some margin when computing the segment
endpoints at entry and exit: we now start above max elevation and end
below min elevation so the intersection should be really on the line.
---
 .../core/duvenhage/DuvenhageAlgorithm.java    |  4 ++--
 .../rugged/core/AbstractAlgorithmTest.java    | 10 ++++-----
 .../duvenhage/DuvenhageAlgorithmTest.java     | 21 +++++++++++++++++++
 3 files changed, 28 insertions(+), 7 deletions(-)

diff --git a/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java b/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java
index 12e3d106..acdf0faf 100644
--- a/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java
+++ b/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java
@@ -72,7 +72,7 @@ public class DuvenhageAlgorithm implements IntersectionAlgorithm {
             while (current == null) {
 
                 // find where line-of-sight crosses tile max altitude
-                final Vector3D entryP = ellipsoid.pointAtAltitude(position, los, tile.getMaxElevation());
+                final Vector3D entryP = ellipsoid.pointAtAltitude(position, los, tile.getMaxElevation() + STEP);
                 if (Vector3D.dotProduct(entryP.subtract(position), los) < 0) {
                     // the entry point is behind spacecraft!
                     throw new RuggedException(RuggedMessages.DEM_ENTRY_POINT_IS_BEHIND_SPACECRAFT);
@@ -267,7 +267,7 @@ public class DuvenhageAlgorithm implements IntersectionAlgorithm {
         throws RuggedException, OrekitException {
 
         // look for an exit at bottom
-        Vector3D exitP = ellipsoid.pointAtAltitude(position, los, tile.getMinElevation());
+        Vector3D exitP = ellipsoid.pointAtAltitude(position, los, tile.getMinElevation() - STEP);
         GeodeticPoint exitGP = ellipsoid.transform(exitP, ellipsoid.getBodyFrame(), null);
 
         switch (tile.getLocation(exitGP.getLatitude(), exitGP.getLongitude())) {
diff --git a/rugged-core/src/test/java/org/orekit/rugged/core/AbstractAlgorithmTest.java b/rugged-core/src/test/java/org/orekit/rugged/core/AbstractAlgorithmTest.java
index 204260fa..d940603b 100644
--- a/rugged-core/src/test/java/org/orekit/rugged/core/AbstractAlgorithmTest.java
+++ b/rugged-core/src/test/java/org/orekit/rugged/core/AbstractAlgorithmTest.java
@@ -149,7 +149,7 @@ public abstract class AbstractAlgorithmTest {
 
     }
 
-    private void setUpMayonVolcanoContext()
+    protected void setUpMayonVolcanoContext()
         throws RuggedException, OrekitException {
 
         // Mayon Volcano location according to Wikipedia: 13°15′24″N 123°41′6″E
@@ -196,7 +196,7 @@ public abstract class AbstractAlgorithmTest {
 
     }
 
-    private void setUpCliffsOfMoherContext()
+    protected void setUpCliffsOfMoherContext()
         throws RuggedException, OrekitException {
 
         // cliffs of Moher location according to Wikipedia: 52°56′10″N 9°28′15″ W
@@ -267,8 +267,8 @@ public abstract class AbstractAlgorithmTest {
         state     = null;
     }
 
-    private ExtendedEllipsoid earth;
-    private TileUpdater       updater;
-    private SpacecraftState   state;
+    protected ExtendedEllipsoid earth;
+    protected TileUpdater       updater;
+    protected SpacecraftState   state;
 
 }
diff --git a/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java b/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java
index 535bc08a..40ed68fa 100644
--- a/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java
+++ b/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java
@@ -17,6 +17,12 @@
 package org.orekit.rugged.core.duvenhage;
 
 
+import org.apache.commons.math3.geometry.euclidean.threed.Vector3D;
+import org.junit.Assert;
+import org.junit.Test;
+import org.orekit.bodies.GeodeticPoint;
+import org.orekit.errors.OrekitException;
+import org.orekit.rugged.api.RuggedException;
 import org.orekit.rugged.core.AbstractAlgorithmTest;
 import org.orekit.rugged.core.raster.IntersectionAlgorithm;
 
@@ -26,4 +32,19 @@ public class DuvenhageAlgorithmTest extends AbstractAlgorithmTest {
         return new DuvenhageAlgorithm();
     }
 
+    @Test
+    public void testNumericalIssueAtTileExit() throws RuggedException, OrekitException {
+        setUpMayonVolcanoContext();
+        final IntersectionAlgorithm algorithm = createAlgorithm();
+        algorithm.setUpTilesManagement(updater, 8);
+        GeodeticPoint intersection = algorithm.intersection(earth,
+                                                            new Vector3D(-3787079.6453602533,
+                                                                          5856784.405679551,
+                                                                          1655869.0582939098),
+                                                            new Vector3D( 0.5127552821932051,
+                                                                         -0.8254313129088879,
+                                                                         -0.2361041470463311));
+        Assert.assertEquals(16.0, intersection.getAltitude(), 1.0e-10);
+    }
+
 }
-- 
GitLab