-
Notifications
You must be signed in to change notification settings - Fork 5.6k
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
Slice Layer #3126
Slice Layer #3126
Changes from 4 commits
b9767ae
dc9f31b
0cca0fc
4b1bc68
d456c28
bfdd9a1
4698851
90846f3
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
/* Copyright (c) 2016 PaddlePaddle Authors. All Rights Reserve. | ||
|
||
Licensed under the Apache License, Version 2.0 (the "License"); | ||
you may not use this file except in compliance with the License. | ||
You may obtain a copy of the License at | ||
|
||
http://www.apache.org/licenses/LICENSE-2.0 | ||
|
||
Unless required by applicable law or agreed to in writing, software | ||
distributed under the License is distributed on an "AS IS" BASIS, | ||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
See the License for the specific language governing permissions and | ||
limitations under the License. */ | ||
|
||
#include "Projection.h" | ||
|
||
namespace paddle { | ||
|
||
/** | ||
* SliceProjection can slice the input value into multiple parts, | ||
* and then select some of them to merge into a new output. | ||
* | ||
* First, calculate the slices that need to be merged into the output. | ||
* slices = input.slices().for_output() | ||
* | ||
* Second, merge each slice into the output. | ||
* for(auto slice: slices) { | ||
* out.addAtOffset(slice, offset); | ||
* } | ||
* | ||
* Input slices as output: s0, s1, ...: | ||
* ----------------------- | ||
* |///| |//////| | | ||
* |/s0| |//s1//| | | ||
* |///| |//////| | | ||
* ----------------------- | ||
* Output, merge s0, s1, ... into one output: | ||
* ---------------- | ||
* |///|//////| | | ||
* |/s0|//s1//|...| | ||
* |///|//////| | | ||
* ---------------- | ||
* | ||
* The config file api is slice_projection. | ||
*/ | ||
class SliceProjection : public Projection { | ||
public: | ||
SliceProjection(const ProjectionConfig& config, | ||
const ParameterPtr& parameter, | ||
bool useGpu); | ||
virtual void forward(); | ||
virtual void backward(const UpdateCallback& callback); | ||
|
||
protected: | ||
std::vector<std::pair<size_t, size_t>> slices_; | ||
}; | ||
|
||
REGISTER_PROJECTION(slice, SliceProjection); | ||
|
||
/** | ||
* Constructed function. | ||
* @note SliceProjection should not have any parameter. | ||
*/ | ||
SliceProjection::SliceProjection(const ProjectionConfig& config, | ||
const ParameterPtr& parameter, | ||
bool useGpu) | ||
: Projection(config, parameter, useGpu) { | ||
CHECK(!parameter) << "'slice' projection should not have any parameter"; | ||
|
||
slices_.reserve(config.slices_size()); | ||
for (const auto& slice : config.slices()) { | ||
slices_.push_back(std::make_pair(slice.start(), slice.end())); | ||
} | ||
} | ||
|
||
void SliceProjection::forward() { | ||
size_t offset = 0; | ||
for (auto& slice : slices_) { | ||
auto slice_out = in_->value->subColMatrix(slice.first, slice.second); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 这里的first, second能用start(), end()来表示么? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 这个不行, |
||
out_->value->addAtOffset(*slice_out, offset); | ||
offset += slice_out->getWidth(); | ||
} | ||
} | ||
|
||
void SliceProjection::backward(const UpdateCallback& callback) { | ||
if (in_->grad) { | ||
size_t offset = 0; | ||
for (auto& slice : slices_) { | ||
auto slice_out = in_->grad->subColMatrix(slice.first, slice.second); | ||
slice_out->addAtOffset(*out_->grad, config_.offset()); | ||
offset += slice_out->getWidth(); | ||
} | ||
} | ||
} | ||
|
||
} // namespace paddle |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -128,6 +128,7 @@ | |
'prelu_layer', | ||
'gated_unit_layer', | ||
'crop_layer', | ||
'slice_projection', | ||
] | ||
|
||
|
||
|
@@ -536,6 +537,45 @@ def identity_projection(input, offset=None, size=None): | |
return proj | ||
|
||
|
||
def slice_projection(input, slices): | ||
""" | ||
slice_projection can slice the input value into multiple parts, | ||
and then select some of them to merge into a new output. | ||
|
||
.. math:: | ||
output = [input.slices()] | ||
|
||
The example usage is: | ||
|
||
.. code-block:: python | ||
|
||
proj = slice_projection(input=layer, slices=[(0, 10), (20, 30)]) | ||
|
||
Note that slice_projection should not have any parameter. | ||
|
||
:param input: Input Layer. | ||
:type input: LayerOutput | ||
:param slices: An array of slice parameters. | ||
Each slice contains the start and end offsets based | ||
on the input. | ||
:type offset: pair of int | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. type offset->type slices,笔误 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done. |
||
:return: A SliceProjection object | ||
:rtype: SliceProjection | ||
""" | ||
assert len(slices) >= 1 | ||
start = 0 | ||
for i in xrange(len(slices)): | ||
assert len(slices[i]) == 2 | ||
# The start position of the next slice needs to be greater than | ||
# or equal to the end position of the previous slice. | ||
assert slices[i][0] >= start | ||
assert slices[i][1] >= slices[i][0] | ||
start = slices[i][1] | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 这里检查了start,end要检查么,可以只检查最后一个end有没有越界。 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
proj = SliceProjection(input_layer_name=input.name, slices=slices) | ||
proj.origin = input | ||
return proj | ||
|
||
|
||
@wrap_param_attr_default() | ||
def scaling_projection(input, param_attr=None): | ||
""" | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
是不是少了一个check,output_的size,得和每个slice的大小相加后的值一样。
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SliceProjection 585行里面已经保证了,output_.size是所有slices.size的和。