improve cpu version of SoftCascade:
- remove division - remove cvRound - cache feature area
This commit is contained in:
parent
2d2c46e717
commit
2914f24521
@ -126,6 +126,7 @@ struct Feature
|
|||||||
{
|
{
|
||||||
int channel;
|
int channel;
|
||||||
cv::Rect rect;
|
cv::Rect rect;
|
||||||
|
float rarea;
|
||||||
|
|
||||||
static const char * const SC_F_CHANNEL;
|
static const char * const SC_F_CHANNEL;
|
||||||
static const char * const SC_F_RECT;
|
static const char * const SC_F_RECT;
|
||||||
@ -136,6 +137,9 @@ struct Feature
|
|||||||
cv::FileNode rn = fn[SC_F_RECT];
|
cv::FileNode rn = fn[SC_F_RECT];
|
||||||
cv::FileNodeIterator r_it = rn.end();
|
cv::FileNodeIterator r_it = rn.end();
|
||||||
rect = cv::Rect(*(--r_it), *(--r_it), *(--r_it), *(--r_it));
|
rect = cv::Rect(*(--r_it), *(--r_it), *(--r_it), *(--r_it));
|
||||||
|
|
||||||
|
// 1 / area
|
||||||
|
rarea = 1.f / ((rect.width - rect.x) * (rect.height - rect.y));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -192,10 +196,13 @@ struct Level
|
|||||||
float origScale;
|
float origScale;
|
||||||
float relScale;
|
float relScale;
|
||||||
float shrScale; // used for marking detection
|
float shrScale; // used for marking detection
|
||||||
|
int scaleshift;
|
||||||
|
|
||||||
cv::Size workRect;
|
cv::Size workRect;
|
||||||
cv::Size objSize;
|
cv::Size objSize;
|
||||||
|
|
||||||
|
enum { R_SHIFT = 1 << 15 };
|
||||||
|
|
||||||
float scaling[2];
|
float scaling[2];
|
||||||
|
|
||||||
Level(const Octave& oct, const float scale, const int shrinkage, const int w, const int h)
|
Level(const Octave& oct, const float scale, const int shrinkage, const int w, const int h)
|
||||||
@ -203,8 +210,9 @@ struct Level
|
|||||||
workRect(cv::Size(cvRound(w / (float)shrinkage),cvRound(h / (float)shrinkage))),
|
workRect(cv::Size(cvRound(w / (float)shrinkage),cvRound(h / (float)shrinkage))),
|
||||||
objSize(cv::Size(cvRound(oct.size.width * relScale), cvRound(oct.size.height * relScale)))
|
objSize(cv::Size(cvRound(oct.size.width * relScale), cvRound(oct.size.height * relScale)))
|
||||||
{
|
{
|
||||||
scaling[0] = CascadeIntrinsics::getFor(0, relScale);
|
scaling[0] = CascadeIntrinsics::getFor(0, relScale) / (relScale * relScale);
|
||||||
scaling[1] = CascadeIntrinsics::getFor(9, relScale);
|
scaling[1] = CascadeIntrinsics::getFor(9, relScale) / (relScale * relScale);
|
||||||
|
scaleshift = relScale * (1 << 16);
|
||||||
}
|
}
|
||||||
|
|
||||||
void markDetection(const int x, const int y, float confidence, std::vector<Object>& detections) const
|
void markDetection(const int x, const int y, float confidence, std::vector<Object>& detections) const
|
||||||
@ -214,6 +222,25 @@ struct Level
|
|||||||
|
|
||||||
detections.push_back(Object(rect, confidence));
|
detections.push_back(Object(rect, confidence));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float rescale(cv::Rect& scaledRect, const float threshold, int idx) const
|
||||||
|
{
|
||||||
|
// rescale
|
||||||
|
// scaledRect.x = cvRound(relScale * scaledRect.x);
|
||||||
|
// scaledRect.y = cvRound(relScale * scaledRect.y);
|
||||||
|
// scaledRect.width = cvRound(relScale * scaledRect.width);
|
||||||
|
// scaledRect.height = cvRound(relScale * scaledRect.height);
|
||||||
|
|
||||||
|
scaledRect.x = (scaleshift * scaledRect.x + R_SHIFT) >> 16;
|
||||||
|
scaledRect.y = (scaleshift * scaledRect.y + R_SHIFT) >> 16;
|
||||||
|
scaledRect.width = (scaleshift * scaledRect.width + R_SHIFT) >> 16;
|
||||||
|
scaledRect.height = (scaleshift * scaledRect.height + R_SHIFT) >> 16;
|
||||||
|
|
||||||
|
float sarea = (scaledRect.width - scaledRect.x) * (scaledRect.height - scaledRect.y);
|
||||||
|
|
||||||
|
// compensation areas rounding
|
||||||
|
return (threshold * scaling[idx] * sarea);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template< typename T>
|
template< typename T>
|
||||||
@ -416,41 +443,6 @@ struct cv::SoftCascade::Filds
|
|||||||
|
|
||||||
typedef std::vector<Octave>::iterator octIt_t;
|
typedef std::vector<Octave>::iterator octIt_t;
|
||||||
|
|
||||||
float rescale(const Feature& feature, const float scaling, const float relScale,
|
|
||||||
cv::Rect& scaledRect, const float threshold) const
|
|
||||||
{
|
|
||||||
scaledRect = feature.rect;
|
|
||||||
|
|
||||||
dprintf("feature %d box %d %d %d %d\n", feature.channel, scaledRect.x, scaledRect.y,
|
|
||||||
scaledRect.width, scaledRect.height);
|
|
||||||
|
|
||||||
dprintf("rescale: %d %f %f\n",feature.channel, relScale, scaling);
|
|
||||||
|
|
||||||
float farea = (scaledRect.width - scaledRect.x) * (scaledRect.height - scaledRect.y);
|
|
||||||
// rescale
|
|
||||||
scaledRect.x = cvRound(relScale * scaledRect.x);
|
|
||||||
scaledRect.y = cvRound(relScale * scaledRect.y);
|
|
||||||
scaledRect.width = cvRound(relScale * scaledRect.width);
|
|
||||||
scaledRect.height = cvRound(relScale * scaledRect.height);
|
|
||||||
|
|
||||||
dprintf("feature %d box %d %d %d %d\n", feature.channel, scaledRect.x, scaledRect.y,
|
|
||||||
scaledRect.width, scaledRect.height);
|
|
||||||
|
|
||||||
float sarea = (scaledRect.width - scaledRect.x) * (scaledRect.height - scaledRect.y);
|
|
||||||
|
|
||||||
const float expected_new_area = farea * relScale * relScale;
|
|
||||||
float approx = sarea / expected_new_area;
|
|
||||||
|
|
||||||
dprintf(" rel areas %f %f\n", expected_new_area, sarea);
|
|
||||||
|
|
||||||
// compensation areas rounding
|
|
||||||
float rootThreshold = threshold * approx * scaling;
|
|
||||||
|
|
||||||
dprintf("approximation %f %f -> %f %f\n", approx, threshold, rootThreshold, scaling);
|
|
||||||
|
|
||||||
return rootThreshold;
|
|
||||||
}
|
|
||||||
|
|
||||||
void detectAt(const Level& level, const int dx, const int dy, const ChannelStorage& storage,
|
void detectAt(const Level& level, const int dx, const int dy, const ChannelStorage& storage,
|
||||||
std::vector<Object>& detections) const
|
std::vector<Object>& detections) const
|
||||||
{
|
{
|
||||||
@ -477,10 +469,9 @@ struct cv::SoftCascade::Filds
|
|||||||
// work with root node
|
// work with root node
|
||||||
const Node& node = nodes[nId];
|
const Node& node = nodes[nId];
|
||||||
const Feature& feature = features[node.feature];
|
const Feature& feature = features[node.feature];
|
||||||
cv::Rect scaledRect;
|
cv::Rect scaledRect(feature.rect);
|
||||||
float threshold = rescale(feature, level.scaling[(int)(feature.channel > 6)],
|
|
||||||
level.relScale, scaledRect, node.threshold);
|
|
||||||
|
|
||||||
|
float threshold = level.rescale(scaledRect, node.threshold,(int)(feature.channel > 6)) * feature.rarea;
|
||||||
|
|
||||||
float sum = storage.get(dx, dy, feature.channel, scaledRect);
|
float sum = storage.get(dx, dy, feature.channel, scaledRect);
|
||||||
|
|
||||||
@ -494,10 +485,10 @@ struct cv::SoftCascade::Filds
|
|||||||
const Node& leaf = nodes[nId + next];
|
const Node& leaf = nodes[nId + next];
|
||||||
const Feature& fLeaf = features[leaf.feature];
|
const Feature& fLeaf = features[leaf.feature];
|
||||||
|
|
||||||
threshold = rescale(fLeaf, level.scaling[(int)(fLeaf.channel > 6)],
|
scaledRect = fLeaf.rect;
|
||||||
level.relScale, scaledRect, leaf.threshold);
|
threshold = level.rescale(scaledRect, leaf.threshold, (int)(fLeaf.channel > 6)) * fLeaf.rarea;
|
||||||
sum = storage.get(dx, dy, fLeaf.channel, scaledRect);
|
|
||||||
|
|
||||||
|
sum = storage.get(dx, dy, fLeaf.channel, scaledRect);
|
||||||
|
|
||||||
int lShift = (next - 1) * 2 + ((sum >= threshold) ? 1 : 0);
|
int lShift = (next - 1) * 2 + ((sum >= threshold) ? 1 : 0);
|
||||||
float impact = leaves[(st * 4) + lShift];
|
float impact = leaves[(st * 4) + lShift];
|
||||||
|
Loading…
x
Reference in New Issue
Block a user