Skip to content

Commit

Permalink
Merge branch 'develop' into pr/226
Browse files Browse the repository at this point in the history
  • Loading branch information
raphaelDkhn committed Oct 6, 2023
2 parents 8ed7b65 + a1af0e4 commit 1231a26
Show file tree
Hide file tree
Showing 47 changed files with 10,889 additions and 10 deletions.
1 change: 1 addition & 0 deletions docs/SUMMARY.md
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,7 @@
* [tensor.squeeze](framework/operators/tensor/tensor.squeeze.md)
* [tensor.unsqueeze](framework/operators/tensor/tensor.unsqueeze.md)
* [tensor.sign](framework/operators/tensor/tensor.sign.md)
* [tensor.clip](framework/operators/tensor/tensor.clip.md)
* [Neural Network](framework/operators/neural-network/README.md)
* [nn.relu](framework/operators/neural-network/nn.relu.md)
* [nn.leaky\_relu](framework/operators/neural-network/nn.leaky\_relu.md)
Expand Down
5 changes: 3 additions & 2 deletions docs/framework/compatibility.md
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ You can see below the list of current supported ONNX Operators:
| [Nonzero](operators/tensor/tensor.nonzero.md) | :white\_check\_mark: |
| [Squeeze](operators/tensor/tensor.squeeze.md) | :white\_check\_mark: |
| [Unsqueeze](operators/tensor/tensor.unsqueeze.md) | :white\_check\_mark: |
| [Sign](operators/tensor/tensor.sign.md) | :white\_check\_mark: |
| [Sign](operators/tensor/tensor.sign.md) | :white\_check\_mark: |
| [Clip](operators/tensor/tensor.clip.md) | :white\_check\_mark: |

Current Operators support: **51/156 (33%)**
Current Operators support: **50/156 (32%)**
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
# Tree Regressor

`TreeRegressorTrait` provides a trait definition for decision tree regression.
This trait offers functionalities to build a decision tree and predict target values based on input features.
`TreeRegressorTrait` provides a trait definition for decision tree regression. This trait offers functionalities to build a decision tree and predict target values based on input features.

```rust
use orion::operators::ml::TreeRegressorTrait;
Expand All @@ -21,4 +20,3 @@ Orion supports currently only fixed point data types for `TreeRegressorTrait`.
| --- | --- |
| [`tree.fit`](tree.fit.md) | Constructs a decision tree regressor based on the provided data and target values. |
| [`tree.predict`](tree.predict.md) | Given a set of features, predicts the target value using the constructed decision tree. |

Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# TreeRegressorTrait::predict
# tree.predict

```rust
```rust
fn predict(ref self: TreeNode<T>, features: Span<T>) -> T;
```

Expand Down
1 change: 1 addition & 0 deletions docs/framework/operators/tensor/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,7 @@ use orion::operators::tensor::TensorTrait;
| [`tensor.squeeze`](tensor.squeeze.md) | Removes dimensions of size 1 from the shape of a tensor. |
| [`tensor.unsqueeze`](tensor.unsqueeze.md) | Inserts single-dimensional entries to the shape of an input tensor. |
| [`tensor.sign`](tensor.sign.md) | Calculates the sign of the given input tensor element-wise. |
| [`tensor.clip`](tensor.clip.md) | Clip operator limits the given input within an interval. |

## Arithmetic Operations

Expand Down
39 changes: 39 additions & 0 deletions docs/framework/operators/tensor/tensor.clip.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
# tensor.clip

```rust
fn clip(self: @Tensor<T>, min: T, max: T) -> Tensor<T>;
```

Clip operator limits the given input within an interval.

## Args

* `self`(`@Tensor<T>`) - Input tensor whose elements to be clipped.
* `min`(`Option<T>`) - Minimum value, under which element is replaced by min.
* `max`(`Option<T>`) - Maximum value, above which element is replaced by max.

## Returns

Output `Tensor<T>` with clipped input elements.

## Example

```rust
use array::{ArrayTrait, SpanTrait};

use orion::operators::tensor::{TensorTrait, Tensor, U32Tensor};

fn clip_example() -> Tensor<u32> {
let tensor = TensorTrait::<u32>::new(
shape: array![2, 3].span(),
data: array![[ 1, 2, 3],[4, 5, 6]].span(),
);

return tensor.clip(
min: Option::None(()),
max: Option::Some(3),
);
}
>>> [[1. 2. 3.]
[3. 3. 3.]]
```
155 changes: 155 additions & 0 deletions nodegen/node/clip.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,155 @@
import numpy as np
from nodegen.node import RunAll
from ..helpers import make_node, make_test, to_fp, Tensor, Dtype, FixedImpl


class Clip(RunAll):
@staticmethod
def clip_u32():
def clip_2D():
x = np.random.randint(0, 255, (2, 4)).astype(np.uint32)
y = np.clip(x, np.uint32(10), np.uint32(20))

x = Tensor(Dtype.U32, x.shape, x.flatten())
y = Tensor(Dtype.U32, y.shape, y.flatten())

name = "clip_u32_2d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(u32 { mag: 10, sign: false }), Option::Some(u32 { mag: 20, sign: false }))", name)

def clip_3D():
x = np.random.randint(0, 255, (20, 10, 5)).astype(np.uint32)
y = np.clip(x, np.uint32(10), np.uint32(20))

x = Tensor(Dtype.U32, x.shape, x.flatten())
y = Tensor(Dtype.U32, y.shape, y.flatten())

name = "clip_u32_3d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(u32 { mag: 10, sign: false }), Option::Some(u32 { mag: 20, sign: false }))", name)

clip_2D()
clip_3D()

@staticmethod
def clip_i32():
def clip_2D():
x = np.random.randint(-127, 127, (2, 4)).astype(np.int32)
y = np.clip(x, np.int32(-10), np.int32(20))

x = Tensor(Dtype.I32, x.shape, x.flatten())
y = Tensor(Dtype.I32, y.shape, y.flatten())

name = "clip_i32_2d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(i32 { mag: 10, sign: true }), Option::Some(i32 { mag: 20, sign: false }))", name)

def clip_3D():
x = np.random.randint(-127, 127, (20, 10, 5)).astype(np.int32)
y = np.clip(x, np.int32(-10), np.int32(20))

x = Tensor(Dtype.I32, x.shape, x.flatten())
y = Tensor(Dtype.I32, y.shape, y.flatten())

name = "clip_i32_3d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(i32 { mag: 10, sign: true }), Option::Some(i32 { mag: 20, sign: false }))", name)


clip_2D()
clip_3D()

@staticmethod
def clip_i8():
def clip_2D():
x = np.random.randint(-127, 127, (2, 4)).astype(np.int8)
y = np.clip(x, np.int8(-10), np.int8(20))

x = Tensor(Dtype.I8, x.shape, x.flatten())
y = Tensor(Dtype.I8, y.shape, y.flatten())

name = "clip_i8_2d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(i8 { mag: 10, sign: true }), Option::Some(i8 { mag: 20, sign: false }))", name)

def clip_3D():
x = np.random.randint(-127, 127, (20, 10, 5)).astype(np.int8)
y = np.clip(x, np.int8(-10), np.int8(20))

x = Tensor(Dtype.I8, x.shape, x.flatten())
y = Tensor(Dtype.I8, y.shape, y.flatten())

name = "clip_i8_3d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(i8 { mag: 10, sign: true }), Option::Some(i8 { mag: 20, sign: false }))", name)

clip_2D()
clip_3D()

@staticmethod
def clip_fp8x23():
def clip_2D():
x = to_fp(np.random.randint(-127, 127, (2, 4)
).astype(np.int64), FixedImpl.FP8x23)
y = np.clip(x, to_fp(np.int64(-10), FixedImpl.FP8x23), to_fp(np.int64(20), FixedImpl.FP8x23))

x = Tensor(Dtype.FP8x23, x.shape, x.flatten())
y = Tensor(Dtype.FP8x23, y.shape, y.flatten())

name = "clip_fp8x23_2d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(FP8x23 { mag: 83886080, sign: true }), Option::Some(FP8x23 { mag: 167772160, sign: false }))", name)

def clip_3D():
x = to_fp(np.random.randint(-127, 127, (20, 10, 5)
).astype(np.int64), FixedImpl.FP8x23)
y = np.clip(x, to_fp(np.int64(-10), FixedImpl.FP8x23), to_fp(np.int64(20), FixedImpl.FP8x23))

x = Tensor(Dtype.FP8x23, x.shape, x.flatten())
y = Tensor(Dtype.FP8x23, y.shape, y.flatten())

name = "clip_fp8x23_3d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(FP8x23 { mag: 83886080, sign: true }), Option::Some(FP8x23 { mag: 167772160, sign: false }))", name)

clip_2D()
clip_3D()

@staticmethod
def clip_fp16x16():
def clip_2D():
x = to_fp(np.random.randint(-127, 127, (2, 4)
).astype(np.int64), FixedImpl.FP16x16)
y = np.clip(x, to_fp(np.int64(-10), FixedImpl.FP16x16), to_fp(np.int64(20), FixedImpl.FP16x16))

x = Tensor(Dtype.FP16x16, x.shape, x.flatten())
y = Tensor(Dtype.FP16x16, y.shape, y.flatten())

name = "clip_fp16x16_2d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(FP16x16 { mag: 655360, sign: true }), Option::Some(FP16x16 { mag: 1310720, sign: false }))", name)

def clip_3D():
x = to_fp(np.random.randint(-127, 127, (20, 10, 5)
).astype(np.int64), FixedImpl.FP16x16)
y = np.clip(x, to_fp(np.int64(-10), FixedImpl.FP16x16), to_fp(np.int64(20), FixedImpl.FP16x16))

x = Tensor(Dtype.FP16x16, x.shape, x.flatten())
y = Tensor(Dtype.FP16x16, y.shape, y.flatten())

name = "clip_fp16x16_3d"
make_node([x], [y], name)
make_test(
[x], y, "input_0.clip(Option::Some(FP16x16 { mag: 655360, sign: true }), Option::Some(FP16x16 { mag: 1310720, sign: false }))", name)

clip_2D()
clip_3D()
85 changes: 83 additions & 2 deletions src/operators/tensor/core.cairo
Original file line number Diff line number Diff line change
Expand Up @@ -2598,6 +2598,47 @@ trait TensorTrait<T> {
self: @Tensor<T>,
axes: Option<Span<i32>>
) -> Tensor<T>;
/// # tensor.clip
///
/// ```rust
/// fn clip(self: @Tensor<T>, min: T, max: T) -> Tensor<T>;
/// ```
///
/// Clip operator limits the given input within an interval.
///
/// ## Args
///
/// * `self`(`@Tensor<T>`) - Input tensor whose elements to be clipped.
/// * `min`(`Option<T>`) - Minimum value, under which element is replaced by min.
/// * `max`(`Option<T>`) - Maximum value, above which element is replaced by max.
///
/// ## Returns
///
/// Output `Tensor<T>` with clipped input elements.
///
/// ## Example
///
/// ```rust
/// use array::{ArrayTrait, SpanTrait};
///
/// use orion::operators::tensor::{TensorTrait, Tensor, U32Tensor};
///
/// fn clip_example() -> Tensor<u32> {
/// let tensor = TensorTrait::<u32>::new(
/// shape: array![2, 3].span(),
/// data: array![[ 1, 2, 3],[4, 5, 6]].span(),
/// );
///
/// return tensor.clip(
/// min: Option::None(()),
/// max: Option::Some(3),
/// );
/// }
/// >>> [[1. 2. 3.]
/// [3. 3. 3.]]
/// ```
///
fn clip(self: @Tensor<T>, min: Option<T>, max: Option<T>) -> Tensor<T>;
/// # tensor.sign
///
/// ```rust
Expand Down Expand Up @@ -3117,7 +3158,7 @@ fn unsqueeze<T>(self: @Tensor<T>, axes: Span<usize>) -> Tensor<T> {
return Tensor::<T> { shape: output_shape.span(), data: *self.data };
}


/// Cf: TensorTrait::sign docstring
fn sign<T,
MAG,
impl TNumber: NumberTrait<T, MAG>,
Expand Down Expand Up @@ -3147,4 +3188,44 @@ fn sign<T,
}
};
}
}
}

/// Cf: TensorTrait::clip docstring
fn clip<T, MAG, impl TCopy: Copy<T>, impl TDrop: Drop<T>, impl TTensor: TensorTrait<T>, impl TPartialOrd: PartialOrd<T>, impl TNumber: NumberTrait<T, MAG>>(self: @Tensor<T>, min: Option<T>, max: Option<T>) -> Tensor<T> {
let min = match min {
Option::Some(min) => min,
Option::None(_) => {
NumberTrait::min_value()
},
};
let max = match max {
Option::Some(max) => max,
Option::None(_) => {
NumberTrait::max_value()
},
};

let mut return_data: Array<T> = ArrayTrait::new();
let mut self_data_copy = *self.data;

loop {
match self_data_copy.pop_front() {
Option::Some(val) => {
if *val < min {
return_data.append(min);
}
else if *val > max {
return_data.append(max);
}
else {
return_data.append(*val);
}
},
Option::None(_) => {
break ();
}
};
};

return Tensor::<T> {shape: *self.shape, data: return_data.span()};
}
4 changes: 4 additions & 0 deletions src/operators/tensor/implementations/tensor_fp16x16.cairo
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,10 @@ impl FP16x16Tensor of TensorTrait<FP16x16> {
fn sign(self: @Tensor<FP16x16>) -> Tensor<FP16x16> {
math::sign::sign(*self)
}

fn clip(self: @Tensor<FP16x16>, min: Option<FP16x16>, max: Option<FP16x16>) -> Tensor<FP16x16> {
core::clip(self, min, max)
}
}

/// Implements addition for `Tensor<FP16x16>` using the `Add` trait.
Expand Down
4 changes: 4 additions & 0 deletions src/operators/tensor/implementations/tensor_fp32x32.cairo
Original file line number Diff line number Diff line change
Expand Up @@ -226,6 +226,10 @@ impl FP32x32Tensor of TensorTrait<FP32x32> {
fn sign(self: @Tensor<FP32x32>) -> Tensor<FP32x32> {
math::sign::sign(*self)
}

fn clip(self: @Tensor<FP32x32>, min: Option<FP32x32>, max: Option<FP32x32>) -> Tensor<FP32x32> {
core::clip(self, min, max)
}
}

/// Implements addition for `Tensor<FP32x32>` using the `Add` trait.
Expand Down
4 changes: 4 additions & 0 deletions src/operators/tensor/implementations/tensor_fp64x64.cairo
Original file line number Diff line number Diff line change
Expand Up @@ -226,6 +226,10 @@ impl FP64x64Tensor of TensorTrait<FP64x64> {
fn sign(self: @Tensor<FP64x64>) -> Tensor<FP64x64> {
math::sign::sign(*self)
}

fn clip(self: @Tensor<FP64x64>, min: Option<FP64x64>, max: Option<FP64x64>) -> Tensor<FP64x64> {
core::clip(self, min, max)
}
}

/// Implements addition for `Tensor<FP64x64>` using the `Add` trait.
Expand Down
4 changes: 4 additions & 0 deletions src/operators/tensor/implementations/tensor_fp8x23.cairo
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,10 @@ impl FP8x23Tensor of TensorTrait<FP8x23> {
fn sign(self: @Tensor<FP8x23>) -> Tensor<FP8x23> {
math::sign::sign(*self)
}

fn clip(self: @Tensor<FP8x23>, min: Option<FP8x23>, max: Option<FP8x23>) -> Tensor<FP8x23> {
core::clip(self, min, max)
}
}

/// Implements addition for `Tensor<FP8x23>` using the `Add` trait.
Expand Down
Loading

0 comments on commit 1231a26

Please sign in to comment.