/src/aom/third_party/fastfeat/nonmax.c
Line | Count | Source (jump to first uncovered line) |
1 | | // Copyright (c) 2006, 2008 Edward Rosten |
2 | | // All rights reserved. |
3 | | // |
4 | | // Redistribution and use in source and binary forms, with or without |
5 | | // modification, are permitted provided that the following conditions |
6 | | // are met: |
7 | | // |
8 | | // *Redistributions of source code must retain the above copyright |
9 | | // notice, this list of conditions and the following disclaimer. |
10 | | // |
11 | | // *Redistributions in binary form must reproduce the above copyright |
12 | | // notice, this list of conditions and the following disclaimer in the |
13 | | // documentation and/or other materials provided with the distribution. |
14 | | // |
15 | | // *Neither the name of the University of Cambridge nor the names of |
16 | | // its contributors may be used to endorse or promote products derived |
17 | | // from this software without specific prior written permission. |
18 | | // |
19 | | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
20 | | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
21 | | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
22 | | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER |
23 | | // OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
24 | | // EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
25 | | // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
26 | | // PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
27 | | // LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
28 | | // NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
29 | | // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
30 | | |
31 | | // clang-format off |
32 | | #include <assert.h> |
33 | | #include <stdlib.h> |
34 | | #include "fast.h" |
35 | | |
36 | | |
37 | 0 | #define Compare(X, Y) ((X)>=(Y)) |
38 | | |
39 | | xy* aom_nonmax_suppression(const xy* corners, const int* scores, int num_corners, |
40 | | int** ret_scores, int* ret_num_nonmax) |
41 | 0 | { |
42 | 0 | int num_nonmax=0; |
43 | 0 | int last_row; |
44 | 0 | int* row_start; |
45 | 0 | int i, j; |
46 | 0 | xy* ret_nonmax; |
47 | 0 | int* nonmax_scores; |
48 | 0 | const int sz = (int)num_corners; |
49 | | |
50 | | /*Point above points (roughly) to the pixel above the one of interest, if there |
51 | | is a feature there.*/ |
52 | 0 | int point_above = 0; |
53 | 0 | int point_below = 0; |
54 | |
|
55 | 0 | *ret_scores = 0; |
56 | 0 | *ret_num_nonmax = -1; |
57 | 0 | if(!(corners && scores) || num_corners < 1) |
58 | 0 | { |
59 | 0 | *ret_num_nonmax = 0; |
60 | 0 | return 0; |
61 | 0 | } |
62 | | |
63 | 0 | ret_nonmax = (xy*)malloc(num_corners * sizeof(xy)); |
64 | 0 | if(!ret_nonmax) |
65 | 0 | { |
66 | 0 | return 0; |
67 | 0 | } |
68 | | |
69 | 0 | nonmax_scores = (int*)malloc(num_corners * sizeof(*nonmax_scores)); |
70 | 0 | if (!nonmax_scores) |
71 | 0 | { |
72 | 0 | free(ret_nonmax); |
73 | 0 | return 0; |
74 | 0 | } |
75 | | |
76 | | /* Find where each row begins |
77 | | (the corners are output in raster scan order). A beginning of -1 signifies |
78 | | that there are no corners on that row. */ |
79 | 0 | last_row = corners[num_corners-1].y; |
80 | 0 | row_start = (int*)malloc((last_row+1)*sizeof(int)); |
81 | 0 | if(!row_start) |
82 | 0 | { |
83 | 0 | free(ret_nonmax); |
84 | 0 | free(nonmax_scores); |
85 | 0 | return 0; |
86 | 0 | } |
87 | | |
88 | 0 | for(i=0; i < last_row+1; i++) |
89 | 0 | row_start[i] = -1; |
90 | |
|
91 | 0 | { |
92 | 0 | int prev_row = -1; |
93 | 0 | for(i=0; i< num_corners; i++) |
94 | 0 | if(corners[i].y != prev_row) |
95 | 0 | { |
96 | 0 | row_start[corners[i].y] = i; |
97 | 0 | prev_row = corners[i].y; |
98 | 0 | } |
99 | 0 | } |
100 | | |
101 | | |
102 | |
|
103 | 0 | for(i=0; i < sz; i++) |
104 | 0 | { |
105 | 0 | int score = scores[i]; |
106 | 0 | xy pos = corners[i]; |
107 | 0 | assert(pos.y <= last_row); |
108 | | |
109 | | /*Check left */ |
110 | 0 | if(i > 0) |
111 | 0 | if(corners[i-1].x == pos.x-1 && corners[i-1].y == pos.y && Compare(scores[i-1], score)) |
112 | 0 | continue; |
113 | | |
114 | | /*Check right*/ |
115 | 0 | if(i < (sz - 1)) |
116 | 0 | if(corners[i+1].x == pos.x+1 && corners[i+1].y == pos.y && Compare(scores[i+1], score)) |
117 | 0 | continue; |
118 | | |
119 | | /*Check above (if there is a valid row above)*/ |
120 | 0 | if(pos.y > 0 && row_start[pos.y - 1] != -1) |
121 | 0 | { |
122 | | /*Make sure that current point_above is one |
123 | | row above.*/ |
124 | 0 | if(corners[point_above].y < pos.y - 1) |
125 | 0 | point_above = row_start[pos.y-1]; |
126 | | |
127 | | /*Make point_above point to the first of the pixels above the current point, |
128 | | if it exists.*/ |
129 | 0 | for(; corners[point_above].y < pos.y && corners[point_above].x < pos.x - 1; point_above++) |
130 | 0 | {} |
131 | | |
132 | |
|
133 | 0 | for(j=point_above; corners[j].y < pos.y && corners[j].x <= pos.x + 1; j++) |
134 | 0 | { |
135 | 0 | int x = corners[j].x; |
136 | 0 | if( (x == pos.x - 1 || x ==pos.x || x == pos.x+1) && Compare(scores[j], score)) |
137 | 0 | goto cont; |
138 | 0 | } |
139 | |
|
140 | 0 | } |
141 | | |
142 | | /*Check below (if there is anything below)*/ |
143 | 0 | if (pos.y + 1 < last_row+1 && row_start[pos.y + 1] != -1 && point_below < sz) /*Nothing below*/ |
144 | 0 | { |
145 | 0 | if(corners[point_below].y < pos.y + 1) |
146 | 0 | point_below = row_start[pos.y+1]; |
147 | | |
148 | | /* Make point below point to one of the pixels belowthe current point, if it |
149 | | exists.*/ |
150 | 0 | for(; point_below < sz && corners[point_below].y == pos.y+1 && corners[point_below].x < pos.x - 1; point_below++) |
151 | 0 | {} |
152 | |
|
153 | 0 | for(j=point_below; j < sz && corners[j].y == pos.y+1 && corners[j].x <= pos.x + 1; j++) |
154 | 0 | { |
155 | 0 | int x = corners[j].x; |
156 | 0 | if( (x == pos.x - 1 || x ==pos.x || x == pos.x+1) && Compare(scores[j],score)) |
157 | 0 | goto cont; |
158 | 0 | } |
159 | 0 | } |
160 | | |
161 | 0 | ret_nonmax[num_nonmax] = corners[i]; |
162 | 0 | nonmax_scores[num_nonmax] = scores[i]; |
163 | 0 | num_nonmax++; |
164 | 0 | cont: |
165 | 0 | ; |
166 | 0 | } |
167 | | |
168 | 0 | free(row_start); |
169 | 0 | *ret_scores = nonmax_scores; |
170 | 0 | *ret_num_nonmax = num_nonmax; |
171 | 0 | return ret_nonmax; |
172 | 0 | } |
173 | | |
174 | | // clang-format on |