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

first version of OpenCL support for operations on rasters (and soon f… #32

Merged
merged 39 commits into from
Oct 12, 2020
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
b381cd4
first version of OpenCL support for operations on rasters (and soon f…
michaelmattig Sep 8, 2020
24031cb
add code for remaining raster data types
michaelmattig Sep 9, 2020
5176f98
Add raster info and nodata functions to kernels
michaelmattig Sep 10, 2020
6819e8e
OCL error handling
michaelmattig Sep 10, 2020
b0179ca
use static reference to OpenCl device to circumvent concurrency problems
michaelmattig Sep 10, 2020
3140ad0
first WIP version of Expression operator.
michaelmattig Sep 11, 2020
e3e389b
calculate raster index from 2d work dims
michaelmattig Sep 15, 2020
7edcb11
fix expression kernel
michaelmattig Sep 15, 2020
a301d2e
macro for calling a function on a raster and the corresponding enum v…
michaelmattig Sep 15, 2020
1dc1455
macros and doc
michaelmattig Sep 16, 2020
b95c003
Merge remote-tracking branch 'remotes/origin/master' into ocl
michaelmattig Sep 16, 2020
d50467c
fix lib.rs
michaelmattig Sep 16, 2020
cb0d6b4
towars opencl support for feature collections
michaelmattig Sep 16, 2020
0a848c2
code clean up
michaelmattig Sep 18, 2020
eaa9745
Arrow OpenCL integration WIP
michaelmattig Sep 22, 2020
4e2fc62
Merge remote-tracking branch 'remotes/origin/master' into ocl
michaelmattig Sep 22, 2020
45634bd
projection to spatial reference
michaelmattig Sep 23, 2020
c5ddd14
OpenCL feature collection buffer handling
michaelmattig Sep 23, 2020
bd4ed71
OpenCL feature collection column handling
michaelmattig Sep 24, 2020
76706d1
clean up some todos
michaelmattig Sep 24, 2020
422b1ff
fixed "offsets do not start at zero" bug
michaelmattig Sep 25, 2020
91ef24f
implement suggestions
michaelmattig Oct 1, 2020
f818167
Merge branch 'master' into ocl
michaelmattig Oct 1, 2020
6d007dd
adjust expression operator to master
michaelmattig Oct 1, 2020
bad20b7
merged column names and types together
michaelmattig Oct 2, 2020
9654a39
changed buffer data type to Double2
michaelmattig Oct 2, 2020
b7fb597
clippy
michaelmattig Oct 2, 2020
88017ba
make condition more readable
michaelmattig Oct 2, 2020
0c148c7
add missing methods
michaelmattig Oct 2, 2020
3b07edb
infer vector data type from CollectionType
michaelmattig Oct 2, 2020
72ac38a
get start and end time only once
michaelmattig Oct 2, 2020
39100f0
remove ProQue to avoid hanging test
michaelmattig Oct 2, 2020
1dbd674
clippy
michaelmattig Oct 2, 2020
7ed0dcf
docs
michaelmattig Oct 7, 2020
ec9ee74
renaming
michaelmattig Oct 8, 2020
f67bb46
pass queue explicitly
michaelmattig Oct 9, 2020
d11f857
use i8/char for transferring bool to opencl device
michaelmattig Oct 9, 2020
1f4f5dd
generify conversion of rasters and typed rasters
michaelmattig Oct 9, 2020
3fe3ae0
require nodata value specification for expression output
michaelmattig Oct 9, 2020
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
29 changes: 29 additions & 0 deletions datatypes/src/collections/data_types.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
use crate::collections::{
DataCollection, MultiLineStringCollection, MultiPointCollection, MultiPolygonCollection,
};
use serde::{Deserialize, Serialize};

/// An enum that contains all possible vector data types
Expand All @@ -9,3 +12,29 @@ pub enum VectorDataType {
MultiLineString,
MultiPolygon,
}

pub enum TypedFeatureCollection {
Data(DataCollection),
MultiPoint(MultiPointCollection),
MultiLineString(MultiLineStringCollection),
MultiPolygon(MultiPolygonCollection),
}

impl TypedFeatureCollection {
jdroenner marked this conversation as resolved.
Show resolved Hide resolved
pub fn vector_data_type(&self) -> VectorDataType {
match self {
TypedFeatureCollection::Data(_) => VectorDataType::Data,
TypedFeatureCollection::MultiPoint(_) => VectorDataType::MultiPoint,
TypedFeatureCollection::MultiLineString(_) => VectorDataType::MultiLineString,
TypedFeatureCollection::MultiPolygon(_) => VectorDataType::MultiPolygon,
}
}

// TODO: other types
ChristianBeilschmidt marked this conversation as resolved.
Show resolved Hide resolved
pub fn get_points(self) -> Option<MultiPointCollection> {
if let TypedFeatureCollection::MultiPoint(points) = self {
return Some(points);
}
None
}
}
22 changes: 22 additions & 0 deletions datatypes/src/collections/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,3 +24,25 @@ pub use data_types::VectorDataType;
pub use multi_line_string_collection::MultiLineStringCollection;
pub use multi_point_collection::MultiPointCollection;
pub use multi_polygon_collection::MultiPolygonCollection;

pub use data_types::TypedFeatureCollection;

/// Calls a function on a `TypedFeatureCollection` by calling it on its variant.
/// Call via `call_generic_features!(input, features => function)`.
#[macro_export]
macro_rules! call_generic_features {
($input_features:expr, $features:ident => $function_call:expr) => {
call_generic_features!(
@variants $input_features, $features => $function_call,
Data, MultiPoint, MultiLineString, MultiPolygon
)
};

(@variants $input_features:expr, $features:ident => $function_call:expr, $($variant:tt),+) => {
match $input_features {
$(
$crate::collections::TypedFeatureCollection::$variant($features) => $function_call,
)+
}
};
}
81 changes: 81 additions & 0 deletions datatypes/src/collections/multi_point_collection.rs
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,44 @@ impl GeoFeatureCollectionRowBuilder<MultiPoint> for FeatureCollectionRowBuilder<
}
}

impl MultiPointCollection {
pub fn coordinates(&self) -> &[Coordinate2D] {
ChristianBeilschmidt marked this conversation as resolved.
Show resolved Hide resolved
let geometries_ref = self
.table
.column_by_name(Self::GEOMETRY_COLUMN_NAME)
.expect("There must exist a geometry column");
let geometries: &ListArray = downcast_array(geometries_ref);

let coordinates_ref = geometries.values();
let coordinates: &FixedSizeListArray = downcast_array(&coordinates_ref);

let number_of_coordinates = coordinates.data().len();

let floats_ref = coordinates.values();
let floats: &Float64Array = downcast_array(&floats_ref);

unsafe {
slice::from_raw_parts(
floats.raw_values() as *const Coordinate2D,
number_of_coordinates,
)
}
}

pub fn coordinates_offsets(&self) -> &[i32] {
let geometries_ref = self
.table
.column_by_name(Self::GEOMETRY_COLUMN_NAME)
.expect("There must exist a geometry column");
let geometries: &ListArray = downcast_array(geometries_ref);

let data = geometries.data();
let buffer = &data.buffers()[0];
ChristianBeilschmidt marked this conversation as resolved.
Show resolved Hide resolved

unsafe { slice::from_raw_parts(buffer.raw_data() as *const i32, geometries.len() + 1) }
}
}

#[cfg(test)]
mod tests {
use super::*;
Expand Down Expand Up @@ -864,4 +902,47 @@ mod tests {

assert_eq!(collection, deserialized);
}

#[test]
fn coordinates() {
let pc = MultiPointCollection::from_data(
MultiPoint::many(vec![
vec![(0., 0.)],
vec![(1., 1.), (1.1, 1.1)],
vec![(2., 2.)],
])
.unwrap(),
vec![
TimeInterval::new_unchecked(0, 1),
TimeInterval::new_unchecked(1, 2),
TimeInterval::new_unchecked(2, 3),
],
{
let mut map = HashMap::new();
map.insert("numbers".into(), FeatureData::Number(vec![0., 1., 2.]));
map.insert(
"number_nulls".into(),
FeatureData::NullableNumber(vec![Some(0.), None, Some(2.)]),
);
map
},
)
.unwrap();

let coords = pc.coordinates();
assert_eq!(coords.len(), 4);
assert_eq!(
coords,
&[
[0., 0.].into(),
[1., 1.].into(),
[1.1, 1.1].into(),
[2., 2.].into(),
]
);

let offsets = pc.coordinates_offsets();
assert_eq!(offsets.len(), 4);
assert_eq!(offsets, &[0, 1, 3, 4]);
}
}
1 change: 1 addition & 0 deletions datatypes/src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -124,6 +124,7 @@ pub enum Error {
ParseU32 {
source: <u32 as std::str::FromStr>::Err,
},
InvalidTypedRasterConversion,
}

impl From<arrow::error::ArrowError> for Error {
Expand Down
5 changes: 4 additions & 1 deletion datatypes/src/primitives/coordinate.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,19 @@ use crate::util::arrow::ArrowTyped;
use arrow::array::{BooleanArray, Float64Builder};
use arrow::datatypes::DataType;
use arrow::error::ArrowError;
use ocl::OclPrm;
use serde::{Deserialize, Serialize};
use std::{fmt, slice};

#[derive(Clone, Copy, Debug, Deserialize, PartialEq, PartialOrd, Serialize)]
#[derive(Clone, Copy, Debug, Deserialize, PartialEq, PartialOrd, Serialize, Default)]
ChristianBeilschmidt marked this conversation as resolved.
Show resolved Hide resolved
#[repr(C)]
pub struct Coordinate2D {
pub x: f64,
pub y: f64,
}

unsafe impl OclPrm for Coordinate2D {}
ChristianBeilschmidt marked this conversation as resolved.
Show resolved Hide resolved

impl Coordinate2D {
/// Creates a new coordinate
///
Expand Down
2 changes: 2 additions & 0 deletions datatypes/src/raster/data_type.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
use crate::operations::image::RgbaTransmutable;
use crate::raster::typed_raster::TypedRasterConversion;
use num_traits::{AsPrimitive, Num};
use serde::{Deserialize, Serialize};

Expand Down Expand Up @@ -35,6 +36,7 @@ pub trait Pixel:
+ FromPrimitive<Self>
+ StaticRasterDataType
+ RgbaTransmutable
+ TypedRasterConversion
{
}

Expand Down
72 changes: 69 additions & 3 deletions datatypes/src/raster/helpers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ macro_rules! call_generic_raster2d {
}

/// Calls a function on two `TypedRaster2D`s by calling it on their variant combination.
/// Call via `call_bi_generic_raster2d!(input, (raster_a, raster_b) => function)`.
/// Call via `call_bi_generic_raster2d!(input_a, input_b, (raster_a, raster_b) => function)`.
#[macro_export]
macro_rules! call_bi_generic_raster2d {
(
Expand Down Expand Up @@ -84,7 +84,7 @@ macro_rules! call_bi_generic_raster2d {
}

/// Calls a function on two `TypedRaster2D`s by calling it on their variant combination.
/// Call via `call_bi_generic_raster2d_same!(input, (raster_a, raster_b) => function)`.
/// Call via `call_bi_generic_raster2d_same!(input_a, input_b, (raster_a, raster_b) => function)`.
/// The resulting call requires the rasters to be of the same type.
/// Otherwise, the last optional parameter is a catch-all function (or it just panics).
#[macro_export]
Expand Down Expand Up @@ -134,7 +134,7 @@ macro_rules! call_bi_generic_raster2d_same {
}

/// Calls a function on two `TypedRaster2D`s by calling it on their variant combination.
/// Call via `call_bi_generic_raster2d_staircase!(input, (raster_a, raster_b) => function)`.
/// Call via `call_bi_generic_raster2d_staircase!(input_a, input_b, (raster_a, raster_b) => function)`.
/// This macro requires the first raster type to be greater or equal to the second one.
/// Otherwise, the last optional parameter is a catch-all function (or it just panics).
#[macro_export]
Expand Down Expand Up @@ -214,10 +214,35 @@ macro_rules! generate_generic_raster2d {
};
}

/// Calls a function on a `TypedRaster2D` and some `RasterDataType`-like enum, effectively matching
/// the raster with the corresponding enum value of the other enum.
/// Call via `call_generic_raster2d_ext!(input, (raster, e) => function)`.
#[macro_export]
macro_rules! call_generic_raster2d_ext {
jdroenner marked this conversation as resolved.
Show resolved Hide resolved
($input_raster:expr, $other_enum:ty, ($raster:ident, $enum:ident) => $func:expr) => {
call_generic_raster2d_ext!(
@variants $input_raster, $other_enum, ($raster, $enum) => $func,
U8, U16, U32, U64, I8, I16, I32, I64, F32, F64
)
};

(@variants $input_raster:expr, $other_enum:ty, ($raster:ident, $enum:ident) => $func:expr, $($variant:tt),+) => {
match $input_raster {
$(
$crate::raster::TypedRaster2D::$variant($raster) => {
let $enum = <$other_enum>::$variant;
$func
}
)+
}
};
}

#[cfg(test)]
mod tests {
use crate::raster::{GridPixelAccess, Pixel, Raster2D, RasterDataType, TypedRaster2D};
use crate::util::test::catch_unwind_silent;
use serde::export::PhantomData;

#[test]
fn map_generic_raster2d() {
Expand Down Expand Up @@ -446,4 +471,45 @@ mod tests {
))
.is_err());
}

#[test]
fn ext() {
struct T<S> {
s: PhantomData<S>,
}

enum Foo {
U8(T<u8>),
U16(T<u16>),
U32(T<u32>),
U64(T<u64>),
I8(T<i8>),
I16(T<i16>),
I32(T<i32>),
I64(T<i64>),
F32(T<f32>),
F64(T<f64>),
}

fn foo<S>(_f: fn(T<S>) -> Foo, _r: &Raster2D<S>)
where
S: Pixel,
{
}

let typed_raster_a = TypedRaster2D::U32(
Raster2D::new(
[3, 2].into(),
vec![1, 2, 3, 4, 5, 6],
None,
Default::default(),
[1.0, 1.0, 0.0, 1.0, 0.0, 1.0].into(),
)
.unwrap(),
);

call_generic_raster2d_ext!(typed_raster_a, Foo, (raster, e) => {
foo(e, &raster)
});
}
}
Loading