1
// Tencent is pleased to support the open source community by making ncnn available.
3
// Copyright (C) 2021 THL A29 Limited, a Tencent company. All rights reserved.
5
// Licensed under the BSD 3-Clause License (the "License"); you may not use this file except
6
// in compliance with the License. You may obtain a copy of the License at
8
// https://opensource.org/licenses/BSD-3-Clause
10
// Unless required by applicable law or agreed to in writing, software distributed
11
// under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12
// CONDITIONS OF ANY KIND, either express or implied. See the License for the
13
// specific language governing permissions and limitations under the License.
15
#include "concat_mips.h"
19
Concat_mips::Concat_mips()
22
support_packing = true;
26
int Concat_mips::forward(const std::vector<Mat>& bottom_blobs, std::vector<Mat>& top_blobs, const Option& opt) const
28
int dims = bottom_blobs[0].dims;
29
int positive_axis = axis < 0 ? dims + axis : axis;
31
if (dims == 1) // positive_axis == 0
35
size_t elemsize = bottom_blobs[0].elemsize;
36
int elempack = bottom_blobs[0].elempack;
38
for (size_t b = 0; b < bottom_blobs.size(); b++)
40
const Mat& bottom_blob = bottom_blobs[b];
41
top_w += bottom_blob.w * bottom_blob.elempack;
44
int out_elempack = opt.use_packing_layout && top_w % 4 == 0 ? 4 : 1;
45
size_t out_elemsize = elemsize / elempack * out_elempack;
47
Mat& top_blob = top_blobs[0];
48
top_blob.create(top_w / out_elempack, out_elemsize, out_elempack, opt.blob_allocator);
52
float* outptr = top_blob;
53
for (size_t b = 0; b < bottom_blobs.size(); b++)
55
const Mat& bottom_blob = bottom_blobs[b];
57
const float* ptr = bottom_blob;
58
memcpy(outptr, ptr, bottom_blob.w * bottom_blob.elemsize);
60
outptr += bottom_blob.w * bottom_blob.elempack;
64
if (dims == 2 && positive_axis == 0)
67
int w = bottom_blobs[0].w;
70
size_t elemsize = bottom_blobs[0].elemsize;
71
int elempack = bottom_blobs[0].elempack;
73
for (size_t b = 0; b < bottom_blobs.size(); b++)
75
const Mat& bottom_blob = bottom_blobs[b];
76
elemsize = std::min(elemsize, bottom_blob.elemsize);
77
elempack = std::min(elempack, bottom_blob.elempack);
78
top_h += bottom_blob.h * bottom_blob.elempack;
81
int out_elempack = opt.use_packing_layout && top_h % 4 == 0 ? 4 : 1;
82
size_t out_elemsize = elemsize / elempack * out_elempack;
84
Mat& top_blob = top_blobs[0];
85
top_blob.create(w, top_h / out_elempack, out_elemsize, out_elempack, opt.blob_allocator);
89
Mat top_blob_unpacked = top_blob;
90
if (elempack < out_elempack)
92
top_blob_unpacked.create(w, top_h / elempack, elemsize, elempack, opt.workspace_allocator);
93
if (top_blob_unpacked.empty())
97
float* outptr = top_blob_unpacked;
98
for (size_t b = 0; b < bottom_blobs.size(); b++)
100
const Mat& bottom_blob = bottom_blobs[b];
102
if (bottom_blob.elempack == 4 && elempack == 1)
104
for (int i = 0; i < bottom_blob.h; i++)
106
const float* r0 = bottom_blob.row(i);
108
float* outptr0 = outptr;
109
float* outptr1 = outptr + w;
110
float* outptr2 = outptr + w * 2;
111
float* outptr3 = outptr + w * 3;
113
for (int j = 0; j < w; j++)
126
else // if (bottom_blob.elempack == 1 && elempack == 1) if (bottom_blob.elempack == 4 && elempack == 4)
128
int size = w * bottom_blob.h;
130
const float* ptr = bottom_blob;
131
memcpy(outptr, ptr, size * bottom_blob.elemsize);
133
outptr += size * bottom_blob.elempack;
138
if (elempack < out_elempack)
140
convert_packing(top_blob_unpacked, top_blob, out_elempack, opt);
144
if (dims == 2 && positive_axis == 1)
146
// interleave image row
147
int h = bottom_blobs[0].h;
148
size_t elemsize = bottom_blobs[0].elemsize;
149
int elempack = bottom_blobs[0].elempack;
153
for (size_t b = 0; b < bottom_blobs.size(); b++)
155
const Mat& bottom_blob = bottom_blobs[b];
156
top_w += bottom_blob.w;
159
Mat& top_blob = top_blobs[0];
160
top_blob.create(top_w, h, elemsize, elempack, opt.blob_allocator);
161
if (top_blob.empty())
164
#pragma omp parallel for num_threads(opt.num_threads)
165
for (int i = 0; i < h; i++)
167
float* outptr = top_blob.row(i);
168
for (size_t b = 0; b < bottom_blobs.size(); b++)
170
const Mat& bottom_blob = bottom_blobs[b];
172
const float* ptr = bottom_blob.row(i);
173
memcpy(outptr, ptr, bottom_blob.w * elemsize);
175
outptr += bottom_blob.w * elempack;
180
if ((dims == 3 || dims == 4) && positive_axis == 0)
183
int w = bottom_blobs[0].w;
184
int h = bottom_blobs[0].h;
185
int d = bottom_blobs[0].d;
188
size_t elemsize = bottom_blobs[0].elemsize;
189
int elempack = bottom_blobs[0].elempack;
190
int top_channels = 0;
191
for (size_t b = 0; b < bottom_blobs.size(); b++)
193
const Mat& bottom_blob = bottom_blobs[b];
194
elemsize = std::min(elemsize, bottom_blob.elemsize);
195
elempack = std::min(elempack, bottom_blob.elempack);
196
top_channels += bottom_blob.c * bottom_blob.elempack;
199
int out_elempack = opt.use_packing_layout && top_channels % 4 == 0 ? 4 : 1;
200
size_t out_elemsize = elemsize / elempack * out_elempack;
202
Mat& top_blob = top_blobs[0];
203
top_blob.create(w, h, d, top_channels / out_elempack, out_elemsize, out_elempack, opt.blob_allocator);
204
if (top_blob.empty())
207
top_blob.dims = dims;
209
Mat top_blob_unpacked = top_blob;
210
if (elempack < out_elempack)
212
top_blob_unpacked.create(w, h, d, top_channels / elempack, elemsize, elempack, opt.workspace_allocator);
213
if (top_blob_unpacked.empty())
216
top_blob_unpacked.dims = dims;
220
for (size_t b = 0; b < bottom_blobs.size(); b++)
222
const Mat& bottom_blob = bottom_blobs[b];
224
if (bottom_blob.elempack == 4 && elempack == 1)
226
int size = bottom_blob.w * bottom_blob.h * bottom_blob.d;
228
for (int q = 0; q < bottom_blob.c; q++)
230
const float* r0 = bottom_blob.channel(q);
232
float* outptr0 = top_blob_unpacked.channel(p);
233
float* outptr1 = top_blob_unpacked.channel(p + 1);
234
float* outptr2 = top_blob_unpacked.channel(p + 2);
235
float* outptr3 = top_blob_unpacked.channel(p + 3);
237
for (int i = 0; i < size; i++)
250
else // if (bottom_blob.elempack == 1 && elempack == 1) if (bottom_blob.elempack == 4 && elempack == 4)
252
int size = bottom_blob.total();
254
const float* ptr = bottom_blob;
255
float* outptr = top_blob_unpacked.channel(p);
256
memcpy(outptr, ptr, size * bottom_blob.elemsize);
263
if (elempack < out_elempack)
265
convert_packing(top_blob_unpacked, top_blob, out_elempack, opt);
269
if ((dims == 3 && positive_axis == 1) || (dims == 4 && positive_axis == 2))
271
// interleave dim height
272
int w = bottom_blobs[0].w;
273
int d = bottom_blobs[0].d;
274
int channels = bottom_blobs[0].c;
275
size_t elemsize = bottom_blobs[0].elemsize;
276
int elempack = bottom_blobs[0].elempack;
280
for (size_t b = 0; b < bottom_blobs.size(); b++)
282
const Mat& bottom_blob = bottom_blobs[b];
283
top_h += bottom_blob.h;
286
Mat& top_blob = top_blobs[0];
287
top_blob.create(w, top_h, d, channels, elemsize, elempack, opt.blob_allocator);
288
if (top_blob.empty())
291
top_blob.dims = dims;
293
#pragma omp parallel for num_threads(opt.num_threads)
294
for (int q = 0; q < channels; q++)
296
float* outptr = top_blob.channel(q);
298
for (int i = 0; i < d; i++)
300
for (size_t b = 0; b < bottom_blobs.size(); b++)
302
const Mat& bottom_blob = bottom_blobs[b];
304
int size = bottom_blob.w * bottom_blob.h;
306
const float* ptr = bottom_blob.channel(q).depth(i);
307
memcpy(outptr, ptr, size * elemsize);
309
outptr += size * elempack;
315
if ((dims == 3 && positive_axis == 2) || (dims == 4 && positive_axis == 3))
317
// interleave dim width
318
int h = bottom_blobs[0].h;
319
int d = bottom_blobs[0].d;
320
int channels = bottom_blobs[0].c;
321
size_t elemsize = bottom_blobs[0].elemsize;
322
int elempack = bottom_blobs[0].elempack;
326
for (size_t b = 0; b < bottom_blobs.size(); b++)
328
const Mat& bottom_blob = bottom_blobs[b];
329
top_w += bottom_blob.w;
332
Mat& top_blob = top_blobs[0];
333
top_blob.create(top_w, h, d, channels, elemsize, elempack, opt.blob_allocator);
334
if (top_blob.empty())
337
top_blob.dims = dims;
339
#pragma omp parallel for num_threads(opt.num_threads)
340
for (int q = 0; q < channels; q++)
342
float* outptr = top_blob.channel(q);
344
for (int i = 0; i < d; i++)
346
for (int j = 0; j < h; j++)
348
for (size_t b = 0; b < bottom_blobs.size(); b++)
350
const Mat& bottom_blob = bottom_blobs[b];
352
const float* ptr = bottom_blob.channel(q).depth(i).row(j);
353
memcpy(outptr, ptr, bottom_blob.w * elemsize);
355
outptr += bottom_blob.w * elempack;
362
if (dims == 4 && positive_axis == 1)
364
// interleave dim depth
365
int w = bottom_blobs[0].w;
366
int h = bottom_blobs[0].h;
367
int channels = bottom_blobs[0].c;
368
size_t elemsize = bottom_blobs[0].elemsize;
369
int elempack = bottom_blobs[0].elempack;
373
for (size_t b = 0; b < bottom_blobs.size(); b++)
375
const Mat& bottom_blob = bottom_blobs[b];
376
top_d += bottom_blob.d;
379
Mat& top_blob = top_blobs[0];
380
top_blob.create(w, h, top_d, channels, elemsize, elempack, opt.blob_allocator);
381
if (top_blob.empty())
384
#pragma omp parallel for num_threads(opt.num_threads)
385
for (int q = 0; q < channels; q++)
387
float* outptr = top_blob.channel(q);
389
for (size_t b = 0; b < bottom_blobs.size(); b++)
391
const Mat& bottom_blob = bottom_blobs[b];
393
int size = bottom_blob.w * bottom_blob.h * bottom_blob.d;
395
const float* ptr = bottom_blob.channel(q);
396
memcpy(outptr, ptr, size * elemsize);
398
outptr += size * elempack;