Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(346)

Side by Side Diff: webrtc/modules/video_processing/video_denoiser.cc

Issue 1871853003: External VNR speed improvement and more. (Closed) Base URL: https://chromium.googlesource.com/external/webrtc.git@master
Patch Set: Refactor. Created 4 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 /* 1 /*
2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved. 2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved.
3 * 3 *
4 * Use of this source code is governed by a BSD-style license 4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source 5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found 6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may 7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree. 8 * be found in the AUTHORS file in the root of the source tree.
9 */ 9 */
10
10 #include "webrtc/common_video/libyuv/include/scaler.h" 11 #include "webrtc/common_video/libyuv/include/scaler.h"
11 #include "webrtc/common_video/libyuv/include/webrtc_libyuv.h" 12 #include "webrtc/common_video/libyuv/include/webrtc_libyuv.h"
12 #include "webrtc/modules/video_processing/video_denoiser.h" 13 #include "webrtc/modules/video_processing/video_denoiser.h"
13 14
15 #if DISPLAY // Rectangle diagnostics
16 static void CopyMem8x8(const uint8_t* src,
17 int src_stride,
18 uint8_t* dst,
19 int dst_stride) {
20 for (int i = 0; i < 8; i++) {
21 memcpy(dst, src, 8);
22 src += src_stride;
23 dst += dst_stride;
24 }
25 }
26
27 static void ShowRect(const std::unique_ptr<DenoiserFilter>& filter,
28 const std::unique_ptr<uint8_t[]>& d_status,
29 const std::unique_ptr<uint8_t[]>& d_status_red,
30 const std::unique_ptr<uint8_t[]>& x_density,
31 const std::unique_ptr<uint8_t[]>& y_density,
32 const uint8_t* u_src,
33 const uint8_t* v_src,
34 uint8_t* u_dst,
35 uint8_t* v_dst,
36 int mb_rows_,
37 int mb_cols_,
38 int stride_u_,
39 int stride_v_) {
40 for (int mb_row = 0; mb_row < mb_rows_; ++mb_row) {
41 for (int mb_col = 0; mb_col < mb_cols_; ++mb_col) {
42 int mb_index = mb_row * mb_cols_ + mb_col;
43 const uint8_t* mb_src_u =
44 u_src + (mb_row << 3) * stride_u_ + (mb_col << 3);
45 const uint8_t* mb_src_v =
46 v_src + (mb_row << 3) * stride_v_ + (mb_col << 3);
47 uint8_t* mb_dst_u = u_dst + (mb_row << 3) * stride_u_ + (mb_col << 3);
48 uint8_t* mb_dst_v = v_dst + (mb_row << 3) * stride_v_ + (mb_col << 3);
49 uint8_t uv_tmp[8 * 8];
50 memset(uv_tmp, 200, 8 * 8);
51 if (d_status[mb_index] == 1) {
52 // Paint to red.
53 CopyMem8x8(mb_src_u, stride_u_, mb_dst_u, stride_u_);
54 CopyMem8x8(uv_tmp, 8, mb_dst_v, stride_v_);
55 } else if (d_status_red[mb_row * mb_cols_ + mb_col] &&
56 x_density[mb_col] * y_density[mb_row]) {
57 // Paint to blue.
58 CopyMem8x8(uv_tmp, 8, mb_dst_u, stride_u_);
59 CopyMem8x8(mb_src_v, stride_v_, mb_dst_v, stride_v_);
60 } else {
61 CopyMem8x8(mb_src_u, stride_u_, mb_dst_u, stride_u_);
62 CopyMem8x8(mb_src_v, stride_v_, mb_dst_v, stride_v_);
63 }
64 }
65 }
66 }
67 #endif
68
14 namespace webrtc { 69 namespace webrtc {
15 70
16 VideoDenoiser::VideoDenoiser(bool runtime_cpu_detection) 71 VideoDenoiser::VideoDenoiser(bool runtime_cpu_detection)
17 : width_(0), 72 : width_(0),
18 height_(0), 73 height_(0),
19 filter_(DenoiserFilter::Create(runtime_cpu_detection, &cpu_type_)), 74 filter_(DenoiserFilter::Create(runtime_cpu_detection, &cpu_type_)),
20 ne_(new NoiseEstimation()) {} 75 ne_(new NoiseEstimation()) {}
21 76
22 #if EXPERIMENTAL 77 // Check the mb position, return 1: close to the frame center, 3: close to the
23 // Check the mb position(1: close to the center, 3: close to the border). 78 // border, 2: in between.
24 static int PositionCheck(int mb_row, int mb_col, int mb_rows, int mb_cols) { 79 int VideoDenoiser::PositionCheck(int mb_row, int mb_col, int noise_level) {
25 if ((mb_row >= (mb_rows >> 3)) && (mb_row <= (7 * mb_rows >> 3)) && 80 if (noise_level == 0)
26 (mb_col >= (mb_cols >> 3)) && (mb_col <= (7 * mb_cols >> 3)))
27 return 1; 81 return 1;
28 else if ((mb_row >= (mb_rows >> 4)) && (mb_row <= (15 * mb_rows >> 4)) && 82 if ((mb_row <= (mb_rows_ >> 4)) || (mb_col <= (mb_cols_ >> 4)) ||
29 (mb_col >= (mb_cols >> 4)) && (mb_col <= (15 * mb_cols >> 4))) 83 (mb_col >= (15 * mb_cols_ >> 4)))
84 return 3;
85 else if ((mb_row <= (mb_rows_ >> 3)) || (mb_col <= (mb_cols_ >> 3)) ||
86 (mb_col >= (7 * mb_cols_ >> 3)))
30 return 2; 87 return 2;
31 else 88 else
32 return 3; 89 return 1;
33 } 90 }
34 91
35 static void ReduceFalseDetection(const std::unique_ptr<uint8_t[]>& d_status, 92 // To reduce false detection in moving object detection (MOD).
36 std::unique_ptr<uint8_t[]>* d_status_tmp1, 93 void VideoDenoiser::ReduceFalseDetection(
37 std::unique_ptr<uint8_t[]>* d_status_tmp2, 94 const std::unique_ptr<uint8_t[]>& d_status,
38 int noise_level, 95 std::unique_ptr<uint8_t[]>* d_status_red,
39 int mb_rows, 96 int noise_level) {
40 int mb_cols) { 97 // From up left corner.
41 // Draft. This can be optimized. This code block is to reduce false detection 98 int mb_col_stop = mb_cols_ - 1;
42 // in moving object detection. 99 for (int mb_row = 0; mb_row <= mb_rows_ - 1; ++mb_row) {
43 int mb_row_min = noise_level ? mb_rows >> 3 : 1; 100 for (int mb_col = 0; mb_col <= mb_col_stop; ++mb_col) {
44 int mb_col_min = noise_level ? mb_cols >> 3 : 1; 101 if (d_status[mb_row * mb_cols_ + mb_col]) {
45 int mb_row_max = noise_level ? (7 * mb_rows >> 3) : mb_rows - 2; 102 mb_col_stop = mb_col - 1;
46 int mb_col_max = noise_level ? (7 * mb_cols >> 3) : mb_cols - 2; 103 break;
47 memcpy((*d_status_tmp1).get(), d_status.get(), mb_rows * mb_cols); 104 }
48 // Up left. 105 (*d_status_red)[mb_row * mb_cols_ + mb_col] = 0;
49 for (int mb_row = mb_row_min; mb_row <= mb_row_max; ++mb_row) { 106 }
50 for (int mb_col = mb_col_min; mb_col <= mb_col_max; ++mb_col) { 107 }
51 (*d_status_tmp1)[mb_row * mb_cols + mb_col] |= 108 // From bottom left corner.
52 ((*d_status_tmp1)[(mb_row - 1) * mb_cols + mb_col] | 109 mb_col_stop = mb_cols_ - 1;
53 (*d_status_tmp1)[mb_row * mb_cols + mb_col - 1]); 110 for (int mb_row = mb_rows_ - 1; mb_row >= 0; --mb_row) {
54 } 111 for (int mb_col = 0; mb_col <= mb_col_stop; ++mb_col) {
55 } 112 if (d_status[mb_row * mb_cols_ + mb_col]) {
56 memcpy((*d_status_tmp2).get(), (*d_status_tmp1).get(), mb_rows * mb_cols); 113 mb_col_stop = mb_col - 1;
57 memcpy((*d_status_tmp1).get(), d_status.get(), mb_rows * mb_cols); 114 break;
58 // Bottom left. 115 }
59 for (int mb_row = mb_row_max; mb_row >= mb_row_min; --mb_row) { 116 (*d_status_red)[mb_row * mb_cols_ + mb_col] = 0;
60 for (int mb_col = mb_col_min; mb_col <= mb_col_max; ++mb_col) { 117 }
61 (*d_status_tmp1)[mb_row * mb_cols + mb_col] |= 118 }
62 ((*d_status_tmp1)[(mb_row + 1) * mb_cols + mb_col] | 119 // From up right corner.
63 (*d_status_tmp1)[mb_row * mb_cols + mb_col - 1]); 120 mb_col_stop = 0;
64 (*d_status_tmp2)[mb_row * mb_cols + mb_col] &= 121 for (int mb_row = 0; mb_row <= mb_rows_ - 1; ++mb_row) {
65 (*d_status_tmp1)[mb_row * mb_cols + mb_col]; 122 for (int mb_col = mb_cols_ - 1; mb_col >= mb_col_stop; --mb_col) {
66 } 123 if (d_status[mb_row * mb_cols_ + mb_col]) {
67 } 124 mb_col_stop = mb_col + 1;
68 memcpy((*d_status_tmp1).get(), d_status.get(), mb_rows * mb_cols); 125 break;
69 // Up right. 126 }
70 for (int mb_row = mb_row_min; mb_row <= mb_row_max; ++mb_row) { 127 (*d_status_red)[mb_row * mb_cols_ + mb_col] = 0;
71 for (int mb_col = mb_col_max; mb_col >= mb_col_min; --mb_col) { 128 }
72 (*d_status_tmp1)[mb_row * mb_cols + mb_col] |= 129 }
73 ((*d_status_tmp1)[(mb_row - 1) * mb_cols + mb_col] | 130 // From bottom right corner.
74 (*d_status_tmp1)[mb_row * mb_cols + mb_col + 1]); 131 mb_col_stop = 0;
75 (*d_status_tmp2)[mb_row * mb_cols + mb_col] &= 132 for (int mb_row = mb_rows_ - 1; mb_row >= 0; --mb_row) {
76 (*d_status_tmp1)[mb_row * mb_cols + mb_col]; 133 for (int mb_col = mb_cols_ - 1; mb_col >= mb_col_stop; --mb_col) {
77 } 134 if (d_status[mb_row * mb_cols_ + mb_col]) {
78 } 135 mb_col_stop = mb_col + 1;
79 memcpy((*d_status_tmp1).get(), d_status.get(), mb_rows * mb_cols); 136 break;
80 // Bottom right. 137 }
81 for (int mb_row = mb_row_max; mb_row >= mb_row_min; --mb_row) { 138 (*d_status_red)[mb_row * mb_cols_ + mb_col] = 0;
82 for (int mb_col = mb_col_max; mb_col >= mb_col_min; --mb_col) { 139 }
83 (*d_status_tmp1)[mb_row * mb_cols + mb_col] |= 140 }
84 ((*d_status_tmp1)[(mb_row + 1) * mb_cols + mb_col] | 141 }
85 (*d_status_tmp1)[mb_row * mb_cols + mb_col + 1]); 142
86 (*d_status_tmp2)[mb_row * mb_cols + mb_col] &= 143 // Check if a neighbor block is a moving edge block.
87 (*d_status_tmp1)[mb_row * mb_cols + mb_col]; 144 bool VideoDenoiser::IsTrailingBlock(const std::unique_ptr<uint8_t[]>& d_status,
88 } 145 int mb_row,
89 } 146 int mb_col) {
90 } 147 bool ret = false;
91 148 int mb_index = mb_row * mb_cols_ + mb_col;
92 static bool TrailingBlock(const std::unique_ptr<uint8_t[]>& d_status, 149 if (!mb_row || !mb_col || mb_row == mb_rows_ - 1 || mb_col == mb_cols_ - 1)
93 int mb_row, 150 ret = false;
94 int mb_col, 151 else
95 int mb_rows, 152 ret = d_status[mb_index + 1] || d_status[mb_index - 1] ||
96 int mb_cols) { 153 d_status[mb_index + mb_cols_] || d_status[mb_index - mb_cols_];
97 int mb_index = mb_row * mb_cols + mb_col; 154 return ret;
98 if (!mb_row || !mb_col || mb_row == mb_rows - 1 || mb_col == mb_cols - 1) 155 }
99 return false; 156
100 return d_status[mb_index + 1] || d_status[mb_index - 1] || 157 void VideoDenoiser::DenoiserReset(const VideoFrame& frame,
101 d_status[mb_index + mb_cols] || d_status[mb_index - mb_cols]; 158 VideoFrame* denoised_frame,
102 } 159 VideoFrame* denoised_frame_prev) {
103 #endif 160 width_ = frame.width();
104 161 height_ = frame.height();
105 #if DISPLAY 162 mb_cols_ = width_ >> 4;
106 void ShowRect(const std::unique_ptr<DenoiserFilter>& filter, 163 mb_rows_ = height_ >> 4;
107 const std::unique_ptr<uint8_t[]>& d_status, 164 stride_y_ = frame.stride(kYPlane);
108 const std::unique_ptr<uint8_t[]>& d_status_tmp2, 165 stride_u_ = frame.stride(kUPlane);
109 const std::unique_ptr<uint8_t[]>& x_density, 166 stride_v_ = frame.stride(kVPlane);
110 const std::unique_ptr<uint8_t[]>& y_density, 167
111 const uint8_t* u_src, 168 // Allocate an empty buffer for denoised_frame_prev.
112 const uint8_t* v_src, 169 denoised_frame_prev->CreateEmptyFrame(width_, height_, stride_y_, stride_u_,
113 uint8_t* u_dst, 170 stride_v_);
114 uint8_t* v_dst, 171 // Allocate and initialize denoised_frame with key frame.
115 int mb_rows, 172 denoised_frame->CreateFrame(frame.buffer(kYPlane), frame.buffer(kUPlane),
116 int mb_cols, 173 frame.buffer(kVPlane), width_, height_, stride_y_,
117 int stride_u, 174 stride_u_, stride_v_, kVideoRotation_0);
118 int stride_v) { 175 // Set time parameters to the output frame.
119 for (int mb_row = 0; mb_row < mb_rows; ++mb_row) { 176 denoised_frame->set_timestamp(frame.timestamp());
120 for (int mb_col = 0; mb_col < mb_cols; ++mb_col) { 177 denoised_frame->set_render_time_ms(frame.render_time_ms());
121 int mb_index = mb_row * mb_cols + mb_col; 178
122 const uint8_t* mb_src_u = 179 // Init noise estimator and allocate buffers.
123 u_src + (mb_row << 3) * stride_u + (mb_col << 3); 180 ne_->Init(width_, height_, cpu_type_);
124 const uint8_t* mb_src_v = 181 d_status_.reset(new uint8_t[mb_cols_ * mb_rows_]);
125 v_src + (mb_row << 3) * stride_v + (mb_col << 3); 182 mb_filter_decision_.reset(new DenoiserDecision[mb_cols_ * mb_rows_]);
126 uint8_t* mb_dst_u = u_dst + (mb_row << 3) * stride_u + (mb_col << 3); 183 x_density_.reset(new uint8_t[mb_cols_]);
127 uint8_t* mb_dst_v = v_dst + (mb_row << 3) * stride_v + (mb_col << 3); 184 y_density_.reset(new uint8_t[mb_rows_]);
128 uint8_t y_tmp_255[8 * 8]; 185 mod_.reset(new uint8_t[mb_cols_ * mb_rows_]);
129 memset(y_tmp_255, 200, 8 * 8); 186 }
130 // x_density_[mb_col] * y_density_[mb_row] 187
131 if (d_status[mb_index] == 1) { 188 void VideoDenoiser::CopySrcOnMOB(const uint8_t* y_src, uint8_t* y_dst) {
132 // Paint to red. 189 // Loop over to copy src block if the block is marked as moving object block
133 filter->CopyMem8x8(mb_src_u, stride_u, mb_dst_u, stride_u); 190 // or if the block may cause trailing artifacts.
134 filter->CopyMem8x8(y_tmp_255, 8, mb_dst_v, stride_v); 191 for (int mb_row = 0; mb_row < mb_rows_; ++mb_row) {
135 #if EXPERIMENTAL 192 const int mb_index_base = mb_row * mb_cols_;
136 } else if (d_status_tmp2[mb_row * mb_cols + mb_col] && 193 const int offset_base = (mb_row << 4) * stride_y_;
137 x_density[mb_col] * y_density[mb_row]) { 194 const uint8_t* mb_src_base = y_src + offset_base;
138 #else 195 uint8_t* mb_dst_base = y_dst + offset_base;
139 } else if (x_density[mb_col] * y_density[mb_row]) { 196 for (int mb_col = 0; mb_col < mb_cols_; ++mb_col) {
140 #endif 197 const int mb_index = mb_index_base + mb_col;
141 // Paint to blue. 198 const uint32_t offset_col = mb_col << 4;
142 filter->CopyMem8x8(y_tmp_255, 8, mb_dst_u, stride_u); 199 const uint8_t* mb_src = mb_src_base + offset_col;
143 filter->CopyMem8x8(mb_src_v, stride_v, mb_dst_v, stride_v); 200 uint8_t* mb_dst = mb_dst_base + offset_col;
144 } else { 201 // Check if the block is a moving object block or may cause a trailing
145 filter->CopyMem8x8(mb_src_u, stride_u, mb_dst_u, stride_u); 202 // artifacts.
146 filter->CopyMem8x8(mb_src_v, stride_v, mb_dst_v, stride_v); 203 if (mb_filter_decision_[mb_index] != FILTER_BLOCK ||
147 } 204 IsTrailingBlock(d_status_, mb_row, mb_col) ||
148 } 205 (x_density_[mb_col] * y_density_[mb_row] &&
149 } 206 mod_[mb_row * mb_cols_ + mb_col])) {
150 } 207 // Copy y source.
151 #endif 208 filter_->CopyMem16x16(mb_src, stride_y_, mb_dst, stride_y_);
209 }
210 }
211 }
212 }
152 213
153 void VideoDenoiser::DenoiseFrame(const VideoFrame& frame, 214 void VideoDenoiser::DenoiseFrame(const VideoFrame& frame,
154 VideoFrame* denoised_frame, 215 VideoFrame* denoised_frame,
155 VideoFrame* denoised_frame_prev, 216 VideoFrame* denoised_frame_prev,
156 int noise_level_prev) { 217 bool noise_estimation_enabled) {
157 int stride_y = frame.stride(kYPlane); 218 // If previous width and height are different from current frame's, need to
158 int stride_u = frame.stride(kUPlane); 219 // reallocate the buffers and no denoising for the current frame.
159 int stride_v = frame.stride(kVPlane);
160 // If previous width and height are different from current frame's, then no
161 // denoising for the current frame.
162 if (width_ != frame.width() || height_ != frame.height()) { 220 if (width_ != frame.width() || height_ != frame.height()) {
163 width_ = frame.width(); 221 DenoiserReset(frame, denoised_frame, denoised_frame_prev);
164 height_ = frame.height();
165 denoised_frame->CreateFrame(frame.buffer(kYPlane), frame.buffer(kUPlane),
166 frame.buffer(kVPlane), width_, height_,
167 stride_y, stride_u, stride_v, kVideoRotation_0);
168 denoised_frame_prev->CreateFrame(
169 frame.buffer(kYPlane), frame.buffer(kUPlane), frame.buffer(kVPlane),
170 width_, height_, stride_y, stride_u, stride_v, kVideoRotation_0);
171 // Setting time parameters to the output frame.
172 denoised_frame->set_timestamp(frame.timestamp());
173 denoised_frame->set_render_time_ms(frame.render_time_ms());
174 ne_->Init(width_, height_, cpu_type_);
175 return; 222 return;
176 } 223 }
177 // For 16x16 block. 224
178 int mb_cols = width_ >> 4; 225 // Set buffer pointers.
179 int mb_rows = height_ >> 4; 226 const uint8_t* y_src = frame.buffer(kYPlane);
180 if (metrics_.get() == nullptr) 227 const uint8_t* u_src = frame.buffer(kUPlane);
181 metrics_.reset(new DenoiseMetrics[mb_cols * mb_rows]()); 228 const uint8_t* v_src = frame.buffer(kVPlane);
182 if (d_status_.get() == nullptr) {
183 d_status_.reset(new uint8_t[mb_cols * mb_rows]());
184 #if EXPERIMENTAL
185 d_status_tmp1_.reset(new uint8_t[mb_cols * mb_rows]());
186 d_status_tmp2_.reset(new uint8_t[mb_cols * mb_rows]());
187 #endif
188 x_density_.reset(new uint8_t[mb_cols]());
189 y_density_.reset(new uint8_t[mb_rows]());
190 }
191
192 // Denoise on Y plane.
193 uint8_t* y_dst = denoised_frame->buffer(kYPlane); 229 uint8_t* y_dst = denoised_frame->buffer(kYPlane);
194 uint8_t* u_dst = denoised_frame->buffer(kUPlane); 230 uint8_t* u_dst = denoised_frame->buffer(kUPlane);
195 uint8_t* v_dst = denoised_frame->buffer(kVPlane); 231 uint8_t* v_dst = denoised_frame->buffer(kVPlane);
196 uint8_t* y_dst_prev = denoised_frame_prev->buffer(kYPlane); 232 uint8_t* y_dst_prev = denoised_frame_prev->buffer(kYPlane);
197 const uint8_t* y_src = frame.buffer(kYPlane); 233 memset(x_density_.get(), 0, mb_cols_);
198 const uint8_t* u_src = frame.buffer(kUPlane); 234 memset(y_density_.get(), 0, mb_rows_);
199 const uint8_t* v_src = frame.buffer(kVPlane); 235 memset(mod_.get(), 1, mb_cols_ * mb_rows_);
200 uint8_t noise_level = noise_level_prev == -1 ? 0 : ne_->GetNoiseLevel(); 236
201 // Temporary buffer to store denoising result. 237 uint8_t noise_level = noise_estimation_enabled ? ne_->GetNoiseLevel() : 0;
202 uint8_t y_tmp[16 * 16] = {0}; 238 int thr_var_base = 16 * 16 * 5;
203 memset(x_density_.get(), 0, mb_cols);
204 memset(y_density_.get(), 0, mb_rows);
205
206 // Loop over blocks to accumulate/extract noise level and update x/y_density 239 // Loop over blocks to accumulate/extract noise level and update x/y_density
207 // factors for moving object detection. 240 // factors for moving object detection.
208 for (int mb_row = 0; mb_row < mb_rows; ++mb_row) { 241 for (int mb_row = 0; mb_row < mb_rows_; ++mb_row) {
209 for (int mb_col = 0; mb_col < mb_cols; ++mb_col) { 242 const int mb_index_base = mb_row * mb_cols_;
210 const uint8_t* mb_src = y_src + (mb_row << 4) * stride_y + (mb_col << 4); 243 const int offset_base = (mb_row << 4) * stride_y_;
211 uint8_t* mb_dst_prev = 244 const uint8_t* mb_src_base = y_src + offset_base;
212 y_dst_prev + (mb_row << 4) * stride_y + (mb_col << 4); 245 uint8_t* mb_dst_base = y_dst + offset_base;
213 int mb_index = mb_row * mb_cols + mb_col; 246 uint8_t* mb_dst_prev_base = y_dst_prev + offset_base;
214 #if EXPERIMENTAL 247 for (int mb_col = 0; mb_col < mb_cols_; ++mb_col) {
215 int pos_factor = PositionCheck(mb_row, mb_col, mb_rows, mb_cols); 248 const int mb_index = mb_index_base + mb_col;
216 uint32_t thr_var_adp = 16 * 16 * 5 * (noise_level ? pos_factor : 1); 249 const bool ne_enable = (mb_index % NOISE_SUBSAMPLE_INTERVAL == 0);
217 #else 250 const int pos_factor = PositionCheck(mb_row, mb_col, noise_level);
218 uint32_t thr_var_adp = 16 * 16 * 5; 251 const uint32_t thr_var_adp = thr_var_base * pos_factor;
219 #endif 252 const uint32_t offset_col = mb_col << 4;
220 int brightness = 0; 253 const uint8_t* mb_src = mb_src_base + offset_col;
221 for (int i = 0; i < 16; ++i) { 254 uint8_t* mb_dst = mb_dst_base + offset_col;
222 for (int j = 0; j < 16; ++j) { 255 uint8_t* mb_dst_prev = mb_dst_prev_base + offset_col;
223 brightness += mb_src[i * stride_y + j]; 256
257 // TODO(jackychen): Need SSE2/NEON opt.
258 int luma = 0;
259 if (ne_enable) {
260 for (int i = 4; i < 12; ++i) {
261 for (int j = 4; j < 12; ++j) {
262 luma += mb_src[i * stride_y_ + j];
263 }
224 } 264 }
225 } 265 }
226 266
227 // Get the denoised block. 267 // Get the filtered block and filter_decision.
228 filter_->MbDenoise(mb_dst_prev, stride_y, y_tmp, 16, mb_src, stride_y, 0, 268 mb_filter_decision_[mb_index] =
229 1, true); 269 filter_->MbDenoise(mb_dst_prev, stride_y_, mb_dst, stride_y_, mb_src,
230 // The variance is based on the denoised blocks in time T and T-1. 270 stride_y_, 0, noise_level);
231 metrics_[mb_index].var = filter_->Variance16x8( 271
232 mb_dst_prev, stride_y, y_tmp, 16, &metrics_[mb_index].sad); 272 // If filter decision is FILTER_BLOCK, no need to check moving edge.
233 273 // It is unlikely for a moving edge block to be filtered in current
234 if (metrics_[mb_index].var > thr_var_adp) { 274 // setting.
235 ne_->ResetConsecLowVar(mb_index); 275 if (mb_filter_decision_[mb_index] == FILTER_BLOCK) {
236 d_status_[mb_index] = 1; 276 uint32_t sse_t = 0;
237 #if EXPERIMENTAL 277 if (ne_enable) {
238 if (noise_level == 0 || pos_factor < 3) { 278 // The variance used in noise estimation is based on the src block in
239 x_density_[mb_col] += 1; 279 // time t (mb_src) and filtered block in time t-1 (mb_dist_prev).
240 y_density_[mb_row] += 1; 280 uint32_t noise_var = filter_->Variance16x8(mb_dst_prev, stride_y_,
281 mb_src, stride_y_, &sse_t);
282 ne_->GetNoise(mb_index, noise_var, luma);
241 } 283 }
242 #else 284 d_status_[mb_index] = 0; // Not a moving edge block.
243 x_density_[mb_col] += 1;
244 y_density_[mb_row] += 1;
245 #endif
246 } else { 285 } else {
247 uint32_t sse_t = 0; 286 uint32_t sse_t = 0;
248 // The variance is based on the src blocks in time T and denoised block 287 // The variance used in MOD is based on the filtered blocks in time
249 // in time T-1. 288 // T (mb_dst) and T-1 (mb_dst_prev).
250 uint32_t noise_var = filter_->Variance16x8(mb_dst_prev, stride_y, 289 uint32_t noise_var = filter_->Variance16x8(mb_dst_prev, stride_y_,
251 mb_src, stride_y, &sse_t); 290 mb_dst, stride_y_, &sse_t);
252 ne_->GetNoise(mb_index, noise_var, brightness); 291 if (noise_var > thr_var_adp) { // Moving edge checking.
253 d_status_[mb_index] = 0; 292 if (ne_enable) {
254 } 293 ne_->ResetConsecLowVar(mb_index);
255 // Track denoised frame. 294 }
256 filter_->CopyMem16x16(y_tmp, 16, mb_dst_prev, stride_y); 295 d_status_[mb_index] = 1; // Mark as moving edge block.
257 } 296 x_density_[mb_col] += (pos_factor < 3);
258 } 297 y_density_[mb_row] += (pos_factor < 3);
259
260 #if EXPERIMENTAL
261 ReduceFalseDetection(d_status_, &d_status_tmp1_, &d_status_tmp2_, noise_level,
262 mb_rows, mb_cols);
263 #endif
264
265 // Denoise each MB based on the results of moving objects detection.
266 for (int mb_row = 0; mb_row < mb_rows; ++mb_row) {
267 for (int mb_col = 0; mb_col < mb_cols; ++mb_col) {
268 const uint8_t* mb_src = y_src + (mb_row << 4) * stride_y + (mb_col << 4);
269 uint8_t* mb_dst = y_dst + (mb_row << 4) * stride_y + (mb_col << 4);
270 const uint8_t* mb_src_u =
271 u_src + (mb_row << 3) * stride_u + (mb_col << 3);
272 const uint8_t* mb_src_v =
273 v_src + (mb_row << 3) * stride_v + (mb_col << 3);
274 uint8_t* mb_dst_u = u_dst + (mb_row << 3) * stride_u + (mb_col << 3);
275 uint8_t* mb_dst_v = v_dst + (mb_row << 3) * stride_v + (mb_col << 3);
276 #if EXPERIMENTAL
277 if ((!d_status_tmp2_[mb_row * mb_cols + mb_col] ||
278 x_density_[mb_col] * y_density_[mb_row] == 0) &&
279 !TrailingBlock(d_status_, mb_row, mb_col, mb_rows, mb_cols)) {
280 #else
281 if (x_density_[mb_col] * y_density_[mb_row] == 0) {
282 #endif
283 if (filter_->MbDenoise(mb_dst, stride_y, y_tmp, 16, mb_src, stride_y, 0,
284 noise_level, false) == FILTER_BLOCK) {
285 filter_->CopyMem16x16(y_tmp, 16, mb_dst, stride_y);
286 } else { 298 } else {
287 // Copy y source. 299 d_status_[mb_index] = 0;
288 filter_->CopyMem16x16(mb_src, stride_y, mb_dst, stride_y); 300 if (ne_enable) {
301 // The variance used in noise estimation is based on the src block
302 // in time t (mb_src) and filtered block in time t-1 (mb_dist_prev).
303 uint32_t noise_var = filter_->Variance16x8(
304 mb_dst_prev, stride_y_, mb_src, stride_y_, &sse_t);
305 ne_->GetNoise(mb_index, noise_var, luma);
306 }
289 } 307 }
290 } else { 308 }
291 // Copy y source. 309 } // End of for loop
292 filter_->CopyMem16x16(mb_src, stride_y, mb_dst, stride_y); 310 } // End of for loop
293 } 311
294 filter_->CopyMem8x8(mb_src_u, stride_u, mb_dst_u, stride_u); 312 ReduceFalseDetection(d_status_, &mod_, noise_level);
295 filter_->CopyMem8x8(mb_src_v, stride_v, mb_dst_v, stride_v); 313
296 } 314 CopySrcOnMOB(y_src, y_dst);
297 } 315
316 // TODO(jackychen): Need SSE2/NEON opt.
317 // Copy u/v planes.
318 memcpy(u_dst, u_src, (height_ >> 1) * stride_u_);
319 memcpy(v_dst, v_src, (height_ >> 1) * stride_v_);
320
321 // Set time parameters to the output frame.
322 denoised_frame->set_timestamp(frame.timestamp());
323 denoised_frame->set_render_time_ms(frame.render_time_ms());
298 324
299 #if DISPLAY // Rectangle diagnostics 325 #if DISPLAY // Rectangle diagnostics
300 // Show rectangular region 326 // Show rectangular region
301 ShowRect(filter_, d_status_, d_status_tmp2_, x_density_, y_density_, u_src, 327 ShowRect(filter_, d_status_, mod_, x_density_, y_density_, u_src, v_src,
302 v_src, u_dst, v_dst, mb_rows, mb_cols, stride_u, stride_v); 328 u_dst, v_dst, mb_rows_, mb_cols_, stride_u_, stride_v_);
303 #endif 329 #endif
304
305 // Setting time parameters to the output frame.
306 denoised_frame->set_timestamp(frame.timestamp());
307 denoised_frame->set_render_time_ms(frame.render_time_ms());
308 return;
309 } 330 }
310 331
311 } // namespace webrtc 332 } // namespace webrtc
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698