| // Copyright (c) 2006, 2008 Edward Rosten |
| // All rights reserved. |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions |
| // are met: |
| // |
| // *Redistributions of source code must retain the above copyright |
| // notice, this list of conditions and the following disclaimer. |
| // |
| // *Redistributions in binary form must reproduce the above copyright |
| // notice, this list of conditions and the following disclaimer in the |
| // documentation and/or other materials provided with the distribution. |
| // |
| // *Neither the name of the University of Cambridge nor the names of |
| // its contributors may be used to endorse or promote products derived |
| // from this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER |
| // OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| // EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| // PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
| // LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
| // NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
| // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| // clang-format off |
| #include <assert.h> |
| #include <stdlib.h> |
| #include "fast.h" |
| |
| |
| #define Compare(X, Y) ((X)>=(Y)) |
| |
| xy* aom_nonmax_suppression(const xy* corners, const int* scores, int num_corners, |
| int** ret_scores, int* ret_num_nonmax) |
| { |
| int num_nonmax=0; |
| int last_row; |
| int* row_start; |
| int i, j; |
| xy* ret_nonmax; |
| int* nonmax_scores; |
| const int sz = (int)num_corners; |
| |
| /*Point above points (roughly) to the pixel above the one of interest, if there |
| is a feature there.*/ |
| int point_above = 0; |
| int point_below = 0; |
| |
| *ret_scores = 0; |
| *ret_num_nonmax = -1; |
| if(!(corners && scores) || num_corners < 1) |
| { |
| *ret_num_nonmax = 0; |
| return 0; |
| } |
| |
| ret_nonmax = (xy*)malloc(num_corners * sizeof(xy)); |
| if(!ret_nonmax) |
| { |
| return 0; |
| } |
| |
| nonmax_scores = (int*)malloc(num_corners * sizeof(*nonmax_scores)); |
| if (!nonmax_scores) |
| { |
| free(ret_nonmax); |
| return 0; |
| } |
| |
| /* Find where each row begins |
| (the corners are output in raster scan order). A beginning of -1 signifies |
| that there are no corners on that row. */ |
| last_row = corners[num_corners-1].y; |
| row_start = (int*)malloc((last_row+1)*sizeof(int)); |
| if(!row_start) |
| { |
| free(ret_nonmax); |
| free(nonmax_scores); |
| return 0; |
| } |
| |
| for(i=0; i < last_row+1; i++) |
| row_start[i] = -1; |
| |
| { |
| int prev_row = -1; |
| for(i=0; i< num_corners; i++) |
| if(corners[i].y != prev_row) |
| { |
| row_start[corners[i].y] = i; |
| prev_row = corners[i].y; |
| } |
| } |
| |
| |
| |
| for(i=0; i < sz; i++) |
| { |
| int score = scores[i]; |
| xy pos = corners[i]; |
| assert(pos.y <= last_row); |
| |
| /*Check left */ |
| if(i > 0) |
| if(corners[i-1].x == pos.x-1 && corners[i-1].y == pos.y && Compare(scores[i-1], score)) |
| continue; |
| |
| /*Check right*/ |
| if(i < (sz - 1)) |
| if(corners[i+1].x == pos.x+1 && corners[i+1].y == pos.y && Compare(scores[i+1], score)) |
| continue; |
| |
| /*Check above (if there is a valid row above)*/ |
| if(pos.y > 0 && row_start[pos.y - 1] != -1) |
| { |
| /*Make sure that current point_above is one |
| row above.*/ |
| if(corners[point_above].y < pos.y - 1) |
| point_above = row_start[pos.y-1]; |
| |
| /*Make point_above point to the first of the pixels above the current point, |
| if it exists.*/ |
| for(; corners[point_above].y < pos.y && corners[point_above].x < pos.x - 1; point_above++) |
| {} |
| |
| |
| for(j=point_above; corners[j].y < pos.y && corners[j].x <= pos.x + 1; j++) |
| { |
| int x = corners[j].x; |
| if( (x == pos.x - 1 || x ==pos.x || x == pos.x+1) && Compare(scores[j], score)) |
| goto cont; |
| } |
| |
| } |
| |
| /*Check below (if there is anything below)*/ |
| if (pos.y + 1 < last_row+1 && row_start[pos.y + 1] != -1 && point_below < sz) /*Nothing below*/ |
| { |
| if(corners[point_below].y < pos.y + 1) |
| point_below = row_start[pos.y+1]; |
| |
| /* Make point below point to one of the pixels belowthe current point, if it |
| exists.*/ |
| for(; point_below < sz && corners[point_below].y == pos.y+1 && corners[point_below].x < pos.x - 1; point_below++) |
| {} |
| |
| for(j=point_below; j < sz && corners[j].y == pos.y+1 && corners[j].x <= pos.x + 1; j++) |
| { |
| int x = corners[j].x; |
| if( (x == pos.x - 1 || x ==pos.x || x == pos.x+1) && Compare(scores[j],score)) |
| goto cont; |
| } |
| } |
| |
| ret_nonmax[num_nonmax] = corners[i]; |
| nonmax_scores[num_nonmax] = scores[i]; |
| num_nonmax++; |
| cont: |
| ; |
| } |
| |
| free(row_start); |
| *ret_scores = nonmax_scores; |
| *ret_num_nonmax = num_nonmax; |
| return ret_nonmax; |
| } |
| |
| // clang-format on |