Performance Considerations
Reliability in Exposure Time
Zivid cameras queue the exposures from each acquisition back to back when capturing using HDR mode. The camera is done capturing the scene when the projector has finished flashing the patterns. This time is called the Acquisition Time and it can be found in the Information panel in Zivid Studio. Knowing this time allows you to reliably estimate when the capturing of the scene is done and the scene can be changed. This can be, for example, either moving the robot or moving the objects in the scene.
경고
Be aware that using the Zivid camera at a high duty cycle can trigger thermal throttling to prevent overheating. For more on this topic check out the projector brightness page.
Which frame member functions returns when?
캡처 후에도 포인트 클라우드 처리는 백그라운드에서 계속 실행됩니다.
const auto frame = camera.capture(settings);
var frame = camera.Capture(settings);
frame = camera.capture(settings)
다음 Zivid::Frame
멤버 함수는 완료될 포인트 클라우드 처리에 의존하지 않으므로 즉시 반환됩니다.
const auto frameSettings = frame.settings();
const auto cameraInfo = frame.cameraInfo();
var frameSettings = frame.Settings;
var cameraInfo = frame.CameraInfo;
frame_settings = frame.settings
camera_info = frame.camera_info
반면 다음 API는 포인트 클라우드 데이터를 사용할 수 있을 때까지 차단하고 대기합니다.
const auto pointCloud = frame.pointCloud();
const auto frameInfo = frame.info();
const auto cameraState = frame.state();
frame.save("Frame.zdf");
var pointCloud = frame.PointCloud;
var frameInfo = frame.Info;
var cameraState = frame.State;
frame.Save("Frame.zdf");
point_cloud = frame.point_cloud()
frame_info = frame.info
camera_state = frame.state
frame.save("Frame.zdf")
Optimizing Robot Cycle Times
로봇에 장착된 카메라가 있는 픽 앤 플레이스 애플리케이션의 경우 로봇이 물체를 픽한 후 캡처하는 것이 좋습니다. 이 전략을 사용하면 로봇은 획득 시간 동안에만 정지하면 됩니다. 획득이 완료되면 즉, 캡처 API가 반환되는 즉시 로봇이 물체를 플레이싱을 시작할 수 있습니다. 병렬로 다음 주기에 대한 픽 포즈를 얻기 위한 포인트 클라우드 처리를 시작할 수 있습니다. 아래 구현 예를 확인하십시오.
참고
속도를 최적화하려면 캡처 기능이 반환된 후 API를 호출하여 포인트 클라우드를 가져오기 전에 로봇을 이동하는 것이 중요합니다.
#include <Zivid/Zivid.h>
#include <future>
#include <iostream>
namespace
{
struct DummyRobot
{
void Move(Zivid::Matrix4x4 pose, std::chrono::milliseconds duration)
{
std::this_thread::sleep_for(duration);
}
};
DummyRobot robot;
Zivid::Matrix4x4 computePickPose(const Zivid::Frame &frame)
{
const auto pointCloud = frame.pointCloud();
const auto data = pointCloud.copyData<Zivid::PointXYZColorRGBA>();
// This is where you should run your processing to get a pick pose from the point cloud
return Zivid::Matrix4x4{};
}
void placeObject()
{
const auto placePose = Zivid::Matrix4x4{};
robot.Move(placePose, std::chrono::milliseconds(2000));
const auto outOfCameraFOVPose = Zivid::Matrix4x4{};
robot.Move(outOfCameraFOVPose, std::chrono::milliseconds(1000));
}
} // namespace
int main()
{
try
{
Zivid::Application zivid;
auto camera = zivid.connectCamera();
std::cout << "Moving the robot mounted camera to the capture pose" << std::endl;
const auto capturePose = Zivid::Matrix4x4{};
robot.Move(capturePose, std::chrono::milliseconds(2000));
std::cout << "Capturing frame" << std::endl;
const auto settings = Zivid::Settings{ Zivid::Settings::Acquisitions{ Zivid::Settings::Acquisition{} } };
const auto frame = camera.capture(settings);
std::cout << "Processing to get the first pick pose" << std::endl;
auto pickPose = computePickPose(frame);
while(true)
{
std::cout << "Picking the new object" << std::endl;
robot.Move(pickPose, std::chrono::milliseconds(2000));
std::cout << "Object picked" << std::endl;
std::cout << "Moving the robot mounted camera to the capture pose" << std::endl;
robot.Move(capturePose, std::chrono::milliseconds(2000));
// At this moment the robot should be at the capture pose
std::cout << "Capturing frame" << std::endl;
const auto settings = Zivid::Settings{ Zivid::Settings::Acquisitions{ Zivid::Settings::Acquisition{} } };
const auto frame = camera.capture(settings);
std::cout << "Acquisition completed" << std::endl;
std::cout
<< "Starting, at the same time (in two threads), robot motion to place the object and capturing and processing to get the new pick pose"
<< std::endl;
auto futurePickPose = std::async(std::launch::async, computePickPose, frame);
auto placingCompleted = std::async(std::launch::async, placeObject);
pickPose =
futurePickPose
.get(); // This (processing to get the new pick pose) should be faster so the robot would not have to stop and wait
placingCompleted.get();
std::cout
<< "Both robot motion to place the object and capturing and processing to get the new pick pose completed"
<< std::endl;
}
}
catch(const std::exception &e)
{
std::cerr << "Error: " << Zivid::toString(e) << std::endl;
return EXIT_FAILURE;
}
}
팁
The same strategy is recommended if you are performing a 2D capture instead of a 3D capture. However, if you need both 2D and 3D, then you should check out our 2D + 3D Capture Strategy.
개체를 플레이스하는 로봇 동작이 다음 주기에 대한 픽킹 포즈를 얻기 위한 포인트 클라우드 처리보다 빠르면 로봇은 기다려야 합니다.
로봇이 멈추는 것을 방지하기 위한 대안은 매 사이클마다 로봇이 이전 사이클에서 계산된 픽 포즈로 이동하는 것입니다. 이 전략을 사용하면 로봇이 물체를 선택하는 동안 데이터를 계속 처리할 수 있습니다. 아래 구현 예를 확인하십시오. 이 전략은 항상 여러 개의 픽 포즈를 추정할 수 있다고 가정합니다. 또한 현재 사이클에서 픽킹할 개체가 이전 픽 사이클 동안 이동하지 않았다고 가정합니다. 현재 픽킹 사이클의 개체가 이전 사이클에서 픽킹된 개체와 다른 빈 영역에 있는지 확인합니다. 이러한 객체 선택은 성공적인 픽킹의 기회를 증가시킬 것입니다.
#include <Zivid/Zivid.h>
#include <future>
#include <iostream>
namespace
{
struct DummyRobot
{
void Move(Zivid::Matrix4x4 pose, std::chrono::milliseconds duration)
{
std::this_thread::sleep_for(duration);
}
};
DummyRobot robot;
Zivid::Array2D<Zivid::PointXYZColorRGBA> getPointCloudInThread(const Zivid::Frame &frame)
{
const auto pointCloud = frame.pointCloud();
const auto data = pointCloud.copyData<Zivid::PointXYZColorRGBA>();
return data;
}
void placeObjectInThread()
{
const auto placePose = Zivid::Matrix4x4{};
robot.Move(placePose, std::chrono::milliseconds(2000));
const auto outOfCameraFOVPose = Zivid::Matrix4x4{};
robot.Move(outOfCameraFOVPose, std::chrono::milliseconds(1000));
}
Zivid::Matrix4x4 computePickPoseInThread(const Zivid::Array2D<Zivid::PointXYZColorRGBA> &data)
{
// This is where you should run your processing to get a pick pose from the point cloud
return Zivid::Matrix4x4{};
}
} // namespace
int main()
{
try
{
Zivid::Application zivid;
auto camera = zivid.connectCamera();
std::cout << "Moving the robot mounted camera to the capture pose" << std::endl;
const auto capturePose = Zivid::Matrix4x4{};
robot.Move(capturePose, std::chrono::milliseconds(2000));
// At this moment the robot should be at the capture pose
std::cout << "Capturing frame" << std::endl;
const auto settings = Zivid::Settings{ Zivid::Settings::Acquisitions{ Zivid::Settings::Acquisition{} } };
const auto frame = camera.capture(settings);
std::cout << "Acquisition completed" << std::endl;
std::cout
<< "Starting, at the same time (in two threads), robot motion to place the object and waiting for the point cloud to be processed and copied"
<< std::endl;
auto futureData = std::async(std::launch::async, getPointCloudInThread, frame);
auto placingCompleted = std::async(std::launch::async, placeObjectInThread);
const auto data =
futureData
.get(); // This (remaining time to process the point cloud) should be faster so the robot would not have to stop and wait
placingCompleted.get();
std::cout << "Both robot motion to place the object and point cloud processing and copying completed"
<< std::endl;
std::cout << "Starting the processing (in a separate thread) to get the pick pose for the next cycle"
<< std::endl;
auto nextPickPose = std::async(std::launch::async, computePickPoseInThread, data);
std::cout
<< "Picking the object from the previous pick cycle (becasue computing the pick new pose is not completed yet at this moment)"
<< std::endl;
const auto previousCyclePickPose = Zivid::Matrix4x4{};
robot.Move(previousCyclePickPose, std::chrono::milliseconds(2000));
std::cout << "Object picked" << std::endl;
}
catch(const std::exception &e)
{
std::cerr << "Error: " << Zivid::toString(e) << std::endl;
return EXIT_FAILURE;
}
}
카메라가 고정식으로 장착된 픽 앤 플레이스 애플리케이션의 경우 로봇이 물체를 선택하고 카메라의 FOV 외부로 이동한 후 캡처하는 것이 좋습니다. 이 전략을 사용하면 다음 작업 중 실행 시간이 더 오래 걸리는 작업에 따라 카메라가 캡처하기 위해 로봇을 중지할 필요가 없습니다.
개체를 감지하고 픽 포즈를 추정하기 위해 데이터를 처리하는 시간과 함께 총 캡처 시간입니다.
카메라 FOV 밖에 있는 포즈에서 플레이스 포즈로, 다시 카메라 FOV에 진입하기 전의 포즈로 로봇 모션.
로봇 동작이 더 오래 지속되면 로봇은 멈추고 기다릴 필요가 없습니다. 그러나 픽 포즈를 얻는 프로세스(캡처 + 감지 + 포즈 추정)를 실행하는 데 시간이 더 오래 걸리면 로봇을 멈춰야 합니다. 아래 구현 예를 확인하십시오.
#include <Zivid/Zivid.h>
#include <future>
#include <iostream>
namespace
{
struct DummyRobot
{
void Move(Zivid::Matrix4x4 pose, std::chrono::milliseconds duration)
{
std::this_thread::sleep_for(duration);
}
};
DummyRobot robot;
Zivid::Matrix4x4 captureAndProcess(Zivid::Camera &camera)
{
const auto settings = Zivid::Settings{ Zivid::Settings::Acquisitions{ Zivid::Settings::Acquisition{} } };
const auto frame = camera.capture(settings);
const auto pointCloud = frame.pointCloud();
const auto data = pointCloud.copyData<Zivid::PointXYZColorRGBA>();
const auto pickPose =
Zivid::Matrix4x4{}; // This is where you should run your processing to get a pick pose from the point cloud
return pickPose;
}
void placeObject()
{
const auto placePose = Zivid::Matrix4x4{};
robot.Move(placePose, std::chrono::milliseconds(2000));
const auto outOfCameraFOV = Zivid::Matrix4x4{};
robot.Move(outOfCameraFOV, std::chrono::milliseconds(1000));
}
} // namespace
int main()
{
try
{
Zivid::Application zivid;
auto camera = zivid.connectCamera();
auto pickPose = captureAndProcess(camera);
while(true)
{
std::cout << "Picking the new object" << std::endl;
robot.Move(pickPose, std::chrono::milliseconds(2000));
std::cout << "Object picked" << std::endl;
std::cout << "Moving the robot with a picked object outside the FOV of the stationary camera" << std::endl;
const auto outOfCameraFOV = Zivid::Matrix4x4{};
robot.Move(outOfCameraFOV, std::chrono::milliseconds(2000));
// At this moment the robot should be outside the camera FOV
std::cout
<< "Starting, at the same time (in two threads), robot motion to place the object and capturing and processing to get the new pick pose"
<< std::endl;
auto futurePickPose = std::async(std::launch::async, captureAndProcess, std::ref(camera));
auto placingCompleted = std::async(std::launch::async, placeObject);
pickPose =
futurePickPose
.get(); // This (capturing and processing to get the new pick pose) should be faster so the robot would not have to stop and wait
placingCompleted.get();
std::cout
<< "Both robot motion to place the object and capturing and processing to get the new pick pose completed"
<< std::endl;
}
}
catch(const std::exception &e)
{
std::cerr << "Error: " << Zivid::toString(e) << std::endl;
return EXIT_FAILURE;
}
}
로봇이 멈추는 것을 방지하기 위한 대안은 매 사이클마다 로봇이 이전 사이클에서 계산된 픽 포즈로 이동하는 것입니다. 이 전략을 사용하면 로봇이 물체를 픽킹하는 동안 데이터를 계속 처리할 수 있습니다. 이 경우 개체를 플레이싱하는 동안 그냥 캡처하거나 캡처하여 포인트 클라우드를 가져와서 CPU 메모리에 복사할 수 있습니다. 아래 구현 예를 확인하십시오. 이 전략은 여러 픽 포즈를 추정하는 것이 항상 가능하다고 가정합니다. 또한 현재 사이클에서 픽킹할 객체가 이전 픽킹 사이클 동안 이동하지 않았다고 가정합니다. 현재 픽킹 사이클의 개체가 이전 사이클에서 픽킹한 개체와 다른 저장소 영역에 있는지 확인합니다. 이러한 개체 선택은 성공적인 픽킹의 기회를 증가시킵니다.
#include <Zivid/Zivid.h>
#include <future>
#include <iostream>
namespace
{
struct DummyRobot
{
void Move(Zivid::Matrix4x4 pose, std::chrono::milliseconds duration)
{
std::this_thread::sleep_for(duration);
}
};
DummyRobot robot;
Zivid::Frame captureInThread(Zivid::Camera &camera)
{
const auto settings = Zivid::Settings{ Zivid::Settings::Acquisitions{ Zivid::Settings::Acquisition{} } };
const auto frame = camera.capture(settings);
return frame;
}
void placeObjectInThread()
{
const auto placePose = Zivid::Matrix4x4{};
robot.Move(placePose, std::chrono::milliseconds(2000));
const auto outOfCameraFOVPose = Zivid::Matrix4x4{};
robot.Move(outOfCameraFOVPose, std::chrono::milliseconds(1000));
}
Zivid::Matrix4x4 computePickPoseInThread(const Zivid::Frame &frame)
{
const auto pointCloud = frame.pointCloud();
const auto data = pointCloud.copyData<Zivid::PointXYZColorRGBA>();
// If you know your robot motion to place the object is slow enough, you can move the above two lines to captureInThread
// This is where you should run your processing to get a pick pose from the point cloud
return Zivid::Matrix4x4{};
}
} // namespace
int main()
{
try
{
Zivid::Application zivid;
auto camera = zivid.connectCamera();
std::cout << "Moving the robot with a picked object outside the FOV of the stationary camera" << std::endl;
const auto outOfCameraFOV = Zivid::Matrix4x4{};
robot.Move(outOfCameraFOV, std::chrono::milliseconds(2000));
// At this moment the robot should be outside the camera FOV
std::cout
<< "Starting, at the same time (in two threads), robot motion to place the object and capturing the point cloud"
<< std::endl;
auto futureFrame = std::async(std::launch::async, captureInThread, std::ref(camera));
auto placingCompleted = std::async(std::launch::async, placeObjectInThread);
const auto frame =
futureFrame
.get(); // This (capturing the point cloud) should be faster so the robot would not have to stop and wait
placingCompleted.get();
std::cout << "Both robot motion to place the object and capturing the point cloud completed" << std::endl;
std::cout << "Starting the processing (in a separate thread) to get the pick pose for the next cycle"
<< std::endl;
auto nextPickPose = std::async(std::launch::async, computePickPoseInThread, frame);
std::cout
<< "Picking the object from the previous pick cycle (becasue computing the pick new pose is not completed yet at this moment)"
<< std::endl;
const auto previousCyclePickPose = Zivid::Matrix4x4{};
robot.Move(previousCyclePickPose, std::chrono::milliseconds(2000));
std::cout << "Object picked" << std::endl;
}
catch(const std::exception &e)
{
std::cerr << "Error: " << Zivid::toString(e) << std::endl;
return EXIT_FAILURE;
}
}
Benefits of using Zivid Point Cloud Manipulation Functions
Zivid SDK에서 포인트 클라우드에 대한 연산(예: Downsample, Transform, Normals)은 포인트 클라우드가 GPU 메모리에 남아 있는 동안 GPU에서 계산됩니다. 따라서 사용자는 연산을 위해 GPU와 CPU 사이를 오가며 데이터를 이동하는 추가 시간을 피할 수 있습니다. 지비드 API로 GPU에서 Downsampling하면 CPU로 복사할 데이터도 줄어듭니다. 따라서 성능상의 이유로 이러한 연산에는 Zivid SDK를 사용하는 것이 유리합니다. 타사 라이브러리로 이러한 포인트 클라우드 연산을 구현하면 시간이 더 많이 소요될 수 있습니다. 평균적으로 CPU에서 연산을 수행하면 속도가 느려집니다. GPU의 다른 소프트웨어에서 동일한 연산을 수행하는 경우 데이터를 GPU에서 CPU로 복사하고 다시 복사해야 합니다.
Version History
SDK |
Changes |
---|---|
2.12.0 |
Zivid One+ has reached its End-of-Life and is no longer supported. |
2.11.0 |
2 and 2+ now support concurrent processing and acquisition, and switching between capture modes has been optimized. |