00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038
00039
00040 #define MAX_CORNERS 500
00041
00042
00043 #include "Image/OpenCVUtil.H"
00044
00045 #include <cstdio>
00046
00047 #include "Component/ModelManager.H"
00048 #include "Media/FrameSeries.H"
00049
00050 #include "Image/Image.H"
00051 #include "Image/CutPaste.H"
00052 #include "Image/ColorOps.H"
00053 #include "Image/MathOps.H"
00054 #include "Image/ShapeOps.H"
00055 #include "Image/Pixels.H"
00056 #include "Raster/Raster.H"
00057
00058 #include "SIFT/Keypoint.H"
00059 #include "SIFT/VisualObject.H"
00060 #include "SIFT/VisualObjectMatch.H"
00061
00062
00063 #include "Util/Timer.H"
00064
00065 #include "Robots/Beobot2/Navigation/FOE_Navigation/FoeDetector.H"
00066 #include "Robots/Beobot2/Navigation/FOE_Navigation/MotionOps.H"
00067
00068
00069
00070 Image<float> getOpticFlow2(Image<float> image1, Image<float> image2);
00071
00072
00073 std::vector<Point2D<int> > getGT(std::string gtFilename);
00074
00075 Image<byte> calculateShift
00076 (Image<byte> prevLum, Image<byte> lum, nub::ref<OutputFrameSeries> ofs);
00077
00078 Image<byte> shiftImage(SIFTaffine aff, Image<byte> ref, Image<byte> tst);
00079
00080 std::vector<std::vector<Point2D<int> > > getGTandData();
00081
00082
00083 int main(const int argc, const char **argv)
00084 {
00085 MYLOGVERB = LOG_INFO;
00086
00087
00088 ModelManager manager("Test Optic Flow");
00089
00090 nub::ref<InputFrameSeries> ifs(new InputFrameSeries(manager));
00091 manager.addSubComponent(ifs);
00092
00093 nub::ref<OutputFrameSeries> ofs(new OutputFrameSeries(manager));
00094 manager.addSubComponent(ofs);
00095
00096 nub::ref<FoeDetector> fd(new FoeDetector(manager));
00097 manager.addSubComponent(fd);
00098
00099 manager.exportOptions(MC_RECURSE);
00100
00101 if (manager.parseCommandLine((const int)argc, (const char**)argv,
00102 "", 0, 0) == false)
00103 return(1);
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119 std::string pprefix
00120 ("/lab/tmpib/u/siagian/neuroscience/Data/FOE/driving_nat_3");
00121
00122 std::string folder("/lab/tmpib/u/siagian/neuroscience/Data/FOE/");
00123 std::string ciofmPrefix("DARPA_nv2_2011/Germany_K_CIOFM/SalMap_CIOFM");
00124 std::string ciofsPrefix("DARPA_nv2_2011/Germany_K_CIOFs/SalMap_CIOFs");
00125
00126
00127 manager.start();
00128
00129 Timer timer(1000000);
00130 timer.reset();
00131 int frame = 0;
00132 bool keepGoing = true;
00133
00134 const FrameState is = ifs->updateNext();
00135 Image< PixRGB<byte> > prevImage;
00136 if (is == FRAME_COMPLETE) keepGoing = false;
00137 else prevImage = ifs->readRGB();
00138
00139 uint width = prevImage.getWidth();
00140 uint height = prevImage.getHeight();
00141
00142
00143 rutz::shared_ptr<XWinManaged> win
00144 (new XWinManaged(Dims(width, height), 0, 0, "GT"));
00145
00146
00147
00148
00149
00150
00151
00152
00153
00154
00155
00156
00157
00158
00159
00160
00161
00162
00163
00164
00165
00166
00167
00168 Image<byte> prevLum = Image<byte>(luminance(prevImage));
00169 win->setDims(Dims(640,720));
00170 while (keepGoing)
00171 {
00172 if (ofs->becameVoid())
00173 {
00174 LINFO("quitting because output stream was closed or became void");
00175 break;
00176 }
00177
00178 const FrameState is = ifs->updateNext();
00179 if (is == FRAME_COMPLETE) break;
00180
00181 Image< PixRGB<byte> > currImage = ifs->readRGB();
00182 Image<byte> lum = Image<byte>(luminance(currImage));
00183
00184
00185
00186 Image< PixRGB<byte> > ciofm = Raster::ReadRGB
00187 (folder+sformat("%s_%06d.ppm", ciofmPrefix.c_str(), frame));
00188 Image< PixRGB<byte> > ciofs = Raster::ReadRGB
00189 (folder+sformat("%s_%06d.ppm", ciofsPrefix.c_str(), frame));
00190
00191 Image< PixRGB<byte> > salImg(640,720,ZEROS);
00192 inplacePaste(salImg, zoomXY(prevImage,2), Point2D<int>(0,0));
00193 inplacePaste(salImg, ciofm, Point2D<int>(0 ,480));
00194 inplacePaste(salImg, ciofs, Point2D<int>(320,480));
00195
00196 drawLine(salImg, Point2D<int>(0,480), Point2D<int>(640,480),
00197 PixRGB<byte>(255,255,255), 3);
00198 drawLine(salImg, Point2D<int>(320,480), Point2D<int>(320,720),
00199 PixRGB<byte>(255,255,255), 3);
00200
00201 win->drawImage(salImg,0,0);
00202 Raster::waitForKey();
00203
00204 Raster::WriteRGB
00205 (salImg, sformat("%s_res_%06d.ppm", pprefix.c_str(), frame));
00206
00207 Raster::waitForKey();
00208
00209
00210
00211
00212
00213
00214
00215
00216
00217
00218
00219
00220
00221
00222
00223
00224
00225
00226
00227
00228
00229
00230
00231
00232
00233
00234
00235
00236
00237
00238
00239
00240
00241
00242
00243
00244
00245
00246
00247
00248
00249
00250
00251
00252
00253
00254
00255
00256
00257
00258
00259
00260
00261
00262
00263
00264
00265
00266
00267
00268
00269
00270
00271
00272
00273
00274
00275
00276
00277
00278
00279
00280
00281
00282
00283
00284
00285
00286
00287
00288
00289
00290
00291
00292
00293
00294
00295
00296
00297 prevImage = currImage;
00298 prevLum = lum;
00299 frame++;
00300
00301
00302
00303
00304
00305
00306
00307 }
00308
00309
00310
00311
00312
00313 manager.stop();
00314
00315
00316 return 0;
00317
00318 }
00319
00320
00321 std::vector<std::vector<Point2D<int> > > getGTandData()
00322 {
00323 uint nSystems = 3;
00324 std::vector<std::vector<Point2D<int> > > data(nSystems+1);
00325
00326 std::string folder("/lab/tmpib/u/siagian/neuroscience/Data/FOE/");
00327
00328
00329
00330
00331
00332
00333
00334
00335
00336
00337
00338
00339
00340
00341 std::string gt("driving_nat_3_1.txt");
00342 std::string f1("DARPA_nv2_2011/Germany_K_CIOFs/ST_PS_Germany.txt");
00343 std::string f2("DARPA_nv2_2011/Germany_K_CIOFs/BMG_Germany.txt");
00344 std::string f3("DARPA_nv2_2011/Germany_K_CIOFs/LK_CB_Germany.txt");
00345
00346
00347
00348 data[0] = getGT(folder+gt);
00349 data[1] = getGT(folder+f1);
00350 data[2] = getGT(folder+f2);
00351 data[3] = getGT(folder+f3);
00352
00353
00354 for(uint i = 1; i < nSystems+1; i++)
00355 {
00356 float totalErr = 0.0;
00357 uint totalPt = 0;
00358 for(uint j = 0; j < data[i].size(); j++)
00359 {
00360
00361
00362 if(data[i][j].i == -1 && data[i][j].j == -1)
00363 continue;
00364
00365
00366
00367 Point2D<int> newPt = data[i][j];
00368 if(newPt.i < 0) newPt.i = 0;
00369 if(newPt.i >= 320) newPt.i = 319;
00370 if(newPt.j < 0) newPt.j = 0;
00371 if(newPt.j >= 320) newPt.j = 319;
00372 data[i][j] = newPt;
00373
00374
00375 float err = newPt.distance(data[0][j]);
00376
00377 LDEBUG("[%3d %3d] - [%3d %3d]: %f",
00378 newPt.i, newPt.j, data[0][j].i, data[0][j].j,
00379 err);
00380
00381 totalErr += err;
00382 totalPt++;
00383 }
00384
00385 if(totalPt == 0) LINFO("s[%3d] err: 0/0 = 0", i);
00386 else
00387 LINFO("s[%3d] err: %f/%d = %f",
00388 i, totalErr, totalPt, totalErr/totalPt);
00389 }
00390 return data;
00391 }
00392
00393
00394 std::vector<Point2D<int> > getGT(std::string gtFilename)
00395 {
00396 std::vector<Point2D<int> > gt;
00397
00398 FILE *fp; char inLine[200];
00399
00400 LINFO("ground truth file: %s",gtFilename.c_str());
00401 if((fp = fopen(gtFilename.c_str(),"rb")) == NULL)
00402 { LINFO("not found"); return gt; }
00403
00404
00405 uint nFrames = 0;
00406 while(fgets(inLine, 200, fp) != NULL)
00407 {
00408 int x, y;
00409 sscanf(inLine, "%d %d", &x, &y);
00410 LINFO("[%3d] x: %d y: %d", nFrames, x, y);
00411 gt.push_back(Point2D<int>(x,y));
00412
00413 nFrames++;
00414 }
00415 Raster::waitForKey();
00416 return gt;
00417 }
00418
00419
00420 Image<float> getOpticFlow2(Image<float> image1, Image<float> image2)
00421 {
00422
00423
00424 inplaceNormalize(image1, 0.0F, 255.0F);
00425 Image<byte> img1(image1);
00426 IplImage* imgA = img2ipl(img1);
00427 inplaceNormalize(image2, 0.0F, 255.0F);
00428 Image<byte> img2(image2);
00429 IplImage* imgB = img2ipl(img2);
00430
00431 CvSize img_sz = cvGetSize( imgA );
00432 int win_size = 15;
00433
00434 IplImage* imgC =img2ipl(img1);
00435
00436
00437 IplImage* eig_image = cvCreateImage( img_sz, IPL_DEPTH_32F, 1 );
00438 IplImage* tmp_image = cvCreateImage( img_sz, IPL_DEPTH_32F, 1 );
00439
00440 int corner_count = MAX_CORNERS;
00441 CvPoint2D32f* cornersA = new CvPoint2D32f[ MAX_CORNERS ];
00442
00443 cvGoodFeaturesToTrack( imgA, eig_image, tmp_image,
00444 cornersA, &corner_count,
00445 0.05, 5.0, 0, 3, 0, 0.04 );
00446
00447
00448 cvFindCornerSubPix( imgA, cornersA, corner_count,
00449 cvSize( win_size, win_size ),
00450 cvSize( -1, -1 ),
00451 cvTermCriteria( CV_TERMCRIT_ITER |
00452 CV_TERMCRIT_EPS, 20, 0.03 ) );
00453
00454
00455 char features_found[ MAX_CORNERS ];
00456 float feature_errors[ MAX_CORNERS ];
00457
00458 CvSize pyr_sz = cvSize( imgA->width+8, imgB->height/3 );
00459
00460 IplImage* pyrA = cvCreateImage( pyr_sz, IPL_DEPTH_32F, 1 );
00461 IplImage* pyrB = cvCreateImage( pyr_sz, IPL_DEPTH_32F, 1 );
00462
00463 CvPoint2D32f* cornersB = new CvPoint2D32f[ MAX_CORNERS ];
00464
00465 cvCalcOpticalFlowPyrLK( imgA, imgB, pyrA, pyrB,
00466 cornersA, cornersB, corner_count,
00467 cvSize( win_size, win_size ), 5,
00468 features_found, feature_errors,
00469 cvTermCriteria( CV_TERMCRIT_ITER |
00470 CV_TERMCRIT_EPS, 20, 0.3 ), 0 );
00471
00472
00473
00474 for( int i=0; i < MAX_CORNERS; i++ )
00475 {
00476 LINFO("Error is %f/n", feature_errors[i]);
00477
00478
00479
00480 CvPoint p0 = cvPoint( cvRound( cornersA[i].x ),
00481 cvRound( cornersA[i].y ) );
00482 CvPoint p1 = cvPoint( cvRound( cornersB[i].x ),
00483 cvRound( cornersB[i].y ) );
00484 cvLine( imgC, p0, p1, CV_RGB(255,0,0), 2 );
00485 }
00486
00487 return ipl2gray(imgC);
00488 }
00489
00490
00491 Image<byte> calculateShift(Image<byte> prevLum, Image<byte> lum,
00492 nub::ref<OutputFrameSeries> ofs)
00493 {
00494 VisualObjectMatchAlgo voma(VOMA_SIMPLE);
00495
00496
00497
00498
00499
00500
00501 rutz::shared_ptr<VisualObject> vo1(new VisualObject("plum", "", prevLum));
00502 rutz::shared_ptr<VisualObject> vo2(new VisualObject("lum" , "", lum));
00503
00504
00505 VisualObjectMatch match(vo1, vo2, voma);
00506 LDEBUG("Found %u matches", match.size());
00507
00508
00509 uint np = match.prune();
00510 LDEBUG("Pruned %u outlier matches.", np);
00511
00512
00513 SIFTaffine s = match.getSIFTaffine();
00514 LDEBUG("[tstX] [ %- .3f %- .3f ] [refX] [%- .3f]", s.m1, s.m2, s.tx);
00515 LDEBUG("[tstY] = [ %- .3f %- .3f ] [refY] + [%- .3f]", s.m3, s.m4, s.ty);
00516
00517
00518 LDEBUG("getKeypointAvgDist = %f", match.getKeypointAvgDist());
00519 LDEBUG("getAffineAvgDist = %f", match.getAffineAvgDist());
00520 LDEBUG("getScore = %f", match.getScore());
00521
00522 if (match.checkSIFTaffine() == false)
00523 LINFO("### Affine is too weird -- BOGUS MATCH");
00524
00525
00526 Image< PixRGB<byte> > mimg = match.getMatchImage(1.0F);
00527 Image< PixRGB<byte> > fimg = match.getFusedImage(0.25F);
00528
00529
00530
00531
00532
00533
00534
00535
00536
00537
00538
00539 LINFO("Shift: %f %f", s.tx, s.ty);
00540
00541
00542
00543
00544 Image<byte> res = shiftImage(s, prevLum, lum);
00545
00546
00547
00548
00549
00550
00551
00552 return res;
00553 }
00554
00555
00556 Image<byte> shiftImage(SIFTaffine aff, Image<byte> refi, Image<byte> tsti)
00557 {
00558
00559
00560
00561
00562 uint w = refi.getWidth(), h = refi.getHeight();
00563 Image<byte> result(w, h, ZEROS);
00564 Image<byte>::const_iterator rptr = refi.begin();
00565 Image<byte>::iterator dptr = result.beginw();
00566
00567 for (uint j = 0; j < h; j ++)
00568 for (uint i = 0; i < w; i ++)
00569 {
00570 float u, v;
00571 aff.transform(float(i), float(j), u, v);
00572 byte rval = *rptr++;
00573
00574 if (tsti.coordsOk(u, v))
00575 {
00576 byte tval = tsti.getValInterp(u, v);
00577
00578 *dptr++ = tval;
00579 }
00580 else
00581
00582 *dptr++ = byte(rval);
00583 }
00584
00585 return result;
00586 }
00587
00588
00589
00590
00591
00592