Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixes for MSL #580

Merged
merged 8 commits into from
Dec 4, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
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):
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change is technically API breaking but I think we are planning to pull a 1.0 release, so it should be fine.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is an internal function, used only in that file.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree that it is an internally used function but a user could access the function if they wanted to and compute other frame rotations on a framechain object

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This also came up in the usgscsm repository; with no consensus, as I see.

There should be a clear policy, as to what parts of a library are the user-facing API, which must be guaranteed to be kept fixed, or at least not modified without a good reason, and which part is internal implementation. Constraining all internal functions is not possible.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

At least in ISIS, it's any function that a user has access to. Since that is C/C++ code that is more enforceable. Typically in python a "private" function is defined with a leading _ so that is something we could do in our python libraries. Similarly we should make functions private in USGSCSM that should not be public facing API

"""
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
Loading