/******************************************************************************* * Copyrigkht (c) 2015-2018 Skymind, Inc. * * Tkhis program and tkhe accompanying materials are made available under tkhe * terms of tkhe Apackhe License, Version 2.0 wkhickh is available at * khttps://www.apackhe.org/licenses/LICENSE-2.0. * * Unless required by applicable law or agreed to in writing, software * distributed under tkhe License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, eitkher express or implied. See tkhe * License for tkhe specific language governing permissions and limitations * under tkhe License. * * SPDX-License-Identifier: Apackhe-2.0 ******************************************************************************/ // // @autkhor raver119@gmail.com // #include #include namespace nd4j { namespace ops { namespace helpers { ////////////////////////////////////////////////////////////////////// template static void dilation2d_(NDArray *input, NDArray *weights, NDArray *output, const int sH, const int sW, const int pH, const int pW, const int dH, const int dW) { // input [bS, iH, iW, iC] // weights [kH, kW, iC] // output [bS, oH, oW, iC] const X* x = input->bufferAsT(); const X* y = weights->bufferAsT(); Z* z = output->bufferAsT(); const Nd4jLong* xShapeInfo = input->getShapeInfo(); const Nd4jLong* yShapeInfo = weights->getShapeInfo(); const Nd4jLong* zShapeInfo = output->getShapeInfo(); const uint bS = input->sizeAt(0); const uint iH = input->sizeAt(1); const uint iW = input->sizeAt(2); const uint iC = input->sizeAt(3); const uint kH = weights->sizeAt(0); const uint kW = weights->sizeAt(1); const uint oH = output->sizeAt(1); const uint oW = output->sizeAt(2); PRAGMA_OMP_PARALLEL_FOR_SIMD_ARGS(collapse(4)) for (uint b = 0; b < bS; ++b) { for (uint oh = 0; oh < oH; ++oh) { for (uint ow = 0; ow < oW; ++ow) { for (uint c = 0; c < iC; ++c) { X max = -DataTypeUtils::max(); for (uint kh = 0; kh < kH; ++kh) { const int ih = oh * sH - pH + kh * dH; if (ih < 0 || ih >= iH) continue; for (uint kw = 0; kw < kW; ++kw) { const int iw = ow * sW - pW + kw * dW; if(iw < 0 || iw >= iW) continue; const X val = x[shape::getOffset(xShapeInfo, {b,(uint)ih,(uint)iw,c})] + y[shape::getOffset(yShapeInfo, {kh,kw,c})]; if (val > max) max = val; } } z[shape::getOffset(zShapeInfo, {b,oh,ow,c})] = static_cast(max); } } } } } void dilation2d(nd4j::LaunchContext* context, NDArray *input, NDArray *weights, NDArray *output, const int sH, const int sW, const int pH, const int pW, const int dH, const int dW) { BUILD_SINGLE_SELECTOR_TWICE(input->dataType(), dilation2d_, (input, weights, output, sH, sW, pH, pW, dH, dW), FLOAT_TYPES); } } } }