我使用英特尔顾问来分析我的并行应用程序。我有这个代码,这是我的程序的主循环,大部分时间花在哪里:
for(size_t i=0; i<wrapperIndexes.size(); i++){
const int r = wrapperIndexes[i].r;
const int c = wrapperIndexes[i].c;
const float val = localWrappers[wrapperIndexes[i].i].cur.at<float>(wrapperIndexes[i].r,wrapperIndexes[i].c);
if ( (val > positiveThreshold && (isMax(val, localWrappers[wrapperIndexes[i].i].cur, r, c) && isMax(val, localWrappers[wrapperIndexes[i].i].low, r, c) && isMax(val, localWrappers[wrapperIndexes[i].i].high, r, c))) ||
(val < negativeThreshold && (isMin(val, localWrappers[wrapperIndexes[i].i].cur, r, c) && isMin(val, localWrappers[wrapperIndexes[i].i].low, r, c) && isMin(val, localWrappers[wrapperIndexes[i].i].high, r, c))) )
// either positive -> local max. or negative -> local min.
ANNOTATE_ITERATION_TASK(localizeKeypoint);
localizeKeypoint(r, c, localCurSigma[wrapperIndexes[i].i], localPixelDistances[wrapperIndexes[i].i], localWrappers[wrapperIndexes[i].i]);
}
正如您所看到的,localizeKeypoint
是循环花费的大部分时间(如果您不考虑if
子句)。我想做一个适用性报告来估算上面的循环并行化的收益。所以我写了这个:
ANNOTATE_SITE_BEGIN(solve);
for(size_t i=0; i<wrapperIndexes.size(); i++){
const int r = wrapperIndexes[i].r;
const int c = wrapperIndexes[i].c;
const float val = localWrappers[wrapperIndexes[i].i].cur.at<float>(wrapperIndexes[i].r,wrapperIndexes[i].c);
if ( (val > positiveThreshold && (isMax(val, localWrappers[wrapperIndexes[i].i].cur, r, c) && isMax(val, localWrappers[wrapperIndexes[i].i].low, r, c) && isMax(val, localWrappers[wrapperIndexes[i].i].high, r, c))) ||
(val < negativeThreshold && (isMin(val, localWrappers[wrapperIndexes[i].i].cur, r, c) && isMin(val, localWrappers[wrapperIndexes[i].i].low, r, c) && isMin(val, localWrappers[wrapperIndexes[i].i].high, r, c))) )
// either positive -> local max. or negative -> local min.
ANNOTATE_ITERATION_TASK(localizeKeypoint);
localizeKeypoint(r, c, localCurSigma[wrapperIndexes[i].i], localPixelDistances[wrapperIndexes[i].i], localWrappers[wrapperIndexes[i].i]);
}
ANNOTATE_SITE_END();
适用性报告的增益为6.69倍,如下所示:
但是,启动依赖项检查,我收到此问题消息:
特别参见&#34;缺少启动任务&#34;。
另外,如果我将ANNOTATE_ITERATION_TASK
放在循环的开头,就像这样:
ANNOTATE_SITE_BEGIN(solve);
for(size_t i=0; i<wrapperIndexes.size(); i++){
ANNOTATE_ITERATION_TASK(localizeKeypoint);
const int r = wrapperIndexes[i].r;
const int c = wrapperIndexes[i].c;
const float val = localWrappers[wrapperIndexes[i].i].cur.at<float>(wrapperIndexes[i].r,wrapperIndexes[i].c);
if ( (val > positiveThreshold && (isMax(val, localWrappers[wrapperIndexes[i].i].cur, r, c) && isMax(val, localWrappers[wrapperIndexes[i].i].low, r, c) && isMax(val, localWrappers[wrapperIndexes[i].i].high, r, c))) ||
(val < negativeThreshold && (isMin(val, localWrappers[wrapperIndexes[i].i].cur, r, c) && isMin(val, localWrappers[wrapperIndexes[i].i].low, r, c) && isMin(val, localWrappers[wrapperIndexes[i].i].high, r, c))) )
// either positive -> local max. or negative -> local min.
localizeKeypoint(r, c, localCurSigma[wrapperIndexes[i].i], localPixelDistances[wrapperIndexes[i].i], localWrappers[wrapperIndexes[i].i]);
}
ANNOTATE_SITE_END();
收益太可怕了:
我做错了吗?
INTEL_OPT=-O3 -simd -xCORE-AVX2 -parallel -qopenmp -fargument-noalias -ansi-alias -no-prec-div -fp-model fast=2
INTEL_PROFILE=-g -qopt-report=5 -Bdynamic -shared-intel -debug inline-debug-info -qopenmp-link dynamic -parallel-source-info=2 -ldl
答案 0 :(得分:1)
你必须使用第二种方法,在循环注释的最开始处放置ANNOTATE_ITERATION_TASK。否则,你会得到(a)在适合性方面的错误表现预测,(b)在正确性中缺少启动任务。
如果为第二个变量运行Correctness(在循环体的最开始处放置迭代任务),那么正确性应该没问题。
你的第二张适应性图表并不可怕。它只是说你必须关注任务分块(点击工具中的“分块”链接以了解更多信息)。幸运的是,在新的OpenMP分块中,默认情况下“足够好”,请参阅https://software.intel.com/en-us/articles/openmp-loop-scheduling。因此,为了看到带有分块ON的Advisor投影,您只需要打开相应的复选框,它就不会那么糟糕。