/** * File: FORB.cpp * Date: June 2012 * Author: Dorian Galvez-Lopez * Description: functions for ORB descriptors * License: see the LICENSE.txt file * * Distance function has been modified * */ #include #include #include #include #include "FORB.h" using namespace std; namespace DBoW2 { // -------------------------------------------------------------------------- const int FORB::L=32; void FORB::meanValue(const std::vector &descriptors, FORB::TDescriptor &mean) { if(descriptors.empty()) { mean.release(); return; } else if(descriptors.size() == 1) { mean = descriptors[0]->clone(); } else { vector sum(FORB::L * 8, 0); for(size_t i = 0; i < descriptors.size(); ++i) { const cv::Mat &d = *descriptors[i]; const unsigned char *p = d.ptr(); for(int j = 0; j < d.cols; ++j, ++p) { if(*p & (1 << 7)) ++sum[ j*8 ]; if(*p & (1 << 6)) ++sum[ j*8 + 1 ]; if(*p & (1 << 5)) ++sum[ j*8 + 2 ]; if(*p & (1 << 4)) ++sum[ j*8 + 3 ]; if(*p & (1 << 3)) ++sum[ j*8 + 4 ]; if(*p & (1 << 2)) ++sum[ j*8 + 5 ]; if(*p & (1 << 1)) ++sum[ j*8 + 6 ]; if(*p & (1)) ++sum[ j*8 + 7 ]; } } mean = cv::Mat::zeros(1, FORB::L, CV_8U); unsigned char *p = mean.ptr(); const int N2 = (int)descriptors.size() / 2 + descriptors.size() % 2; for(size_t i = 0; i < sum.size(); ++i) { if(sum[i] >= N2) { // set bit *p |= 1 << (7 - (i % 8)); } if(i % 8 == 7) ++p; } } } // -------------------------------------------------------------------------- int FORB::distance(const FORB::TDescriptor &a, const FORB::TDescriptor &b) { // Bit set count operation from // http://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetParallel const int *pa = a.ptr(); const int *pb = b.ptr(); int dist=0; for(int i=0; i<8; i++, pa++, pb++) { unsigned int v = *pa ^ *pb; v = v - ((v >> 1) & 0x55555555); v = (v & 0x33333333) + ((v >> 2) & 0x33333333); dist += (((v + (v >> 4)) & 0xF0F0F0F) * 0x1010101) >> 24; } return dist; } // -------------------------------------------------------------------------- std::string FORB::toString(const FORB::TDescriptor &a) { stringstream ss; const unsigned char *p = a.ptr(); for(int i = 0; i < a.cols; ++i, ++p) { ss << (int)*p << " "; } return ss.str(); } // -------------------------------------------------------------------------- void FORB::fromString(FORB::TDescriptor &a, const std::string &s) { a.create(1, FORB::L, CV_8U); unsigned char *p = a.ptr(); stringstream ss(s); for(int i = 0; i < FORB::L; ++i, ++p) { int n; ss >> n; if(!ss.fail()) *p = (unsigned char)n; } } // -------------------------------------------------------------------------- void FORB::toMat32F(const std::vector &descriptors, cv::Mat &mat) { if(descriptors.empty()) { mat.release(); return; } const size_t N = descriptors.size(); mat.create(N, FORB::L*8, CV_32F); float *p = mat.ptr(); for(size_t i = 0; i < N; ++i) { const int C = descriptors[i].cols; const unsigned char *desc = descriptors[i].ptr(); for(int j = 0; j < C; ++j, p += 8) { p[0] = (desc[j] & (1 << 7) ? 1 : 0); p[1] = (desc[j] & (1 << 6) ? 1 : 0); p[2] = (desc[j] & (1 << 5) ? 1 : 0); p[3] = (desc[j] & (1 << 4) ? 1 : 0); p[4] = (desc[j] & (1 << 3) ? 1 : 0); p[5] = (desc[j] & (1 << 2) ? 1 : 0); p[6] = (desc[j] & (1 << 1) ? 1 : 0); p[7] = desc[j] & (1); } } } // -------------------------------------------------------------------------- void FORB::toMat8U(const std::vector &descriptors, cv::Mat &mat) { mat.create(descriptors.size(), 32, CV_8U); unsigned char *p = mat.ptr(); for(size_t i = 0; i < descriptors.size(); ++i, p += 32) { const unsigned char *d = descriptors[i].ptr(); std::copy(d, d+32, p); } } // -------------------------------------------------------------------------- } // namespace DBoW2