#include <mil.h>
#if M_MIL_USE_LINUX
#define SAVE_PATH MIL_TEXT("")
#else
#define SAVE_PATH MIL_TEXT("")
#endif
void PrintHeader()
{
MosPrintf(MIL_TEXT("[EXAMPLE NAME]\n")
MIL_TEXT("CameraOnRobotArmCalibration\n\n")
MIL_TEXT("[SYNOPSIS]\n")
MIL_TEXT("This example shows how to calibrate a 3D robotics setup where a camera is\n")
MIL_TEXT("attached to a robotic arm. The calibration module is used to:\n")
MIL_TEXT(" - Calibrate the camera.\n")
MIL_TEXT(" - Find the pose of the camera coordinate system with respect to the\n")
MIL_TEXT(" robot tool coordinate system.\n")
MIL_TEXT(" - Find the pose of the robot base coordinate system with respect to the\n")
MIL_TEXT(" absolute coordinate system.\n\n")
MIL_TEXT("[MODULES USED]\n")
MIL_TEXT("Modules used: Application, System, Display, Buffer, Calibration, 3D Display\n")
MIL_TEXT("and 3D Graphics.\n\n")
MIL_TEXT("Press <Enter> to start.\n\n"));
MosGetch();
}
struct SCalibrationData
{
MIL_CONST_TEXT_PTR m_ImageFile;
MIL_DOUBLE m_GridCornerHintX;
MIL_DOUBLE m_GridCornerHintY;
MIL_DOUBLE m_ToolPositionX;
MIL_DOUBLE m_ToolPositionY;
MIL_DOUBLE m_ToolPositionZ;
MIL_DOUBLE m_ToolRotationX;
MIL_DOUBLE m_ToolRotationY;
MIL_DOUBLE m_ToolRotationZ;
};
#define EXAMPLE_IMAGE_PATH M_IMAGE_PATH MIL_TEXT("CameraOnRobotArmCalibration/")
static MIL_CONST_TEXT_PTR const OUTPUT_CALIBRATION_FILE = SAVE_PATH MIL_TEXT("MilRobotCalibration.mca");
static const MIL_INT NB_CALIBRATION_IMAGES = 7;
static const SCalibrationData CALIBRATION_DATA[NB_CALIBRATION_IMAGES] =
{
{EXAMPLE_IMAGE_PATH MIL_TEXT("CalGrid0.mim"), M_NONE, M_NONE, -29.999479, 700.000122, 510.000092, 174.405594, 28.591669, 91.206627},
{EXAMPLE_IMAGE_PATH MIL_TEXT("CalGrid1.mim"), M_NONE, M_NONE, -51.989830, 599.020020, 505.920288, 173.120300, 20.788210, 95.883133},
{EXAMPLE_IMAGE_PATH MIL_TEXT("CalGrid2.mim"), M_NONE, M_NONE, 118.010101, 629.020020, 515.919983, -169.119003, 24.478680, 79.661667},
{EXAMPLE_IMAGE_PATH MIL_TEXT("CalGrid3.mim"), M_NONE, M_NONE, 118.009903, 719.020020, 505.920105, -167.463898, 31.302469, 85.128510},
{EXAMPLE_IMAGE_PATH MIL_TEXT("CalGrid4.mim"), M_NONE, M_NONE, -11.990170, 519.020081, 415.920013, 179.393494, 16.471180, 91.697990},
{EXAMPLE_IMAGE_PATH MIL_TEXT("CalGrid5.mim"), M_NONE, M_NONE, -29.999969, 399.999786, 509.999786, 175.392303, 0.067751, 97.142853},
{EXAMPLE_IMAGE_PATH MIL_TEXT("CalGrid6.mim"), M_NONE, M_NONE, -130.000000, 399.999786, 510.000000, 164.944305, 7.392438, 115.798599}
};
static const SCalibrationData TEST_DATA =
{EXAMPLE_IMAGE_PATH MIL_TEXT("TestGrid.mim"), M_NONE, M_NONE, 18.009741, 629.019775, 505.919891, 178.899307, 22.548679, 88.419952};
static const MIL_INT ROW_NUMBER = 20;
static const MIL_INT COLUMN_NUMBER = 20;
static const MIL_DOUBLE ROW_SPACING = 10.05;
static const MIL_DOUBLE COLUMN_SPACING = 10.00;
static const MIL_INT GRID_TYPE = M_CHESSBOARD_GRID;
static const MIL_DOUBLE PIXEL_COLOR = M_COLOR_GREEN;
static const MIL_DOUBLE WORLD_COLOR = M_COLOR_RED;
static const MIL_INT M3D_DISPLAY_POSITION_X = 600;
static MIL_CONST_TEXT_PTR const SEPARATOR = MIL_TEXT("--------------------\n\n");
void MoveRobotPose(MIL_ID MilCalibration, const SCalibrationData& rData);
void AddCalibrationGrid(MIL_ID MilDisplay,
MIL_ID MilCalibration,
MIL_ID MilDisplayImage,
MIL_ID MilOverlayImage,
MIL_INT ImageIndex);
void ShowCalibrationResults(MIL_ID MilDisplay, MIL_ID MilCalibration, MIL_ID MilDisplayImage, MIL_ID MilOverlayImage);
void TestCalibration(MIL_ID MilDisplay,
MIL_ID MilCalibration,
MIL_ID MilDisplayImage,
MIL_ID MilOverlayImage);
MIL_ID Alloc3dDisplayId(MIL_ID MilSystem);
bool CheckForRequiredMILFile(MIL_CONST_TEXT_PTR FileName);
int MosMain(void)
{
PrintHeader();
MIL_ID MilApplication = MappAlloc(M_NULL, M_DEFAULT, M_NULL);
if(!CheckForRequiredMILFile(TEST_DATA.m_ImageFile))
{
MappFree(MilApplication);
return -1;
}
MIL_ID MilSystem = MsysAlloc(M_DEFAULT, M_SYSTEM_HOST, M_DEFAULT, M_DEFAULT, M_NULL);
MIL_ID MilDisplay = MdispAlloc(MilSystem, M_DEFAULT, MIL_TEXT("M_DEFAULT"), M_WINDOWED, M_NULL);
MIL_ID MilCalibration = McalAlloc(MilSystem, M_3D_ROBOTICS, M_DEFAULT, M_NULL);
MIL_INT SizeX = MbufDiskInquire(TEST_DATA.m_ImageFile, M_SIZE_X, M_NULL);
MIL_INT SizeY = MbufDiskInquire(TEST_DATA.m_ImageFile, M_SIZE_Y, M_NULL);
MIL_ID MilDisplayImage = MbufAlloc2d(MilSystem, SizeX, SizeY, 8+M_UNSIGNED, M_IMAGE+M_PROC+M_DISP, M_NULL);
MbufClear(MilDisplayImage, 0.0);
MdispSelect(MilDisplay, MilDisplayImage);
MIL_ID MilOverlayImage;
MdispControl(MilDisplay, M_OVERLAY, M_ENABLE);
MdispInquire(MilDisplay, M_OVERLAY_ID, &MilOverlayImage);
for (MIL_INT ImageIndex = 0; ImageIndex < NB_CALIBRATION_IMAGES; ++ImageIndex)
{
MosPrintf(MIL_TEXT("The robot arm is at pose #%d.\n\n"), (int)ImageIndex);
MoveRobotPose(MilCalibration, CALIBRATION_DATA[ImageIndex]);
AddCalibrationGrid(MilDisplay, MilCalibration, MilDisplayImage, MilOverlayImage, ImageIndex);
MosPrintf(SEPARATOR);
}
MosPrintf(MIL_TEXT("The 3D robotics calibration is performed using all the accumulated data.\n"));
McalGrid(MilCalibration, M_NULL, M_NULL, M_NULL, M_NULL, M_NULL, M_NULL, M_NULL, M_NULL, M_DEFAULT, M_DEFAULT);
MIL_INT CalibrationStatus = McalInquire(MilCalibration, M_CALIBRATION_STATUS, M_NULL);
if (CalibrationStatus == M_CALIBRATED)
{
MosPrintf(MIL_TEXT("The calibration is successful.\n\n"));
ShowCalibrationResults(MilDisplay, MilCalibration, MilDisplayImage, MilOverlayImage);
MosPrintf(SEPARATOR);
TestCalibration(MilDisplay, MilCalibration, MilDisplayImage, MilOverlayImage);
}
else
{
MosPrintf(MIL_TEXT("The calibration failed.\n\n")
MIL_TEXT("Press <Enter> to exit.\n\n"));
MosGetch();
}
MbufFree(MilDisplayImage);
McalFree(MilCalibration);
MdispFree(MilDisplay);
MsysFree(MilSystem);
MappFree(MilApplication);
return 0;
}
void MoveRobotPose(MIL_ID MilCalibration, const SCalibrationData& rData)
{
McalSetCoordinateSystem(MilCalibration,
M_TOOL_COORDINATE_SYSTEM,
M_ROBOT_BASE_COORDINATE_SYSTEM,
M_TRANSLATION+M_ASSIGN,
M_NULL,
rData.m_ToolPositionX,
rData.m_ToolPositionY,
rData.m_ToolPositionZ,
M_DEFAULT);
McalSetCoordinateSystem(MilCalibration,
M_TOOL_COORDINATE_SYSTEM,
M_TOOL_COORDINATE_SYSTEM,
M_ROTATION_ZYX+M_COMPOSE_WITH_CURRENT,
M_NULL,
rData.m_ToolRotationZ,
rData.m_ToolRotationY,
rData.m_ToolRotationX,
M_DEFAULT);
}
void AddCalibrationGrid(MIL_ID MilDisplay,
MIL_ID MilCalibration,
MIL_ID MilDisplayImage,
MIL_ID MilOverlayImage,
MIL_INT ImageIndex)
{
MosPrintf(MIL_TEXT("An image of the calibration grid is taken at that position, and used for\n")
MIL_TEXT("calibration.\n\n")
MIL_TEXT("Calling McalGrid(): "));
const SCalibrationData& rData = CALIBRATION_DATA[ImageIndex];
MbufLoad(rData.m_ImageFile, MilDisplayImage);
McalControl(MilCalibration, M_GRID_HINT_PIXEL_X, rData.m_GridCornerHintX);
McalControl(MilCalibration, M_GRID_HINT_PIXEL_Y, rData.m_GridCornerHintY);
McalGrid(MilCalibration, MilDisplayImage,
0.0, 0.0, 0.0,
ROW_NUMBER, COLUMN_NUMBER,
ROW_SPACING, COLUMN_SPACING,
M_ACCUMULATE, GRID_TYPE);
MIL_INT CalibrationStatus = McalInquire(MilCalibration, M_CALIBRATION_STATUS, M_NULL);
if (CalibrationStatus == M_CALIBRATING)
{
MgraColor(M_DEFAULT, PIXEL_COLOR);
McalDraw(M_DEFAULT, MilCalibration, MilOverlayImage, M_DRAW_IMAGE_POINTS, ImageIndex, M_DEFAULT);
MosPrintf(MIL_TEXT("extracted features are displayed in green.\n")
MIL_TEXT("Press <Enter> to continue.\n\n"));
MosGetch();
MdispControl(MilDisplay, M_OVERLAY_CLEAR, M_DEFAULT);
}
else
{
MosPrintf(MIL_TEXT("the grid was not found.\n")
MIL_TEXT("Press <Enter> to continue.\n\n"));
MosGetch();
}
}
void ShowCalibrationResults(MIL_ID MilDisplay, MIL_ID MilCalibration, MIL_ID MilDisplayImage, MIL_ID MilOverlayImage)
{
MIL_DOUBLE AveragePixelError, MaximumPixelError, AverageWorldError, MaximumWorldError;
McalInquire( MilCalibration, M_GLOBAL_AVERAGE_PIXEL_ERROR, &AveragePixelError );
McalInquire( MilCalibration, M_GLOBAL_MAXIMUM_PIXEL_ERROR, &MaximumPixelError );
McalInquire( MilCalibration, M_GLOBAL_AVERAGE_WORLD_ERROR, &AverageWorldError );
McalInquire( MilCalibration, M_GLOBAL_MAXIMUM_WORLD_ERROR, &MaximumWorldError );
MosPrintf(MIL_TEXT("Global pixel error\n average: %.3g pixels\n maximum: %.3g pixels\n"),
AveragePixelError, MaximumPixelError);
MosPrintf(MIL_TEXT("Global world error\n average: %.3g mm\n maximum: %.3g mm\n\n"),
AverageWorldError, MaximumWorldError);
McalSave(OUTPUT_CALIBRATION_FILE, MilCalibration, M_DEFAULT);
MosPrintf(MIL_TEXT("The calibration context was saved as '"));
MosPrintf(OUTPUT_CALIBRATION_FILE);
MosPrintf(MIL_TEXT("'.\nPress <Enter> to verify the calibration accuracy for each pose.\n\n"));
MosGetch();
MosPrintf(SEPARATOR);
MIL_ID MilDisplay3d = Alloc3dDisplayId(M_DEFAULT_HOST);
MIL_INT64 DrawLabel = 0;
MIL_ID MilGraphicList = M_NULL;
MIL_UNIQUE_CAL_ID MilDrawContextId;
if(MilDisplay3d)
{
M3ddispControl(MilDisplay3d,M_WINDOW_INITIAL_POSITION_X, M3D_DISPLAY_POSITION_X);
M3ddispSetView(MilDisplay3d, M_AZIM_ELEV_ROLL, 120.0, 220.0, 0.0, M_DEFAULT);
M3ddispInquire(MilDisplay3d, M_3D_GRAPHIC_LIST_ID, &MilGraphicList);
M3dgraControl(MilGraphicList, M_DEFAULT_SETTINGS, M_FONT_SIZE, 18);
MilDrawContextId = McalAlloc(M_DEFAULT_HOST, M_DRAW_3D_CONTEXT, M_DEFAULT, M_UNIQUE_ID);
McalControl(MilDrawContextId, M_DRAW_RELATIVE_XY_PLANE, M_ENABLE);
McalControl(MilDrawContextId, M_DRAW_RELATIVE_XY_PLANE_COLOR_FILL, M_TEXTURE_IMAGE);
McalControl(MilDrawContextId, M_DRAW_RELATIVE_XY_PLANE_COLOR_OUTLINE, M_COLOR_WHITE);
McalControl(MilDrawContextId, M_DRAW_RELATIVE_COORDINATE_SYSTEM, M_DISABLE);
McalControl(MilDrawContextId, M_DRAW_TOOL_COORDINATE_SYSTEM, M_ENABLE);
MIL_INT64 MilGrid = M3dgraGrid(MilGraphicList, M_ROOT_NODE, M_SIZE_AND_SPACING, M_DEFAULT, 1000, 1000, 50, 50, M_DEFAULT);
M3dgraControl(MilGraphicList, MilGrid, M_OPACITY, 10);
}
for (MIL_INT ImageIndex = 0; ImageIndex < NB_CALIBRATION_IMAGES; ++ImageIndex)
{
MosPrintf(MIL_TEXT("Pose #%d\n"), (int)ImageIndex);
MosPrintf(MIL_TEXT("-------\n\n"));
MbufLoad(CALIBRATION_DATA[ImageIndex].m_ImageFile, MilDisplayImage);
MoveRobotPose(MilCalibration, CALIBRATION_DATA[ImageIndex]);
McalAssociate(MilCalibration, MilDisplayImage, M_DEFAULT);
McalInquireSingle( MilCalibration, ImageIndex, M_AVERAGE_PIXEL_ERROR, &AveragePixelError );
McalInquireSingle( MilCalibration, ImageIndex, M_MAXIMUM_PIXEL_ERROR, &MaximumPixelError );
McalInquireSingle( MilCalibration, ImageIndex, M_AVERAGE_WORLD_ERROR, &AverageWorldError );
McalInquireSingle( MilCalibration, ImageIndex, M_MAXIMUM_WORLD_ERROR, &MaximumWorldError );
MosPrintf(MIL_TEXT("Pixel error\n average: %.3g pixels\n maximum: %.3g pixels\n"),
AveragePixelError, MaximumPixelError);
MosPrintf(MIL_TEXT("World error\n average: %.3g mm\n maximum: %.3g mm\n\n"),
AverageWorldError, MaximumWorldError);
MgraColor(M_DEFAULT, PIXEL_COLOR);
McalDraw(M_DEFAULT, MilCalibration, MilOverlayImage, M_DRAW_IMAGE_POINTS, ImageIndex, M_DEFAULT);
MgraColor(M_DEFAULT, WORLD_COLOR);
McalDraw(M_DEFAULT, MilCalibration, MilOverlayImage, M_DRAW_WORLD_POINTS, ImageIndex, M_DEFAULT);
MosPrintf(MIL_TEXT("Green: extracted features (pixels).\n")
MIL_TEXT("Red: world points converted to pixels using the calibration context.\n\n"));
if(MilDisplay3d)
{
if(DrawLabel != 0)
M3dgraRemove(MilGraphicList, DrawLabel, M_DEFAULT);
DrawLabel = McalDraw3d(MilDrawContextId, MilDisplayImage, M_DEFAULT, MilGraphicList, M_DEFAULT, MilDisplayImage, M_DEFAULT);
MosPrintf(MIL_TEXT("The 3D display shows the coordinate systems for calibration pose #%d.\n\n"), (int)ImageIndex);
if(ImageIndex == 0)
{
M3ddispSelect(MilDisplay3d, M_NULL, M_OPEN, M_DEFAULT);
M3ddispSetView(MilDisplay3d, M_ZOOM, 2, M_DEFAULT, M_DEFAULT, M_DEFAULT);
}
}
MosPrintf(MIL_TEXT("Press <Enter> to continue.\n\n"));
MosGetch();
MdispControl(MilDisplay, M_OVERLAY_CLEAR, M_DEFAULT);
}
if(MilDisplay3d)
{ M3ddispFree(MilDisplay3d); }
}
void TestCalibration(MIL_ID MilDisplay,
MIL_ID MilCalibration,
MIL_ID MilDisplayImage,
MIL_ID MilOverlayImage)
{
MoveRobotPose(MilCalibration, TEST_DATA);
MbufLoad(TEST_DATA.m_ImageFile, MilDisplayImage);
MgraColor(M_DEFAULT, WORLD_COLOR);
McalDraw(M_DEFAULT, MilCalibration, MilOverlayImage, M_DRAW_WORLD_POINTS, M_DEFAULT, M_DEFAULT);
MosPrintf(MIL_TEXT("The robot arm is moved to a new position. McalSetCoordinateSystem() is used\n")
MIL_TEXT("to provide the new tool pose to the calibration module, thus the system\n")
MIL_TEXT("remains fully calibrated.\n\n")
MIL_TEXT("The calibration grid is grabbed at the new camera position. This grabbed\n")
MIL_TEXT("image was not used during calibration. The corners of the grid are not\n")
MIL_TEXT("extracted from this image; no image processing is performed.\n\n")
MIL_TEXT("Instead, the world points of the calibration grid are converted to pixels,\n")
MIL_TEXT("using the calibration module with the new tool pose. Since these points\n")
MIL_TEXT("(displayed in red) coincide with the corners of the grabbed grid, the\n")
MIL_TEXT("calibration is accurate.\n\n")
MIL_TEXT("Press <Enter> to exit.\n\n"));
MosGetch();
MdispControl(MilDisplay, M_OVERLAY_CLEAR, M_DEFAULT);
}
MIL_ID Alloc3dDisplayId(MIL_ID MilSystem)
{
MappControl(M_DEFAULT, M_ERROR, M_PRINT_DISABLE);
MIL_ID MilDisplay3D = M3ddispAlloc(MilSystem, M_DEFAULT, MIL_TEXT("M_DEFAULT"), M_DEFAULT, M_NULL);
MappControl(M_DEFAULT, M_ERROR, M_PRINT_ENABLE);
if(!MilDisplay3D)
{
MosPrintf(MIL_TEXT("\n")
MIL_TEXT("The current system does not support the 3D display.\n\n"));
}
return MilDisplay3D;
}
bool CheckForRequiredMILFile(MIL_CONST_TEXT_PTR FileName)
{
MIL_INT FilePresent;
MappFileOperation(M_DEFAULT, FileName, M_NULL, M_NULL, M_FILE_EXISTS, M_DEFAULT, &FilePresent);
if(FilePresent == M_NO)
{
MosPrintf(MIL_TEXT("\n")
MIL_TEXT("The files needed to run this example are missing. You need \n")
MIL_TEXT("to obtain and apply a separate specific update to have it.\n\n"));
MosPrintf(MIL_TEXT("Press <Enter> to end.\n\n"));
MosGetch();
}
return (FilePresent == M_YES);
}