Commit 35b935c9 authored by Pavel Rojtberg's avatar Pavel Rojtberg Committed by Pavel Rojtberg

aruco: simplify Dictionary by not interleaving the marker rotations

store marker internally as
"r0_b0, r0_b1, .. , r0_bn, r2_b0, .." instead of
"r0_b0, r1_b0, .. , rn_b0, r1_b0, .."
i.e. the same as in predefined_dictionaries.cpp.
This makes loading a dictionary a simple memcpy and also allows to get
rid of the custom hammingDistance implementation in favor of
hal::normHamming.
parent 4f860dcf
...@@ -48,24 +48,6 @@ namespace aruco { ...@@ -48,24 +48,6 @@ namespace aruco {
using namespace std; using namespace std;
/**
* Hamming weight look up table from 0 to 255
*/
const unsigned char hammingWeightLUT[] = {
0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5,
1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8
};
/** /**
*/ */
Dictionary::Dictionary(const unsigned char *bytes, int _markerSize, int dictsize, int _maxcorr) { Dictionary::Dictionary(const unsigned char *bytes, int _markerSize, int dictsize, int _maxcorr) {
...@@ -75,14 +57,9 @@ Dictionary::Dictionary(const unsigned char *bytes, int _markerSize, int dictsize ...@@ -75,14 +57,9 @@ Dictionary::Dictionary(const unsigned char *bytes, int _markerSize, int dictsize
if((markerSize * markerSize) % 8 != 0) nbytes++; if((markerSize * markerSize) % 8 != 0) nbytes++;
// save bytes in internal format // save bytes in internal format
// bytesList.at<Vec4b>(i, j)[k] is j-th byte of i-th marker, in its k-th rotation // bytesList.ptr(i)[k*nbytes + j] is j-th byte of i-th marker, in its k-th rotation
bytesList = Mat(dictsize, nbytes, CV_8UC4); bytesList = Mat(dictsize, nbytes, CV_8UC4);
for(int i = 0; i < dictsize; i++) { memcpy(bytesList.data, bytes, dictsize*nbytes*4);
for(int j = 0; j < nbytes; j++) {
for(int k = 0; k < 4; k++)
bytesList.at< Vec4b >(i, j)[k] = bytes[i * (4 * nbytes) + k * nbytes + j];
}
}
} }
...@@ -106,13 +83,10 @@ bool Dictionary::identify(const Mat &onlyBits, int &idx, int &rotation, ...@@ -106,13 +83,10 @@ bool Dictionary::identify(const Mat &onlyBits, int &idx, int &rotation,
int currentMinDistance = markerSize * markerSize + 1; int currentMinDistance = markerSize * markerSize + 1;
int currentRotation = -1; int currentRotation = -1;
for(unsigned int r = 0; r < 4; r++) { for(unsigned int r = 0; r < 4; r++) {
int currentHamming = 0; int currentHamming = hal::normHamming(
// for each byte, calculate XOR result and then sum the Hamming weight from the LUT bytesList.ptr(m)+r*candidateBytes.cols,
for(unsigned int b = 0; b < candidateBytes.total(); b++) { candidateBytes.ptr(),
unsigned char xorRes = candidateBytes.cols);
bytesList.ptr< Vec4b >(m)[b][r] ^ candidateBytes.ptr< Vec4b >(0)[b][0];
currentHamming += hammingWeightLUT[xorRes];
}
if(currentHamming < currentMinDistance) { if(currentHamming < currentMinDistance) {
currentMinDistance = currentHamming; currentMinDistance = currentHamming;
...@@ -147,12 +121,10 @@ int Dictionary::getDistanceToId(InputArray bits, int id, bool allRotations) cons ...@@ -147,12 +121,10 @@ int Dictionary::getDistanceToId(InputArray bits, int id, bool allRotations) cons
Mat candidateBytes = getByteListFromBits(bits.getMat()); Mat candidateBytes = getByteListFromBits(bits.getMat());
int currentMinDistance = int(bits.total() * bits.total()); int currentMinDistance = int(bits.total() * bits.total());
for(unsigned int r = 0; r < nRotations; r++) { for(unsigned int r = 0; r < nRotations; r++) {
int currentHamming = 0; int currentHamming = hal::normHamming(
for(unsigned int b = 0; b < candidateBytes.total(); b++) { bytesList.ptr(id) + r*candidateBytes.cols,
unsigned char xorRes = candidateBytes.ptr(),
bytesList.ptr< Vec4b >(id)[b][r] ^ candidateBytes.ptr< Vec4b >(0)[b][0]; candidateBytes.cols);
currentHamming += hammingWeightLUT[xorRes];
}
if(currentHamming < currentMinDistance) { if(currentHamming < currentMinDistance) {
currentMinDistance = currentHamming; currentMinDistance = currentHamming;
...@@ -202,26 +174,25 @@ Mat Dictionary::getByteListFromBits(const Mat &bits) { ...@@ -202,26 +174,25 @@ Mat Dictionary::getByteListFromBits(const Mat &bits) {
Mat candidateByteList(1, nbytes, CV_8UC4, Scalar::all(0)); Mat candidateByteList(1, nbytes, CV_8UC4, Scalar::all(0));
unsigned char currentBit = 0; unsigned char currentBit = 0;
int currentByte = 0; int currentByte = 0;
// the 4 rotations
uchar* rot0 = candidateByteList.ptr();
uchar* rot1 = candidateByteList.ptr() + 1*nbytes;
uchar* rot2 = candidateByteList.ptr() + 2*nbytes;
uchar* rot3 = candidateByteList.ptr() + 3*nbytes;
for(int row = 0; row < bits.rows; row++) { for(int row = 0; row < bits.rows; row++) {
for(int col = 0; col < bits.cols; col++) { for(int col = 0; col < bits.cols; col++) {
// circular shift // circular shift
candidateByteList.ptr< Vec4b >(0)[currentByte][0] = rot0[currentByte] <<= 1;
candidateByteList.ptr< Vec4b >(0)[currentByte][0] << 1; rot1[currentByte] <<= 1;
candidateByteList.ptr< Vec4b >(0)[currentByte][1] = rot2[currentByte] <<= 1;
candidateByteList.ptr< Vec4b >(0)[currentByte][1] << 1; rot3[currentByte] <<= 1;
candidateByteList.ptr< Vec4b >(0)[currentByte][2] = // set bit
candidateByteList.ptr< Vec4b >(0)[currentByte][2] << 1; rot0[currentByte] |= bits.at<uchar>(row, col);
candidateByteList.ptr< Vec4b >(0)[currentByte][3] = rot1[currentByte] |= bits.at<uchar>(col, bits.cols - 1 - row);
candidateByteList.ptr< Vec4b >(0)[currentByte][3] << 1; rot2[currentByte] |= bits.at<uchar>(bits.rows - 1 - row, bits.cols - 1 - col);
// increment if bit is 1 rot3[currentByte] |= bits.at<uchar>(bits.rows - 1 - col, row);
if(bits.at< unsigned char >(row, col))
candidateByteList.ptr< Vec4b >(0)[currentByte][0]++;
if(bits.at< unsigned char >(col, bits.cols - 1 - row))
candidateByteList.ptr< Vec4b >(0)[currentByte][1]++;
if(bits.at< unsigned char >(bits.rows - 1 - row, bits.cols - 1 - col))
candidateByteList.ptr< Vec4b >(0)[currentByte][2]++;
if(bits.at< unsigned char >(bits.rows - 1 - col, row))
candidateByteList.ptr< Vec4b >(0)[currentByte][3]++;
currentBit++; currentBit++;
if(currentBit == 8) { if(currentBit == 8) {
// next byte // next byte
...@@ -247,7 +218,7 @@ Mat Dictionary::getBitsFromByteList(const Mat &byteList, int markerSize) { ...@@ -247,7 +218,7 @@ Mat Dictionary::getBitsFromByteList(const Mat &byteList, int markerSize) {
unsigned char base2List[] = { 128, 64, 32, 16, 8, 4, 2, 1 }; unsigned char base2List[] = { 128, 64, 32, 16, 8, 4, 2, 1 };
int currentByteIdx = 0; int currentByteIdx = 0;
// we only need the bytes in normal rotation // we only need the bytes in normal rotation
unsigned char currentByte = byteList.ptr< Vec4b >(0)[0][0]; unsigned char currentByte = byteList.ptr()[0];
int currentBit = 0; int currentBit = 0;
for(int row = 0; row < bits.rows; row++) { for(int row = 0; row < bits.rows; row++) {
for(int col = 0; col < bits.cols; col++) { for(int col = 0; col < bits.cols; col++) {
...@@ -258,7 +229,7 @@ Mat Dictionary::getBitsFromByteList(const Mat &byteList, int markerSize) { ...@@ -258,7 +229,7 @@ Mat Dictionary::getBitsFromByteList(const Mat &byteList, int markerSize) {
currentBit++; currentBit++;
if(currentBit == 8) { if(currentBit == 8) {
currentByteIdx++; currentByteIdx++;
currentByte = byteList.ptr< Vec4b >(0)[currentByteIdx][0]; currentByte = byteList.ptr()[currentByteIdx];
// if not enough bits for one more byte, we are in the end // if not enough bits for one more byte, we are in the end
// update bit position accordingly // update bit position accordingly
if(8 * (currentByteIdx + 1) > (int)bits.total()) if(8 * (currentByteIdx + 1) > (int)bits.total())
...@@ -369,12 +340,8 @@ static Mat _generateRandomMarker(int markerSize) { ...@@ -369,12 +340,8 @@ static Mat _generateRandomMarker(int markerSize) {
static int _getSelfDistance(const Mat &marker) { static int _getSelfDistance(const Mat &marker) {
Mat bytes = Dictionary::getByteListFromBits(marker); Mat bytes = Dictionary::getByteListFromBits(marker);
int minHamming = (int)marker.total() + 1; int minHamming = (int)marker.total() + 1;
for(int i = 1; i < 4; i++) { for(int r = 1; r < 4; r++) {
int currentHamming = 0; int currentHamming = hal::normHamming(bytes.ptr(), bytes.ptr() + bytes.cols*r, bytes.cols);
for(int j = 0; j < bytes.cols; j++) {
unsigned char xorRes = bytes.ptr< Vec4b >()[j][0] ^ bytes.ptr< Vec4b >()[j][i];
currentHamming += hammingWeightLUT[xorRes];
}
if(currentHamming < minHamming) minHamming = currentHamming; if(currentHamming < minHamming) minHamming = currentHamming;
} }
return minHamming; return minHamming;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment