-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathTrackingModelFF.cpp
166 lines (151 loc) · 5.93 KB
/
TrackingModelFF.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
//-------------------------------------------------------------
// ____ _ _
// / ___|____ _ _ ____ ____| |__ | |
// | | / ___| | | | _ \/ ___| _ \| |
// | |___| | | |_| | | | | |___| | | ||_|
// \____|_| \_____|_| |_|\____|_| |_|(_) Media benchmarks
//
// 2006, Intel Corporation, licensed under Apache 2.0
//
// file : TrackingModelFF.cpp
// author : Scott Ettinger - [email protected]
// description : Observation model for kinematic tree body
// tracking threaded with OpenMP.
//
// modified : FastFlow version written by Daniele De Sensi
// ([email protected])
//--------------------------------------------------------------
#if defined(HAVE_CONFIG_H)
# include "config.h"
#endif
#include "TrackingModelFF.h"
#include <vector>
#include <string>
#include "system.h"
using namespace std;
//------------------------ Threaded versions of image filters --------------------
//FF threaded - 1D filter Row wise 1 channel any type data or kernel valid pixels only
template<class T, class T2>
bool FlexFilterRowVFF(ff::ParallelFor* pf, int mThreads, FlexImage<T,1> &src, FlexImage<T,1> &dst, T2 *kernel, int kernelSize, bool allocate = true)
{
if(kernelSize % 2 == 0) //enforce odd length kernels
return false;
if(allocate)
dst.Reallocate(src.Size());
dst.Set((T)0);
int n = kernelSize / 2, h = src.Height();
pf->parallel_for(0, h, [&](const int y)
{ T *psrc = &src(n, y), *pdst = &dst(n, y);
for(int x = n; x < src.Width() - n; x++)
{ int k = 0;
T2 acc = 0;
for(int i = -n; i <= n; i++)
acc += (T2)(psrc[i] * kernel[k++]);
*pdst = (T)acc;
pdst++;
psrc++;
}
},
mThreads);
return true;
}
//FF threaded - 1D filter Column wise 1 channel any type data or kernel valid pixels only
template<class T, class T2>
bool FlexFilterColumnVFF(ff::ParallelFor* pf, int mThreads, FlexImage<T,1> &src, FlexImage<T,1> &dst, T2 *kernel, int kernelSize, bool allocate = true)
{
if(kernelSize % 2 == 0) //enforce odd length kernels
return false;
if(allocate)
dst.Reallocate(src.Size());
dst.Set((T)0);
int n = kernelSize / 2;
int sb = src.StepBytes(), h = src.Height() - n;
pf->parallel_for(0, h, [&](const int y)
{ T *psrc = &src(0, y), *pdst = &dst(0, y);
for(int x = 0; x < src.Width(); x++)
{ int k = 0;
T2 acc = 0;
for(int i = -n; i <= n; i++)
acc += (T2)(*(T *)((char *)psrc + sb * i) * kernel[k++]);
*pdst = (T)acc;
pdst++;
psrc++;
}
},
mThreads);
return true;
}
// ----------------------------------------------------------------------------------
//Generate an edge map from the original camera image
//Separable 7x7 gaussian filter - threaded
inline void GaussianBlurFF(ff::ParallelFor* pf, int mThreads, FlexImage8u &src, FlexImage8u &dst)
{
float k[] = {0.12149085090552f, 0.14203719483447f, 0.15599734045770f, 0.16094922760463f, 0.15599734045770f, 0.14203719483447f, 0.12149085090552f};
FlexImage8u tmp;
FlexFilterRowVFF(pf, mThreads, src, tmp, k, 7); //separable gaussian convolution using kernel k
FlexFilterColumnVFF(pf, mThreads, tmp, dst, k, 7);
}
//Calculate gradient magnitude and threshold to binarize - threaded
inline FlexImage8u GradientMagThresholdFF(ff::ParallelFor* pf, int mThreads, FlexImage8u &src, float threshold)
{
FlexImage8u r(src.Size());
ZeroBorder(r);
pf->parallel_for(1, src.Height() - 1, [&](const int y)
{ Im8u *p = &src(1,y), *ph = &src(1,y - 1), *pl = &src(1,y + 1), *pr = &r(1,y);
for(int x = 1; x < src.Width() - 1; x++)
{ float xg = -0.125f * ph[-1] + 0.125f * ph[1] - 0.250f * p[-1] + 0.250f * p[1] - 0.125f * pl[-1] + 0.125f * pl[1]; //calc x and y gradients
float yg = -0.125f * ph[-1] - 0.250f * ph[0] - 0.125f * ph[1] + 0.125f * pl[-1] + 0.250f * pl[0] + 0.125f * pl[1];
float mag = xg * xg + yg * yg; //calc magnitude and threshold
*pr = (mag < threshold) ? 0 : 255;
p++; ph++; pl++; pr++;
}
},
mThreads);
return r;
}
//Generate an edge map from the original camera image
void TrackingModelFF::CreateEdgeMap(FlexImage8u &src, FlexImage8u &dst)
{
FlexImage8u gr = GradientMagThresholdFF(_pf, mThreads, src, 16.0f); //calc gradient magnitude and threshold
GaussianBlurFF(_pf, mThreads, gr, dst); //Blur to create distance error map
}
//templated conversion to string with field width
template<class T>
inline string str(T n, int width = 0, char pad = '0')
{ stringstream ss;
ss << setw(width) << setfill(pad) << n;
return ss.str();
}
//load and process all images for new observation at a given time(frame)
//Overloaded from base class for future threading to overlap disk I/O with
//generating the edge maps
bool TrackingModelFF::GetObservation(float timeval)
{
int frame = (int)timeval; //generate image filenames
int n = mCameras.GetCameraCount();
vector<string> FGfiles(n), ImageFiles(n);
for(int i = 0; i < n; i++)
{ FGfiles[i] = mPath + "FG" + str(i + 1) + DIR_SEPARATOR + "image" + str(frame, 4) + ".bmp";
ImageFiles[i] = mPath + "CAM" + str(i + 1) + DIR_SEPARATOR + "image" + str(frame, 4) + ".bmp";
}
FlexImage8u im;
for(int i = 0; i < (int)FGfiles.size(); i++)
{ if(!FlexLoadBMP(FGfiles[i].c_str(), im)) //Load foreground maps and raw images
{ cout << "Unable to load image: " << FGfiles[i].c_str() << endl;
return false;
}
mFGMaps[i].ConvertToBinary(im); //binarize foreground maps to 0 and 1
if(!FlexLoadBMP(ImageFiles[i].c_str(), im))
{ cout << "Unable to load image: " << ImageFiles[i].c_str() << endl;
return false;
}
CreateEdgeMap(im, mEdgeMaps[i]); //Create edge maps
}
return true;
}
ff::ParallelFor* TrackingModelFF::getParallelFor(){
return _pf;
}
void TrackingModelFF::setParallelFor(ff::ParallelFor* pf){
_pf = pf;
}