Expand description
๐ท ๐ Geometric models of cameras for photogrammetry
ยงAbout
The crate implements geometric models of cameras which may be useful for photogrammetry.
The crate provides a couple camera models, the pinhole perspective
camera and the
orthographic
camera. Adding
another camera model entails implementing the
IntrinsicParameters
trait. See the
opencv_ros_camera
crate
for one example.
Also provided is the function
best_intersection_of_rays()
which
determines the best 3D point corresponding to the intersection of multiple
rays. Thus, this crate is also useful for multiple view geometry.
Characteristics:
- Extensive use of static typing to ensure no unpleasant runtime surprises with coordinate system, matrix dimensions, and so on.
- Serialization and deserialization using
serde
. Enable with theserde-serialize
cargo feature. - Linear algebra and types from the
nalgebra
crate. - Possible to create new camera models by implementing the
IntrinsicParameters
trait. While the camera models implemented in this crate are linear, there is no requirement that implementations are linear. For example, theopencv_ros_camera
crate exhibits distortion. ExtrinsicParameters
based on thenalgebra::Isometry3
type to handle the camera pose.- No standard library is required (disable the default features to disable
use of
std
) and no heap allocations. In other words, this can run on a bare-metal microcontroller with no OS. - Extensive documentation and tests.
- Requires rust version 1.40 or greater.
ยงTesting
ยงUnit tests
To run the basic unit tests:
cargo test
To run all unit tests:
cargo test --features serde-serialize
ยงTest for no_std
Since the thumbv7em-none-eabihf
target does not have std
available, we
can build for it to check that our crate does not inadvertently pull in
std. The unit tests require std, so cannot be run on a no_std
platform.
The following will fail if a std dependency is present:
# install target with: "rustup target add thumbv7em-none-eabihf"
cargo build --no-default-features --target thumbv7em-none-eabihf
ยงCode of conduct
Anyone who interacts with this software in any space, including but not limited to this GitHub repository, must follow our code of conduct.
ยงLicense
Licensed under either of these:
- Apache License, Version 2.0, (LICENSE-APACHE or https://www.apache.org/licenses/LICENSE-2.0)
- MIT license (LICENSE-MIT or https://opensource.org/licenses/MIT)
ยงExamples
ยงExample - projecting 3D world coordinates to 2D pixel coordinates.
use cam_geom::*;
use nalgebra::{Matrix2x3, Unit, Vector3};
// Create two points in the world coordinate frame.
let world_coords = Points::new(Matrix2x3::new(
1.0, 0.0, 0.0, // point 1
0.0, 1.0, 0.0, // point 2
));
// perepective parameters - focal length of 100, no skew, pixel center at (640,480)
let intrinsics = IntrinsicParametersPerspective::from(PerspectiveParams {
fx: 100.0,
fy: 100.0,
skew: 0.0,
cx: 640.0,
cy: 480.0,
});
// Set extrinsic parameters - camera at (10,0,0), looing at (0,0,0), up (0,0,1)
let camcenter = Vector3::new(10.0, 0.0, 0.0);
let lookat = Vector3::new(0.0, 0.0, 0.0);
let up = Unit::new_normalize(Vector3::new(0.0, 0.0, 1.0));
let pose = ExtrinsicParameters::from_view(&camcenter, &lookat, &up);
// Create a `Camera` with both intrinsic and extrinsic parameters.
let camera = Camera::new(intrinsics, pose);
// Project the original 3D coordinates to 2D pixel coordinates.
let pixel_coords = camera.world_to_pixel(&world_coords);
// Print the results.
for i in 0..world_coords.data.nrows() {
let wc = world_coords.data.row(i);
let pix = pixel_coords.data.row(i);
println!("{} -> {}", wc, pix);
}
This will print:
โ โ
โ 1 0 0 โ
โ โ
->
โ โ
โ 640 480 โ
โ โ
โ โ
โ 0 1 0 โ
โ โ
->
โ โ
โ 650 480 โ
โ โ
ยงExample - intersection of rays
use cam_geom::*;
use nalgebra::RowVector3;
// Create the first ray.
let ray1 = Ray::<WorldFrame, _>::new(
RowVector3::new(1.0, 0.0, 0.0), // origin
RowVector3::new(0.0, 1.0, 0.0), // direction
);
// Create the second ray.
let ray2 = Ray::<WorldFrame, _>::new(
RowVector3::new(0.0, 1.0, 0.0), // origin
RowVector3::new(1.0, 0.0, 0.0), // direction
);
// Compute the best intersection.
let result = best_intersection_of_rays(&[ray1, ray2]).unwrap();
// Print the result.
println!("result: {}", result.data);
This will print:
result:
โ โ
โ 1 1 0 โ
โ โ
Re-exportsยง
pub use coordinate_system::CameraFrame;
pub use coordinate_system::WorldFrame;
Modulesยง
- Implementations of
CoordinateSystem
. - Utilities for testing
cam_geom
implementations. - Linearize camera models by computing the Jacobian matrix.
- Defines the different possible types of ray bundles.
Structsยง
- A camera model that can convert world coordinates into pixel coordinates.
- Defines the pose of a camera in the world coordinate system.
- An orthographic camera model. Implements
IntrinsicParameters
. - A pinhole perspective camera model. Implements
IntrinsicParameters
. - Parameters defining the intrinsic part of an orthographic camera model.
- Parameters defining a pinhole perspective camera model.
- 2D pixel locations on the image sensor.
- 3D points. Can be in any
CoordinateSystem
. - A single ray. Can be in any
CoordinateSystem
. - 3D rays. Can be in any
CoordinateSystem
.
Enumsยง
- All possible errors.
Traitsยง
- Specifies operations which any RayBundle must implement.
- A coordinate system in which points and rays can be defined.
- A geometric model of camera coordinates to pixels (and vice versa).
Functionsยง
- Return the 3D point which is the best intersection of rays.