| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463 |
- // Copyright 2020-2021 Espressif Systems (Shanghai) PTE LTD
- //
- // 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 <stdio.h>
- #include <esp_nn_defs.h>
- #include <common_functions.h>
- static int16_t *scratch_buffer = NULL;
- extern void esp_nn_conv_s8_mult8_1x1_esp32s3(const int8_t *input_data,
- const uint16_t input_wd,
- const uint16_t input_ht,
- const uint16_t in_channels,
- const int32_t input_offset,
- const int8_t *filter_aligned,
- const int32_t *bias,
- int8_t *out_data,
- const uint16_t out_wd,
- const uint16_t out_ht,
- const uint16_t out_channels,
- const int32_t out_offset,
- const int32_t *out_shift,
- const int32_t *out_mult,
- const int32_t activation_min,
- const int32_t activation_max,
- void *buffer /* scratch buffer */);
- extern void esp_nn_conv_s16_mult4_1x1_esp32s3(const int16_t *input_data,
- const uint16_t input_wd,
- const uint16_t input_ht,
- const uint16_t in_channels,
- const int16_t *filter_data,
- const int32_t *bias,
- int8_t *out_data,
- const uint16_t out_wd,
- const uint16_t out_ht,
- const uint16_t out_channels,
- const int32_t out_offset,
- const int32_t *out_shift,
- const int32_t *out_mult,
- const int32_t activation_min,
- const int32_t activation_max,
- void *buffer /* scratch buffer */);
- extern void esp_nn_conv_s16_mult8_esp32s3(const int16_t *input_data,
- const uint16_t input_wd,
- const uint16_t input_ht,
- const uint16_t in_channels,
- const uint16_t pad_wd,
- const uint16_t pad_ht,
- const uint16_t stride_wd,
- const uint16_t stride_ht,
- const int16_t *filter_data,
- const uint16_t filter_wd,
- const uint16_t filter_ht,
- const int32_t *bias,
- int8_t *out_data,
- const uint16_t out_wd,
- const uint16_t out_ht,
- const uint16_t out_channels,
- const int32_t out_offset,
- const int32_t *out_shift,
- const int32_t *out_mult,
- const int32_t activation_min,
- const int32_t activation_max);
- extern void esp_nn_aligned_s8_to_s16_with_offset_esp32s3(const int8_t *src, int16_t *dst,
- const int size, const int32_t offset);
- extern void esp_nn_s8_to_s16_esp32s3(const int8_t *src, int16_t *dst, const int size);
- static void esp_nn_conv_s8_unrolled(const data_dims_t *input_dims,
- const int8_t *input_data,
- const data_dims_t *filter_dims,
- const int8_t *filter_data,
- const int32_t *bias,
- const data_dims_t *output_dims,
- int8_t *out_data,
- const conv_params_t *conv_params,
- const quant_data_t *quant_data)
- {
- const uint16_t input_wd = input_dims->width;
- const uint16_t input_ht = input_dims->height;
- const uint16_t in_ch = input_dims->channels;
- const int32_t input_offset = conv_params->in_offset;
- const int32_t out_offset = conv_params->out_offset;
- const uint16_t pad_wd = conv_params->padding.width;
- const uint16_t pad_ht = conv_params->padding.height;
- const uint16_t stride_wd = conv_params->stride.width;
- const uint16_t stride_ht = conv_params->stride.height;
- const uint16_t filter_wd = filter_dims->width;
- const uint16_t filter_ht = filter_dims->height;
- const uint16_t out_wd = output_dims->width;
- const uint16_t out_ht = output_dims->height;
- const uint16_t out_ch = output_dims->channels;
- const int32_t *out_shift = quant_data->shift;
- const int32_t *out_mult = quant_data->mult;
- const int32_t activation_min = conv_params->activation.min;
- const int32_t activation_max = conv_params->activation.max;
- int32_t out_ch_idx, out_y, out_x, in_ch_idx, filter_y_idx, filter_x_idx;
- for (out_y = 0; out_y < out_ht; out_y++) {
- for (out_x = 0; out_x < out_wd; out_x++) {
- for (out_ch_idx = 0; out_ch_idx < out_ch; out_ch_idx++) {
- int32_t conv_out = 0;
- const int32_t base_y = stride_ht * out_y - pad_ht;
- const int32_t base_x = stride_wd * out_x - pad_wd;
- const int32_t filter_y_start = max(0, -base_y);
- const int32_t filter_x_start = max(0, -base_x);
- const int32_t filter_y_end = min(filter_ht, input_ht - base_y);
- const int32_t filter_x_end = min(filter_wd, input_wd - base_x);
- for (filter_y_idx = filter_y_start; filter_y_idx < filter_y_end; filter_y_idx++) {
- for (filter_x_idx = filter_x_start; filter_x_idx < filter_x_end; filter_x_idx++) {
- const int32_t in_row = base_y + filter_y_idx;
- const int32_t in_col = base_x + filter_x_idx;
- int32_t input_base_offset = (in_row * input_wd + in_col) * in_ch;
- int32_t filter_base_offset = out_ch_idx * in_ch * filter_ht * filter_wd +
- (filter_y_idx * filter_wd + filter_x_idx) * in_ch;
- for (in_ch_idx = 0; in_ch_idx < in_ch; in_ch_idx++) {
- conv_out +=
- (input_data[input_base_offset + in_ch_idx] + input_offset) *
- filter_data[filter_base_offset + in_ch_idx];
- }
- }
- }
- if (bias) {
- conv_out += bias[out_ch_idx];
- }
- conv_out = esp_nn_multiply_by_quantized_mult_fast(conv_out, out_mult[out_ch_idx], out_shift[out_ch_idx]);
- conv_out += out_offset;
- conv_out = max(conv_out, activation_min);
- conv_out = min(conv_out, activation_max);
- *out_data++ = (int8_t) conv_out;
- }
- }
- }
- }
- static void esp_nn_conv_s8_pad_valid(const int8_t *input_data,
- const uint16_t input_wd,
- const uint16_t input_ht,
- const uint16_t in_channels,
- const int32_t input_offset,
- const uint16_t stride_wd,
- const uint16_t stride_ht,
- const int8_t *filter_data,
- const uint16_t filter_wd,
- const uint16_t filter_ht,
- const int32_t *bias,
- int8_t *out_data,
- const uint16_t out_wd,
- const uint16_t out_ht,
- const uint16_t out_channels,
- const int32_t out_offset,
- const int32_t *out_shift,
- const int32_t *out_mult,
- const int32_t activation_min,
- const int32_t activation_max)
- {
- int32_t out_ch_idx, out_y, out_x, in_ch_idx, filter_y_idx, filter_x_idx;
- for (out_y = 0; out_y < out_ht; out_y++) {
- for (out_x = 0; out_x < out_wd; out_x++) {
- for (out_ch_idx = 0; out_ch_idx < out_channels; out_ch_idx++) {
- int32_t conv_out = 0;
- const int32_t base_y = stride_ht * out_y;
- const int32_t base_x = stride_wd * out_x;
- for (filter_y_idx = 0; filter_y_idx < filter_ht; filter_y_idx++) {
- for (filter_x_idx = 0; filter_x_idx < filter_wd; filter_x_idx++) {
- const int32_t in_row = base_y + filter_y_idx;
- const int32_t in_col = base_x + filter_x_idx;
- int32_t input_base_offset = (in_row * input_wd + in_col) * in_channels;
- int32_t filter_base_offset = out_ch_idx * in_channels * filter_ht * filter_wd +
- (filter_y_idx * filter_wd + filter_x_idx) * in_channels;
- const int8_t *input_data_ptr = input_data + input_base_offset;
- const int8_t *filter_data_ptr = filter_data + filter_base_offset;
- for (in_ch_idx = 0; in_ch_idx < in_channels; in_ch_idx++) {
- conv_out += (*input_data_ptr++ + input_offset) * *filter_data_ptr++;
- }
- }
- }
- if (bias) {
- conv_out += bias[out_ch_idx];
- }
- conv_out = esp_nn_multiply_by_quantized_mult_fast(conv_out, out_mult[out_ch_idx], out_shift[out_ch_idx]);
- conv_out += out_offset;
- conv_out = max(conv_out, activation_min);
- conv_out = min(conv_out, activation_max);
- *out_data++ = (int8_t) conv_out;
- }
- }
- }
- }
- static void esp_nn_conv_s8_pad_valid_3x3(const int8_t *input_data,
- const uint16_t input_wd,
- const uint16_t input_ht,
- const uint16_t in_channels,
- const int32_t input_offset,
- const uint16_t stride_wd,
- const uint16_t stride_ht,
- const int8_t *filter_data,
- const int32_t *bias,
- int8_t *out_data,
- const uint16_t out_wd,
- const uint16_t out_ht,
- const uint16_t out_channels,
- const int32_t out_offset,
- const int32_t *out_shift,
- const int32_t *out_mult,
- const int32_t activation_min,
- const int32_t activation_max)
- {
- int32_t out_ch_idx, out_y, out_x, in_ch_idx, filter_y_idx, filter_x_idx;
- for (out_y = 0; out_y < out_ht; out_y++) {
- for (out_x = 0; out_x < out_wd; out_x++) {
- const int32_t base_y = stride_ht * out_y;
- const int32_t base_x = stride_wd * out_x;
- for (out_ch_idx = 0; out_ch_idx < out_channels; out_ch_idx++) {
- int32_t conv_out = 0;
- for (filter_y_idx = 0; filter_y_idx < 3; filter_y_idx++) {
- for (filter_x_idx = 0; filter_x_idx < 3; filter_x_idx++) {
- const int32_t in_row = base_y + filter_y_idx;
- const int32_t in_col = base_x + filter_x_idx;
- int32_t input_base_offset = (in_row * input_wd + in_col) * in_channels;
- int32_t filter_base_offset = out_ch_idx * in_channels * 3 * 3 +
- (filter_y_idx * 3 + filter_x_idx) * in_channels;
- const int8_t *input_data_ptr = input_data + input_base_offset;
- const int8_t *filter_data_ptr = filter_data + filter_base_offset;
- for (in_ch_idx = 0; in_ch_idx < in_channels; in_ch_idx++) {
- conv_out += (*input_data_ptr++ + input_offset) * *filter_data_ptr++;
- }
- }
- }
- if (bias) {
- conv_out += bias[out_ch_idx];
- }
- conv_out = esp_nn_multiply_by_quantized_mult_fast(conv_out, out_mult[out_ch_idx], out_shift[out_ch_idx]);
- conv_out += out_offset;
- conv_out = max(conv_out, activation_min);
- conv_out = min(conv_out, activation_max);
- *out_data++ = (int8_t) conv_out;
- }
- }
- }
- }
- static void esp_nn_conv_s8_pad_valid_ch3_3x3(const int8_t *input_data,
- const uint16_t input_wd,
- const uint16_t input_ht,
- const int32_t input_offset,
- const uint16_t stride_wd,
- const uint16_t stride_ht,
- const int8_t *filter_data,
- const int32_t *bias,
- int8_t *out_data,
- const uint16_t out_wd,
- const uint16_t out_ht,
- const uint16_t out_channels,
- const int32_t out_offset,
- const int32_t *out_shift,
- const int32_t *out_mult,
- const int32_t activation_min,
- const int32_t activation_max)
- {
- int32_t out_ch_idx, out_y, out_x, filter_y_idx;
- /* use scratch_buffer to pre-compute offset factor */
- int16_t *filter_sum = (int16_t *) scratch_buffer;
- const int8_t *filter_ptr = filter_data;
- for (out_ch_idx = 0; out_ch_idx < out_channels; out_ch_idx++) {
- int16_t sum_val = 0;
- for (int i = 0; i < 9; i++) {
- sum_val += *filter_ptr++;
- sum_val += *filter_ptr++;
- sum_val += *filter_ptr++;
- }
- *filter_sum++ = sum_val;
- }
- for (out_y = 0; out_y < out_ht; out_y++) {
- for (out_x = 0; out_x < out_wd; out_x++) {
- const int8_t *filter_data_ptr = filter_data;
- const int32_t base_y = stride_ht * out_y;
- const int32_t base_x = stride_wd * out_x;
- const int8_t *input_base_ptr = input_data + (base_y * input_wd + base_x) * 3;
- int16_t *filter_sum = (int16_t *) scratch_buffer;
- for (out_ch_idx = 0; out_ch_idx < out_channels; out_ch_idx++) {
- int32_t conv_out = 0;
- for (filter_y_idx = 0; filter_y_idx < 3; filter_y_idx++) {
- const int8_t *input_data_ptr = input_base_ptr + (filter_y_idx * input_wd) * 3;
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- conv_out += (*input_data_ptr++) * (*filter_data_ptr++);
- }
- conv_out += *filter_sum++ * input_offset;
- if (bias) {
- conv_out += bias[out_ch_idx];
- }
- conv_out = esp_nn_multiply_by_quantized_mult_fast(conv_out, out_mult[out_ch_idx], out_shift[out_ch_idx]);
- conv_out += out_offset;
- conv_out = max(conv_out, activation_min);
- conv_out = min(conv_out, activation_max);
- *out_data++ = (int8_t) conv_out;
- }
- }
- }
- }
- int esp_nn_get_conv_scratch_size_esp32s3(const data_dims_t *input_dims,
- const data_dims_t *filter_dims,
- const data_dims_t *output_dims,
- const conv_params_t *conv_params)
- {
- const uint16_t input_wd = input_dims->width;
- const uint16_t input_ht = input_dims->height;
- const uint16_t in_ch = input_dims->channels;
- const uint16_t filter_wd = filter_dims->width;
- const uint16_t filter_ht = filter_dims->height;
- const uint16_t out_ch = output_dims->channels;
- const uint16_t pad_wd = conv_params->padding.width;
- const uint16_t pad_ht = conv_params->padding.height;
- const uint16_t stride_wd = conv_params->stride.width;
- const uint16_t stride_ht = conv_params->stride.height;
- int filter_size = filter_wd * filter_ht * in_ch * out_ch;
- int input_size = input_wd * input_ht * in_ch;
- int transpose_buf_size = 2 * (8 * in_ch); /* to store intermediate data */
- if (input_wd * input_ht < 8) {
- transpose_buf_size = 0; // not using this for leftover
- }
- int align_buf_size = 32; /* extra buffer for alignment */
- if (in_ch % 8 == 0 && filter_wd == 1 && filter_ht == 1 &&
- pad_wd == 0 && pad_ht == 0 && stride_wd == 1 && stride_ht == 1) {
- return filter_size + transpose_buf_size + align_buf_size;
- }
- return 2 * (filter_size + input_size) + transpose_buf_size + align_buf_size;
- }
- void esp_nn_set_conv_scratch_buf_esp32s3(void *buf)
- {
- scratch_buffer = (int16_t *) buf;
- }
- void esp_nn_conv_s8_esp32s3(const data_dims_t *input_dims,
- const int8_t *input,
- const data_dims_t *filter_dims,
- const int8_t *filter_data,
- const int32_t *bias,
- const data_dims_t *output_dims,
- int8_t *out_data,
- const conv_params_t *conv_params,
- const quant_data_t *quant_data)
- {
- const uint16_t input_wd = input_dims->width;
- const uint16_t input_ht = input_dims->height;
- const uint16_t channels = input_dims->channels;
- const int32_t input_offset = conv_params->in_offset;
- const int32_t out_offset = conv_params->out_offset;
- const uint16_t pad_wd = conv_params->padding.width;
- const uint16_t pad_ht = conv_params->padding.height;
- const uint16_t stride_wd = conv_params->stride.width;
- const uint16_t stride_ht = conv_params->stride.height;
- const uint16_t filter_wd = filter_dims->width;
- const uint16_t filter_ht = filter_dims->height;
- const uint16_t out_wd = output_dims->width;
- const uint16_t out_ht = output_dims->height;
- const uint16_t out_channels = output_dims->channels;
- const int32_t *out_shift = quant_data->shift;
- const int32_t *out_mult = quant_data->mult;
- const int32_t activation_min = conv_params->activation.min;
- const int32_t activation_max = conv_params->activation.max;
- int filter_size = filter_wd * filter_ht * channels * out_channels;
- int input_size = input_wd * input_ht * channels;
- int align_len = 16 - (filter_size & 15);
- int16_t *filter_data16 = scratch_buffer;
- int16_t *input_data16 = scratch_buffer + filter_size + align_len;
- if (scratch_buffer == NULL) {
- printf("esp_nn_conv error! scratch_buffer not set!\n");
- return;
- }
- if (channels % 8 == 0 && filter_wd == 1 && filter_ht == 1 &&
- pad_wd == 0 && pad_ht == 0 && stride_wd == 1 && stride_ht == 1) {
- int8_t *filter_aligned = (int8_t *) scratch_buffer;
- int scratch_offset = (int) (filter_aligned + filter_size);
- void *scratch_buf = (void *) (scratch_offset + 16 - (scratch_offset & 15));
- memcpy(filter_aligned, filter_data, filter_size); // copy to aligned address
- esp_nn_conv_s8_mult8_1x1_esp32s3(
- input, input_wd, input_ht, channels, input_offset, filter_aligned,
- bias, out_data, out_wd, out_ht, out_channels, out_offset,
- out_shift, out_mult, activation_min, activation_max, scratch_buf);
- } else if (channels % 4 == 0 && filter_wd == 1 && filter_ht == 1 &&
- (input_wd * input_ht) % 4 == 0 && /* TODO: remove this check */
- pad_wd == 0 && pad_ht == 0 && stride_wd == 1 && stride_ht == 1) {
- int scratch_offset = (int) (input_data16 + input_size);
- void *scratch_buf = (void *) (scratch_offset + 16 - (scratch_offset & 15));
- esp_nn_s8_to_s16_esp32s3(filter_data, filter_data16, filter_size);
- esp_nn_aligned_s8_to_s16_with_offset_esp32s3(input, input_data16, input_size, input_offset);
- esp_nn_conv_s16_mult4_1x1_esp32s3(
- input_data16, input_wd, input_ht, channels, filter_data16,
- bias, out_data, out_wd, out_ht, out_channels, out_offset,
- out_shift, out_mult, activation_min, activation_max, scratch_buf);
- } else if (channels % 8 == 0) {
- esp_nn_s8_to_s16_esp32s3(filter_data, filter_data16, filter_size);
- esp_nn_aligned_s8_to_s16_with_offset_esp32s3(input, input_data16, input_size, input_offset);
- esp_nn_conv_s16_mult8_esp32s3(
- input_data16, input_wd, input_ht, channels, pad_wd, pad_ht,
- stride_wd, stride_ht, filter_data16, filter_wd, filter_ht, bias,
- out_data, out_wd, out_ht, out_channels, out_offset, out_shift,
- out_mult, activation_min, activation_max);
- } else if (pad_wd == 0 && pad_ht == 0) {
- if (filter_wd == 3 && filter_ht == 3 && channels == 3) {
- esp_nn_conv_s8_pad_valid_ch3_3x3(input, input_wd, input_ht, input_offset,
- stride_wd, stride_ht, filter_data, bias,
- out_data, out_wd, out_ht, out_channels, out_offset,
- out_shift, out_mult, activation_min, activation_max);
- } else {
- esp_nn_conv_s8_pad_valid(input, input_wd, input_ht, channels, input_offset,
- stride_wd, stride_ht, filter_data, filter_wd, filter_ht, bias,
- out_data, out_wd, out_ht, out_channels, out_offset, out_shift,
- out_mult, activation_min, activation_max);
- }
- } else {
- /* Basic unrolled version */
- esp_nn_conv_s8_unrolled(input_dims, input, filter_dims, filter_data,
- bias, output_dims, out_data, conv_params, quant_data);
- }
- }
|