Skip to content

Commit

Permalink
Fixes for MSL (DOI-USGS#580)
Browse files Browse the repository at this point in the history
* Some potential fixes for MSL

Post-conflict fixes

More conflict merging

More reconciliation with recent ale

More reconciliation

Fix for pixel size

* Bugfix for focal length lookup for nav cam

* Make some MSL tests pass

Undo some tests

* Fix for nadir mode with MSL cameras

* Spellcheck

* Add changelog note for MSL

* CHANGELOG: Spell

* Move changelog note
  • Loading branch information
oleg-alexandrov authored Dec 4, 2023
1 parent cf11ad7 commit 5124f92
Show file tree
Hide file tree
Showing 11 changed files with 221 additions and 113 deletions.
4 changes: 4 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ release.

### Fixed
- Fixed LRO MiniRF drivers naif keywords focal to pixel and pixel to focal translations to be correct. [#569](https://github.com/DOI-USGS/ale/pull/569)
- Bugfix for position and orientation for MSL cameras (driver MslMastcamPds3NaifSpiceDriver). Validated that Nav and Mast LBL files (for both left and right sensor) produce correctly positioned and oriented CSM cameras, that are self-consistent and consistent with a prior DEM for the site. [#580](https://github.com/DOI-USGS/ale/pull/580)

### Changed
- Removed the affine6p library and replaced affine6p's affine transformation with a numpy solution [#579](https://github.com/DOI-USGS/ale/pull/579)
Expand Down Expand Up @@ -77,3 +78,6 @@ release.
- Chandrayaan1_mrffr IsisLabelNaifSpice driver, tests and test data [#519](https://github.com/DOI-USGS/ale/pull/519)
- MGS MOC Narrow Angle IsisLabelNaifSpice driver, tests, and test data [#517](https://github.com/DOI-USGS/ale/pull/517)
- Hayabusa NIRS IsisLabelNaifSpice driver, tests and test data [#532](https://github.com/DOI-USGS/ale/pull/532)



74 changes: 59 additions & 15 deletions ale/base/type_sensor.py
Original file line number Diff line number Diff line change
Expand Up @@ -373,9 +373,8 @@ def line_times(self):

class Cahvor():
"""
Mixin for largely ground based sensors to add an
extra step in the frame chain to go from ground camera to
the Camera
Mixin for ground-based sensors to add to the position and rotation
the components going from rover frame to camera frame.
"""

@property
Expand All @@ -396,17 +395,50 @@ def final_inst_frame(self):

@property
def sensor_position(self):
positions, velocities, times = super().sensor_position
positions += self.cahvor_camera_dict["C"]
if self._props.get("landed", False):
positions = np.array([[0, 0, 0]] * len(times))
velocities = np.array([[0, 0, 0]] * len(times))
return positions, velocities, times
"""
Find the rover position, then add the camera position relative to the
rover. The returned position is in ECEF.
Returns
-------
: (positions, velocities, times)
a tuple containing a list of positions, a list of velocities, and a
list of times.
"""

# Rover position in ECEF
positions, velocities, times = super().sensor_position

nadir = self._props.get("nadir", False)
if nadir:
# For nadir applying the rover-to-camera offset runs into
# problems, so return early. TBD
return positions, velocities, times

# Rover-to-camera offset in rover frame
cam_ctr = self.cahvor_center

# Rover-to-camera offset in ECEF
ecef_frame = self.target_frame_id
rover_frame = self.final_inst_frame
frame_chain = self.frame_chain
rover2ecef_rotation = \
frame_chain.compute_rotation(rover_frame, ecef_frame)
cam_ctr = rover2ecef_rotation.apply_at([cam_ctr], times)[0]

# Go from rover position to camera position
positions[0] += cam_ctr

if self._props.get("landed", False):
positions = np.array([[0, 0, 0]] * len(times))
velocities = np.array([[0, 0, 0]] * len(times))

return positions, velocities, times

def compute_h_c(self):
"""
Computes the h_c element of a cahvor model for the conversion
to a photogrametric model
to a photogrammetric model
Returns
-------
Expand All @@ -418,7 +450,7 @@ def compute_h_c(self):
def compute_h_s(self):
"""
Computes the h_s element of a cahvor model for the conversion
to a photogrametric model
to a photogrammetric model
Returns
-------
Expand All @@ -430,7 +462,7 @@ def compute_h_s(self):
def compute_v_c(self):
"""
Computes the v_c element of a cahvor model for the conversion
to a photogrametric model
to a photogrammetric model
Returns
-------
Expand All @@ -442,7 +474,7 @@ def compute_v_c(self):
def compute_v_s(self):
"""
Computes the v_s element of a cahvor model for the conversion
to a photogrametric model
to a photogrammetric model
Returns
-------
Expand Down Expand Up @@ -474,6 +506,18 @@ def cahvor_rotation_matrix(self):
self._cahvor_rotation_matrix = np.array([H_prime, V_prime, self.cahvor_camera_dict['A']])
return self._cahvor_rotation_matrix

@property
def cahvor_center(self):
"""
Computes the cahvor center for the sensor relative to the rover frame
Returns
-------
: array
Cahvor center as a 1D numpy array
"""
return self.cahvor_camera_dict['C']

@property
def frame_chain(self):
"""
Expand Down Expand Up @@ -566,6 +610,6 @@ def pixel_size(self):
Returns
-------
: float
Focal length of a cahvor model instrument
Pixel size of a cahvor model instrument
"""
return self.focal_length/self.compute_h_s()
return -self.focal_length/self.compute_h_s()
2 changes: 1 addition & 1 deletion ale/drivers/co_drivers.py
Original file line number Diff line number Diff line change
Expand Up @@ -205,7 +205,7 @@ def focal_length(self):
"""
NAC uses multiple filter pairs, each filter combination has a different focal length.
NAIF's Cassini kernels do not contain focal lengths for NAC filters and
so we aquired updated NAC filter data from ISIS's IAK kernel.
so we acquired updated NAC filter data from ISIS's IAK kernel.
"""
# default focal defined by IAK kernel
Expand Down
4 changes: 2 additions & 2 deletions ale/drivers/lro_drivers.py
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ def odtk(self):
@property
def light_time_correction(self):
"""
Returns the type of light time correction and abberation correction to
Returns the type of light time correction and aberration correction to
use in NAIF calls.
LROC is specifically set to not use light time correction because it is
Expand All @@ -112,7 +112,7 @@ def light_time_correction(self):
Returns
-------
: str
The light time and abberation correction string for use in NAIF calls.
The light time and aberration correction string for use in NAIF calls.
See https://naif.jpl.nasa.gov/pub/naif/toolkit_docs/C/req/abcorr.html
for the different options available.
"""
Expand Down
89 changes: 77 additions & 12 deletions ale/drivers/msl_drivers.py
Original file line number Diff line number Diff line change
Expand Up @@ -42,10 +42,28 @@ def instrument_id(self):
"""
lookup = {
"MAST_RIGHT": 'MASTCAM_RIGHT',
"MAST_LEFT": 'MASTCAM_LEFT'
"MAST_LEFT": 'MASTCAM_LEFT',
"NAV_RIGHT_B": 'NAVCAM_RIGHT_B',
"NAV_LEFT_B": 'NAVCAM_LEFT_B'
}
return self.instrument_host_id + "_" + lookup[super().instrument_id]

@property
def is_navcam(self):
"""
Returns True if the camera is a nav cam, False otherwise.
Need to handle nav cam differently as its focal length
cannot be looked up in the spice data. Use instead
a focal length in pixels computed from the CAHVOR model,
and a pixel size of 1.
Returns
-------
: bool
True if the camera is a nav cam, False otherwise
"""
return 'NAVCAM' in self.instrument_id

@property
def cahvor_camera_dict(self):
"""
Expand All @@ -72,14 +90,14 @@ def cahvor_camera_dict(self):
@property
def final_inst_frame(self):
"""
Defines MSLs last naif frame before the cahvor model frame
Defines the rover frame, relative to which the MSL cahvor camera is defined
Returns
-------
: int
Naif frame code for MSL_RSM_HEAD
Naif frame code for MSL_ROVER
"""
return spice.bods2c("MSL_RSM_HEAD")
return spice.bods2c("MSL_ROVER")

@property
def sensor_frame_id(self):
Expand Down Expand Up @@ -108,10 +126,7 @@ def focal2pixel_lines(self):
: list<double>
focal plane to detector lines
"""
if self._props.get("landed", False):
return [0, 0, -1/self.pixel_size]
else:
return [0, 0, 1/self.pixel_size]
return [0, 0, -1/self.pixel_size]

@property
def focal2pixel_samples(self):
Expand All @@ -123,10 +138,7 @@ def focal2pixel_samples(self):
: list<double>
focal plane to detector samples
"""
if (self._props.get("nadir", False)):
return [0, 1/self.pixel_size, 0]
else:
return [0, -1/self.pixel_size, 0]
return [0, -1/self.pixel_size, 0]

@property
def sensor_model_version(self):
Expand All @@ -137,3 +149,56 @@ def sensor_model_version(self):
ISIS sensor model version
"""
return 1

@property
def light_time_correction(self):
"""
Returns the type of light time correction and aberration correction to
use in NAIF calls.
For MSL using such a correction returns wrong results, so turn it off.
Returns
-------
: str
The light time and aberration correction string for use in NAIF calls.
See https://naif.jpl.nasa.gov/pub/naif/toolkit_docs/C/req/abcorr.html
for the different options available.
"""
return 'NONE'

@property
def focal_length(self):
"""
Returns the focal length of the sensor with a negative sign.
This was tested to work with MSL mast and nav cams.
Returns
-------
: float
focal length
"""
if self.is_navcam:
# Focal length in pixel as computed for a cahvor model.
# See is_navcam() for an explanation.
return -(self.compute_h_s() + self.compute_v_s())/2.0

# For mast cam
return -super().focal_length

@property
def pixel_size(self):
"""
Returns the pixel size.
Returns
-------
: float
pixel size
"""
if self.is_navcam:
# See is_navcam() for an explanation.
return 1.0

# For mast cam
return super().pixel_size
22 changes: 11 additions & 11 deletions ale/formatters/formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ def to_isd(driver):
meta_data = {}

meta_data['isis_camera_version'] = driver.sensor_model_version

# general information
meta_data['image_lines'] = driver.image_lines
meta_data['image_samples'] = driver.image_samples
Expand Down Expand Up @@ -89,14 +89,15 @@ def to_isd(driver):
frame_chain = driver.frame_chain
target_frame = driver.target_frame_id

J2000 = 1 # J2000 frame id
body_rotation = {}
source_frame, destination_frame, time_dependent_target_frame = frame_chain.last_time_dependent_frame_between(target_frame, 1)

if source_frame != 1:
source_frame, destination_frame, time_dependent_target_frame = frame_chain.last_time_dependent_frame_between(target_frame, J2000)
if source_frame != J2000:
# Reverse the frame order because ISIS orders frames as
# (destination, intermediate, ..., intermediate, source)
body_rotation['time_dependent_frames'] = shortest_path(frame_chain, source_frame, 1)
time_dependent_rotation = frame_chain.compute_rotation(1, source_frame)
body_rotation['time_dependent_frames'] = shortest_path(frame_chain, source_frame, J2000)
time_dependent_rotation = frame_chain.compute_rotation(J2000, source_frame)
body_rotation['ck_table_start_time'] = time_dependent_rotation.times[0]
body_rotation['ck_table_end_time'] = time_dependent_rotation.times[-1]
body_rotation['ck_table_original_size'] = len(time_dependent_rotation.times)
Expand All @@ -122,8 +123,8 @@ def to_isd(driver):

# Reverse the frame order because ISIS orders frames as
# (destination, intermediate, ..., intermediate, source)
instrument_pointing['time_dependent_frames'] = shortest_path(frame_chain, destination_frame, 1)
time_dependent_rotation = frame_chain.compute_rotation(1, destination_frame)
instrument_pointing['time_dependent_frames'] = shortest_path(frame_chain, destination_frame, J2000)
time_dependent_rotation = frame_chain.compute_rotation(J2000, destination_frame)
instrument_pointing['ck_table_start_time'] = time_dependent_rotation.times[0]
instrument_pointing['ck_table_end_time'] = time_dependent_rotation.times[-1]
instrument_pointing['ck_table_original_size'] = len(time_dependent_rotation.times)
Expand All @@ -141,7 +142,7 @@ def to_isd(driver):
instrument_pointing['constant_rotation'] = constant_rotation.rotation_matrix().flatten()
meta_data['instrument_pointing'] = instrument_pointing

# interiror orientation
# interior orientation
meta_data['naif_keywords'] = driver.naif_keywords

if isinstance(driver,LineScanner) or isinstance(driver, Framer) or isinstance(driver, PushFrame):
Expand All @@ -162,9 +163,8 @@ def to_isd(driver):

meta_data['starting_detector_line'] = driver.detector_start_line
meta_data['starting_detector_sample'] = driver.detector_start_sample


j2000_rotation = frame_chain.compute_rotation(target_frame, 1)
j2000_rotation = frame_chain.compute_rotation(target_frame, J2000)

instrument_position = {}
positions, velocities, times = driver.sensor_position
Expand Down
6 changes: 3 additions & 3 deletions ale/transformation.py
Original file line number Diff line number Diff line change
Expand Up @@ -123,8 +123,8 @@ def from_spice(cls, sensor_frame, target_frame, center_ephemeris_time, ephemeris

constant_frames.extend(target_constant_frames)

frame_chain.compute_time_dependent_rotiations(sensor_time_dependent_frames, sensor_times, inst_time_bias)
frame_chain.compute_time_dependent_rotiations(target_time_dependent_frames, target_times, 0)
frame_chain.compute_time_dependent_rotations(sensor_time_dependent_frames, sensor_times, inst_time_bias)
frame_chain.compute_time_dependent_rotations(target_time_dependent_frames, target_times, 0)

for s, d in constant_frames:
quats = np.zeros(4)
Expand Down Expand Up @@ -380,7 +380,7 @@ def extract_exact_ck_times(observStart, observEnd, targetFrame):

return times

def compute_time_dependent_rotiations(self, frames, times, time_bias):
def compute_time_dependent_rotations(self, frames, times, time_bias):
"""
Computes the time dependent rotations based on a list of tuples that define the
relationships between frames as (source, destination) and a list of times to
Expand Down
Loading

0 comments on commit 5124f92

Please sign in to comment.